Java 后端实战——基于 Dubbo 的分布式系统架构

来源:互联网 发布:华为杭州研究所知乎 编辑:程序博客网 时间:2024/06/08 07:08

转载请注明出处:http://blog.csdn.net/smartbetter/article/details/64190798

分布式架构就是把系统按照模块拆分成多个子系统。Dubbo(http://dubbo.io/)是一个开源分布式服务框架,致力于提供高性能和透明化的RPC远程服务调用方案,以及SOA服务治理方案。Dubbo 采用全 Spring 配置方式,透明化接入应用,对应用没有任何 API 侵入,只需要 Spring 加载 Dubbo 的配置即可(Dubbo 基于 Spring 的 Schema 扩展进行加载)。其核心部分包含:远程通讯(提供对多种基于长连接的NIO框架抽象封装,包括多种线程模型,序列化,以及“请求-响应”模式的信息交换方式);集群容错(提供基于接口方法的透明远程过程调用,包括多协议支持,以及软负载均衡,失败容错,地址路由,动态配置等集群支持),自动发现(基于注册中心目录服务,使服务消费方能动态的查找服务提供方,使地址透明,使服务提供方可以平滑增加或减少机器)。Dubbo 推荐的注册中心是 ZooKeeper。Dubbo架构如下:

dubbo-architecture

1、服务容器负责启动,加载,运行服务提供者,服务提供者在启动时,向注册中心注册自己提供的服务;
2、服务消费者在启动时,向注册中心订阅自己所需的服务;
3、注册中心返回服务提供者地址列表给消费者,如果有变更,注册中心将基于长链接推送变更数据给消费者;
4、服务消费者,从提供者地址列表中,基于软负载均衡算法,选一台提供者进行调用,如果调用失败,再选另一台调用;
5、服务消费者和提供者,在内存中累计调用次数和调用时间,定时每分钟发送一次统计数据到监控中心。

关于 Dubbo 管理控制台的安装(Dubbo管理控制台的主要作用是服务治理,包含服务注册、服务降级、路由规则、访问控制、动态配置、权重调节、负载均衡、服务负责人、等管理功能。)请参考文章 http://blog.csdn.net/chenli522/article/details/47083287。

192.168.30.10  zookeeper(注册中心)192.168.30.11  dubbo-consumer(服务消费方)192.168.30.12  dubbo-provider(服务提供方)

1.Spring Boot集成Dubbo

2.Spring集成Dubbo

服务提供方和服务消费方斗需要添加 Dubbo 的 Maven 依赖:

<!--================================================================== --><!--|| 配置dubbo || --><!--================================================================== --><dependency>    <groupId>com.alibaba</groupId>    <artifactId>dubbo</artifactId>    <version>2.5.4</version>    <exclusions>        <exclusion>            <groupId>org.springframework</groupId>            <artifactId>spring</artifactId>        </exclusion>    </exclusions></dependency><dependency>    <groupId>org.apache.zookeeper</groupId>    <artifactId>zookeeper</artifactId>    <version>3.4.10</version>    <type>pom</type></dependency><dependency>    <groupId>com.101tec</groupId>    <artifactId>zkclient</artifactId>    <version>0.10</version></dependency><dependency>    <groupId>org.jboss.netty</groupId>    <artifactId>netty</artifactId>    <version>3.2.10.Final</version></dependency>

1.dubbo-demo-provider工程

编辑 resources/spring/dubbo-provider.xml:

<?xml version="1.0" encoding="UTF-8"?><beans xmlns="http://www.springframework.org/schema/beans"       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"       xmlns:dubbo="http://code.alibabatech.com/schema/dubbo"       xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://code.alibabatech.com/schema/dubbo http://code.alibabatech.com/schema/dubbo/dubbo.xsd">    <!-- 提供方应用信息,用于计算依赖关系 -->    <dubbo:application name="dubbo-demo-provider"/>    <!-- 使用zookeeper注册中心暴露服务地址 -->    <dubbo:registry address="zookeeper://192.168.20.11:2181" />    <!-- 用dubbo协议在20880端口暴露服务 -->    <dubbo:protocol name="dubbo" port="20880"/>    <!-- 声明需要暴露的服务接口 -->    <dubbo:service interface="com.example.spring.service.UserService" ref="userService"/>    <!-- 具体的实现bean -->    <bean id="userService" class="com.example.spring.service.impl.UserServiceImpl"/></beans>

打包 jar 运行不需要在 resources/spring/applicationContext.xml 文件中导入上面的配置,但是除了 dubbo-provider 外其他配置需要在 resources/spring/applicationContext.xml 文件中导入。下面添加实体代码:

public class User implements Serializable {    private Long id;    private String loginName;    private String password;    //省略setter、getter方法和toString方法}

添加服务接口类:

public interface UserService {    //增    boolean save(User user);    //删    boolean deleteUser(Long userId);    //改    boolean updateUser(User user);    //查    User selectUserById(Long userId);    //查询所有    List<User> selectAllUser();}

添加服务接口实现类:

@Servicepublic class UserServiceImpl implements UserService{    //忽略具体实现}

最后在服务器端运行服务,打包可运行 jar 参考文章 https://yq.aliyun.com/articles/114344,在Linux 上部署 dubbo 服务方法将在下面给出。Maven 打包 dubbo 可运行 jar 的通用配置 参考代码 https://gitee.com/smartbetter/codes/roa2p537efbx9tv1s6uj474

2.dubbo-demo-consumer工程

编辑 resources/spring/dubbo-consumer.xml:

<?xml version="1.0" encoding="UTF-8"?><beans xmlns="http://www.springframework.org/schema/beans"       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"       xmlns:dubbo="http://code.alibabatech.com/schema/dubbo"       xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://code.alibabatech.com/schema/dubbo http://code.alibabatech.com/schema/dubbo/dubbo.xsd">    <!-- 提供方应用信息,用于计算依赖关系 -->    <dubbo:application name="dubbo-demo-consumer"/>    <!-- 使用zookeeper注册中心暴露服务地址 -->    <dubbo:registry address="zookeeper://192.168.20.11:2181" />    <!-- 生成远程服务代理 -->    <dubbo:reference id="userService" interface="com.example.spring.service.UserService"/></beans>

然后在 resources/spring/applicationContext.xml 文件中导入上面的配置:

<import resource="dubbo-consumer.xml" />

添加实体代码:

public class User implements Serializable {    private Long id;    private String loginName;    private String password;    //省略setter、getter方法和toString方法}

添加服务接口类(不需要编写接口实现类):

public interface UserService {    //增    boolean save(User user);    //删    boolean deleteUser(Long userId);    //改    boolean updateUser(User user);    //查    User selectUserById(Long userId);    //查询所有    List<User> selectAllUser();}

部署项目到服务器,然后 dubbo-demo-consumer 项目中就可以调用 dubbo-demo-provider 中 UserService 服务的方法了。

2.在Linux上部署Dubbo服务

部署目录规范(要避免应用迁移时的路径冲突):

/home/wwwroot/example/                |- /app             //部署app                     |- /app1                     |- /app2                |- /service         //部署分布式服务                     |- /order      //订单服务                     |- /queue      //消息队列服务                     |- /user       //用户服务                |- /timer           //部署定时任务                     |- /report     //报表                |- /web             //部署web工程                     |- /gateway                     |- /operation                     |- /portal

手工维护 Dubbo 服务(有时会报映射错误,主机名ip映射可通过 vim /etc/hosts 配置):

# java -jar xxx.jar &       //启动服务# nohup java -jar xxx.jar &                 //加日志启动# tail -f nohup.out                         //查看启动日志# ps -ef | grep java                        //第一个数字就是PID# kill PID                                  //杀死服务# kill -9 PID

自定义 Dubbo 服务维护的 Shell 脚本(命名规范:/home/wwwroot/example/xxx/service-xxx.sh):

#!/bin/sh## java envexport JAVA_HOME=/usr/local/jdkexport JRE_HOME=$JAVA_HOME/jre## service nameAPP_NAME=userSERVICE_DIR=/home/wwwroot/example/service/$APP_NAMESERVICE_NAME=dubbo-service-$APP_NAMEJAR_NAME=$SERVICE_NAME\.jarPID=$SERVICE_NAME\.pidcd $SERVICE_DIRcase "$1" in    start)        ## debug:-Xms256m-Xmx512m,formal:-Xms512m-Xmx2048m        nohup $JRE_HOME/bin/java -Xms256m -Xmx512m -jar $JAR_NAME >/dev/null 2>&1 &        echo $! > $SERVICE_DIR/$PID        echo "=== start $SERVICE_NAME"        ;;    stop)        kill `cat $SERVICE_DIR/$PID`        rm -rf $SERVICE_DIR/$PID        echo "=== stop $SERVICE_NAME"        sleep 5        ##        ## dubbo-service-aa.jar        ## dubbo-service-aa-bb.jar        P_ID=`ps -ef | grep -w "$SERVICE_NAME" | grep -v "grep" | awk '{print $2}'`        if [ "$P_ID" == "" ]; then              echo "=== $SERVICE_NAME process not exists or stop success"        else            echo "=== $SERVICE_NAME process pid is:$P_ID"            echo "=== begin kill $SERVICE_NAME process, pid is:$P_ID"            kill -9 $P_ID        fi        ;;    restart)        $0 stop        sleep 2        $0 start        echo "=== restart $SERVICE_NAME"        ;;    *)        ## restart        $0 stop        sleep 2        $0 start        ;;esacexit 0
# cd /home/wwwroot/example/service/user# chmod 777 *.sh                             //切换到root用户,对*.sh赋可执行的权限# ./service-user.sh start# ./service-user.sh stop# ./service-user.sh restart# ps -ef | grep java                        //查看运行状态
8 2