`
m635674608
  • 浏览: 5053573 次
  • 性别: Icon_minigender_1
  • 来自: 南京
社区版块
存档分类
最新评论

JGroups 入门实践

 
阅读更多

前言

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 ID。

  • 注意的是,每次viewAccepted(View view)方法被调用时,view参数都不同,其View ID也会增长。如果没有名字,名字是机器名+随机数,后面跟|,以及自增长的View ID。View内的第一个实例被称为coordinator。

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 class SimpleChat extends ReceiverAdapter {
    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 class ChatClient01 {
    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 class ChatClient02 {
    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);
        }
    }
}
程序员是彻头彻尾的脑力工作者(Mind Worker),怠于思考者绝对不能成为好的程序员!!!
 
http://www.cnblogs.com/wtclosyn/articles/4456565.html
分享到:
评论

相关推荐

    JavaEE源代码 jgroups-2.2.8

    JavaEE源代码 jgroups-2.2.8JavaEE源代码 jgroups-2.2.8JavaEE源代码 jgroups-2.2.8JavaEE源代码 jgroups-2.2.8JavaEE源代码 jgroups-2.2.8JavaEE源代码 jgroups-2.2.8JavaEE源代码 jgroups-2.2.8JavaEE源代码 ...

    Jgroups 教程

    ### JGroups教程:深入理解与应用 #### 一、安装JGroups JGroups是一款高性能、...总之,JGroups为开发者提供了构建复杂分布式系统的基石,通过学习和实践,可以充分发挥其潜力,实现高效稳定的分布式通信解决方案。

    Ehcache通过Jgroups做集群

    在Ehcache通过Jgroups进行集群配置时,首先需要理解Jgroups的配置文件——`jgroups.xml`。这个文件定义了集群中节点如何相互发现、通信以及故障检测的规则。配置文件中的关键元素包括: 1. **Transport**: 定义了...

    JGroups_集群.pdf

    JGroups集群技术概述 JGroups是一个用于建立可靠的组播通信的工具包,它提供了灵活的、可定制的协议栈,以满足不同的需求。JGroups支持多种传输协议,包括UDP、TCP和JMS等。在JGroups中,消息传输可以保证可靠性,...

    jgroups-2.2.7.jar

    jgroups-2.2.7.jar jgroups-2.2.7.jar

    jgroups.part1

    jgroups.part1

    JGroups的Raft实现jgroups-raft.zip

    jgroups-raft 项目是 JGroups 框架对 Raft 的实现。Maven:<groupId>org.jgroups <artifactId>jgroups-raft <version>0.2</version>Raft 是一个容易理解的共识算法。在容错和性能方面它相当于 Paxos(Google 的一致...

    jgroups源代码

    通过对JGroups源代码的深入学习,开发者不仅可以掌握其工作原理,还能了解到分布式系统中的许多通用设计模式和最佳实践,这对于构建自己的分布式解决方案具有极大的价值。无论是理解JGroups本身,还是提升分布式系统...

    jgroups

    本教程将深入探讨如何安装配置JGroups,并指导你通过实践编写一个简单的群集应用。 ##### 下载与解压 JGroups可在其官方SourceForge页面下载,对于本次教程,我们将使用二进制版本的JGroups 2.5。下载的ZIP文件名...

    jgroups-3.0.2

    JGroups是一个开源的纯java编写的可靠的群组通讯工具。其是一个可靠的组播通讯工具集(需要说明的是,这并不是说必须要使用IP Multicast,JGroups也可以使用TCP来实现)。其工作模式基于IP多播,但可以在可靠性和群组...

    Jgroups中的UNICAST3协议中文翻译

    Jgroups 中的 UNICAST3 协议详解 Jgroups 是一种基于 IP 多播的可靠的组播中间件,UNICAST3 协议是 Jgroups 中的一种单播协议,旨在保持单播和 UNICAST2 的正面特征,而修正负面特征。 UNICAST3 协议的主要特点是...

    jgroups官方帮助文档html格式打包2.X版本

    此文档主要针对JGroups 2.X版本的官方帮助文档进行详细解读,旨在帮助开发者深入理解并有效地利用JGroups。 一、JGroups简介 JGroups的核心目标是确保在分布式环境中数据的一致性。它提供了一套完整的工具,用于...

    jgroups.part3

    jgroups.part3

    JGroups-jdk.zip_jgroups

    《JGroups:构建高效可靠的组通信系统》 JGroups是一个用Java编程语言编写的开源库,专注于实现基于IP组播的高效、可配置的组通信协议栈。它为分布式系统提供了一种健壮且灵活的方式来实现节点间的通信,是构建大...

    jgroups-3.2

    JGroups是一个开源的纯java编写的可靠的群组通讯工具。其是一个可靠的组播通讯工具集(需要说明的是,这并不是说必须要使用IP Multicast,JGroups也可以使用TCP来实现)。其工作模式基于IP多播,但可以在可靠性和群组...

    jgroups-2.6.8.GA.jar

    jgroups-2.6.8.GA.jar jgroups-2.6.8.GA.jar

    Android代码-jgroups-android

    JGroups - A Framework for Group Communication in Java ======================================================== March 3, 1998 Bela Ban 4114 Upson Hall Cornell University Ithaca, NY 14853 bba@...

    Java多播通讯框架 JGroups

    Java多播通讯框架JGroups是Java开发者用于构建高可用、高性能和可伸缩的集群通信系统的重要工具。它提供了一套全面的协议栈,能够处理网络中的节点发现、消息传递、故障检测和恢复等问题,从而使得开发分布式应用变...

    JGROUPS集群框架源码分析之消息发送、处理、接收

    对于有兴趣深入了解JGROUPS工作原理的开发者,阅读源码和实践案例是获取深入理解的宝贵途径。 参考文献: - 博文链接:https://hxy-go.iteye.com/blog/518795 请注意,以上内容基于提供的标题和标签信息,详细分析...

Global site tag (gtag.js) - Google Analytics