spring cloud是一系列框架的有序集合。它利用spring boot的開發便利性巧妙地簡化了分布式系統基礎設施的開發,如服務發現注冊、配置中心、消息總線、負載均衡、斷路器、數據監控等,都可以用spring boot的開發風格做到一鍵啟動和部署。spring并沒有重復制造輪子,它只是將目前各家公司開發的比較成熟、經得起實際考驗的服務框架組合起來,通過spring boot風格進行再封裝屏蔽掉了復雜的配置和實現原理,最終給開發者留出了一套簡單易懂、易部署和易維護的分布式系統開發工具包。
接下來我們就使用springcloud實現一套簡單的微服務架構。
以下所有代碼都已開源到github上了,地址:https://github.com/lynnlovemin/softservice
eureka(服務注冊與發現)
首先引入相關的依賴包
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
|
<parent> <groupid>org.springframework.boot</groupid> <artifactid>spring-boot-starter-parent</artifactid> <version> 1.5 . 9 .release</version> <relativepath/> <!-- lookup parent from repository --> </parent> <properties> <project.build.sourceencoding>utf- 8 </project.build.sourceencoding> <project.reporting.outputencoding>utf- 8 </project.reporting.outputencoding> <java.version> 1.8 </java.version> </properties> <dependencies> <dependency> <groupid>org.springframework.cloud</groupid> <artifactid>spring-cloud-starter-eureka</artifactid> </dependency> <dependency> <groupid>org.springframework.cloud</groupid> <artifactid>spring-cloud-starter-eureka-server</artifactid> </dependency> <dependency> <groupid>org.springframework.boot</groupid> <artifactid>spring-boot-starter-web</artifactid> </dependency> <dependency> <groupid>org.springframework.boot</groupid> <artifactid>spring-boot-starter-test</artifactid> <scope>test</scope> </dependency> </dependencies> <dependencymanagement> <dependencies> <dependency> <groupid>org.springframework.cloud</groupid> <artifactid>spring-cloud-dependencies</artifactid> <version>dalston.rc1</version> <type>pom</type> <scope> import </scope> </dependency> </dependencies> </dependencymanagement> <build> <plugins> <plugin> <groupid>org.springframework.boot</groupid> <artifactid>spring-boot-maven-plugin</artifactid> </plugin> </plugins> </build> <repositories> <repository> <id>spring-milestones</id> <name>spring milestones</name> <url>https: //repo.spring.io/milestone</url> <snapshots> <enabled> false </enabled> </snapshots> </repository> </repositories> |
配置application.yml
1
2
3
4
5
6
7
8
9
10
|
server: port: 8761 eureka: instance: hostname: localhost client: registerwitheureka: false fetchregistry: false serviceurl: defaultzone: http: //${eureka.instance.hostname}:${server.port}/eureka/ |
創建啟動類application
1
2
3
4
5
6
7
8
|
@enableeurekaserver @springbootapplication public class application { public static void main(string[] args) { springapplication.run(application. class , args); } } |
運行main方法,瀏覽器訪問:http://localhost:8761,我們就能在瀏覽器看到如下界面:
說明eureka啟動成功。
接下來,我們實現負債均衡、斷路器、網關、客戶端,所有的服務都應該注冊到eureka中,并且訪問eureka就能看到所有注冊的服務
client(客戶端)
pom.xml
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
|
<parent> <groupid>org.springframework.boot</groupid> <artifactid>spring-boot-starter-parent</artifactid> <version> 1.5 . 9 .release</version> <relativepath/> <!-- lookup parent from repository --> </parent> <properties> <project.build.sourceencoding>utf- 8 </project.build.sourceencoding> <project.reporting.outputencoding>utf- 8 </project.reporting.outputencoding> <java.version> 1.8 </java.version> </properties> <dependencies> <dependency> <groupid>org.springframework.cloud</groupid> <artifactid>spring-cloud-starter-eureka</artifactid> </dependency> <dependency> <groupid>org.springframework.boot</groupid> <artifactid>spring-boot-starter-web</artifactid> </dependency> <dependency> <groupid>org.springframework.boot</groupid> <artifactid>spring-boot-starter-test</artifactid> <scope>test</scope> </dependency> </dependencies> <dependencymanagement> <dependencies> <dependency> <groupid>org.springframework.cloud</groupid> <artifactid>spring-cloud-dependencies</artifactid> <version>dalston.rc1</version> <type>pom</type> <scope> import </scope> </dependency> </dependencies> </dependencymanagement> <build> <plugins> <plugin> <groupid>org.springframework.boot</groupid> <artifactid>spring-boot-maven-plugin</artifactid> </plugin> </plugins> </build> <repositories> <repository> <id>spring-milestones</id> <name>spring milestones</name> <url>https: //repo.spring.io/milestone</url> <snapshots> <enabled> false </enabled> </snapshots> </repository> </repositories> |
application.yml
1
2
3
4
5
6
7
8
9
|
eureka: client: serviceurl: defaultzone: http: //localhost:8761/eureka/ #這里注冊到eureka中 server: port: 8763 spring: application: name: service-hi |
application類
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
|
@springbootapplication @enableeurekaclient @restcontroller public class application { public static void main(string[] args) { springapplication.run(applicatioin. class , args); } @value ( "${server.port}" ) string port; //這里我們提供一個接口 @requestmapping ( "/hi" ) public string home( @requestparam string name) { return "hi " +name+ ",i am from port:" +port; } } |
feign(負債均衡、斷路器)
pom.xml
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
|
<parent> <groupid>org.springframework.boot</groupid> <artifactid>spring-boot-starter-parent</artifactid> <version> 1.5 . 9 .release</version> <relativepath/> <!-- lookup parent from repository --> </parent> <properties> <project.build.sourceencoding>utf- 8 </project.build.sourceencoding> <project.reporting.outputencoding>utf- 8 </project.reporting.outputencoding> <java.version> 1.8 </java.version> </properties> <dependencies> <dependency> <groupid>org.springframework.cloud</groupid> <artifactid>spring-cloud-starter-eureka</artifactid> </dependency> <dependency> <groupid>org.springframework.boot</groupid> <artifactid>spring-boot-starter-web</artifactid> </dependency> <dependency> <groupid>org.springframework.cloud</groupid> <artifactid>spring-cloud-starter-feign</artifactid> </dependency> <dependency> <groupid>org.springframework.boot</groupid> <artifactid>spring-boot-starter-actuator</artifactid> </dependency> <dependency> <groupid>org.springframework.cloud</groupid> <artifactid>spring-cloud-starter-hystrix-dashboard</artifactid> </dependency> <dependency> <groupid>org.springframework.boot</groupid> <artifactid>spring-boot-starter-test</artifactid> <scope>test</scope> </dependency> </dependencies> <dependencymanagement> <dependencies> <dependency> <groupid>org.springframework.cloud</groupid> <artifactid>spring-cloud-dependencies</artifactid> <version>dalston.rc1</version> <type>pom</type> <scope> import </scope> </dependency> </dependencies> </dependencymanagement> <build> <plugins> <plugin> <groupid>org.springframework.boot</groupid> <artifactid>spring-boot-maven-plugin</artifactid> </plugin> </plugins> </build> <repositories> <repository> <id>spring-milestones</id> <name>spring milestones</name> <url>https: //repo.spring.io/milestone</url> <snapshots> <enabled> false </enabled> </snapshots> </repository> </repositories> |
application.yml
1
2
3
4
5
6
7
8
9
10
11
12
|
eureka: client: serviceurl: defaultzone: http: //localhost:8761/eureka/ server: port: 8765 spring: application: name: service-feign feign: hystrix: enabled: true |
application類
1
2
3
4
5
6
7
8
9
10
|
@springbootapplication @enablediscoveryclient @enablefeignclients @enablehystrixdashboard public class application { public static void main(string[] args) { springapplication.run(application. class , args); } } |
然后再提供一個service,他的作用就是做負債均衡和斷路器功能
1
2
3
4
5
|
@feignclient (value = "service-hi" ,fallback = schedualservicehihystric. class ) public interface schedualservicehi { @requestmapping (value = "/hi" ,method = requestmethod.get) string sayhifromclientone( @requestparam (value = "name" ) string name); } |
1
2
3
4
5
6
7
|
@component public class schedualservicehihystric implements schedualservicehi { @override public string sayhifromclientone(string name) { return "sorry " +name; } } |
feignclient我們指定之前創建client時指定的name:service-hi,fallback指定服務不可用時的返回數據,這樣我們啟動多個client時就可以看到http請求時會交替訪問不同的feign端口,當停掉clien時再訪問接口會返回錯誤信息。
zuul(服務網關)
在一般情況下,我們不會直接暴露客戶端給外部,而是通過服務網關來轉發,內部服務都是在局域網內通信,外部訪問不了,通過服務網關,我們還可以統一做接口的安全性校驗,統一攔截,請看代碼:
pom.xml
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
|
<parent> <groupid>org.springframework.boot</groupid> <artifactid>spring-boot-starter-parent</artifactid> <version> 1.5 . 9 .release</version> <relativepath/> <!-- lookup parent from repository --> </parent> <properties> <project.build.sourceencoding>utf- 8 </project.build.sourceencoding> <project.reporting.outputencoding>utf- 8 </project.reporting.outputencoding> <java.version> 1.8 </java.version> </properties> <dependencies> <dependency> <groupid>org.springframework.cloud</groupid> <artifactid>spring-cloud-starter-eureka</artifactid> </dependency> <dependency> <groupid>org.springframework.boot</groupid> <artifactid>spring-boot-starter-web</artifactid> </dependency> <dependency> <groupid>org.springframework.cloud</groupid> <artifactid>spring-cloud-starter-zuul</artifactid> </dependency> <dependency> <groupid>org.springframework.boot</groupid> <artifactid>spring-boot-starter-test</artifactid> <scope>test</scope> </dependency> </dependencies> <dependencymanagement> <dependencies> <dependency> <groupid>org.springframework.cloud</groupid> <artifactid>spring-cloud-dependencies</artifactid> <version>dalston.rc1</version> <type>pom</type> <scope> import </scope> </dependency> </dependencies> </dependencymanagement> <build> <plugins> <plugin> <groupid>org.springframework.boot</groupid> <artifactid>spring-boot-maven-plugin</artifactid> </plugin> </plugins> </build> <repositories> <repository> <id>spring-milestones</id> <name>spring milestones</name> <url>https: //repo.spring.io/milestone</url> <snapshots> <enabled> false </enabled> </snapshots> </repository> |
application.yml
1
2
3
4
5
6
7
8
9
10
11
12
13
14
|
eureka: client: serviceurl: defaultzone: http: //localhost:8761/eureka/ server: port: 8080 spring: application: name: service-zuul zuul: routes: api-b: path: /api/** serviceid: service-feign #凡是以api開始的請求都訪問service-feign服務 |
application類
1
2
3
4
5
6
7
8
9
|
@enablezuulproxy @enableeurekaclient @springbootapplication public class application { public static void main(string[] args) { springapplication.run(application. class , args); } } |
啟動application,訪問:http://localhost:8080/api/hi,就能訪問到之前我們定義的接口,接下來我們做接口的攔截:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
|
/** * filtertype:返回一個字符串代表過濾器的類型,在zuul中定義了四種不同生命周期的過濾器類型,具體如下: pre:路由之前 routing:路由之時 post: 路由之后 error:發送錯誤調用 filterorder:過濾的順序 shouldfilter:這里可以寫邏輯判斷,是否要過濾,本文true,永遠過濾。 run:過濾器的具體邏輯。可用很復雜,包括查sql,nosql去判斷該請求到底有沒有權限訪問。 */ @component public class myfilter extends zuulfilter{ private static logger log = loggerfactory.getlogger(myfilter. class ); @override public string filtertype() { return "pre" ; } @override public int filterorder() { return 0 ; } @override public boolean shouldfilter() { return true ; } @override public object run() { requestcontext ctx = requestcontext.getcurrentcontext(); httpservletrequest request = ctx.getrequest(); log.info(string.format( "%s >>> %s" , request.getmethod(), request.getrequesturl().tostring())); object accesstoken = request.getparameter( "token" ); if (accesstoken == null ) { log.warn( "token is empty" ); ctx.setsendzuulresponse( false ); ctx.setresponsestatuscode( 401 ); try { ctx.getresponse().getwriter().write( "token is empty" ); } catch (exception e){} return null ; } log.info( "ok" ); return null ; } } |
這樣我們在調用接口前會先執行myfilter類中的run方法,在這個方法里可以做一系列安全驗證,比如token。
好了,一個簡單的微服務架構就已經搭建完成了。
以上所有代碼都已開源到github上了,地址:https://github.com/lynnlovemin/softservice
以上所述是小編給大家介紹的springcloud實現簡單的微服務架構,希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時回復
原文鏈接:http://blog.csdn.net/lynnlovemin/article/details/79019680