在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
相关推荐
当用户聊天时,将当前用户名、聊天对象、聊天内容、聊天语气和是否私聊进行封装,然后与服务器建立Socket连接,再用对象输出流包装Socket的输出流将聊天信息对象发送给服务器端 当用户发送聊天信息时,服务端将会...
没问题,socket.io-java-client 处理。 有意义的异常- 如果出现问题,SocketIO 会尝试抛出有意义的异常并提供修复提示。 状态:与 Websocket 连接已准备就绪。 XHR 处于测试阶段。 如何使用 使用 socket.io-java-...
1.8 Java Socket编程 1.9 Java的内存泄漏 1.10 抽象类与接口的区别 1.11 Java变量类型间的相互转换 2 JAVA与WEB 2.1 JMX规范 2.1.1 JMX概述 2.1.2 设备层(Instrumentation Level) 2.1.3 代理层 2.1.4 分布服务层 ...
用java写的socket服务端接收代码,用swift写的socket连接和接收代码,客户端建立socket套接字和服务端连接都是没有问题的
这样当客户建立Socket的同时。服务器也会使用这一根连线来先后通讯,那么既然如此只要我们存在多条连线就可以了。那么我们的程序可以变为如下: 服务器: import java.io.*; import java.net.*; public...
telnet客户端,访问系统的telnet服务实质上是与telnet服务建立socket连接,默认的telnet服务的端口是23,TelnetClient.java; UDP编程,包括收发udp报文; 聊天室服务器端,Chatserver.java;聊天室客户端,...
1.文件传输速度太慢,可以创建发送和接收缓存提高传输速度,最简单的办法就是加大UDP包大小,设置MyPacket.java 文件里变量packetLength = 1024*50;后速度会有很大提高.. 2.在文件传输过程中,再发消息会丢包(接收方收...
[目录]Java简单问题 2. [目录]为什么Applet修改后在浏览器中不发生变化 3. [目录]Java中的类型转换 4. [目录]怎样找到编译时缺少的类 2. [目录]Java疑难解答 1. [目录]Java原理问题 ...
1.8 Java Socket编程 80 1.9 Java的内存泄漏 85 1.10 抽象类与接口的区别 86 1.11 Java变量类型间的相互转换 87 2 JAVA与WEB 87 2.1 JMX规范 87 2.1.1 JMX概述 87 2.1.2 设备层(Instrumentation Level) 88 2.1.3 ...
当用户聊天时,将当前用户名、聊天对象、聊天内容、聊天语气和是否私聊进行封装,然后与服务器建立Socket连接,再用对象输出流包装Socket的输出流将聊天信息对象发送给服务器端 当用户发送聊天信息时,服务端将会...
java多线程socket通信、java网络编程学习心得,基本实现多线程通信,其他问题的欢饮反馈。应付作业有余了: 应用多线程实现服务器与多客户端之间的通信: ① 服务器端创建ServerSocket,循环调用accept()等待客户端...
flash使用socket安全沙箱问题,需要建立安全策略服务器,java完成的一个jar,建立安全策略服务器,需要servlet-api.jar
因为该课题需要用到很多以前从未接触到的知识,如需要用JMF,RTP协议传输视频,JAVA socket编程,如何采集图像等知识。在用JAVA语言编码过程中,因编程规范及其他各种语法问题出现了很多错误,通过老师指导和查阅...
JAVA多服务器通讯框架是基于NIO开发的Socket通讯框架,实现了客户端和服务器,服务器与服务器之间的通讯功能,适合应用于大型聊天服务器,大型游戏服务器。 本演示程序实现了一个基本的命令行聊天功能,以演示...
6.5.2 生产者、消费者问题.. 137 6.6 小结 140 第7章 我要彩色照片——SWING的基本概念 141 7.1 讲解 142 7.1.1 Swing的基本概念 142 7.1.2 Swing组件继承关系 142 7.1.3 Swing组件一览 143 7.1.4 Swing和MVC...
虽然Unity为socket编程和多人游戏实现提供了网络api和协议,但有时你需要建立自己的裸socket和通信协议。 由于当时我没有找到一个可以使用的、清晰的、多线程的 c# 套接字在 unity 中的使用示例,因此我开始为 udp ...
使用socket建立客户端与服务器的通信的过程 60.JAVA语言国际化应用,Locale类,Unicode 61.描述反射机制的作用 62.如何读写一个文件? 63.在图形界面中,一个按钮如何处理鼠标点击事件? 64.在图形界面中,一个...
发送方和接收方成对的两个socket之间必须建立连接,以便在TCP协议的基础上 进行通信,当一个socket(通常都是server socket)等待建立连接时,另一个socket可以要求进行连接,一旦这两个socket连接起 来,它们就可以...
虽然Unity为socket编程和多人游戏实现提供了网络api和协议,但有时你需要建立自己的裸socket和通信协议。 由于当时我没有找到一个可以使用的、清晰的、多线程的 c# 套接字在 unity 中的使用示例,因此我开始为 udp ...
当用户聊天时,将当前用户名、聊天对象、聊天内容、聊天语气和是否私聊进行封装,然后与服务器建立Socket连接,再用对象输出流包装Socket的输出流将聊天信息对象发送给服务器端 当用户发送聊天信息时,服务端将会...