Java网络编程学习A轮_07_基于Buffer的Socket编程
示例代码:
https://github.com/gordonklg/study,socket module
A. LineSeparate
基于 Buffer 实现逐行读取的 EchoServer 比传统 Socket 编程困难,相当于需要自己通过 Buffer 实现 BufferedReader 的 readLine 功能。
代码如下,假设单行不超过256字节,支持 Win 和 Linux(不支持单 \r 作为换行符)系统,空行忽略。
代码就不分析了,写了好久才跑对测试,分包粘包真是麻烦,要去刷 LeetCode 基本题提高编码能力了,不能整天都 CTRL C CTRL V 啊。
gordon.study.socket.nio.basic.LineSeparateBlockingEchoServer.java
public class LineSeparateBlockingEchoServer {
public static void main(String[] args) throws Exception {
for (int i = 0; i < 3; i++) {
new Thread(new Client()).start();
}
ServerSocketChannel serverSocketChannel = ServerSocketChannel.open();
serverSocketChannel.bind(new InetSocketAddress(8888));
while (true) {
SocketChannel socketChannel = serverSocketChannel.accept();
new Thread(new ServerHandler(socketChannel)).start();
}
}
private static class ServerHandler implements Runnable {
private SocketChannel socketChannel;
private int lastScannedPos = 0;
public ServerHandler(SocketChannel socketChannel) {
this.socketChannel = socketChannel;
}
@Override
public void run() {
try {
ByteBuffer buf = ByteBuffer.allocate(256);
ByteBuffer writeBuf = ByteBuffer.allocate(256);
byte[] content = null;
while (true) {
if (socketChannel.read(buf) > 0) {
do {
content = extractLine(buf);
if (content != null) {
echo(writeBuf, content);
}
} while (content != null && buf.position() > 0);
}
}
} catch (Exception e) {
e.printStackTrace();
}
}
private byte[] extractLine(ByteBuffer buf) {
byte[] result = null;
int totalLen = buf.position();
buf.position(lastScannedPos);
for (int index = lastScannedPos; index < totalLen; index++) {
if (buf.get() == '\n') {
result = new byte[index - (hasSlashRBeforeSlashN(buf) ? 1 : 0)];
buf.position(0);
buf.get(result);
buf.position(index + 1);
buf.limit(totalLen);
buf.compact();
lastScannedPos = 0;
return result.length == 0 ? null : result;
}
}
lastScannedPos = buf.position();
return result;
}
private boolean hasSlashRBeforeSlashN(ByteBuffer buf) {
int posOfSlashN = buf.position() - 1;
if (posOfSlashN > 0) {
return (buf.get(posOfSlashN - 1) == '\r');
}
return false;
}
private void echo(ByteBuffer writeBuf, byte[] content) throws IOException {
System.out.println("ECHO: " + new String(content));
writeBuf.clear();
writeBuf.put(content);
writeBuf.put("\n".getBytes());
writeBuf.flip();
while (writeBuf.hasRemaining()) {
socketChannel.write(writeBuf);
}
}
}
private static class Client implements Runnable {
@Override
public void run() {
try (Socket socket = new Socket()) {
socket.connect(new InetSocketAddress(8888));
DataOutputStream dos = new DataOutputStream(socket.getOutputStream());
dos.write("hello\n".getBytes());
Thread.sleep(100);
dos.write("\n".getBytes());
Thread.sleep(100);
dos.write("你瞅啥?\r\n".getBytes());
Thread.sleep(100);
dos.write("\r\nchi".getBytes());
Thread.sleep(100);
dos.write(" le ".getBytes());
Thread.sleep(100);
dos.write("ma?\nni hao\n\nhi d".getBytes());
Thread.sleep(100);
dos.write("ude\r\n".getBytes());
Thread.sleep(100);
dos.close();
} catch (Exception e) {
e.printStackTrace();
}
}
}
}
B.CustomProtocol
最经典的自定义协议就是基于 TLV (Type, Length, Value) 格式编码。我们约定 Type 占用1字节,0表示通讯结束,1表示文本消息;Length 占用2字节。
代码依然很难写,而且极不优雅,留给自己以后吐槽用吧。
gordon.study.socket.nio.basic.CustomProtocolBlockingPrintServer.java
public class CustomProtocolBlockingPrintServer {
public static void main(String[] args) throws Exception {
for (int i = 0; i < 3; i++) {
new Thread(new Client()).start();
}
ServerSocketChannel serverSocketChannel = ServerSocketChannel.open();
serverSocketChannel.bind(new InetSocketAddress(8888));
while (true) {
SocketChannel socketChannel = serverSocketChannel.accept();
new Thread(new ServerHandler(socketChannel)).start();
}
}
private static class ServerHandler implements Runnable {
private SocketChannel socketChannel;
private int nextMsgLen = 0;
public ServerHandler(SocketChannel socketChannel) {
this.socketChannel = socketChannel;
}
@Override
public void run() {
try {
ByteBuffer buf = ByteBuffer.allocate(256);
while (!Thread.currentThread().isInterrupted()) {
if (socketChannel.read(buf) > 0) {
extractMessageAndPrint(buf);
}
}
System.out.println("===============exit==============");
} catch (Exception e) {
e.printStackTrace();
}
}
private void extractMessageAndPrint(ByteBuffer buf) {
if (nextMsgLen == 0) {// means we havn't get full "head" info
buf.flip();
int type = buf.get();
if (type == 0) {
Thread.currentThread().interrupt();
return;
}
if (buf.remaining() < 2) {
buf.rewind();
buf.compact();
} else {
int length = buf.getChar();
if (buf.remaining() < length - 3) {
nextMsgLen = length;
buf.rewind();
buf.compact();
} else {
byte[] content = new byte[length - 3];
buf.get(content);
System.out.println(new String(content));
buf.compact();
if (buf.position() > 0) {
extractMessageAndPrint(buf);
}
}
}
} else {
buf.flip();
if (buf.remaining() >= nextMsgLen) {
byte[] content = new byte[nextMsgLen - 3];
buf.position(3);
buf.get(content);
System.out.println(new String(content));
buf.compact();
nextMsgLen = 0;
if (buf.position() > 0) {
extractMessageAndPrint(buf);
}
} else {
buf.compact();
}
}
}
}
private static class Client implements Runnable {
@Override
public void run() {
try (Socket socket = new Socket()) {
socket.connect(new InetSocketAddress(8888));
DataOutputStream dos = new DataOutputStream(socket.getOutputStream());
print(dos, "hello");
Thread.sleep(100);
print(dos, "");
Thread.sleep(100);
print(dos, "你瞅啥?");
Thread.sleep(100);
dos.writeByte(1);
dos.flush();
Thread.sleep(100);
dos.write((byte) (9 >> 8 & 0xFF));
dos.flush();
Thread.sleep(100);
dos.write((byte) (9 & 0xFF));
dos.flush();
Thread.sleep(100);
dos.write("ni".getBytes());
dos.flush();
Thread.sleep(100);
dos.write(" ".getBytes());
dos.flush();
Thread.sleep(100);
ByteBuffer buf = ByteBuffer.allocate(100);
buf.put("hao".getBytes());
buf.put((byte) 1);
buf.put((byte) 0);
buf.put((byte) 9);
buf.put("abcdef".getBytes());
buf.put((byte) 1);
buf.put((byte) 0);
buf.put((byte) 8);
buf.put("12345".getBytes());
buf.flip();
byte[] bytes = new byte[buf.remaining()];
buf.get(bytes);
dos.write(bytes);
dos.flush();
Thread.sleep(100);
dos.writeByte(0);
dos.close();
} catch (Exception e) {
e.printStackTrace();
}
}
private void print(DataOutputStream dos, String message) throws IOException {
byte[] bytes = message.getBytes();
int totalLength = 3 + bytes.length;
dos.writeByte(1);
dos.write((byte) (totalLength >> 8 & 0xFF));
dos.write((byte) (totalLength & 0xFF));
dos.write(bytes);
dos.flush();
}
}
}
Java网络编程学习A轮_07_基于Buffer的Socket编程的更多相关文章
- Python网络编程03 /缓存区、基于TCP的socket循环通信、执行远程命令、socketserver通信
Python网络编程03 /缓存区.基于TCP的socket循环通信.执行远程命令.socketserver通信 目录 Python网络编程03 /缓存区.基于TCP的socket循环通信.执行远程命 ...
- POCO库中文编程参考指南(8)丰富的Socket编程
POCO库中文编程参考指南(8)丰富的Socket编程 作者:柳大·Poechant 博客:Blog.CSDN.net/Poechant 邮箱:zhongchao.ustc#gmail.com (# ...
- Java网络编程学习A轮_01_目标与基础复习
A. A轮目标 复习网络编程基础知识,重点学习下TCP三次握手四次挥手,以及可能引发的异常情况. 回顾 Socket 编程,好多年没写(chao)过相关代码了. 重学 NIO,以前学的基本忘光了,毕竟 ...
- Java网络编程学习A轮_06_NIO入门
参考资料: 老外写的教程,很适合入门:http://tutorials.jenkov.com/java-nio/index.html 上面教程的译文:http://ifeve.com/overview ...
- Java网络编程学习A轮_08_NIO的Reactor模型
参考资料: 了解 Java NIO 的 Reactor 模型,大神 Doug Lea 的 PPT Scalable IO in Java 必看:http://gee.cs.oswego.edu/dl/ ...
- Java网络编程学习A轮_05_Socket编程
示例代码: https://github.com/gordonklg/study,socket module A. Socket 编程简单例子 最简单的 Socket 编程是通过回车/换行符,整行读取 ...
- Java基于TCP的Socket编程练习
环境:Notpad ++ 6.0 + JDK 6.0.31 问题:使用套接字编写客户-服务器程序,实现客户-服务器交互计算.客户将三角形3个边的长度发给服务器,服务器把计算出的三角形的面积返回给客户. ...
- 基于MFC的socket编程(异步非阻塞通信)
对于许多初学者来说,网络通信程序的开发,普遍的一个现象就是觉得难以入手.许多概念,诸如:同步(Sync)/异步(Async),阻塞(Block)/非阻塞(Unblock)等,初学者往往迷惑不清, ...
- 基于win32的socket编程及程序实现
初步研究了win32平台的Windows Sockets,它是Microsoft Windows的网络程序设计接口,它是从Berkeley Sockets扩展而来的,以动态链接库的形式提供给我们使用. ...
随机推荐
- angularJS表达式详解!
angularJS的表达式很像Javascript里的表达式:它可以包含文字,运算符和变量: angularJS 表达式: - 数字:{{100+100}} - 字符串:{{‘hello’+'angu ...
- c++Template 的辨析
1.在c++Template中很多地方都用到了typename与class这两个关键字,而且好像可以替换,是不是这两个关键字完全一样呢? 答:class用于定义类,在模板引入c++后,最初定义模板的方 ...
- linux注意的一些地方
assert宏的原型定义在<assert.h>中,其作用是如果它的条件返回错误,则终止程序执行 #include <assert.h>void assert( int expr ...
- [MongoDB] 用户权限管理
在新环境按照原来的步骤新装了MongoDB,结果出现了一些错误,才发现版本升到了2.6.1,用户权限相关的内容全部改掉了. 现在使用Role来管理用户,有一些内置的Role,也可以自定义Role. 内 ...
- vue.js实现初了解(一)
1. vue 2.0是用Flow做静态类型检查, 3.0对TypeScript的支持更好了: 2. vue.js是基于Rollup(更轻量,适合js库的构建)构建的,它的构建相关配置都在scripts ...
- jquery插件方式实现table查询功能
1.写插件部分,如下: ;(function($){ $.fn.plugin = function(options){ var defaults = { //各种属性,各种参数 } var optio ...
- sprintf函数详解
转摘声明:选自<CSDN 社区电子杂志——C/C++杂志> 在将各种类型的数据构造成字符串时,sprintf 的强大功能很少会让你失望.由于sprintf 跟printf 在用法上几乎一样 ...
- 徐州网络赛B-BE,GE or NE【记忆化搜索】【博弈论】
In a world where ordinary people cannot reach, a boy named "Koutarou" and a girl named &qu ...
- (2.12)Mysql之SQL基础——存储过程条件定义与错误处理
转自:博客园桦仔 5.存储过程条件定义与错误处理 -- (1)定义 [1]条件定义:declare condition_name condition for condition_value;[2]错误 ...
- 开发一个根据xml创建代理类的小框架
github地址 https://github.com/1367356/GradleTestUseSubModule/tree/master/CreateMyFrameWork 1:定义一些规则