yanf4j引入了客户端非阻塞API
2023-09-11 14:16:05 时间
yanf4j发布一个0.50-beta2版本,这个版本最重要的改进就是引入了客户端连接非阻塞API,主要最近的工作要用到,所以添加了。两个核心类TCPConnectorController和UDPConnectorController分别用于TCP和UDP的客户端连接控制。例如,现在的UDP echo client可以写成:
//客户端echo handler
class EchoClientHandler extends HandlerAdapter {
public void onReceive(Session udpSession, Object t) {
DatagramPacket datagramPacket = (DatagramPacket) t;
System.out.println("recv:" + new String(datagramPacket.getData()));
}
@Override
public void onMessageSent(Session session, Object t) {
System.out.println("send:" + new String((byte[]) t));
}
}
//连接代码,并发送UDP包
UDPConnectorController connector = new UDPConnectorController();
connector.setSoTimeout(1000);
connector.setHandler(new EchoClientHandler());
connector.connect(new InetSocketAddress(InetAddress.getByName(host),
port));
for (int i = 0; i 10000; i++) {
String s = "hello " + i;
DatagramPacket packet = new DatagramPacket(s.getBytes(), s.length());
connector.send(packet);
} UDP不是面向连接的,因此connect方法仅仅是调用了底层DatagramChannel.connect方法,用来限制接收和发送的packet的远程端点。
再来看看TCPConnectorController的使用,同样看Echo Client的实现:
//客户端的echo handler
class EchoHandler extends HandlerAdapter String {
@Override
public void onConnected(Session session) {
try {
//一连接就发送NUM个字符串
for (int i = 0; i NUM; i++)
session.send(generateString(i));
} catch (Exception e) {
}
}
public String generateString(int len) {
StringBuffer sb = new StringBuffer();
for (int i = 0; i MESSAGE_LEN; i++)
sb.append(i);
return sb.toString();
}
@Override
public void onReceive(Session session, String t) {
//打印接收到字符串
if (DEBUG)
System.out.println("recv:" + t);
}
}
//...连接API,TCPConnectorController示例
Configuration configuration = new Configuration();
configuration.setTcpSessionReadBufferSize(256 * 1024); // 设置读的缓冲区大小
TCPConnectorController connector = new TCPConnectorController(configuration,
new StringCodecFactory());
connector.setHandler(new EchoHandler());
connector.setCodecFactory(new StringCodecFactory());
try {
connector.Connect(new InetSocketAddress("localhost", 8080));
} catch (IOExceptione) {
e.printStackTrace();
}
注意,connect方法并不阻塞,而是立即返回,连接是否建立可以通过TCPConnectorController.isConnected()方法来判断,因此通常你可能会这样使用:
try {
connector.Connect(new InetSocketAddress("localhost", 8080));
while(!connector.isConnected())
;
} catch (Exception e) {
e.printStackTrace();
} 来强制确保后面对connector的使用是已经连接上的connector,然而更好的做法是在Handler的onConnected()回调方法中处理逻辑,因为这个方法仅仅在连接建立后才会被调用。
两个ConnectorController都有系列send方法,用于发送数据:
TCPConnectorController.send(Object msg) throws InterruptedException
UDPConnectorController.send(DatagramPacket packet) throws InterruptedException
UDPConnectorController.send(SocketAddress targetAddr, Object msg)throws InterruptedException
0.50-beta2带来的另一个修改就是Session接口添加setReadBufferByteOrder方法,用于设置session接收缓冲区的字节序,默认是网络字节序,也就是大端法。这个方法建议在Handler的onSessionStarted回调方法中调用。
在0.50-beta最重要的修改是引入了session发送队列缓冲区的流量控制选项。默认情况下,session的发送缓冲队列是无界的,队列的push和pop也全然不会阻塞。在设置了缓冲队列的高低水位选项后即引入了发送流量控制,规则如下:
a)当发送队列中的数据总量大于高水位标记(highWaterMark),Session.send将阻塞
b)在条件a的作用下,Session.send的阻塞将持续到发送队列中的数据总量小于于低水位标记(lowWaterMark)才解除。
缓冲队列高低水位的设置通过Controller的下列方法设置:
public void setSessionWriteQueueHighWaterMark(int highWaterMark);
public void setSessionWriteQueueLowWaterMark(int lowWaterMark);
跨端架构下客户端侧API维护方案总结 淘宝App搜索业务侧采用的是局部动态化的跨端技术架构,客户端提供丰富的基础能力与视图组件的API,前端负责业务视图搭建与业务逻辑实现。
国王小组:数字货币交易所开发API客户端的集合 DAPP交易所开发稳定版丨DAPP交易所系统开发(开发案例)丨DAPP交易所系统源码部署 数字货币去中心化交易所开发详情版丨数字货币去中心化交易所系统开发(原生开发)丨数字货币去中心化交易所现成源码模板 去中心化交易所开发丨去中心化交易所系统开发(功能及逻辑)丨去中心化交易所系统源码部署 海外版数字货币交易所系统开发(逻辑及功能)丨多语言数字货币交易所系统开发(案例及源码) 数字货币交易所系统开发(功能版)丨数字货币交易所开发源码案例部署 交易所系统开发(原生开发)丨 交易所系统开发(多语言)丨交易所源码详情 交易所源码案例丨交易所APP开发丨交易所系统开发(海外版) 秒合约交易所系统开发实
etcd通信接口:客户端 API 实践与核心方法 你好,我是 aoho,今天我和你分享的主题是通信接口:客户端 API 实践与核心方法。 我们在前面一课时介绍了 etcd 的整体架构。学习客户端与 etcd 服务端的通信以及 etcd 集群节点的内部通信接口对于我们更好地使用和掌握 etcd 组件很有帮助,也是所必需了解的内容。本课时我们将会介绍 etcd 的 gRPC 通信接口以及客户端的实践。
//客户端echo handler
class EchoClientHandler extends HandlerAdapter {
public void onReceive(Session udpSession, Object t) {
DatagramPacket datagramPacket = (DatagramPacket) t;
System.out.println("recv:" + new String(datagramPacket.getData()));
}
@Override
public void onMessageSent(Session session, Object t) {
System.out.println("send:" + new String((byte[]) t));
}
}
//连接代码,并发送UDP包
UDPConnectorController connector = new UDPConnectorController();
connector.setSoTimeout(1000);
connector.setHandler(new EchoClientHandler());
connector.connect(new InetSocketAddress(InetAddress.getByName(host),
port));
for (int i = 0; i 10000; i++) {
String s = "hello " + i;
DatagramPacket packet = new DatagramPacket(s.getBytes(), s.length());
connector.send(packet);
} UDP不是面向连接的,因此connect方法仅仅是调用了底层DatagramChannel.connect方法,用来限制接收和发送的packet的远程端点。
再来看看TCPConnectorController的使用,同样看Echo Client的实现:
//客户端的echo handler
class EchoHandler extends HandlerAdapter String {
@Override
public void onConnected(Session session) {
try {
//一连接就发送NUM个字符串
for (int i = 0; i NUM; i++)
session.send(generateString(i));
} catch (Exception e) {
}
}
public String generateString(int len) {
StringBuffer sb = new StringBuffer();
for (int i = 0; i MESSAGE_LEN; i++)
sb.append(i);
return sb.toString();
}
@Override
public void onReceive(Session session, String t) {
//打印接收到字符串
if (DEBUG)
System.out.println("recv:" + t);
}
}
//...连接API,TCPConnectorController示例
Configuration configuration = new Configuration();
configuration.setTcpSessionReadBufferSize(256 * 1024); // 设置读的缓冲区大小
TCPConnectorController connector = new TCPConnectorController(configuration,
new StringCodecFactory());
connector.setHandler(new EchoHandler());
connector.setCodecFactory(new StringCodecFactory());
try {
connector.Connect(new InetSocketAddress("localhost", 8080));
} catch (IOExceptione) {
e.printStackTrace();
}
注意,connect方法并不阻塞,而是立即返回,连接是否建立可以通过TCPConnectorController.isConnected()方法来判断,因此通常你可能会这样使用:
try {
connector.Connect(new InetSocketAddress("localhost", 8080));
while(!connector.isConnected())
;
} catch (Exception e) {
e.printStackTrace();
} 来强制确保后面对connector的使用是已经连接上的connector,然而更好的做法是在Handler的onConnected()回调方法中处理逻辑,因为这个方法仅仅在连接建立后才会被调用。
两个ConnectorController都有系列send方法,用于发送数据:
TCPConnectorController.send(Object msg) throws InterruptedException
UDPConnectorController.send(DatagramPacket packet) throws InterruptedException
UDPConnectorController.send(SocketAddress targetAddr, Object msg)throws InterruptedException
0.50-beta2带来的另一个修改就是Session接口添加setReadBufferByteOrder方法,用于设置session接收缓冲区的字节序,默认是网络字节序,也就是大端法。这个方法建议在Handler的onSessionStarted回调方法中调用。
在0.50-beta最重要的修改是引入了session发送队列缓冲区的流量控制选项。默认情况下,session的发送缓冲队列是无界的,队列的push和pop也全然不会阻塞。在设置了缓冲队列的高低水位选项后即引入了发送流量控制,规则如下:
a)当发送队列中的数据总量大于高水位标记(highWaterMark),Session.send将阻塞
b)在条件a的作用下,Session.send的阻塞将持续到发送队列中的数据总量小于于低水位标记(lowWaterMark)才解除。
缓冲队列高低水位的设置通过Controller的下列方法设置:
public void setSessionWriteQueueHighWaterMark(int highWaterMark);
public void setSessionWriteQueueLowWaterMark(int lowWaterMark);
缓冲队列的流量控制想法来自ACE的ACE_Message_Queue,是通过com.google.code.yanf4j.util.MessageQueue类实现的。
0.50-beta还引入了Session.send(Object msg)的重载版本 Session.send(Object msg,long timeout),在超过timeout时间后send仍然阻塞时即终止send。注意,现在Session.send的这两个方法都返回一个bool值来表示send成功与否,并且都将响应中断(仅限启动了流量控制选项)抛出InterruptedException。
文章转自庄周梦蝶 ,原文发布时间2009-02-19
跨端架构下客户端侧API维护方案总结 淘宝App搜索业务侧采用的是局部动态化的跨端技术架构,客户端提供丰富的基础能力与视图组件的API,前端负责业务视图搭建与业务逻辑实现。
国王小组:数字货币交易所开发API客户端的集合 DAPP交易所开发稳定版丨DAPP交易所系统开发(开发案例)丨DAPP交易所系统源码部署 数字货币去中心化交易所开发详情版丨数字货币去中心化交易所系统开发(原生开发)丨数字货币去中心化交易所现成源码模板 去中心化交易所开发丨去中心化交易所系统开发(功能及逻辑)丨去中心化交易所系统源码部署 海外版数字货币交易所系统开发(逻辑及功能)丨多语言数字货币交易所系统开发(案例及源码) 数字货币交易所系统开发(功能版)丨数字货币交易所开发源码案例部署 交易所系统开发(原生开发)丨 交易所系统开发(多语言)丨交易所源码详情 交易所源码案例丨交易所APP开发丨交易所系统开发(海外版) 秒合约交易所系统开发实
etcd通信接口:客户端 API 实践与核心方法 你好,我是 aoho,今天我和你分享的主题是通信接口:客户端 API 实践与核心方法。 我们在前面一课时介绍了 etcd 的整体架构。学习客户端与 etcd 服务端的通信以及 etcd 集群节点的内部通信接口对于我们更好地使用和掌握 etcd 组件很有帮助,也是所必需了解的内容。本课时我们将会介绍 etcd 的 gRPC 通信接口以及客户端的实践。
相关文章
- VS2017 处理 Rdlc , microsoft report viewer 轻量级报表处理(WPF CS客户端版本)
- HBase Thrift客户端Java API实践
- Flume C# Thrift客户端
- Asp.Net Web API 2第三课——.NET客户端调用Web API
- Oracle 客户端连接时报ORA-01019错误总结
- cnblogs_client博客园客户端——雏形
- 客户端同步服务器端时间方案
- linux svn客户端 常用命令
- C# 海康DVR客户端开发系列(2)—— 封装API
- Nginx 获取客户端真实IP $remote_addr与X-Forwarded-For
- 使用Fiddler监控网易云笔记客户端向服务器定时发送的sync请求
- 使用邮件客户端整合日常信息
- Atitit Persistence API持久性标准化法总结 目录 1. 持久性对于大多数企业应用程序都非常要害1 2. 持久化api内容2 2.1. 一种声明式地执行O-R映射的方式。2
- Atitit.常用语言的常用内部api 以及API兼容性对源码级别可移植的重要性 总结
- Atitit.常用语言的常用内部api 以及API兼容性对源码级别可移植的重要性 总结
- 服务器检测到客户端退出或崩溃后,如何优雅地做出反应
- Proxifer+BurpSuite 抓取PC客户端HTTP(s)数据包
- SSL代理——见图,利用SSL代理证书替换加密Web网站的数字证书,并将SSL代理证书发送到客户端的Web浏览器,中间代理以此获得加密通信的明文内容进行和真实服务器的交互转发
- JSP 客户端请求
- Java与WCF交互(一):Java客户端调用WCF服务
- 高性能对象存储MinIO学习&API使用&使用api创建文件夹&MinIO工具类
- Zookeeper客户端Curator使用详解
- Zookeeper(5):Zookeeper的客户端 API 操作