Tcp编程常见问题及解决方法总结
2014-08-12 15:23
344 查看
问题1、粘包问题
解决方法一:TCP提供了强制数据立即传送的操作指令push,TCP软件收到该操作指令后,就立即将本段数据发送出去,而不必等待发送缓冲区满;
解决方法二:发送固定长度的消息
解决方法三:把消息的尺寸与消息一块发送
解决方法四:双方约定每次传送的大小
解决方法五:双方约定使用特殊标记来区分消息间隔
解决方法六:标准协议按协议规则处理,如Sip协议
问题2、字符串编码问题
将中文字符串用utf8编码格式转换为字节数组发送时,一个中文字符可能会占用2~4个字节(假设为3个字节),这3个字节可能分3次接收,接收端每次接收完后用utf8编码格式转换为字符串,就会出现乱码,并导致接收长度计算错误的情况。
解决方法一:以字节数做为消息长度的计算单位,而不是字符个数。
解决方法二:发送方和接收方都采用unicode编码格式。
问题3、长连接的保活问题
标准TCP层协议里把对方超时设为2小时,若服务器端超过了2小时还没收到客户的信息,它就发送探测报文段,若发送了10个探测报文段(每一个相隔75S)还没有收到响应,就假定客户出了故障,并终止这个连接。因此应对tcp长连接进行保活。
以下是异步通信时会遇到的问题:
问题4、缓冲区脏数据问题
同步发送的拷贝,是直接拷贝数据到基础系统缓冲区,拷贝完成后返回;
异步发送消息的拷贝,是将Socket自带的Buffer空间内的所有数据,拷贝到基础系统发送缓冲区,并立即返回;
因此异步发送时缓冲区设置不好会导致接收到脏数据的问题,如下所示:
第一次发送数据:1234567890
第一次接受数据:1234567890
第二次发送数据:abc
第二次接受数据:abc4567890
请参考:/article/4967046.html
解决方法一:将缓冲区的大小设置为实际发送数据的大小。
问题5、内存碎片问题
频繁的申请缓冲区会导致内存碎片的问题。
解决方法一:使用对象池和内存池。
请参考MSDN:http://msdn.microsoft.com/zh-cn/library/bb517542(v=vs.100).aspx
http://msdn.microsoft.com/zh-cn/library/system.net.sockets.socketasynceventargs.socketasynceventargs(v=vs.100).aspx
问题6、乱序问题
多个线程使用异步通信方式向同一个接收端(socket)同时发送数据,会导致接收端接收的数据混乱。如下所示:
线程1第一次发送:123456789,假设未发送完,只发送了123
线程2第一次发送:abcdefgh,假设未发送完,只发送了abc
线程1第二次发送:456789,发送完成
线程2第二次发送:defgh,发送完成
接收端最终接收的数据为:123abc456789defgh。
解决方法一:一个连接的发送端线程排队发送数据。
解决方法一:TCP提供了强制数据立即传送的操作指令push,TCP软件收到该操作指令后,就立即将本段数据发送出去,而不必等待发送缓冲区满;
解决方法二:发送固定长度的消息
解决方法三:把消息的尺寸与消息一块发送
解决方法四:双方约定每次传送的大小
解决方法五:双方约定使用特殊标记来区分消息间隔
解决方法六:标准协议按协议规则处理,如Sip协议
问题2、字符串编码问题
将中文字符串用utf8编码格式转换为字节数组发送时,一个中文字符可能会占用2~4个字节(假设为3个字节),这3个字节可能分3次接收,接收端每次接收完后用utf8编码格式转换为字符串,就会出现乱码,并导致接收长度计算错误的情况。
解决方法一:以字节数做为消息长度的计算单位,而不是字符个数。
解决方法二:发送方和接收方都采用unicode编码格式。
问题3、长连接的保活问题
标准TCP层协议里把对方超时设为2小时,若服务器端超过了2小时还没收到客户的信息,它就发送探测报文段,若发送了10个探测报文段(每一个相隔75S)还没有收到响应,就假定客户出了故障,并终止这个连接。因此应对tcp长连接进行保活。
以下是异步通信时会遇到的问题:
问题4、缓冲区脏数据问题
同步发送的拷贝,是直接拷贝数据到基础系统缓冲区,拷贝完成后返回;
异步发送消息的拷贝,是将Socket自带的Buffer空间内的所有数据,拷贝到基础系统发送缓冲区,并立即返回;
因此异步发送时缓冲区设置不好会导致接收到脏数据的问题,如下所示:
第一次发送数据:1234567890
第一次接受数据:1234567890
第二次发送数据:abc
第二次接受数据:abc4567890
请参考:/article/4967046.html
解决方法一:将缓冲区的大小设置为实际发送数据的大小。
问题5、内存碎片问题
频繁的申请缓冲区会导致内存碎片的问题。
解决方法一:使用对象池和内存池。
请参考MSDN:http://msdn.microsoft.com/zh-cn/library/bb517542(v=vs.100).aspx
http://msdn.microsoft.com/zh-cn/library/system.net.sockets.socketasynceventargs.socketasynceventargs(v=vs.100).aspx
问题6、乱序问题
多个线程使用异步通信方式向同一个接收端(socket)同时发送数据,会导致接收端接收的数据混乱。如下所示:
线程1第一次发送:123456789,假设未发送完,只发送了123
线程2第一次发送:abcdefgh,假设未发送完,只发送了abc
线程1第二次发送:456789,发送完成
线程2第二次发送:defgh,发送完成
接收端最终接收的数据为:123abc456789defgh。
解决方法一:一个连接的发送端线程排队发送数据。
相关文章推荐
- Tcp编程常见问题及解决方法总结(粘包,拆包)
- 自己总结的几个常见问题的解决方法
- QT5常见问题二:应用程序中文乱码解决方法,总结
- 二分查找(Binary Search)常见问题解决方法总结
- IIS6架设网站过程常见问题解决方法总结
- 针对上午的"asp.net最常见的错误总结"的问题的解决方法!
- QT5常见问题二:应用程序中文乱码解决方法,总结
- Hadoop常见错误问题及解决方法总结二
- 黑马程序员--10.网络编程--08.【C_S常见的“双卡”现象和解决--TCP复制文本文件示例II】【阻塞式循环的分析过程 ---总结】
- 扫描仪使用过程中常见问题解决方法总结
- 二分查找(Binary Search) 常见问题解决方法总结
- Asp.net MVC中Razor常见的问题与解决方法总结
- 【JAVA学习】总结classpath常见问题以及解决方法
- 常见的嵌入式Linux内核启动问题总结及解决方法
- 内存混插常见问题和解决方法
- MySQL5.0中文问题及JDBC数据库连接和JSP汉字编码问题解决方法总结
- 兼容性不忽视 内存混插常见问题和解决方法
- 基于消息分发的多线程程序设计,常见的问题,以及解决方法
- 解决java网络编程IPv6带来的问题方法
- 使用Apache Axis部署 Web服务时的常见问题及其解决方法