当前位置: 首页 > news >正文

TCP服务端

出处: https://blog.csdn.net/DGH2430284817/article/details/86653294
问题描述:
       在用socket的通信中,经常会出现这种情况,客户端连接服务器,客户端使用输出流写数据,服务器用输入流读数据,但是服务器会出现read()的阻塞,导致程序一直阻塞跑不下去。

解决方法:
       一  客户端使用flush()方法,刷新缓存。

             结果没用,一样会阻塞。

       二  客户端使用输出流时用write(b, off, len)方法,请求数据多长就输出多长,服务器接收时也按照这个长度接。

             结果没用,而且不实际,服务器读数据时要先判断数据长度,有点麻烦。

       三  客户端在用完write方法后马上用out.close()关闭输出流,这样服务器就不会阻塞。

             可以解决服务器的阻塞,但是现实情况是客户端与服务器的交互时相互的,如果采用这种方法,服务端可以接受客户端的数据,但是客户端无法接受服务端返回的数据,如果协议是单向的可以采用这种方法。

       四  服务器用一个byte当做容器,设置这个容器的长度,每次读数据时,如果读的长度等于这个容器长度,说明后面还可能有数据,当读的数据长度小于这个容器长度时,说明后面没有数据了,就用break来退出read()方法,解决阻塞。

             可以解决服务器的阻塞, 也是没有副作用。

分析原因:
       服务端在用read方法的时候,如用byte[1024]来当容器,当客户端剩余数据不够填满的这个容器时,服务端就会一直读,等读够为止,但是当客户端输出流写完数据时,服务端却不知道读完了,本应read()返回-1的,却一直在阻塞。哪怕客户端用了flush()或者用write(b, off, len),服务端本应知道输出流结束的,但是无法得知,虽然客户端如果主动把输出流关闭,服务端就可以知道输出流结束,继续跑程序,但是缺点明显, 解决方法三时已经说了,所以只有解决方法四才能解决这个问题。

代码示例:
       TCP服务端:

import java.io.DataInputStream;
import java.io.DataOutputStream;
import java.io.InputStream;
import java.net.ServerSocket;
import java.net.Socket;

import org.omg.CORBA.INTERNAL;

public class Server {  
    public static final int PORT = 12345;//监听的端口号     
      
    public static void main(String[] args) {    
        System.out.println("TCP服务器启动:\n");    
        Server server = new Server();    
        server.init();    
    }    
    
    public void init() {    
        try {    
            ServerSocket serverSocket = new ServerSocket(PORT);    
            while (true) {    
                Socket client = serverSocket.accept();    
                System.out.println("接到新连接:" + client.getInetAddress() + "-" + client.getPort());
                // 用线程处理  
                new HandlerThread(client);    
            }    
        } catch (Exception e) {    
            System.out.println("服务器异常: " + e.getMessage());    
        }    
    }    
    
    private class HandlerThread implements Runnable {    
        private Socket socket;    
        public HandlerThread(Socket client) {    
            socket = client;    
            new Thread(this).start();    
        }    
    
        public void run() {    
            try {    
                // 读取客户端数据    
                InputStream input = socket.getInputStream();  
                
                StringBuffer acceptMsg = new StringBuffer();
                int MsgLong = 0;//接收数据总长度
                int len = 0;  //每次容器读时的长度
                byte[] b = new byte[1024]; //容器,存放数据
                
                while ((len = input.read(b)) != -1) {//一直读,读到没数据为止
                    acceptMsg.append(new String(b, 0, len, "GBK"));
                    MsgLong += len;
                    if (len < 1024) {//如果读的长度小于1024,说明是最后一次读,后面已经没有数据,跳出循环
                        break;
                    }
                }  
                // 处理客户端数据    
                System.out.println("客户端发过来的内容长度:" + MsgLong);    
                System.out.println("客户端发过来的内容:" + acceptMsg.toString());    
                // 向客户端回复信息    
                DataOutputStream out = new DataOutputStream(socket.getOutputStream());    
                // 发送键盘输入的一行    
                String s = "server send msg to client";    
                System.out.print("服务端返回数据:\t"+s);  
                out.write(s.getBytes());    
                  
                out.close();    
                input.close();    
            } catch (Exception e) {    
                System.out.println("服务器 run 异常: " + e.getMessage());    
            } finally {    
                if (socket != null) {    
                    try {    
                        socket.close();    
                    } catch (Exception e) {    
                        socket = null;    
                        System.out.println("服务端 finally 异常:" + e.getMessage());    
                    }    
                }    
            }   
        }    
    }    
}

转载于:https://www.cnblogs.com/lingyao/p/11095576.html

相关文章:

  • IBatis.Net学习笔记九--动态选择Dao的设计分析
  • 强化学习基础:蒙特卡罗和时序差分
  • golang 浮点数 取精度的效率对比
  • OpenCV入门指南 人脸检测 haar分类器
  • MySQL主从延时这么长,要怎么优化?
  • 使用 NPOI 导出数据示例
  • WPF Browser 中如何获取当前路径(临时文件中)?
  • 10+优秀“分步引导”jQuery插件(转)
  • 用processing画李萨如曲线
  • MVC笔记 初识模型(二)
  • android 手机网络接入点名称及WAP、NET模式的区别
  • 金蝶osf接口开发_金蝶云·星辰 | ?小微企业服务成长平台
  • 小程序商店刷榜_怎么注册微信小程序商店
  • 中getname_【136期】你能谈谈Java中 synchronized 对象锁和类锁的区别
  • 加到service中无效_给 COLA 做减法:应用架构中的“弯弯绕设计”
  • #Java异常处理
  • AHK 中 = 和 == 等比较运算符的用法
  • JAVA并发编程--1.基础概念
  • JS+CSS实现数字滚动
  • leetcode98. Validate Binary Search Tree
  • Linux后台研发超实用命令总结
  • Mocha测试初探
  • nodejs调试方法
  • Python十分钟制作属于你自己的个性logo
  • Redis的resp协议
  • REST架构的思考
  • Wamp集成环境 添加PHP的新版本
  • 当SetTimeout遇到了字符串
  • 动态魔术使用DBMS_SQL
  • 发布国内首个无服务器容器服务,运维效率从未如此高效
  • 简析gRPC client 连接管理
  • 开年巨制!千人千面回放技术让你“看到”Flutter用户侧问题
  • 坑!为什么View.startAnimation不起作用?
  • 理解 C# 泛型接口中的协变与逆变(抗变)
  • 每个JavaScript开发人员应阅读的书【1】 - JavaScript: The Good Parts
  • 批量截取pdf文件
  • 小李飞刀:SQL题目刷起来!
  • 一个普通的 5 年iOS开发者的自我总结,以及5年开发经历和感想!
  • 怎么把视频里的音乐提取出来
  • 正则学习笔记
  • 中文输入法与React文本输入框的问题与解决方案
  • 自制字幕遮挡器
  • 整理一些计算机基础知识!
  • ​ ​Redis(五)主从复制:主从模式介绍、配置、拓扑(一主一从结构、一主多从结构、树形主从结构)、原理(复制过程、​​​​​​​数据同步psync)、总结
  • ​ 全球云科技基础设施:亚马逊云科技的海外服务器网络如何演进
  • ​MPV,汽车产品里一个特殊品类的进化过程
  • #define,static,const,三种常量的区别
  • #NOIP 2014# day.1 生活大爆炸版 石头剪刀布
  • #控制台大学课堂点名问题_课堂随机点名
  • (3)nginx 配置(nginx.conf)
  • (Python) SOAP Web Service (HTTP POST)
  • (Redis使用系列) Springboot 使用redis实现接口Api限流 十
  • (二)【Jmeter】专栏实战项目靶场drupal部署
  • (附源码)springboot课程在线考试系统 毕业设计 655127
  • (排序详解之 堆排序)