`

Java建立Socket慢的问题

阅读更多

 

在Java编程中,一般都是使用下面的语句来建立Socket

 

String ip ="192.168.0.100";

int port = 8090;

Socket socket = new Socket(ip,port);

//.......

 

 

在有些JDK和JRE的版本中,会发生这个new Socket语句非常缓慢的问题(Linux和windows环境中都可能有这种问题)。

 

我遇到这种问题,还是在Java中使用Tomcat,配置了一个数据库连接池,最小连接数配了10个,在软件一启动时,就会建立这些连接,结果就发现启动过程卡在这里了,悲剧了。。。。

 

 

更有甚者,在某个Java做得项目中,启动时,会建立一个长连接的连接池,一启动时就会建立一些Socket客户端,在启动后并发访问中来使用,结果,等啊等啊。。。

 

(最讨厌的是,不知道到底哪些版本出这样的问题,好像JDK1.5时看到过这个问题,后面的版本也有时候有问题,再有就是IBM的JDK也说不定哪个版本会出这个问题,而且,Linux平台出现问题的时候更多)

 

经过这两次,我下决心找一下这个问题,后来发现网上也有人讨论这个问题,但是很少,后来在sun(当时Java还是sun的)的bug库中有个条目在描述这个问题,链接找不到了,大体上的描述就是:

 

Java中,使用字符串的主机和整数的端口号来构造Socket的构造函数是

 

public Socket(String host, int port) throws UnknownHostException, IOException

 

可以明确的是,第一个参数的含义是host,因此,java将第一个参数字符串会理解成主机名,因此,在建立Socket时会根据主机名来查找主机的IP地址。因此,即便实际给的参数是一个字符串中包含的IP地址,但是,这是Java所无法分辨的。

 

因为机器的DNS配置不大对头,到DNS查找主机对应IP,消耗了太多的时间,所以,建立连接变得非常缓慢,就造成了前边所描述的问题。至于怎样到DNS查找,就不是本文所描述的内容了。

 

如果想通过编程避免这个问题,应该怎么处理呢?我们可以看到,Socket还可以这样建立

 

public Socket(InetAddress address, int port) throws IOException

 

而InetAddress可以通过如下的方法来创建:

 

public static InetAddress getByName(String host) throws UnknownHostException

 

 

public static InetAddress getByAddress(byte[] addr) throws UnknownHostException

 

使用字符串的host来创建InetAddress,可以想象,和使用字符串的host来创建Socket一样会缓慢(因为需要

到DNS),而是要字节数组来创建又是如何呢?我们看这个函数的解释:

 

 

在给定原始 IP 地址的情况下,返回 InetAddress 对象。参数按网络字节顺序:地址的高位字节位于 getAddress()[0] 中。

此方法不会阻塞,即不执行任何反向名称服务查找操作。

IPv4 地址 byte 数组的长度必须为 4 个字节,IPv6 byte 数组的长度必须为 16 个字节

 

首先可以明确看到,不执行反向名称服务查找查找;第二,讲IP地址转换成字节数组。因为Java看到这个函数时已经知道送来的是IP地址,所以没有查DNS的问题了。

 

从这里可以看到,其实这个问题是因为我们在送参数时,没有更加准确的去区分IP地址和主机名这两个概念。送IP地址和送主机名的场景如果清晰的区分开来就不会出这种问题了。

 

一个很糟糕的是Java的JDBC,我记得好像JDBC是建立的Socket上的,而且是字符串方式送的主机或IP地址,所以,这个问题会影响到使用数据库的Java应用,如前边提到的。

 

再有,如果打算使用字符串这种构造地址或Socket的方式,应该配置DNS,使之能够很快的找到主机。说白了,在windows下,你可以在windows\system32\drivers\etc\hosts文件这,加入一行(示例如下):

 

192.168.0.100 192.168.0.100

 

这样就告诉系统,在寻找主机名“192.168.0.100”时,从这里就返回IP地址是192.168.0.200。这样就省了很多的时间。对于Linux,是/etc/hosts文件。

 

结论:

 

(1)如果是自己建立TCP连接这类的应用,在可能的话,使用getByName这种方式,直接传IP地址;

(2)修改hosts文件,有些时候能够修改,有些情况,可能没有办法改别人系统的文件;

(3)配置好的DNS

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

16
5
分享到:
评论
20 楼 windshome 2013-03-29  
谢谢Iiwutao,一不小心写错了,向所有看帖的朋友们致歉!也希望大家发现我的错误,指给我,这也是帮我提高
19 楼 liwutao 2013-03-29  
楼主写的很好!也很用心^
不过有一点楼主笔误了吧
引用


这样就告诉系统,在寻找主机名“192.168.0.100”时,从这里就返回IP地址是192.168.0.200。这样就省了很多的时间。对于Linux,是/etc/hosts文件

windows\system32\drivers\etc\hosts这个文件里面的配置意思是前一个是IP,后一个是域名,这里返回的IP应该是192.168.0.100吧?
18 楼 hbthlw 2013-03-29  
学习就是需要有这么细致的心啊!
17 楼 windshome 2013-03-28  
liye71023326 写道
如果是自己建立TCP连接这类的应用,在可能的话,使用getByName这种方式,直接传IP地址;  按照楼主的解释,是不是应该把getByName  换成getByAddress?

getByName 参数是字符串,不知道是主机名还是IP,所以都会去找DNS啊,如果DNS配置的不好,不就慢了嘛。如果使用getByAddress,JVM知道送的是IP,就直接使用IP建立连接了,省了时间
16 楼 liye71023326 2013-03-28  
如果是自己建立TCP连接这类的应用,在可能的话,使用getByName这种方式,直接传IP地址;  按照楼主的解释,是不是应该把getByName  换成getByAddress?
15 楼 sgp420 2013-03-28  
  
14 楼 laogao3232 2013-03-27  
真没注意,感谢楼主分享。
13 楼 aotian16 2013-03-27  
多谢分享, 长知识了
12 楼 windshome 2013-03-27  
关键是你给Java的Socket送一个字符串的IP地址,它会当成主机名再解析一遍,呵呵 想想也怪怪的,JVM的确也不知道这个字符串竟然是一个IP地址而不是主机名啊!
11 楼 lipeixin 2013-03-27  
mark .
10 楼 wwwcomy 2013-03-27  
windshome 写道
我写的情况都是局域网发生的,有时候很快,但是也有时候很慢。到底为什么慢的原因,我也没有具体查找过,某些操作系统,某些Java版本。

我觉得一是从设计上和部署上避免问题,二是遇到这样的问题了能有个思路就行了。


也是 所以一般局域网都会用计算机名 然后hosts文件都会修改 也必须修改 这样应该会一定程度上减少此类问题
9 楼 darkness_t 2013-03-27  
真的长见识了~看来什么问题都怕钻研,多谢分享~
8 楼 windshome 2013-03-27  
我写的情况都是局域网发生的,有时候很快,但是也有时候很慢。到底为什么慢的原因,我也没有具体查找过,某些操作系统,某些Java版本。

我觉得一是从设计上和部署上避免问题,二是遇到这样的问题了能有个思路就行了。
7 楼 wwwcomy 2013-03-27  
很实用  不过是不是局域网内部可以不需要考虑类似问题  毕竟局域网内部寻址应该很快的吧
6 楼 windshome 2013-03-27  
很高兴能对朋友们有所帮助,十年的研发,对这类问题经历过很多,后面我会继续总结一些。因为时间少,只能在上班和带孩子之余的时间写,写的思路并不顺畅,文笔也一般,希望能越写越好吧!
5 楼 hzieept 2013-03-27  
解决了我多年困惑的一个问题
4 楼 kaiserlu 2013-03-27  
谢谢分享!很实用
3 楼 afunti 2013-03-26  
good不错哦
2 楼 baitian 2013-03-26  
谢谢,很实用
1 楼 zdb_cn 2013-03-26  
谢谢分享 !!!!  

相关推荐

    Java实现Socket长连接和短连接

    在Java编程中,Socket是网络通信的基础,它允许两个应用程序通过TCP/IP协议进行数据传输。Socket连接分为两种类型:长连接和短连接。这两种连接方式各有特点,适用于不同的应用场景。 **1. 短连接(Short ...

    JAVA Socket编程实现文件上传

    Java Socket编程是网络编程的基础,它提供了在两个应用程序之间建立通信连接的能力。在这个场景中,我们讨论的是如何使用Java的Socket来实现文件上传功能,即从客户端将文件发送到服务器,然后保存到服务器的数据库...

    java socket连接池 实现

    2. 获取连接:当应用程序需要建立Socket连接时,从连接池中获取一个闲置的Socket,如果没有,则等待直到有连接被归还或达到最大等待时间。 3. 使用连接:应用程序使用获取到的Socket进行网络通信。 4. 归还连接:...

    java的Socket编程

    Java的Socket编程是Java网络编程的核心部分,它提供了在TCP/IP协议栈上进行通信的机制。Socket接口是对TCP/IP协议的抽象,使得开发者...通过实践和学习,你将能够熟练掌握Java Socket编程,并能解决各种网络编程问题。

    java的Socket实现的多人聊天程序

    Java的Socket实现的多人聊天程序是一个基于网络通信的项目,主要利用了Java的Socket类来构建客户端和服务端的通信桥梁。Socket是TCP/IP协议的一部分,它提供了两台计算机之间进行数据传输的基础。在这个项目中,...

    java socket 客户端代码

    在Java编程语言中,Socket是实现网络通信的基础组件,它为两台计算机之间的通信提供了低级别的接口。在本文中,我们将深入探讨Java Socket客户端代码及其在创建基于Socket的聊天室客户端中的应用。 首先,理解...

    JAVA Socket 经典教程

    Java Socket是Java编程语言中实现网络通信的基础组件,它提供了应用程序与网络协议的接口,使得开发者可以构建基于TCP(传输控制协议)或UDP(用户数据报协议)的应用程序。本教程将深入探讨Java Socket编程的核心...

    Java c++ socket通信

    在Java客户端,一旦Socket连接建立,可以使用`DataOutputStream`或`BufferedWriter`向服务端写入数据,同时使用`DataInputStream`或`BufferedReader`读取服务端的响应。在C++服务端,可以使用`write()`和`read()`...

    java socket 用户真实IP测试

    通常,当用户通过Nginx访问服务时,Nginx会作为客户端与服务器建立Socket连接,这时服务器接收到的Socket连接请求的源IP将是Nginx的IP,而不是用户的IP。为了获取用户的真实IP,我们需要利用HTTP头部信息,特别是"X-...

    JAVA+Socket教程

    Java Socket教程是一个...以上内容只是Java Socket编程的冰山一角,实际应用中还涉及到更复杂的设计模式、网络协议、数据编码解码等问题。通过不断实践和学习,你将能够掌握更多高级技巧,并构建出强大的网络应用程序。

    总结java_socket编程.doc

    发送方和接收方的成对的两个Socket之间必须建立连接,以便在TCP协议的基础上进行通信。 UDP 协议 UDP协议是User Datagram Protocol的简称,是一种无连接的协议,每个数据报都是一个独立的信息,包括完整的源地址或...

    C#和java 之间基于Socket的通信

    本话题主要探讨了两种常用编程语言——Java和C#之间如何利用Socket进行通信。Socket是网络编程的基本接口,允许应用程序通过网络发送和接收数据。以下是关于"Java和C#之间基于Socket的通信"的详细知识点: 1. **...

    java socket 经典版本

    - 在Socket编程中,需要处理各种网络异常,如`IOException`、`SocketException`等,确保程序在遇到问题时能够优雅地关闭连接并释放资源。 4. **关闭连接**: - 当通信完成后,必须关闭Socket和ServerSocket以释放...

    Java基于socket的进程间通信 聊天小程序

    当有客户端连接时,ServerSocket会创建一个新的Socket实例,与客户端建立连接,并通过这个Socket进行数据交换。 5. **客户端实现**: 客户端则需要创建Socket对象,指定服务器的IP地址和端口号,然后通过Socket对象...

    java swing socket写的聊天工具

    一旦连接建立,服务器就可以通过`Socket`对象向客户端发送数据,同时接收客户端发来的数据。在这个聊天工具中,服务端可能使用多线程处理多个客户端连接,确保可以同时与所有在线用户进行通信,实现群发消息的功能。...

    Java_socket 通信(源码+Java包)

    Socket通信包括了ServerSocket和Socket两种类型,ServerSocket用于监听客户端连接,Socket则用于建立与服务器的连接。 2. **服务器端实现**: "Java通信之服务器"源码中,ServerSocket通常会在指定的端口(如9000)...

    java socket 学习资料

    - Java Socket编程中需要处理`IOException`,这包括连接失败、网络中断、超时等问题。 - 使用`try-catch`语句来捕获并处理这些异常,保证程序的健壮性。 8. **性能优化**: - 对于高并发场景,可以考虑使用多...

    java Socket 多线程

    Java Socket 多线程是网络编程中的一个重要概念,它结合了Java的并发处理能力和Socket通信技术,使得服务器能够同时处理多个客户端的连接请求。在Java中,Socket是用于在网络环境中进行双向通信的类,而多线程则允许...

    基于Java的socket的网络聊天程序

    Java的Socket编程是网络通信的基础,它允许两个网络应用程序之间建立连接并进行数据交换。在本项目中,"基于Java的socket的网络聊天程序"是一个初级程序员使用Java Socket API实现的练习项目,旨在理解并应用网络...

    Java基于Socket文件传输示例

    一旦连接建立,服务器读取文件并将其内容通过Socket发送。 在服务器端,使用了`BufferedInputStream`和`DataInputStream`来提高读取文件的效率。`BufferedInputStream`通过内部缓冲区提供流的优化读取,而`...

Global site tag (gtag.js) - Google Analytics