由于项目上的 http 请求量较大,项目上性能跟不上。于是考虑把 短连接的 http 换成 长连接的tcp 形式 试试效果。

先 研究了一下 长连接方式。就是要用到 socket 方面的知识。

package com.bkc.bpmp.common.utils;

import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.io.PrintWriter;
import java.net.ServerSocket;
import java.net.Socket; /**
* 搭建服务器端 a)、创建ServerSocket对象绑定监听端口。 b)、通过accept()方法监听客户端的请求。 c)、建立连接后,通过输入输出流读取客户端发送的请求信息。 d)、通过输出流向客户端发送请求信息。 e)、关闭相关资源。
* @author ppnie
*
*/
public class SocketService
{
// 搭建服务器端
public static void main(String[] args)
throws IOException
{
SocketService socketService = new SocketService();
// 1、a)创建一个服务器端Socket,即SocketService
socketService.oneServer();
} public void oneServer()
{
try
{
ServerSocket server = null;
try
{
server = new ServerSocket(5209);
// b)指定绑定的端口,并监听此端口。
System.out.println("服务器启动成功");
// 创建一个ServerSocket在端口5209监听客户请求
}
catch (Exception e)
{
System.out.println("没有启动监听:" + e);
// 出错,打印出错信息
}
Socket socket = null;
try
{
socket = server.accept();
// 2、调用accept()方法开始监听,等待客户端的连接
// 使用accept()阻塞等待客户请求,有客户
// 请求到来则产生一个Socket对象,并继续执行
}
catch (Exception e)
{
System.out.println("Error." + e);
// 出错,打印出错信息
}
// 3、获取输入流,并读取客户端信息
String line;
BufferedReader in = new BufferedReader(new InputStreamReader(socket.getInputStream()));
// 由Socket对象得到输入流,并构造相应的BufferedReader对象
PrintWriter writer = new PrintWriter(socket.getOutputStream());
// 由Socket对象得到输出流,并构造PrintWriter对象
BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
// 由系统标准输入设备构造BufferedReader对象
System.out.println("Client:" + in.readLine());
// 在标准输出上打印从客户端读入的字符串
line = br.readLine();
// 从标准输入读入一字符串
// 4、获取输出流,响应客户端的请求
while (!line.equals("end"))
{
// 如果该字符串为 "bye",则停止循环
writer.println(line);
// 向客户端输出该字符串
writer.flush();
// 刷新输出流,使Client马上收到该字符串
System.out.println("Server:" + line);
// 在系统标准输出上打印读入的字符串
System.out.println("Client:" + in.readLine());
// 从Client读入一字符串,并打印到标准输出上
line = br.readLine();
// 从系统标准输入读入一字符串
} // 继续循环 // 5、关闭资源
writer.close(); // 关闭Socket输出流
in.close(); // 关闭Socket输入流
socket.close(); // 关闭Socket
server.close(); // 关闭ServerSocket
}
catch (Exception e)
{// 出错,打印出错信息
System.out.println("Error." + e);
}
}
} package com.bkc.bpmp.common.utils; import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.io.PrintWriter;
import java.net.Socket; public class SocketClient
{
// 搭建客户端
public static void main(String[] args)
throws IOException
{
try
{
// 1、创建客户端Socket,指定服务器地址和端口
// Socket socket=new Socket("127.0.0.1",5200);
Socket socket = new Socket("192.168.1.99", 5209);
System.out.println("客户端启动成功");
// 2、获取输出流,向服务器端发送信息
// 向本机的52000端口发出客户请求
BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
// 由系统标准输入设备构造BufferedReader对象
PrintWriter write = new PrintWriter(socket.getOutputStream());
// 由Socket对象得到输出流,并构造PrintWriter对象
// 3、获取输入流,并读取服务器端的响应信息
BufferedReader in = new BufferedReader(new InputStreamReader(socket.getInputStream()));
// 由Socket对象得到输入流,并构造相应的BufferedReader对象
String readline;
readline = br.readLine(); // 从系统标准输入读入一字符串
while (!readline.equals("end"))
{
// 若从标准输入读入的字符串为 "end"则停止循环
write.println(readline);
// 将从系统标准输入读入的字符串输出到Server
write.flush();
// 刷新输出流,使Server马上收到该字符串
System.out.println("Client:" + readline);
// 在系统标准输出上打印读入的字符串
System.out.println("Server:" + in.readLine());
// 从Server读入一字符串,并打印到标准输出上
readline = br.readLine(); // 从系统标准输入读入一字符串
} // 继续循环
// 4、关闭资源
write.close(); // 关闭Socket输出流
in.close(); // 关闭Socket输入流
socket.close(); // 关闭Socket
}
catch (Exception e)
{
System.out.println("can not listen to:" + e);// 出错,打印出错信息
}
} }

这是一个简单的 用java 实现 socket 客户端与服务端的 代码。

但是这样的例子,并不能实际运用与项目之中。在实际项目中需要考虑到 套接字的连接什么时候关闭,流什么时候关闭,参数的传入与信息的响应都需要得到结果。

若是这里使用的方法同 http 的话,应该是 一个请求一个对应的响应结果,若是长连接,应该如何把多线程与长连接 结合起来呢?

现在的项目是 使用C++ 服务端 和 java 客户端  来使用 socket 进行通信。

C++ 与 Java 之间的通信还涉及到了一个 网络字节码 不一致的问题。以及 还有一个 自定义协议 的问题

假设 自定义协议 为

java 客户端代码(至于这里的BytePtr ,就是  网络字节码 的转码)

public class BytePtr {

    public static byte[] toLH(short n) {
byte[] b = new byte[2];
b[0] = (byte) (n & 0xff);
b[1] = (byte) (n >> 8 & 0xff);
return b;
} public static byte [] toLH( int n) {
byte [] b = new byte [ 4 ];
b[0 ] = ( byte ) (n & 0xff );
b[1 ] = ( byte ) (n >> 8 & 0xff );
b[2 ] = ( byte ) (n >> 16 & 0xff );
b[3 ] = ( byte ) (n >> 24 & 0xff );
return b;
} public static byte[] addBytes(byte[] data1, byte[] data2) {
byte[] data3 = new byte[data1.length + data2.length];
System.arraycopy(data1, 0, data3, 0, data1.length);
System.arraycopy(data2, 0, data3, data1.length, data2.length);
return data3; }
} import java.io.DataInputStream;
import java.io.DataOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.net.Socket;
import java.net.UnknownHostException;
import java.util.Scanner; public class ChatClient
{ public static void main(String[] args)
{
try
{
Socket client = new Socket("192.168.1.108", 10008);
OutputStream out = client.getOutputStream();
DataOutputStream outs = new DataOutputStream(out);
Scanner scaner = new Scanner(System.in);
genProtocol(outs, scaner.next());
parseProtocol(client); outs.close();
out.close();
client.close();
}
catch (UnknownHostException e)
{
e.printStackTrace();
}
catch (IOException e)
{
e.printStackTrace();
}
} /**
* 构造协议
*
* @param out
* @param msg
* @throws IOException
*/
private static void genProtocol(DataOutputStream out, String msg)
throws IOException
{
short type = 10001; // 消息类型
byte[] bytes = msg.getBytes(); // 消息内容
int totalLen = bytes.length; // 消息长度
out.write((BytePtr.addBytes(BytePtr.addBytes(BytePtr.toLH(type), BytePtr.toLH(totalLen)), bytes))); // 写入消息内容
out.flush();
} private static void parseProtocol(Socket client)
throws IOException
{
InputStream is = client.getInputStream();
DataInputStream dis = new DataInputStream(is); // 读取Java标准数据类型的输入流 // 协议解析 int totalLen = dis.readInt(); // 读取消息长度
byte[] data = new byte[totalLen]; // 定义存放消息内容的字节数组
int len = is.read(data);
String text = new String(data, 0, len);
System.out.println("发来的内容是:" + text); is.close();
dis.close();
} }

哦,这里还提到 一个工具 Wireshark,用来抓取包,来查看一下网络字节码到底传送的是一些什么

哦,还有  提到的 netty  技术,至于怎么用,还是先看看再说。

至于 有一些电子书之类的,可以 到脚本之家 去下载。

最最重要的一点,下载的电子书、视频、源码 之类的,一定要看啊。

Socket 相关资料(随笔)的更多相关文章

  1. 合宙模块LUA相关资料汇总

    1. 目录 1. 目录 [2. LUA二次开发](#2. LUA二次开发) 2.1 [新手教程](#2.1 新手教程) 2.2 [进阶教程](#2.2 进阶教程) 2.3 [LUA开发环境](#2.3 ...

  2. 全文检索解决方案(lucene工具类以及sphinx相关资料)

    介绍两种全文检索的技术. 1.  lucene+ 中文分词(IK) 关于lucene的原理,在这里可以得到很好的学习. http://www.blogjava.net/zhyiwww/archive/ ...

  3. React Test相关资料

    karma 前端测试驱动器,生产测试报告,多个浏览器 mocha js的测试框架,相当于junit chai,单元测试的断言库,提供expect shudl assert enzyme sinon.j ...

  4. iOS10以及xCode8相关资料收集

    兼容iOS 10 资料整理笔记 源文:http://www.jianshu.com/p/0cc7aad638d9 1.Notification(通知) 自从Notification被引入之后,苹果就不 ...

  5. Nao 类人机器人 相关资料

    Nao 类人机器人 相关资料: 1.兄妹 PEPPER :在山东烟台生产,http://www.robot-china.com/news/201510/30/26564.html 2.国内机器人领先公 ...

  6. GBrowse配置相关资料

    GBrowse配置相关资料(形状.颜色.配置.gff3) http://gmod.org/wiki/Glyphs_and_Glyph_Optionshttp://gmod.org/wiki/GBrow ...

  7. AssetBundle机制相关资料收集

    原地址:http://www.cnblogs.com/realtimepixels/p/3652075.html AssetBundle机制相关资料收集 最近网友通过网站搜索Unity3D在手机及其他 ...

  8. 转:基于IOS上MDM技术相关资料整理及汇总

    一.MDM相关知识: MDM (Mobile Device Management ),即移动设备管理.在21世纪的今天,数据是企业宝贵的资产,安全问题更是重中之重,在移动互联网时代,员工个人的设备接入 ...

  9. smb相关资料

    smb相关资料 看资料就上维基 https://en.wikipedia.org/wiki/Server_Message_Block#Implementation http://www.bing.co ...

随机推荐

  1. BZOJ.2160.拉拉队排练(Manacher)

    题目链接 \(Description\) 求给定字符串中 最长的k个回文串长度的乘积(要求回文串长度为奇数):若奇数长度回文串不足k个则输出-1.(len<=10^6,k<=10^12) ...

  2. 洛谷.3437.[POI2006]TET-Tetris 3D(二维线段树)

    题目链接 下落一个d*s的方块,则要在这个平面区域找一个最高的h' 更新整个平面区域的值为h+h' 对于本题,维护最大高度h和all 对于平面的x轴维护一棵线段树t1,每个t1的节点维护对应y轴的两棵 ...

  3. Flask启动原理,源码流程分析

    1.执行Flask的实例对象.run()方法 from flask import Flask,request,session app = Flask(__name__) app.secret_key ...

  4. 网络名词拾遗--part2

    网络名词拾遗--part2 关于所谓的连接上限 先要明白服务端和客户端的交互逻辑: 服务端创建socket 与提供对外服务的port端口绑定 开始监听 客户端向这个端口提出请求 服务端接收到这个请求后 ...

  5. filter的dispatcher节点

    1.FORWARD (1)a.jsp页面内容如此<jsp:forward page="/b.jsp"></jsp:forward> (2) a.jsp页面内 ...

  6. Oracle 拼接列数据的方法

    select wm_concat(fphone) phone from dq_phone_ndtbdxz wm_concat(列名):把多列值,合并成一列,用,隔开.

  7. Xshell配置ssh免密码登录-密钥公钥(Public key)与私钥(Private Key)登录【已成功实例】

    本文转自https://blog.csdn.net/qjc_501165091/article/details/51278696 ssh登录提供两种认证方式:口令(密码)认证方式和密钥认证方式.其中口 ...

  8. Maven项目搭建-Eclipse版

    一.Maven简单介绍 Maven是基于Java平台的项目构建(mvn clean install).依赖管理(中央仓库,Nexus)和项目信息管理的项目管理工具. Maven是基于项目对象模型(PO ...

  9. C++11 多线程编程 使用lambda创建std::thread (生产/消费者模式)

    要写个tcp server / client的博客,想着先写个c++11多线程程序.方便后面写博客使用. 目前c++11中写多线程已经很方便了,不用再像之前的pthread_create,c++11中 ...

  10. 教育单元测试mock框架优化之路(中)

    转载:https://sq.163yun.com/blog/article/169564470918451200 三.间接依赖的bean的mock替换 对于前面提供的@Mock,@Spy+@Injec ...