设为首页 收藏本站
查看: 877|回复: 0

[经验分享] JGroups 入门实践

[复制链接]

尚未签到

发表于 2017-3-1 10:02:40 | 显示全部楼层 |阅读模式
前言
  JGroups是一个开源的纯java编写的可靠的群组通讯工具。其工作模式基于IP多播,但可以在可靠性和群组成员管理上进行扩展。其结构上设计灵活,提供了一种灵活兼容多种协议的协议栈。
  JGroups 多线程的方式实现了多个协议之间的协同工作,常见工作线程有心跳检测,诊断等等。
  JGroups实现多机器之间的通信一般都会包含维护群组状态、群组通信协议、群组数据可靠性传输这样的一些主题。
  JGroups群组的各个节点是存在"管理节点"的,至少可以说某个节点提供了在一段时间内维护状态信息和消息可靠性检测的功能(一般是最先启动的节点)。
  目前Jboss、Ecache的分布式缓存是基于Groups通信。
  若JGroups通信基于Udp,则可能需要开启机器上UDP相关的设置,比如Open udp。


  • 温馨提示:JGroups各个协议相关的配置文件都可以从JGroups-x.x.x.Final.jar中找到。
JGroups 资料

  http://www.jgroups.org/tutorial/index.html(官网)
  
http://sourceforge.net/projects/javagroups/(JGroups工程&讨论组(Discussion))


tcp/ip与udp协议
  通常我们都知道tcp和udp最大的区别在于可靠性,tcp是基于可靠连接的传输,udp则属非可靠连接,具体可参考百度百科(http://baike.baidu.com/view/1161229.htm?fr=aladdin)。
  JGroups当中,udp是比较推荐的通信方式,其特点是不需要知道另一个节点的ip,通过多播网络发现就可以“找到”相应的节点,而tcp则需要在配置文件中固定配置。

tcp模式

  如果是同一台机器测试,请注意在
  
TCPPING 元素下修改 initial_hosts的配置端口
  
例如
  

"${jgroups.tcpping.initial_hosts:192.168.1.100[7800],192.168.1.100[7801]}  

  如果是多台机器测试,请注意在 TCPPING 元素下修改 initial_hosts 的ip,端口随意
  
例如
  

"${jgroups.tcpping.initial_hosts:192.168.1.100[7800],192.168.1.112[7800]}  


udp模式

  同一台机器的不同端口(端口是动态的)可通信.
  
不同机器之间的ip多播可能会受到一些因素限制而造成节点之间无法彼此发现.


JGroups的可靠性

  1,对所有接收者的消息的无丢失传输(通过丢失消息的重发)
  2,大消息的分割传输和重组
  3,消息的顺序发送和接收
  4,原子性:消息要么被所有接收者接收,要么全不


JavaGroups的成员关系管理

  1,可以知道组内有哪些成员
  2,成员的加入,离开,掉线等的通知


JavaGroups的主要功能特征


  • 组的创建与删除。组成员能在LAN或WAN环境内互相发送消息
  • 组的成员加入或离开
  • 组成员的检测和通知:加入,离开,掉线
  • 检测与移除已掉线的成员
  • 消息的组播 (member-to-group或point-to-multipoint)
  • 消息的点对点发送 (member-to-member或point-to-point)
  • 支持UDP (IP Multicast), TCP, JMS等传输协议
基本概况
  在JGroups中JChannel类提供了主要的API ,用于连接到集群(cluster)、发送和接收消息(Message)和注册listeners等。以相同name调用JChannel.connect(String name)方法的所有实例会连接到同一个集群。
  Message包含消息头(保存地址等信息)和一个字节数组(保存希望传输的数据)。
  org.jgroups.Address接口及其实现类封装了地址信息,它通常包含IP地址和端口号。
  连接到集群中的所有实例(instance)被称为一个视图(org.jgroups.View)。通过View.getMembers()可以得到所有实例的地址。实例只有在连接到集群后才能够发送和接收消息。
  当实例希望离开集群时,可以调用JChannel.disconnect()方法。当希望释放占有的资源时,可以调用JChannel.close()方法。JChannel.close()方法内部会调用JChannel.disconnect()方法。

  通过调用JChannel.setReceiver()方法可以接收消息和得到View改变的通知。每当有实例加入或者离开集群的时候,viewAccepted(View view)方法会被调用。View.toString()方法会打印出View中所有实例的地址,以及View>

    注意的是,每次viewAccepted(View view)方法被调用时,view参数都不同,其View>
  Receiver接口上的getState(),setState()方法用于在实例间传递状态。新的实例通过setState()方法获得通过状态,而这个状态是通过调用集群中其它某个实例上的getState()获得的。

jgroups的应用
  jgroups可以用来做state replication
  以下项目场景都使用了jgroups


  • JBoss Application Server Clustering
  • OSCache Clustering
  • Jetty HTTP session replication
  • Tomcat HTTP session replication
Chat例子
  jgroups的应用很简单,只需要继承 org.jgroups.ReceiverAdapter 实现相应的方法即可
  详细代码参考:https://github.com/wtclosyn/repostories/tree/master/GroupChat
  

package com.losyn.chat;  

  
import java.io.InputStream;
  
import java.nio.charset.Charset;
  
import org.jgroups.Address;
  
import org.jgroups.JChannel;
  
import org.jgroups.Message;
  
import org.jgroups.ReceiverAdapter;
  
import org.jgroups.View;
  

  
/**
  * jGroups
  */

  
public>  private static final String DEFULT_CONFIG_XML = "jgroups-chat-udp.xml";
  /**
  * 配置文件.
  */
  private String confXml;
  

  /**
  * 集群名称.
  */
  private static final String CLUSTER_NAME = "WTCLOSYN-SIMPLE-CHAT";
  /**
  * 字符编码
  */
  private static final Charset CHARSET = Charset.defaultCharset();
  /**
  * 节点通道.
  */
  private JChannel channel = null;
  

  public SimpleChat() {
  this.confXml = DEFULT_CONFIG_XML;
  }
  

  public SimpleChat(String confXml) {
  this.confXml = confXml;
  }
  

  /**
  * 发送消息
  */
  public void start() {
  try {
  InputStream cfg = SimpleChat.class.getClassLoader().getResourceAsStream(confXml);
  channel = new JChannel(cfg);
  //连接到集群
  channel.connect(CLUSTER_NAME);
  channel.setDiscardOwnMessages(true);
  //指定Receiver用来收消息和得到View改变的通知
  channel.setReceiver(this);
  }catch (Exception e){
  System.out.println("启动Chat失败!");
  }
  }
  

  public void sendMessage(Address dst, String text){
  try {
  //Message构造函数的第一个参数代表目的地地址,这里传null代表要发消息给集群内的所有地址
  //第二个参数表示源地址,传null即可,框架会自动赋值
  //第三个参数line会被序列化成byte[]然后发送,推荐自己序列化而不是用java自带的序列化
  Message msg = new Message(dst, null, text.getBytes(CHARSET));
  //发消息到集群
  channel.send(msg);
  } catch (Exception e) {
  System.out.println("Chat发送消息失败!");
  }
  }
  

  @Override
  public void receive(Message msg) {
  String line = msg.getSrc() + ":" + new String(msg.getBuffer(), CHARSET);
  System.out.println(line);
  }
  

  @Override
  public void viewAccepted(View view) {
  System.out.println("A client has changed!" + view.toString());
  }
  
}
  

客户端01
  

package com.losyn.chat;  

  
import java.util.Scanner;
  

  
/**
  * 客户端01
  */

  
public>  private static final String CONFIG_XML = "jgroups-chat-tcp.xml";
  public static void main(String[] args) {
  SimpleChat chat01 = new SimpleChat(CONFIG_XML);
  chat01.start();
  scannerChat(chat01);
  }
  

  private static void scannerChat(SimpleChat chat01){
  Scanner scanner = new Scanner(System.in);
  while (true) {
  System.out.print("> ");
  System.out.flush();
  String line = scanner.next();
  if (line.startsWith("quit") || line.startsWith("exit")) {
  System.exit(-1);
  break;
  }
  chat01.sendMessage(null, line);
  }
  }
  
}
  

客户端02
  

package com.losyn.chat;  

  
import java.util.Scanner;
  

  
/**
  * 客户端02
  */

  
public>  private static final String CONFIG_XML = "jgroups-chat-tcp.xml";
  public static void main(String[] args) {
  SimpleChat chat02 = new SimpleChat(CONFIG_XML);
  chat02.start();
  scannerChat(chat02);
  }
  

  private static void scannerChat(SimpleChat chat01){
  Scanner scanner = new Scanner(System.in);
  while (true) {
  System.out.print("> ");
  System.out.flush();
  String line = scanner.next();
  if (line.startsWith("quit") || line.startsWith("exit")) {
  System.exit(-1);
  break;
  }
  chat01.sendMessage(null, line);
  }
  }
  
}
  

运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-348666-1-1.html 上篇帖子: 软件开发之技能梳理 下篇帖子: Emoji与unicode特殊字符的处理
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表