`
linliangyi2007
  • 浏览: 1012604 次
  • 性别: Icon_minigender_1
  • 来自: 福州
社区版块
存档分类
最新评论

Graceful Java Programming 优雅Java编程 之Socket Client

阅读更多
   老久没有动手写Socket程序了,今天应同事的要求上了一段程序。

这是一段很简单与C++编写的服务端通讯的java客户端,咋一看上去,没有任何问题。

貌似没有问题的程序

	public static String sendSynMsg(String ipAddr, byte[] datas) throws Exception{
		//解析服务器地址和端口号
		int dotPos = ipAddr.indexOf(':');
		String ip = ipAddr.substring(0, dotPos).trim();
		int port = Integer.parseInt(ipAddr.substring(dotPos+1).trim());
		InetSocketAddress endpoint = new InetSocketAddress(ip , port);
		
		Socket socket = null;
		OutputStream out = null;
		InputStream in = null;
		try	{		
			socket = new Socket();
			//设置发送逗留时间2秒
			socket.setSoLinger(true, 2); 
			//设置InputStream上调用 read()阻塞超时时间2秒
			socket.setSoTimeout(2000);
			//设置socket发包缓冲为32k;
			socket.setSendBufferSize(32*1024);
			//设置socket底层接收缓冲为32k
			socket.setReceiveBufferSize(32*1024);
			//关闭Nagle算法.立即发包
			socket.setTcpNoDelay(true);
			//连接服务器
			socket.connect(endpoint);
			//获取输出输入流
			out = socket.getOutputStream();
			in = socket.getInputStream();
			//输出请求			
			out.write(datas);
			out.flush();
			//接收应答
			BufferedReader br = new BufferedReader( new InputStreamReader(in) , 4096);
			StringWriter received = new StringWriter(4096);
			char[] charBuf = new char[4096];
			int size = 0;
			while ((size = br.read(charBuf)) > 0){
				received.write(charBuf, 0, size);
			}
			return received.toString();
			
		} finally {
			if (out != null) {
				try {
					out.close();
				} catch(Exception ex) {
					ex.printStackTrace();
				}
			}
			if (in != null) {
				try {
					in.close();
				} catch(Exception ex) {
					ex.printStackTrace();
				}
			}		
			if (socket != null) {
				try {
					socket.close();
				} catch(Exception ex) {
					ex.printStackTrace();
				}
			}
		}				
	}


但实际的调试中,总是报Read TimeOut异常!!排查原因后发现,数据是接收到了,只是size = br.read(charBuf)不会返回-1(java doc中的说明是读取到结束时size会返回-1)。
对于编写服务器端程序的C++程序员而言,他们通常会在通讯结束的时候,在数据的尾部加上一个\0的结束字符。因此,我们针对此做了修正

出问题的程序段
    while ((size = br.read(charBuf)) > 0){
        received.write(charBuf, 0, size);
    }


修改后的正确写法
  char lastChar = 0;
  do {
      size = br.read(charBuf , 0 , 4096);
      lastChar = charBuf[size-1];
      if(lastChar == 0){
          //去除尾部的\0字符
          received.write(charBuf, 0, size - 1);
      }
  }while(lastChar != 0);


最终的完整程序段:
	private static String sendSynMsg(String ipAddr, byte[] datas) throws Exception{
		//解析服务器地址和端口号
		int dotPos = ipAddr.indexOf(':');
		String ip = ipAddr.substring(0, dotPos).trim();
		int port = Integer.parseInt(ipAddr.substring(dotPos+1).trim());
		InetSocketAddress endpoint = new InetSocketAddress(ip , port);
		
		Socket socket = null;
		OutputStream out = null;
		InputStream in = null;
		try	{		
			socket = new Socket();
			//设置发送逗留时间2秒
			socket.setSoLinger(true, 2); 
			//设置InputStream上调用 read()阻塞超时时间2秒
			socket.setSoTimeout(2000);
			//设置socket发包缓冲为32k;
			socket.setSendBufferSize(32*1024);
			//设置socket底层接收缓冲为32k
			socket.setReceiveBufferSize(32*1024);
			//关闭Nagle算法.立即发包
			socket.setTcpNoDelay(true);
			//连接服务器
			socket.connect(endpoint);
			//获取输出输入流
			out = socket.getOutputStream();
			in = socket.getInputStream();
			//输出请求			
			out.write(datas);
			out.flush();
			//接收应答
			BufferedReader br = new BufferedReader( new InputStreamReader(in) , 4096);
			StringWriter received = new StringWriter(4096);
			char[] charBuf = new char[4096];
			int size = 0;
			char lastChar = 0;
			do {
				size = br.read(charBuf , 0 , 4096);
				lastChar = charBuf[size-1];
				if(lastChar == 0){
					received.write(charBuf, 0, size - 1);
				}
				//System.out.println(received.toString());
			}while(lastChar != 0);
			
			return received.toString();
			
		} finally {
			if (out != null) {
				try {
					out.close();
				} catch(Exception ex) {
					ex.printStackTrace();
				}
			}
			if (in != null) {
				try {
					in.close();
				} catch(Exception ex) {
					ex.printStackTrace();
				}
			}		
			if (socket != null) {
				try {
					socket.close();
				} catch(Exception ex) {
					ex.printStackTrace();
				}
			}
		}				
	}



程序中的关键点:
1. 不直接使用new Socket(String ip , int port)的构造函数,而是设置了socket的环境参数后再连接

2.设置发送逗留时间 socket.setSoLinger(true, 2); 这个参数是socket发送数据时的超时,如果对方在固定时间内不接受,则关闭socket。与socket.setSoTimeout(2000)不同,这个是设置InputStream上调用 read()阻塞超时时间。

3.socket.setTcpNoDelay(true);关闭Nagle算法。这使得在调用out.flush();时总能第一时间的发送数据包(这个适用于你的数据包是完整的一次性发送的前提)。

4.根据应用协议的实际大小,优化你的接收和发送缓冲,这两个参数可以有效提高网络通信的效率。

5.使用char的数组配合StringWriter作为接收数据的写入,这个比使用readLine方法实现更优雅。

一点小经验,和大家分享

分享到:
评论
7 楼 linliangyi2007 2009-02-07  
dennis_zane 写道
客户端主动连接发送第一个syn分节可能会携带你设置的TCP选项,主要是MSS和窗口规模通告,因此如果你要调整socket的缓冲区肯定要在发送第一个syn分节前(也就是connect前)设置了。而SO_LINGER选项是设置本地socket的, connect之前还是之后没有影响。


谢谢楼上大拿。看来一定是做C/C++通讯的高手。对于我们这些学java的,一直为不能接触通讯底层而无奈。。。。
6 楼 dennis_zane 2009-02-07  
客户端主动连接发送第一个syn分节可能会携带你设置的TCP选项,主要是MSS和窗口规模通告,因此如果你要调整socket的缓冲区肯定要在发送第一个syn分节前(也就是connect前)设置了。而SO_LINGER选项是设置本地socket的, connect之前还是之后没有影响。
5 楼 linliangyi2007 2009-02-06  
unsid 写道
恩我喜欢这个话题,问几个问题:
(1) 如果new Socket(String ip , int port),那么之后的socket.setSoLinger(true, 2); 是不是就不管用了???所以必须先socket.setSoLinger(true, 2)再connect???

(2) 我最感兴趣的是br.read(charBuf)这句阻塞到什么时候..按照你写的socket.setSoTimeout(2000);
  1/ 是不是br.read(charBuf)这句最多只等2秒?有可能在服务器端2s还没运算出数据..那样的话就不合理了..
  2/ 如果超时那么br.read(charBuf)是否返回-1跳出循环?

(3) 之前和别人讨论过关于网络通讯如何判断字节流是否接收完毕的话题,有人说"-1"是判断从文件读取的流,不能用来判断网络流,还有有人用in.avaliable()来判断该读取多少,你认为是否合理?我用in.avaliable()判断过,结果是不对的...


关于(1)点,不是先new就不起作用,但后set,我怀疑(仅仅是怀疑),至少要增加后续的网络通信。还请了解socket底层的牛人也来详细说明一下。但后connect,从程序上至少能保证设置一定有效,而且是在connect后立即生效吧!

关于(2),这个timeout参数的作用你说的没有错。当然,这是根据服务器的业务特点设定的,我们的业务完成只有不到50ms,2m钟的超时基本可以任务服务失败了,而前台的用户是希望尽快的高知失败,而不是等上10秒在告诉他任务失败。
其次,超时的时候,read方法不会有-1返回,而是抛出Read TimeOut异常。
4 楼 linliangyi2007 2009-02-06  
dennis_zane 写道
1、setSoLinger的作用完全说错了,这个选项主要是设置socket关闭后的行为,具体请google或者参考unp
2、禁掉nagle算法也不能说总能第一时间发出数据包,只是底层不再帮你组合小的数据包到MSS大小左右再发送,如果你的数据没有小于40个字节,通常不建议关闭这个选项。


关于楼上提到的第一点,我是参考Stevens 的http://mindprod.com/jgloss/socket.html#DISCONNECT 以及他的译稿http://frenchmay.iteye.com/blog/254503。
也许我说的不够清楚,实际上就是指socket的发送等待,在你调用的flush后,又close了socket,这个时候,在socket的底层缓冲可能还有数据尚未发出,这就要求socket等待数据全部发出后关闭。setSoLinger就是设置这个超时的。

关于第二点,楼上说的没有错,关于nagle的算法,我的说明也不够详细,但从效果上看,不关闭nagle,flush常会等待包到足够大后才真的发出,关闭了,则基本上就是立即发送了。


欢迎更多牛人指正!
3 楼 dennis_zane 2009-02-06  
1、setSoLinger的作用完全说错了,这个选项主要是设置socket关闭后的行为,具体请google或者参考unp
2、禁掉nagle算法也不能说总能第一时间发出数据包,只是底层不再帮你组合小的数据包到MSS大小左右再发送,如果你的数据没有小于40个字节,通常不建议关闭这个选项。

2 楼 unsid 2009-02-06  
另外关于IO,网上大多数帖子都是讲java各种io操作,各种包装类方法的,找不到对原理进行深入谈论的,我希望能和搂主多交流.
我有时候也下点源码看,我发现稍微有点规模的网络通讯程序,都没用java默认io,都有自己一套io包,比如下载hassian代码中用的都是HassianOutputStream之类的自定义IO,我很好奇它们把什么风装进去了,也许是hassian转有通讯机制,我想以后写网络通讯程序应该多学那些开源东西里的方法,多学IO原理,争取以后能写程序定义自己的IO库..
1 楼 unsid 2009-02-06  
恩我喜欢这个话题,问几个问题:
(1) 如果new Socket(String ip , int port),那么之后的socket.setSoLinger(true, 2); 是不是就不管用了???所以必须先socket.setSoLinger(true, 2)再connect???

(2) 我最感兴趣的是br.read(charBuf)这句阻塞到什么时候..按照你写的socket.setSoTimeout(2000);
  1/ 是不是br.read(charBuf)这句最多只等2秒?有可能在服务器端2s还没运算出数据..那样的话就不合理了..
  2/ 如果超时那么br.read(charBuf)是否返回-1跳出循环?

(3) 之前和别人讨论过关于网络通讯如何判断字节流是否接收完毕的话题,有人说"-1"是判断从文件读取的流,不能用来判断网络流,还有有人用in.avaliable()来判断该读取多少,你认为是否合理?我用in.avaliable()判断过,结果是不对的...

相关推荐

    Go-graceful优雅的重载http服务器零宕机时间兼容systemdsupervisor

    本文将深入探讨如何使用Go-graceful库来实现优雅的HTTP服务器重载,确保在更新服务时实现零宕机时间,并同时兼容systemd和supervisor这两种常见的系统管理工具。 `Go-graceful`是一个专门设计用于处理HTTP服务器的...

    graceful:写优雅服务的库

    控制优雅的应用程序 优雅的应用程序可以用信号来控制。 主进程支持以下信号 期限/INT 快速关机 放弃正常关机 合肥 重启工人 启动新的工作进程 优雅关闭旧的工作进程 USR2 升级可执行文件(启动新的主进程和工作进程...

    Laravel开发-graceful-cache

    `graceful-cache`是一个针对Laravel的缓存存储库的包装器,其目标是提供一种更加优雅、灵活的方式来管理你的应用缓存。在这个项目中,开发者可以更好地控制缓存的生命周期,提高应用的效率。 `graceful-cache`的...

    前端开源库-graceful-readlink

    优雅的前端开源库——`graceful-readlink`是一个针对`fs.readlink`进行优化的模块,主要用于解决Node.js在处理符号链接时可能出现的问题。在Node.js原生的`fs`模块中,`fs.readlink`函数用于读取符号链接(symbolic ...

    前端开源库-graceful-ncp

    `graceful-ncp`是一个专门为前端开发者设计的开源库,它实现了优雅的、异步的文件和目录的递归复制功能。这个库的核心目标是提供一种更稳定、更易用的方式来处理文件系统操作,特别是在处理大量文件或复杂目录结构时...

    Graceful Graph

    优雅图(Graceful Graph)是图论中的一个概念,它源于数学和计算机科学,特别是网络流和图遍历的问题。这个概念是由美国数学家Edward G. Kolaitis在1970年代提出的,目的是研究图的标号特性。在英特尔线程挑战赛3月...

    前端开源库-graceful-kill

    本文将深入探讨“graceful-kill”的概念、作用以及它如何改善前端应用程序的性能。 1. 优雅地杀死进程 在计算机系统中,进程的正常退出通常涉及清理资源、保存状态和关闭连接等操作。"优雅地杀死"一个进程意味着...

    java程序发布成window后服台

    Java程序在Windows环境下发布为后台服务是常见的需求,这使得应用程序能够在系统启动时自动运行,无需用户交互。本文将详细讲解如何将一个Java程序打包并发布为Windows后台服务,同时提供一个具体的示例。 首先,...

    graceful-fs-fs模块的一个替代拥有各种改进

    优雅的fs(graceful-fs)是Node.js生态系统中一个重要的模块,它作为原生fs模块的一个增强版本,提供了一系列的改进和优化,旨在提高文件系统的操作性能和稳定性。在Node.js开发过程中,文件系统操作是不可或缺的一...

    Laravel开发-graceful-laravel-workers

    3. **优雅关闭(Graceful Shutdown)**:在“graceful-laravel-workers”中,主要目标是实现优雅关闭,即在接收到停止命令时,工作进程会等待当前处理的任务完成后再退出,确保数据的一致性和完整性。 4. **...

    express-graceful-exit, 为了得到零停机,优雅地退出 Express.zip

    express-graceful-exit, 为了得到零停机,优雅地退出 Express express-graceful-exit具有零停机时间的组件为 node.js 使用 Express 部署。 它是为 3. X, 开发的,因此它可能需要与 Express 2.x 兼容。这个模块最初是...

    socket常见错误代码解析.pdf

    Socket Error 10094是指优雅关闭正在进行中,这种错误通常是由于Socket关闭正在进行中所致。解决方法是等待Socket关闭完成。 Socket Error 11001 - Host not found Socket Error 11001是指主机不存在,这种错误...

    Functional Programming in C++

    Mastering the functional style of programming can help you tackle the demands of modern apps and will lead to simpler expression of complex program logic, graceful error handling, and elegant ...

    Scalable IO in Java.pdf

    在讨论可伸缩性目标时,文档强调了在增加负载时的优雅退化(Graceful degradation),即在用户数量增加时,系统能够合理地使用增加的资源(CPU、内存、磁盘、带宽)来提升性能。同时,系统应当满足可用性与性能目标...

    RFC8538 Notification Message Support for BGP Graceful Restart

    **BGP Graceful Restart**(优雅重启)是一种使边界网关协议(BGP)能够在路由器重启时保持部分连接状态的方法,从而减少对网络的影响。在**RFC4724**中定义了这一机制,但它限制了BGP优雅重启的应用范围,即不包括...

    javaAJAX无刷技术

    在实际开发中,为了兼容各种浏览器和提升用户体验,开发者通常会结合使用AJAX和其他技术,例如使用渐进增强(Progressive Enhancement)或优雅降级(Graceful Degradation)策略,确保即使在JavaScript禁用或不支持...

    前端开源库-graceful-fs-extra

    `graceful-fs-extra`是一个针对Node.js环境的`fs`模块扩展库,它在原生`fs`模块的基础上增加了一些实用功能,并提供了更优雅的错误处理机制,使得文件操作更加高效和稳定。本文将深入探讨`graceful-fs-extra`库的...

    Scalable IO in Java -Doug Lea

    在Scalable IO in Java中,Doug Lea首先介绍了网络服务的可扩展性目标,包括在增加负载(即更多客户端)的情况下优雅地降级(graceful degradation),以及在资源(如CPU、内存、磁盘和带宽)增加时持续改进性能。...

    BGP Graceful Restart

    【BGP Graceful Restart】是BGP(Border Gateway Protocol)的一种特性,旨在确保在路由器或协议处理组件(如Route Processor,RP)重启时,路由信息的连续性和稳定性,避免网络中断。这一特性允许BGP Speaker...

    go-graceful-restart-example, 具有优雅重启功能的服务器示例.zip

    go-graceful-restart-example, 具有优雅重启功能的服务器示例 服务器使用 Go graceful重新启动安装并运行服务器$ go get github.com/Scalingo/go-graceful-restart-example$ go-graceful-restart-examp

Global site tag (gtag.js) - Google Analytics