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扩展而来的,以动态链接库的形式提供给我们使用. ...
随机推荐
- 微信小程序 --- action-sheet底部弹框
action-sheet:从屏幕底部弹出一个菜单,选择: 使用的时候,在给不同的 action-sheet-item 添加不同的事件. 效果: (这里的确定可以有多个) 代码: <button ...
- postgresql----SELECT
示例1.简单查询 使用*查询表所有的字段,也可以指定字段名查询 test=# select * from tbl_insert; a | b ---+---- | sd | ff ( rows) te ...
- pta 习题集 5-5 最长连续递增子序列 (dp)
给定一个顺序存储的线性表,请设计一个算法查找该线性表中最长的连续递增子序列.例如,(1,9,2,5,7,3,4,6,8,0)中最长的递增子序列为(3,4,6,8). 输入格式: 输入第1行给出正整数n ...
- nginx挂维护页面
本篇文章摘抄于他人的文章,来自于CSDN的JeremyIT同学,但我还是自己重新敲一遍. 实现的效果是:访问网站的任何页面,都跳转到同一个页面.而这一个页面就是维护页面,可以根据需要修改. serve ...
- Oracle安装部署之RAC安装环境配置脚本
#!/bin/bash#Usage:Log on as the superuser('root'),and then execute the command:#./1preusers.sh group ...
- Ora-1157 ora-1110错误解决案例一枚
1.数据库打开报错如下: SQL> alter database open; alter database open * ERROR at line 1: ORA-01157: cannot i ...
- Pycharm一直报ImportError: No module named requests
1.首先检查是否安装了requests l 安装命令:pip install requests如果出现了Requirement already satisfied 代表安装成功 2.系统含有多个版本的 ...
- Redis的一些结构
- Java中参数传递时值传递的机制分析
参数传递是什么? 在C的函数或是JAVA的方法中,向一个函数或方法内部传递一个参数,比如: void fun( int num ){ num+=2 ; } int a = 3 ...
- C语言头文件#include<stdlib.h>的作用
stdlib 头文件即standard library标准库头文件 stdlib 头文件里包含了C.C++语言的最常用的系统函数 该文件包含了的C语言标准库函数的定义 stdlib.h里面定义了五 ...