示例代码:

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编程的更多相关文章

  1. Python网络编程03 /缓存区、基于TCP的socket循环通信、执行远程命令、socketserver通信

    Python网络编程03 /缓存区.基于TCP的socket循环通信.执行远程命令.socketserver通信 目录 Python网络编程03 /缓存区.基于TCP的socket循环通信.执行远程命 ...

  2. POCO库中文编程参考指南(8)丰富的Socket编程

    POCO库中文编程参考指南(8)丰富的Socket编程 作者:柳大·Poechant 博客:Blog.CSDN.net/Poechant 邮箱:zhongchao.ustc#gmail.com (# ...

  3. Java网络编程学习A轮_01_目标与基础复习

    A. A轮目标 复习网络编程基础知识,重点学习下TCP三次握手四次挥手,以及可能引发的异常情况. 回顾 Socket 编程,好多年没写(chao)过相关代码了. 重学 NIO,以前学的基本忘光了,毕竟 ...

  4. Java网络编程学习A轮_06_NIO入门

    参考资料: 老外写的教程,很适合入门:http://tutorials.jenkov.com/java-nio/index.html 上面教程的译文:http://ifeve.com/overview ...

  5. Java网络编程学习A轮_08_NIO的Reactor模型

    参考资料: 了解 Java NIO 的 Reactor 模型,大神 Doug Lea 的 PPT Scalable IO in Java 必看:http://gee.cs.oswego.edu/dl/ ...

  6. Java网络编程学习A轮_05_Socket编程

    示例代码: https://github.com/gordonklg/study,socket module A. Socket 编程简单例子 最简单的 Socket 编程是通过回车/换行符,整行读取 ...

  7. Java基于TCP的Socket编程练习

    环境:Notpad ++ 6.0 + JDK 6.0.31 问题:使用套接字编写客户-服务器程序,实现客户-服务器交互计算.客户将三角形3个边的长度发给服务器,服务器把计算出的三角形的面积返回给客户. ...

  8. 基于MFC的socket编程(异步非阻塞通信)

       对于许多初学者来说,网络通信程序的开发,普遍的一个现象就是觉得难以入手.许多概念,诸如:同步(Sync)/异步(Async),阻塞(Block)/非阻塞(Unblock)等,初学者往往迷惑不清, ...

  9. 基于win32的socket编程及程序实现

    初步研究了win32平台的Windows Sockets,它是Microsoft Windows的网络程序设计接口,它是从Berkeley Sockets扩展而来的,以动态链接库的形式提供给我们使用. ...

随机推荐

  1. 次小生成树(poj1679)

    The Unique MST Time Limit: 1000MS   Memory Limit: 10000K Total Submissions: 20737   Accepted: 7281 D ...

  2. Hibernate之核心文件

    一个Hibernate项目核心配置文件主要分为以下三个方面:1.配置文件hibernate.cfg.xml:2.配置文件*.hbm.xml,此文件一般包括映射文件的结构以及各种属性的映射方式:3.Hi ...

  3. 170707、springboot编程之监控和管理生产环境

    spring-boot-actuator模块提供了一个监控和管理生产环境的模块,可以使用http.jmx.ssh.telnet等拉管理和监控应用.审计(Auditing). 健康(health).数据 ...

  4. Yii2如何批量添加数据

    批量添加这个操作,在实际开发中经常用得到,今天小编抽空给大家整理些有关yii2批量添加的问题,感兴趣的朋友一起看看吧. 在上篇文章给大家介绍了关于浅析Yii2 gridview实现批量删除教程,当然, ...

  5. nginx挂维护页面

    本篇文章摘抄于他人的文章,来自于CSDN的JeremyIT同学,但我还是自己重新敲一遍. 实现的效果是:访问网站的任何页面,都跳转到同一个页面.而这一个页面就是维护页面,可以根据需要修改. serve ...

  6. C# WinForm实现任务栏程序图标闪烁

    相信大家在用QQ的时候都会知道,你打开了QQ聊天窗口,如果窗口不是当前激活的窗口的话,收到QQ消息时,任务栏(不是托盘图标)上的图标会闪一下变成黄色(Win7默认主题下),用以通知用户有消息进来了,之 ...

  7. cocos2d 特效

    一.特效概念 特效是让精灵(CCSprite)执行某种特殊的效果.其实,特效也是一种动画! 但是,为什么要把特效与动画区分呢?因为,特效是基于网格属性来进行的. 如何区分动画与特效?简单的将,当使用到 ...

  8. ArcEngine和GDAL读写栅格数据机制对比(一)

    最近应用AE开发插值和栅格转等值线的程序,涉及到栅格读写的有关内容.联想到ArcGIS利用了GDAL的某些东西,从AE的OMD中也发现RasterDataset和RasterBand这些命名和GDAL ...

  9. Openstack(十六)实现内外网结构

    类似于阿里云ECS主机的内外网(双网卡不通网段)的结构,最终实现内外网区分隔离. https://www.aliyun.com/product/ecs/?utm_medium=text&utm ...

  10. Spark Core(二)Driver上的Task的生成、分配、调度(转载)

    1. 什么是Task? 在前面的章节里描述过几个角色,Driver(Client),Master,Worker(Executor),Driver会提交Application到Master进行Worke ...