从我开始接触javaweb开发的时间来(很短),也接触了几个架构,有jsp+servlet+mysql这种最原始的架构,代码耦合度非常高,比如jdbc连接都得写好几次,后来开始用框架了,接触ssm,又接触了springboot,springcloud等等,下面就实现一个简单的分布式项目

分布式,就是要在客户和服务端间建立一个注册中心,这个注册中心可以用springcloud来实现,

有三个模块

spring 分布式ID 包_spring

首先建立一个服务端

在开发springboot的时候,遇到一个问题,就是,springboot的实现类,ServiceProvideApplication该类要与其他的包(controller包,mapper包…)在同一级目录,不然就不能成功访问

spring 分布式ID 包_java_02


用springboot来创建该模块,因为要连接到数据库,所以记得勾选jdbc,mybatis,mysql,还有web的,没勾也没关系,直接到pom.xml中添加依赖

<modelVersion>4.0.0</modelVersion>
    <parent>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-parent</artifactId>
        <version>2.2.5.RELEASE</version>
        <relativePath/> <!-- lookup parent from repository -->
    </parent>
    <groupId>com.example</groupId>
    <artifactId>service_provide</artifactId>
    <version>0.0.1-SNAPSHOT</version>
    <name>service_provide</name>
    <description>Demo project for Spring Boot</description>
    <properties>
        <java.version>1.8</java.version>
    </properties>
    <dependencies>
        <!-- 通用mapper,Mybatis使用mapper接口方式来实现,自己引入 -->
        <dependency>
            <groupId>tk.mybatis</groupId>
            <artifactId>mapper-spring-boot-starter</artifactId>
            <version>2.0.2</version>
        </dependency>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-jdbc</artifactId>
        </dependency>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>
        <dependency>
            <groupId>org.mybatis.spring.boot</groupId>
            <artifactId>mybatis-spring-boot-starter</artifactId>
            <version>2.1.2</version>
        </dependency>

        <dependency>
            <groupId>mysql</groupId>
            <artifactId>mysql-connector-java</artifactId>
            <scope>runtime</scope>
        </dependency>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-test</artifactId>
            <scope>test</scope>
            <exclusions>
                <exclusion>
                    <groupId>org.junit.vintage</groupId>
                    <artifactId>junit-vintage-engine</artifactId>
                </exclusion>
            </exclusions>
        </dependency>
        <!-- Eureka客户端,这是后面要用到的 -->
        <dependency>
            <groupId>org.springframework.cloud</groupId>
            <artifactId>spring-cloud-starter-netflix-eureka-client</artifactId>
        </dependency>
    </dependencies>

    <!-- SpringCloud的依赖 -->
    <dependencyManagement>
        <dependencies>
            <dependency>
                <groupId>org.springframework.cloud</groupId>
                <artifactId>spring-cloud-dependencies</artifactId>
                <version>Finchley.SR2</version>
                <type>pom</type>
                <scope>import</scope>
            </dependency>
        </dependencies>
    </dependencyManagement>

这里省略了头尾,不要直接引用

server.port=8081

spring.datasource.url=jdbc:mysql:///springboot
spring.datasource.username=root
spring.datasource.password=123456
spring.datasource.driver-class-name=com.mysql.jdbc.Driver 
#配置包扫描
mybatis.type-aliases-package=com.sb.pojo

#以下是在springcloud中心注册的配置,可以先关注上面的配置
eureka.client.service-url.defaultZone=http://127.0.0.1:10086/eureka
spring.application.name=service-provide
#如果有重名,开启覆盖
spring.main.allow-bean-definition-overriding=true

表的字段如下,大家可以自己建,或者用已有的数据库代替

spring 分布式ID 包_java_03

要实现通过页面访问后,返回数据库中的数据,用json来返回数据,这样一个rest风格的接口,

@RestController
@RequestMapping("user")
public class UserController {
    @Autowired
    public UserService userService;

    @GetMapping("{id}")
    public User queryUserById(@PathVariable("id") long id){
        return this.userService.queryUserById(id);
    }
}

这个控制器,可以直接在http中输入id值来得到想要的数据,@PathVariable注解用来映射URL绑定的占位符,可以直接用url地址如localhost:8080/user/1来获取id为1的值,在以前学习ssm的时候,好像没有这样的方法,这应该也是springmvc的rest的风格的体现,

@Service
public class UserService {

    @Autowired
    public UserMapper userMapper;

    public User queryUserById(long id){
        User user = userMapper.selectByPrimaryKey(id);
        return user;
    }
}

上面有提到过,mybatis用接口的方式来实现数据库操作,但是这里的mapper类,只要继承tk.mybatis.mapper.common.Mapper;就可以实现了,

public interface UserMapper extends Mapper<User> {
}

直接映射到user实体类,可以实现selectByPrimaryKey(id);返回查询到的user数据了。这个依赖要自己引的,上面的pom.xml中已指出
但是现在的mapper和springboot项目还没有关联,要在ServiceProvideApplication中加入

@MapperScan("com.sb.mapper")
@EnableDiscoveryClient

扫描mapper包,下面的那个注解作用是使注册中心能够发现自己(当前的服务端)

这时候大家就可以启动服务器访问下数据了。

建客户端

spring 分布式ID 包_spring_04


看图就可以发现,客户端只管拿数据就行了,不用再做那些查数据库的操作,

@Controller
@RequestMapping("consumer/user")
public class UserController {

    @Autowired
    public RestTemplate restTemplate;

	
    @Autowired
    public DiscoveryClient discoveryClient;

    @GetMapping
    @ResponseBody
    public User queryUserById(@RequestParam("id")Long id){
        
        //	|
        //	\/
        //根据服务器名称,获取服务实例,有可能是集群,所以是service集合
        List<ServiceInstance> instances = discoveryClient.getInstances("service-provide");
        //因为只有一个service-provide所以获取第一个
        ServiceInstance instance = instances.get(0);
        String baseUrl = "http://"+instance.getHost()+":"+instance.getPort()+"/user/"+id;
        //	/\
        //	|
        
        User u = restTemplate.getForObject(baseUrl, User.class);
        return  u;
    } 
}

代码中的注释是和注册中心相关的,可以先不关注,baseUrl就是服务器的地址,这时的服务器是provide,注册中心还没做,直接连上provide,
RestTemplate是Spring提供的用于访问Rest服务的客户端,RestTemplate提供了多种便捷访问远程Http服务的方法,能够大大提高客户端的编写效率。

server.port=8082

#注册中心相关
eureka.client.service-url.defaultZone=http://127.0.0.1:10086/eureka,http://127.0.0.1:10087/eureka
spring.application.name=service-consumer

配置也只要配个端口

启动服务器,然后再启动客户端,这是就可以通过localhost:8082/consumer/user/1来访问服务器,获取数据了。当然了,这时的服务器和客户端是放在同一个电脑上的,所以有些奇怪。

eureka服务器(注册中心)

spring 分布式ID 包_ide_05


在创建springboot项目时勾选eureka服务

spring 分布式ID 包_spring_06

<modelVersion>4.0.0</modelVersion>
    <parent>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-parent</artifactId>
        <version>2.2.5.RELEASE</version>
        <relativePath/> <!-- lookup parent from repository -->
    </parent>
    <groupId>com</groupId>
    <artifactId>eureka</artifactId>
    <version>0.0.1-SNAPSHOT</version>
    <name>eureka</name>
    <description>Demo project for Spring Boot</description>

    <properties>
        <java.version>1.8</java.version>
        <spring-cloud.version>Hoxton.SR3</spring-cloud.version>
    </properties>

    <dependencies>
        <dependency>
            <groupId>org.springframework.cloud</groupId>
            <artifactId>spring-cloud-starter-netflix-eureka-server</artifactId>
        </dependency>

        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-test</artifactId>
            <scope>test</scope>
            <exclusions>
                <exclusion>
                    <groupId>org.junit.vintage</groupId>
                    <artifactId>junit-vintage-engine</artifactId>
                </exclusion>
            </exclusions>
        </dependency>
    </dependencies>
	<!-- SpringCloud的依赖 -->
    <dependencyManagement>
        <dependencies>
            <dependency>
                <groupId>org.springframework.cloud</groupId>
                <artifactId>spring-cloud-dependencies</artifactId>
                <version>${spring-cloud.version}</version>
                <type>pom</type>
                <scope>import</scope>
            </dependency>
        </dependencies>
    </dependencyManagement>

不要忘了加springcloud的统一依赖管理↑
配置信息↓

server.port=10086
#eureka服务的名字
spring.application.name=eureka-server
#默认的地址
eureka.client.service-url.defaultZone=http://127.0.0.1:10086/eureka

在springboot的启动器加上注解@EnableEurekaServer,开启eureka服务

启动服务,并访问:http://127.0.0.1:10086

spring 分布式ID 包_java_07


这时候eureka服务就开启了,接下来就要将服务端和客户端加入这个服务器了

但是在现实生活中,只有一台服务器来处理请求就会有问题,一般都是多台服务器,这里就涉及到eureka的高可用,就是注册多台eureka服务器,这里注册两台,使用相互注册的方式,两台的端口号这里设置的是10086,10087

先将服务端和客户端的配置信息写好

service_provide的配置文件

spring.application.name=service-provide
#如果有重名,开启覆盖
spring.main.allow-bean-definition-overriding=true
#eureka客户端的配置,连接到注册中心的地址
eureka.client.service-url.defaultZone=http://127.0.0.1:10086/eureka,http://127.0.0.1:10087/eureka

在启动服务端时,启动失败,

Description:The bean ‘dataSource’, defined in BeanDefinition defined in class path resource [org/springframework/boot/autoconfigure/jdbc/DataSourceConfigurationspring 分布式ID 包_spring_08Hikari.class] and overriding is disabled.

应该是datasource定义了两次,找不到到底在哪定义了两次,直接在配置文件中配置覆盖,就能启动了。
客户端service_consumer也要加上

spring.application.name=service-consumer

eureka.client.service-url.defaultZone=http://127.0.0.1:10086/eureka,http://127.0.0.1:10087/eureka
高可用

这里要先打开run dashboard窗口,便于操作
在你的项目中,D:…\myData.idea,我的项目名是myData,.idea中找workspace.xml文件,找到RunDashBoard字段,在component便签下添加

<option name="configurationTypes">  
    <set>  
        <option value="SpringBootApplicationConfigurationType" />  
    </set>  
</option>

之后启动项目就会弹出dashboard窗口了,

右键EurekaApplication,点击复制配置,弹出编辑配置的窗口,直接点击确定,就会生成一个EurekaApplication2的复制项,

spring 分布式ID 包_spring 分布式ID 包_09

server.port=10087
spring.application.name=eureka-server
eureka.client.service-url.defaultZone=http://127.0.0.1:10086/eureka

把Eureka模块的配置文件的defaultZone的端口改为10087,因为相互注册,将10086注册给10087,启动原eureka的启动器,这时候会报错,但是dashboard中原eureka服务器后面已经跟着个10086的端口了,注册成功了,之后把配置文件(两个服务共用一个配置,但是加载完原项目后,把配置文件改了),

server.port=10086
spring.application.name=eureka-server
eureka.client.service-url.defaultZone=http://127.0.0.1:10087/eureka

把10087注册给10086,然后启动那个复制的eureka,在我启动时有时候会失败,但是照着上面的那个配置文件的修改顺序来,应该是没错的,

spring 分布式ID 包_spring_10


这时候,两个eureka就都启动了,在把服务端和客户端启动,可以访问http://127.0.0.1:10086来查看

spring 分布式ID 包_spring 分布式ID 包_11


这时候高可用就完成了

spring 分布式ID 包_spring_12

这篇blog只是大体的论述下springcloud的使用,里面很多的细节并没有去细扣,也有许多的错误和不解。