`
dragonsoar
  • 浏览: 205687 次
  • 性别: Icon_minigender_1
  • 来自: 杭州
社区版块
存档分类
最新评论

纯Java的高性能长连接RPC解决方案

 
阅读更多

 

简介:轻量封装Ibatis3

         因为本人在国内最大的电子商务公司工作期间,深感一个好的分表分库框架可以大大提高系统的承载能力及系统的灵活性,而一个不好的分表分库方案,则让系统在大数据量处理的时候非常郁闷。所以, 在根据笔者在从事电子商务开发的这几年中,对各个应用场景而开发的一个轻量封装Ibatis3的一个分表分库框架。

         笔者工作的这几年之中,总结并开发了如下几个框架: summercool(Web 框架,已经应用于某国内大型网络公司的等重要应用)、summercool-hsf(基于Netty实现的RPC框架,已经应用国内某移动互联网公司)、 summercool-ddl(基于Mybaits的分表分库框架,已经应用国内某移动互联网公司);相继缓存方案、和消息系统解决方案也会慢慢开源。 Summercool框架做为笔者的第一个开源框架

summercool-hsf:http://summercool-hsf.googlecode.com/svn/trunk

 

 

1. 什么是HSF框架

    HSF框架是一个高性能远程通信框架,底层基于Netty实现TCP通信,对上层进行封装,提供易于使用和高度可扩展能力。

 

 

名词解译:

    1)Channel:可以理解为一个通道,或者连接

    2)ChannelGroup:多个通道组合成为一个ChannelGroup

 

2.HSF工作流程

 

3.消息协议设计

    消息协议这里是指对消息编码和解码的规范的一种定义,HSF内置的消息协议采用如下结构:

 

 

 

    Length:以4个字节表示,是指ID + Content的长度。

    ID:以1个字节表示,1表示Content部分被压缩,0表示未被压缩。

    Content:真实的消息内容。

 

 

4.处理器

    Netty框架原生提供一个处理器链对事件进行处理,每个处理器均实现ChannelHandler接口,ChannelHandler是个空接口,拥有三个子接口:ChannelDownstreamHandler, ChannelUpstreamHandler和LifeCycleAwareChannelHandler。这里我们主要关注前两个接口,因为它们被用来处理读与写的消息。
    事件主要分为三种:ChannelEvent、MessageEvent和ExceptionEvent,一旦这些事件被触发,它们将从处理器链的一端到另一端,被逐个处理器处理,注意,整个过程是单线程场景。一般而言,ChannelEvent和ExceptionEvent事件都是从底层被触发,因此,它们会被ChannelUpstreamHandler处理。而MessageEvent则需要根据读与写方式的不同,分别从两个方向被ChannelUpstreamHandler和ChannelDownstreamHandler处理。
    HSF内置的编(解)码处理器、压缩(解压)处理器及序列化(反序列化)处理器等都是直接或间接实现ChannelHandler。

    ♦ ChannelDownstreamHandler

 

public interface ChannelDownstreamHandler extends ChannelHandler {
    /**
     * Handles the specified downstream event.
     *
     * @param ctx  the context object for this handler
     * @param e    the downstream event to process or intercept
     */
    void handleDownstream(ChannelHandlerContext ctx, ChannelEvent e) throws Exception;
}

 

    ♦ ChannelUpstreamHandler

 

public interface ChannelUpstreamHandler extends ChannelHandler {
    /**
     * Handles the specified upstream event.
     *
     * @param ctx  the context object for this handler
     * @param e    the upstream event to process or intercept
     */
    void handleUpstream(ChannelHandlerContext ctx, ChannelEvent e) throws Exception;
}

 

 

4.1.Encoding流程

     HSF内置的encoding过程由三个Handler组合完成,流程如下:

 

 

 

    1) SerializeDownstreamHandler

 

/**
 * @Title: SerializeDownstreamHandler.java
 * @Package com.gexin.hsf.netty.channelhandler.downstream
 * @Description: 序列化
 * @author 
 * @date 2011-9-16 下午4:45:59
 * @version V1.0
 */
public class SerializeDownstreamHandler implements ChannelDownstreamHandler {
    Logger logger = LoggerFactory.getLogger(getClass());
    private Serializer serializer = new KryoSerializer();
    public SerializeDownstreamHandler() {
    }
    public void handleDownstream(ChannelHandlerContext ctx, ChannelEvent e) throws Exception {
        if (!(e instanceof MessageEvent)) {
            ctx.sendDownstream(e);
            return;
        }
        MessageEvent event = (MessageEvent) e;
        Object originalMessage = event.getMessage();
        Object encodedMessage = originalMessage;
        if (!(originalMessage instanceof Heartbeat)) {
            encodedMessage = serializer.serialize(originalMessage);
        } else {
            encodedMessage = Heartbeat.BYTES;
        }
        if (originalMessage == encodedMessage) {
            ctx.sendDownstream(e);
        } else if (encodedMessage != null) {
            write(ctx, e.getFuture(), encodedMessage, event.getRemoteAddress());
        }
    }
    public void setSerializer(Serializer serializer) {
        this.serializer = serializer;
    }
}

      2)CompressionDownstreamHandler

 

 

/**
 * @Title: CompressionDownstreamHandler.java
 * @Package com.gexin.hsf.netty.channelhandler.downstream
 * @Description: 压缩处理器
 * @author 
 * @date 2011-9-16 下午4:45:59
 * @version V1.0
 */
public class CompressionDownstreamHandler implements ChannelDownstreamHandler {
    private CompressionStrategy compressionStrategy = new ThresholdCompressionStrategy();
    public void handleDownstream(ChannelHandlerContext ctx, ChannelEvent e) throws Exception {
        if (!(e instanceof MessageEvent)) {
            ctx.sendDownstream(e);
            return;
        }
        MessageEvent event = (MessageEvent) e;
        Object originalMessage = event.getMessage();
        if (originalMessage instanceof byte[]) {
            CompressionResult compressionResult = compressionStrategy.compress((byte[]) originalMessage);
            byte[] resBuffer = compressionResult.getBuffer();
            int length = resBuffer.length;
            byte[] bytes = new byte[length + 1];
            bytes[0] = compressionResult.isCompressed() ? (byte) 1 : (byte) 0;
            for (int i = 0; i < length; i++) {
                bytes[i + 1] = resBuffer[i];
            }
            DownstreamMessageEvent evt = new DownstreamMessageEvent(event.getChannel(), event.getFuture(), bytes,
                    event.getRemoteAddress());
            ctx.sendDownstream(evt);
        } else {
            ctx.sendDownstream(e);
        }
    }
    public void setCompressionStrategy(CompressionStrategy compressionStrategy) {
        this.compressionStrategy = compressionStrategy;
    }
}

       3)LengthBasedEncoder

 

 

/**
 * @ClassName: LengthBasedEncoder
 * @Description: 基于长度的编码器
 * @author 
 * @date 2011-9-29 下午1:43:41
 * 
 */
public class LengthBasedEncoder extends ObjectEncoder {
    Logger logger = LoggerFactory.getLogger(getClass());
    private final int estimatedLength;
    public LengthBasedEncoder() {
        this(512);
    }
    public LengthBasedEncoder(int estimatedLength) {
        if (estimatedLength < 0) {
            throw new IllegalArgumentException("estimatedLength: " + estimatedLength);
        }
        this.estimatedLength = estimatedLength;
    }
    @Override
    protected Object encode(ChannelHandlerContext ctx, Channel channel, Object msg) throws Exception {
        if (msg instanceof byte[]) {
            byte[] bytes = (byte[]) msg;
            ChannelBuffer ob = ChannelBuffers.dynamicBuffer(estimatedLength, channel.getConfig().getBufferFactory());
            ob.writeInt(bytes.length);
            ob.writeBytes(bytes);
            return ob;
        } else {
            throw new IllegalArgumentException("msg must be a byte[], but " + msg);
        }
    }
}

 

 

   4.2.Decoding流程

    decoding流程与encoding正好相反,流程如下:

 

    1)LengthBasedDecoder

    对于TCP通信而言,粘包是很正常的现象,因此decoder必须处理粘包问题。HsfFrameDecoder是一个支持粘包处理的decoder类抽象。

/**
 * @ClassName: LengthBasedDecoder
 * @Description: 基于长度的解码器
 * @author 
 * @date 2011-9-29 下午1:42:59
 * 
 */
public class LengthBasedDecoder extends HsfFrameDecoder {
    private Logger logger = LoggerFactory.getLogger(getClass());
    private int headerFieldLength = 4;
    public LengthBasedDecoder() {
        this(4);
    }
    public LengthBasedDecoder(int headerFieldLength) {
        this.headerFieldLength = headerFieldLength;
    }
    @Override
    protected Object decode(ChannelHandlerContext ctx, Channel channel, ChannelBuffer buffer) throws Exception {
        if (buffer.readableBytes() >= headerFieldLength) {
            buffer.markReaderIndex();
            int length = buffer.readInt();
            if (length < 0) {
                logger.error("msg length must >= 0. but length={}", length);
                return null;
            } else if (length == 0) {
                return Heartbeat.BYTES;
            } else if (buffer.readableBytes() >= length) {
                byte[] bytes = new byte[length];
                buffer.readBytes(bytes);
                return bytes;
            } else {
                buffer.resetReaderIndex();
            }
        }
        return null;
    }
}

     2)DecompressionUpstreamHandler

/**
 * @Title: DecompressionUpstreamHandler.java
 * @Package com.gexin.hsf.netty.channelhandler.downstream
 * @Description: 解压缩处理器
 * @author 
 * @date 2011-9-16 下午4:45:59
 * @version V1.0
 */
public class DecompressionUpstreamHandler extends SimpleChannelUpstreamHandler {
    private CompressionStrategy compressionStrategy = new ThresholdCompressionStrategy();
    @Override
    public void messageReceived(ChannelHandlerContext ctx, MessageEvent e) throws Exception {
        if (e.getMessage() instanceof byte[]) {
            byte[] bytes = (byte[]) e.getMessage();
            int length = bytes.length;
            if (length > 0) {
                byte[] buffer = new byte[length - 1];
                for (int i = 1; i < length; i++) {
                    buffer[i - 1] = bytes[i];
                }
                if (bytes[0] == 1) {
                    buffer = compressionStrategy.decompress(buffer);
                }
                UpstreamMessageEvent event = new UpstreamMessageEvent(e.getChannel(), buffer, e.getRemoteAddress());
                super.messageReceived(ctx, event);
            }
        } else {
            super.messageReceived(ctx, e);
        }
    }
    public void setCompressionStrategy(CompressionStrategy compressionStrategy) {
        this.compressionStrategy = compressionStrategy;
    }
}

     3)DeserializeUpstreamHandler

/**
 * @Title: DeserializeUpstreamHandler.java
 * @Package com.gexin.hsf.netty.channelhandler.downstream
 * @Description: 反序列化
 * @author 
 * @date 2011-9-16 下午4:45:59
 * @version V1.0
 */
public class DeserializeUpstreamHandler extends SimpleChannelUpstreamHandler {
    private Logger logger = LoggerFactory.getLogger(getClass());
    private Serializer serializer = new KryoSerializer();
    @Override
    public void messageReceived(ChannelHandlerContext ctx, MessageEvent e) throws Exception {
        if (e.getMessage() == null) {
            return;
        } else if (e.getMessage() instanceof byte[]) {
            byte[] bytes = (byte[]) e.getMessage();
            Object msg;
            if (bytes.length == 0) {
                msg = Heartbeat.getSingleton();
            } else {
                try {
                    msg = serializer.deserialize(bytes);
                } catch (Exception ex) {
                    throw ex;
                }
            }
            UpstreamMessageEvent event = new UpstreamMessageEvent(e.getChannel(), msg, e.getRemoteAddress());
            super.messageReceived(ctx, event);
        } else {
            super.messageReceived(ctx, e);
        }
    }
    public void setSerializer(Serializer serializer) {
        this.serializer = serializer;
    }
}

    4.3.处理器链的建立

    HSF使用如下的方式构建处理器链:

bootstrap.setPipelineFactory(new ChannelPipelineFactory() {
            public ChannelPipeline getPipeline() throws Exception {
                ChannelPipeline pipeline = Channels.pipeline();
                // 注册各种自定义Handler
                for (String key : handlers.keySet()) {
                    pipeline.addLast(key, handlers.get(key));
                }
                // 注册链路空闲检测Handler
                Integer writeIdleTime = LangUtil.parseInt(options.get(HsfOptions.WRITE_IDLE_TIME));
                Integer readIdleTime = LangUtil.parseInt(options.get(HsfOptions.READ_IDLE_TIME));
                if (writeIdleTime == null) {
                    writeIdleTime = 10;
                }
                if (readIdleTime == null) {
                    // 默认为写空闲的3倍
                    readIdleTime = writeIdleTime * 3;
                }
                pipeline.addLast("timeout", new IdleStateHandler(idleTimer, readIdleTime, writeIdleTime, 0));
                pipeline.addLast("idleHandler", new StateCheckChannelHandler(HsfAcceptorImpl.this));
                // 注册事件分发Handler
                pipeline.addLast("dispatchHandler", new DispatchUpStreamHandler(eventDispatcher));
                return pipeline;
            }
        });

5.Dispatcher

    消息经过Handler链处理后,将被Dispatcher转发,并进入EventListener链处理。

    Dispatcher内置两个线程池:channelExecutor和msgExecutor。

    channelExecutor用于处理通道事件和异常事件,考虑到在通道事件可能需要同步调用远程服务,因此此线程池不设上线(因为同步调用将会阻塞当前线程)。

    msgExecutor用于处理消息事件,根据经验值,缺省最大线程数为150,该值可以通过Option参数修改。

6.EventListener

    EventListener有以下三种:

    1)ChannelEventListener

/**
 * @Title: ChannelEventListener.java
 * @Package com.gexin.hsf.netty.listener
 * @Description: 通道事件监听类
 * @author 
 * @date 2011-9-27 上午11:45:50
 * @version V1.0
 */
public interface ChannelEventListener extends EventListener {
    /**
     * Invoked when a {@link Channel} was closed and all its related resources were released.
     * 
     * @author
     * @param ctx
     * @param channel
     * @param e
     * @return EventBehavior Whether to continue the events deliver
     */
    public EventBehavior channelClosed(ChannelHandlerContext ctx, HsfChannel channel, ChannelStateEvent e);
    /**
     * Invoked when a {@link Channel} is open, bound to a local address, and connected to a remote address.
     * 
     * @author
     * @param ctx
     * @param channel
     * @param e
     * @return EventBehavior Whether to continue the events deliver
     */
    public EventBehavior channelConnected(ChannelHandlerContext ctx, HsfChannel channel, ChannelStateEvent e);
    /**
     * Invoked when a group is created.
     * 
     * @author
     * @param ctx
     * @param channel
     * @param groupName
     * @return EventBehavior Whether to continue the events deliver
     */
    public EventBehavior groupCreated(ChannelHandlerContext ctx, HsfChannel channel, String groupName);
    /**
     * Invoked when a group is removed.
     * 
     * @author
     * @param ctx
     * @param channel
     * @param groupName
     * @return EventBehavior Whether to continue the events deliver
     */
    public EventBehavior groupRemoved(ChannelHandlerContext ctx, HsfChannel channel, String groupName);
}

     2)MessageEventListener

/**
 * @Title: MessageListener.java
 * @Package com.gexin.hsf.netty.listener
 * @Description: 消息监听接口
 * @author
 * @date 2011-9-27 上午11:36:22
 * @version V1.0
 */
public interface MessageEventListener extends EventListener {
    /**
     * Invoked when a message object (e.g: {@link ChannelBuffer}) was received
     * from a remote peer.
     */
    public EventBehavior messageReceived(ChannelHandlerContext ctx, HsfChannel channel, MessageEvent e);
}

     3)ExceptionEventListener

/**
 * @Title: ExceptionEventListener.java
 * @Package com.gexin.hsf.netty.listener
 * @Description: 异常监听接口
 * @author
 * @date 2011-9-27 上午11:48:09
 * @version V1.0
 */
public interface ExceptionEventListener extends EventListener {
    /**
     * Invoked when an exception was raised by an I/O thread or a {@link ChannelHandler}.
     */
    public EventBehavior exceptionCaught(ChannelHandlerContext ctx, Channel channel, ExceptionEvent e);
}

     Hsf框架会预先在EventListener链末端注册ServiceMessageEventListener,该Listener负责调用被注册的Service,并将返回值或异常回传。

    7.Service

    1)RemoteServiceContract注解

    所有实现了拥有RemoteServiceContract注解的Java类都可以直接注册到HsfService,示例如下:

@RemoteServiceContract
public interface TestService {
    String test(String ctx);
}
 
public class TestServiceImpl implements TestService {
    @Override
    public String test(String ctx) {
        return String.valueOf("hello " + ctx);
    }
}

    2)ServiceEntry

    对于未添加RemoteServiceContract注解的接口,Hsf框架使用org.summercool.hsf.pojo.ServiceEntry类实现注册。

   

    3)注册Service

    服务提供方需要向Hsf注册Service方可被远程调用,示例如下:

    ♦ 注册Service

HsfAcceptor acceptor = new HsfAcceptorImpl();
// 注册Service
acceptor.registerService(new TestServiceImpl());
// 监听端口
acceptor.bind(8082);

     ♦ 远程调用Service

HsfConnector connector = new HsfConnectorImpl();
connector.connect(new InetSocketAddress("127.0.0.1",8082));
// 同步方式
TestService testService = ServiceProxyFactory.getRoundFactoryInstance(connector).wrapSyncProxy(TestService.class);
System.out.println(testService.test("HSF"));

    

    3)同步与异步

    4)原理

   7.Handshake

    当通道建立后,Client和Server会进行三次握手,以完成初始化

    初次握手步骤

    1)Client与Server建立连接成功

    2)Client向Server发送握手请求包(handshake request)

    3)Server接收到握手请求包后,生成group信息,然后触发groupCreated事件,接着向client发送握手反馈包(handshake ack)

    4)Client接收到握手反馈包后,生成group信息,然后触发groupCreated事件,接着向server发送握手完成包(handshake finish)

    非初次握手步骤

    1)Client与Server建立连接成功

    2)Client向Server发送握手请求包(handshake request)

    3)Server接收到握手请求包后,添加该连接到Group,接着向client发送握手反馈包(handshake ack)

    4)Client接收到握手反馈包后,添加该连接到Group,接着向server发送握手完成包(handshake finish)

    以上三次握手所发送的包都只包含本身的group信息,但Hsf对外提供了握手的扩展接口,应用可以使用该接口结合自身的业务,以完成连接建立后的初始化工作。

    Client握手扩展接口

   8.Heartbeat、超时及重连机制

    Heartbeat和超时机制依赖于Netty的读空闲和写空闲回调。

    当发生写空闲时,会向对方发送Heartbeat消息,写空闲时间可以通过参数HsfOptions.WRITE_IDLE_TIME设定,缺省为10秒。

    当发生读空闲时,即判定为超时,主动关闭连接,读空闲时间可以通过参数HsfOptions.READ_IDLE_TIME设定,缺省为60秒。

    对于断开的连接,Hsf会为其重连,重连频率通过HsfOptions.RECONNECT_INTERVAL参数设定,缺省为10000毫秒。

 

   9.Option参数

    Hsf支持以参数配置:

参数名 说明 缺省值
HsfOptions.TCP_NO_DELAY TCP参数,是否关闭延迟发送消息包 true
HsfOptions.KEEP_ALIVE TCP参数,是否保持连接 true
HsfOptions.REUSE_ADDRESS TCP参数,是否重用端口 false
HsfOptions.WRITE_IDLE_TIME 写空闲时间(秒) 10
HsfOptions.READ_IDLE_TIME 读空闲时间(秒) 60
HsfOptions.SYNC_INVOKE_TIMEOUT 同步调用超时时间(毫秒) 60000
HsfOptions.HANDSHAKE_TIMEOUT 握手超时时间(毫秒) 15000
HsfOptions.FLOW_LIMIT 流量限额 2000000
HsfOptions.TIMEOUT_WHEN_FLOW_EXCEEDED 申请流量超时时间(毫秒) 3000
HsfOptions.MAX_THREAD_NUM_OF_DISPATCHER 分发器的最大线程数 150
HsfOptions.CHANNEL_NUM_PER_GROUP 每个Group建立的通道数 Runtime.getRuntime().availableProcessors()
HsfOptions.RECONNECT_INTERVAL 重连频率(毫秒) 10000
HsfOptions.CONNECT_TIMEOUT 建立连接超时时间(毫秒) 30000
HsfOptions.HOLD_CALLBACK_MESSAGE 是否缓存Callback方式发送的消息,缓存后将会在发送失败时回调doException方法参数传入

false

    这些参数可以通过如下方式调整:

HsfConnector connector = new HsfConnectorImpl();
connector.setOption(HsfOptions.CHANNEL_NUM_PER_GROUP, 1);

10.Hsf的使用

    待续(大家可以先看代码中的test包)

 

 

 

 

  • 大小: 8.4 KB
  • 大小: 41.4 KB
  • 大小: 7.3 KB
  • 大小: 14.2 KB
  • 大小: 14 KB
  • 大小: 23.8 KB
分享到:
评论
24 楼 allen.lei 2015-10-20  
服务器端如果要向某个客户端发送消息是否是需要服务端通过一个Map保存所有连接上来的客户端SocketChannel,客户端的Id作为Map的key(这个服务端代码中有实现吗?另外如果用map保存的话50w并发是否会爆掉啊?)。每次服务器端如果要向某个客户端发送消息,只需根据ClientId取出对应的SocketChannel,往里面写入message即可?
23 楼 ayanami001 2015-10-19  
请问分布式事务是如何处理的
22 楼 nothing.whoami 2015-05-08  
您好!请问下HSF的代码是开源的吗?想拜读下,谢谢!
21 楼 jussyzou 2015-01-14  
您好,什么时候能迁移完毕呢?希望能下载学习,谢谢。邮箱:jussyzou@163.com
20 楼 dragonsoar 2014-09-02  
lkgforex 写道
你好,我现在也在研究这块,但是现在  http://summercool-hsf.googlecode.com/svn/trunk/ 下载不了代码了。所以请分享一份代码 ,谢谢。
我的邮箱是 lkgforex@163.com

暂时在迁到gitcafe上面,等迁号了会通知你的~
19 楼 dragonsoar 2014-09-02  
ahua186186 写道
thank you  我还没看代码,我想问下你这个可以加SSL吗,可以用来传文件吗。


加不了SSL
18 楼 ahua186186 2014-08-04  
thank you  我还没看代码,我想问下你这个可以加SSL吗,可以用来传文件吗。
17 楼 lkgforex 2014-08-02  
你好,我现在也在研究这块,但是现在  http://summercool-hsf.googlecode.com/svn/trunk/ 下载不了代码了。所以请分享一份代码 ,谢谢。
我的邮箱是 lkgforex@163.com
16 楼 dragonsoar 2014-04-29  
magicyang919 写道
能不能多点文档啊?比如概要设计和详细设计文档。

正在弄中~
15 楼 magicyang919 2014-04-26  
能不能多点文档啊?比如概要设计和详细设计文档。
14 楼 dragonsoar 2013-11-14  
andye 写道
看您的代码就是一种享受!呵呵。
还有您的缓存方案和消息系统框架什么时候开源?

andye 写道
您这个框架是基于点对点调用模式实现的,如果实现多对多的调用,如Dubbo框架,改动会不会比较大?

暂时不会开源,因为堆外大缓存在国内还没有好的解决方案,这个解决方案用处比较广,所以可能会做为我们这样创业公司一个非常好的卖点。
先谢谢各位的支持,呵~
13 楼 jd2bs 2013-10-30  
andye 写道
您这个框架是基于点对点调用模式实现的,如果实现多对多的调用,如Dubbo框架,改动会不会比较大?


怎么是点对点呢? 楼主下面已经说了,configServer包就是处理多点调用的;如果你认为configServer包太简陋,可以改用dubbo的zokeeper方式来管理多个服务提供者,以提供工业强度。不过原理都是一样的。
12 楼 andye 2013-10-28  
看您的代码就是一种享受!呵呵。
还有您的缓存方案和消息系统框架什么时候开源?

andye 写道
您这个框架是基于点对点调用模式实现的,如果实现多对多的调用,如Dubbo框架,改动会不会比较大?
11 楼 andye 2013-10-28  
您这个框架是基于点对点调用模式实现的,如果实现多对多的调用,如Dubbo框架,改动会不会比较大?
10 楼 dragonsoar 2013-06-28  
thinkfly123thinkfly 写道
请问一下,phone的测试包的client和server如何进行交互,client向server发送消息,server接受client的消息,并向client发送消息?

是的,这个只是写的一个思路测试包,不严格,只是能跑通而已
9 楼 thinkfly123thinkfly 2013-06-26  
请问一下,phone的测试包的client和server如何进行交互,client向server发送消息,server接受client的消息,并向client发送消息?
8 楼 dragonsoar 2013-05-15  
yangfuchao418 写道
稳定性和性能怎么样??假如客户端是其他语言调用也可以?

肯定可以了; 稳定性和性能还是不错的,某些大公司内部一直在用的~
7 楼 yangfuchao418 2013-05-06  
稳定性和性能怎么样??假如客户端是其他语言调用也可以?
6 楼 dragonsoar 2013-03-05  
hda 写道
楼主,谢谢分享。提点意见。
你这里提到的HSF似乎与网上其他介绍的tb-hsf方案不一样,
这里提到的似乎没有负载和集群方面的考虑吧?也就是server无法简单的通过增加机器来扩展性能。
对否?

是可以通过加机器来实现扩展性能的,里面有一个configserver包,里面的解决方案和tb的一样的~
5 楼 hda 2013-02-27  
楼主,谢谢分享。提点意见。
你这里提到的HSF似乎与网上其他介绍的tb-hsf方案不一样,
这里提到的似乎没有负载和集群方面的考虑吧?也就是server无法简单的通过增加机器来扩展性能。
对否?

相关推荐

    纯Java的高性能长连接RPC解决方案.docx

    纯Java的高性能长连接RPC解决方案 本文介绍了一个基于Java的高性能长连接RPC解决方案,旨在解决大型电子商务公司系统的承载能力和灵活性问题。该解决方案是基于Netty实现的RPC框架,提供了易于使用和高度可扩展能力...

    java源码:高性能RPC框架 nfs-rpc.rar

    本压缩包“java源码:高性能RPC框架 nfs-rpc.rar”提供了基于Java实现的高性能RPC框架,名为nfs-rpc。这个框架旨在简化服务之间的通信,提高系统的整体效率。 1. **RPC基本概念** - RPC使得客户端能够透明地调用...

    基于Java的高性能RPC框架 nfs-rpc.zip

    Java平台上的RPC框架有很多,如Hadoop的Hadoop RPC、Apache Thrift、Google的gRPC等,而“nfs-rpc”则是一个专门基于Java开发的高性能RPC框架。这个框架的设计目标是提供高效、稳定、易用的跨网络服务调用能力。 ...

    基于java的高性能RPC框架 nfs-rpc.zip

    本教程将深入探讨基于Java实现的高性能RPC框架——nfs-rpc,它旨在提供高效、易用且可扩展的解决方案。 一、RPC框架的核心概念 1. 客户端(Client):发起RPC调用的一方,通常在本地系统上运行,需要调用远程服务...

    基于Java的源码-高性能RPC框架 nfs-rpc.zip

    本项目"nfs-rpc"是一个基于Java实现的高性能RPC框架,旨在提供高效、可靠且易用的服务调用解决方案。 在Java中,RPC框架如Hessian、Dubbo、gRPC等已经非常成熟,而nfs-rpc可能是针对特定场景或需求进行优化的版本。...

    sofa-rpc,SOFARPC是一种高性能、高扩展性、生产级的Java RPC框架。.zip

    SOFARPC是一款由阿里巴巴开源的高性能、高可扩展性的Java Remote Procedure Call (RPC)框架,专为生产环境设计。这个框架旨在简化分布式系统的服务调用,提供了一套完整的解决方案,包括服务注册、服务发现、负载...

    phprpc for java 服务器实例

    Phprpc是一种高效、轻量级的RPC解决方案,它支持多种编程语言,包括PHP和Java。通过这个框架,开发者可以在不同语言之间透明地调用函数和方法,提高了分布式系统开发的灵活性和效率。 2. **Java服务器设置** 在...

    sofa-rpc,sofarpc是一个高性能、高扩展性、生产级的java rpc框架。.zip

    SOFA-RPC 是阿里巴巴开源的一款高性能、高可扩展性、企业级的Java远程过程调用(RPC)框架。它的设计目标是简化分布式服务之间的通信,...如果你正在寻找一个可靠的Java RPC解决方案,SOFA-RPC无疑是值得考虑的选择。

    JavaRPC框架

    在给定的“Java RPC框架”中,我们可以看到它是基于Netty实现的,Netty是一个高性能、异步事件驱动的网络应用程序框架,适用于开发高度可定制和高效率的服务端与客户端应用。 首先,让我们深入了解Java RPC框架的...

    采用Netty4.x网络编程实现高性能高并发的RPC开源框架, 基于Java语言来开发(JDK1.8).zip

    Netty 是一个高性能、异步事件驱动的网络应用程序框架,用于快速开发可维护的高性能协议服务器和客户端。在Java领域,Netty因其高效、稳定和丰富的功能集而被广泛使用,尤其是在开发网络通信相关的系统,如RPC...

    基于Java SPI与高性能反射的Echo-RPC框架设计源码

    该项目是基于Java语言开发的高性能RPC调用框架,源码包含107个文件,其中90个为Java源文件,6个为XML配置文件,其余文件包括Git忽略文件、许可证、标记文件以及图片资源。该框架参考了Dubbo开源项目的设计理念,具备...

    超P7级!JAVA百万并发高性能网关设计 企业级CI & CD平台实践+企业级架构容器云平

    课程全程高能,直击超一线JAVA架构师的最前沿高级架构技术,课程内容包括了RPC核心框架剖析,注册中心深入剖析,消息队列企业级应用及原理剖析,分布式架构整体概括,企业级百万并发高性能网关设计实践,分布式架构...

    利用JSON-RPC-Java构建下一代Web应用程序

    通过本文的介绍,我们可以看到JSON-RPC-Java为构建下一代Web应用程序提供了一种强大而灵活的技术解决方案。它不仅解决了数据序列化与反序列化的难题,还大大提升了Web应用的交互性和性能。随着技术的不断发展和完善...

    Dubbo是一个分布式的高性能的RPC框架

    【标题】:“Dubbo——分布式高性能RPC框架解析” 【描述】:“Dubbo是阿里巴巴开源的一款高性能、轻量级的Java RPC框架,它为服务治理提供了全面的解决方案,旨在实现服务的透明化调用,提高系统间的协同效率。” ...

    forest, 基于netty轻量的高性能分布式RPC服务框架.zip

    森林(forest)是一个基于Netty构建的轻量级、高性能的分布式RPC服务框架。这个框架的设计目标是简化RPC调用,提高服务之间的通信...对于那些寻求高效、易用的RPC解决方案的开发者来说,Forest无疑是一个值得考虑的选择。

    rpc-server.zip

    **网络通信库的选择**:在Java中,如Netty、Mina、Grizzly等NIO框架广泛应用于高性能的网络应用开发,它们能够处理大量并发连接,并且提供灵活的事件驱动模型。`rpc-server.zip`可能使用了其中的一种或自定义的解决...

    基于Java的Thunder分布式RPC框架设计源码

    它支持多协议、多组件和多序列化,为开发者提供了一套全面的分布式RPC解决方案。项目包含554个文件,其中416个Java源文件,以及相关的配置和资源文件,适用于需要高性能、高可用分布式系统的场景。

    RPC应用的java实现

    RPC(Remote Procedure Call)是一种协议,允许运行在一台计算机上的程序调用另一台计算机上的子...通过Java实现RPC,我们可以利用各种框架,根据具体需求选择合适的解决方案,以实现跨网络的高效、透明的远程调用。

    阿里开源的额高性能 RPC 框架.zip

    【描述】Apache Dubbo是一个由阿里巴巴开源的、基于Java的高性能RPC框架。它旨在提供一种简单易用、高性能、可扩展的远程过程调用解决方案,使得服务间的通信变得更加便捷。Dubbo在设计时考虑了大型分布式系统的需求...

    rpc.zip_java_remote

    - **Dubbo**:阿里巴巴开源的高性能RPC框架,支持多种协议,如Dubbo、HTTP、Hessian等,同时集成了服务注册与发现、负载均衡等功能。 - **gRPC**:Google开源的一款基于HTTP/2协议的RPC框架,使用ProtoBuf进行接口...

Global site tag (gtag.js) - Google Analytics