Java网络编程精解.pdf
![资源得分’ title=](/images/score_1.gif)
![资源得分’ title=](/images/score_1.gif)
![资源得分’ title=](/images/score_1.gif)
![资源得分’ title=](/images/score_1.gif)
![资源得分’ title=](/images/score_05.gif)
《Java网络编程精解.pdf》由会员分享,可在线阅读,更多相关《Java网络编程精解.pdf(29页珍藏版)》请在淘文阁 - 分享文档赚钱的网站上搜索。
1、 资料来源:北软教育 www.softedu.org|专业的java 培训、网络培训、网络安全培训基地 在客户/服务器通信模式中,服务器端需要创建监听特定端口的 ServerSocket,ServerSocket 负责接收客户连接请求。本章首先介绍 ServerSocket 类的各个构造方法,以及成员方法的用法,接着介绍服务器如何用多线程来处理与多个客户的通信任务。本章提供线程池的一种实现方式。线程池包括一个工作队列和若干工作线程。服务器程序向工作队列中加入与客户通信的任务,工作线程不断从工作队列中取出任务并执行它。本章还介绍了 java.util.concurrent 包中的线程池类的用法,
2、在服务器程序中可以直接使用它们。3.1 构造构造 ServerSocket ServerSocket 的构造方法有以下几种重载形式:l ServerSocket()throws IOException l ServerSocket(int port)throws IOException l ServerSocket(int port,int backlog)throws IOException l ServerSocket(int port,int backlog,InetAddress bindAddr)throws IOException 在以上构造方法中,参数 port 指定服务器要绑定
3、的端口(服务器要监听的端口),参数 backlog 指定客户连接请求队列的长度,参数 bindAddr 指定服务器要绑定的 IP 地址。3.1.1 绑定端口绑定端口 除了第一个不带参数的构造方法以外,其他构造方法都会使服务器与特定端口绑定,该端口由参数 port 指定。例如,以下代码创建了一个与 80 端口绑定的服务器:ServerSocket serverSocket=new ServerSocket(80);如果运行时无法绑定到 80 端口,以上代码会抛出 IOException,更确切地说,是抛出BindException,它是 IOException 的子类。BindException
4、 一般是由以下原因造成的:l 端口已经被其他服务器进程占用;l 在某些操作系统中,如果没有以超级用户的身份来运行服务器程序,那么操作系统不允许服务器绑定到 11023 之间的端口。如果把参数 port 设为 0,表示由操作系统来为服务器分配一个任意可用的端口。由操作系统分配的端口也称为匿名端口。对于多数服务器,会使用明确的端口,而不会使用匿名端口,因为客户程序需要事先知道服务器的端口,才能方便地访问服务器。在某些场合,匿名端口有着特殊的用途,本章 3.4 节会对此作介绍。3.1.2 设定客户连接请求队列的长度设定客户连接请求队列的长度 当服务器进程运行时,可能会同时监听到多个客户的连接请求。例
5、如,每当一个客户进程执行以下代码:Socket socket=new Socket(www.javathinker.org,80);就意味着在远程 www.javathinker.org 主机的 80 端口上,监听到了一个客户的连接请求。管理客户连接请求的任务是由操作系统来完成的。操作系统把这些连接请求存储在一个先进先出的队列中。许多操作系统限定了队列的最大长度,一般为 50。当队列中的连接请求达到了队列的最大容量时,服务器进程所在的主机会拒绝新的连接请求。只有当服务器进程通UnRegistered 资料来源:北软教育 www.softedu.org|专业的java 培训、网络培训、网络安全培
6、训基地 过 ServerSocket 的 accept()方法从队列中取出连接请求,使队列腾出空位时,队列才能继续加入新的连接请求。对于客户进程,如果它发出的连接请求被加入到服务器的队列中,就意味着客户与服务器的连接建立成功,客户进程从 Socket 构造方法中正常返回。如果客户进程发出的连接请求被服务器拒绝,Socket 构造方法就会抛出ConnectionException。ServerSocket 构造方法的 backlog 参数用来显式设置连接请求队列的长度,它将覆盖操作系统限定的队列的最大长度。值得注意的是,在以下几种情况中,仍然会采用操作系统限定的队列的最大长度:l backlog
7、 参数的值大于操作系统限定的队列的最大长度;l backlog 参数的值小于或等于 0;l 在 ServerSocket 构造方法中没有设置 backlog 参数。以下例程3-1的 Client.java和例程 3-2的Server.java用来演示服务器的连接请求队列的特性。例程 3-1 Client.java import .*;public class Client public static void main(String args)throws Exception final int length=100;String host=localhost;int port=8000;So
8、cket sockets=new Socketlength;for(int i=0;ilength;i+)/试图建立 100 次连接 socketsi=new Socket(host,port);System.out.println(第+(i+1)+次连接成功);Thread.sleep(3000);for(int i=0;ilength;i+)socketsi.close();/断开连接 例程 3-2 Server.java import java.io.*;import .*;public class Server private int port=8000;private ServerS
9、ocket serverSocket;public Server()throws IOException serverSocket=new ServerSocket(port,3);/连接请求队列的长度为 3 System.out.println(服务器启动);UnRegistered 资料来源:北软教育 www.softedu.org|专业的java 培训、网络培训、网络安全培训基地 public void service()while(true)Socket socket=null;try socket=serverSocket.accept();/从连接请求队列中取出一个连接 Syste
10、m.out.println(New connection accepted +socket.getInetAddress()+:+socket.getPort();catch(IOException e)e.printStackTrace();finally try if(socket!=null)socket.close();catch(IOException e)e.printStackTrace();public static void main(String args)throws Exception Server server=new Server();Thread.sleep(60
11、000*10);/睡眠 10 分钟 /server.service();Client 试图与 Server 进行 100 次连接。在 Server 类中,把连接请求队列的长度设为 3。这意味着当队列中有了 3 个连接请求时,如果 Client 再请求连接,就会被 Server 拒绝。下面按照以下步骤运行 Server 和 Client 程序。(1)把 Server 类的 main()方法中的“server.service();”这行程序代码注释掉。这使得服务器与 8 000 端口绑定后,永远不会执行 serverSocket.accept()方法。这意味着队列中的连接请求永远不会被取出。先运行
12、 Server 程序,然后再运行 Client 程序,Client 程序的打印结果如下:第 1 次连接成功 第 2 次连接成功 第 3 次连接成功 Exception in thread main .ConnectException:Connection refused:connect at .PlainSocketImpl.socketConnect(Native Method)at .PlainSocketImpl.doConnect(Unknown Source)at .PlainSocketImpl.connectToAddress(Unknown Source)at .PlainSo
13、cketImpl.connect(Unknown Source)at .SocksSocketImpl.connect(Unknown Source)at .Socket.connect(Unknown Source)at .Socket.connect(Unknown Source)at .Socket.(Unknown Source)UnRegistered 资料来源:北软教育 www.softedu.org|专业的java 培训、网络培训、网络安全培训基地 at .Socket.(Unknown Source)at Client.main(Client.java:10)从以上打印结果可以
14、看出,Client 与 Server 在成功地建立了 3 个连接后,就无法再创建其余的连接了,因为服务器的队列已经满了。(2)把 Server 类的 main()方法按如下方式修改:public static void main(String args)throws Exception Server server=new Server();/Thread.sleep(60000*10);/睡眠 10 分钟 server.service();作了以上修改,服务器与 8 000 端口绑定后,就会在一个 while 循环中不断执行serverSocket.accept()方法,该方法从队列中取出连接
15、请求,使得队列能及时腾出空位,以容纳新的连接请求。先运行 Server 程序,然后再运行 Client 程序,Client 程序的打印结果如下:第 1 次连接成功 第 2 次连接成功 第 3 次连接成功 第 100 次连接成功 从以上打印结果可以看出,此时 Client 能顺利与 Server 建立 100 次连接。3.1.3 设定绑定的设定绑定的 IP 地址地址 如果主机只有一个 IP 地址,那么默认情况下,服务器程序就与该 IP 地址绑定。ServerSocket的第 4 个构造方法 ServerSocket(int port,int backlog,InetAddress bindAdd
16、r)有一个bindAddr 参数,它显式指定服务器要绑定的 IP 地址,该构造方法适用于具有多个 IP 地址的主机。假定一个主机有两个网卡,一个网卡用于连接到 Internet,IP 地址为 222.67.5.94,还有一个网卡用于连接到本地局域网,IP 地址为 192.168.3.4。如果服务器仅仅被本地局域网中的客户访问,那么可以按如下方式创建 ServerSocket:ServerSocket serverSocket=new ServerSocket(8000,10,InetAddress.getByName(192.168.3.4);3.1.4 默认构造方法的作用默认构造方法的作用
17、ServerSocket 有一个不带参数的默认构造方法。通过该方法创建的ServerSocket 不与任何端口绑定,接下来还需要通过 bind()方法与特定端口绑定。这个默认构造方法的用途是,允许服务器在绑定到特定端口之前,先设置ServerSocket 的一些选项。因为一旦服务器与特定端口绑定,有些选项就不能再改变了。在以下代码中,先把 ServerSocket 的 SO_REUSEADDR 选项设为 true,然后再把它与 8000端口绑定:ServerSocket serverSocket=new ServerSocket();serverSocket.setReuseAddress(t
18、rue);/设置 ServerSocket 的选项 UnRegistered 资料来源:北软教育 www.softedu.org|专业的java 培训、网络培训、网络安全培训基地 serverSocket.bind(new InetSocketAddress(8000);/与 8000 端口绑定 如果把以上程序代码改为:ServerSocket serverSocket=new ServerSocket(8000);serverSocket.setReuseAddress(true);/设置 ServerSocket 的选项 那么 serverSocket.setReuseAddress(tr
19、ue)方法就不起任何作用了,因为 SO_ REUSEADDR选项必须在服务器绑定端口之前设置才有效。3.2 接收和关闭与客户的连接接收和关闭与客户的连接 ServerSocket 的 accept()方法从连接请求队列中取出一个客户的连接请求,然后创建与客户连接的 Socket 对象,并将它返回。如果队列中没有连接请求,accept()方法就会一直等待,直到接收到了连接请求才返回。接下来,服务器从 Socket 对象中获得输入流和输出流,就能与客户交换数据。当服务器正在进行发送数据的操作时,如果客户端断开了连接,那么服务器端会抛出一个 IOException的子类 SocketExceptio
20、n 异常:.SocketException:Connection reset by peer 这只是服务器与单个客户通信中出现的异常,这种异常应该被捕获,使得服务器能继续与其他客户通信。以下程序显示了单线程服务器采用的通信流程:public void service()while(true)Socket socket=null;try socket=serverSocket.accept();/从连接请求队列中取出一个连接 System.out.println(New connection accepted +socket.getInetAddress()+:+socket.getPort()
21、;/接收和发送数据 catch(IOException e)/这只是与单个客户通信时遇到的异常,可能是由于客户端过早断开连接引起的 /这种异常不应该中断整个 while 循环 e.printStackTrace();finally try if(socket!=null)socket.close();/与一个客户通信结束后,要关闭Socket catch(IOException e)e.printStackTrace();与单个客户通信的代码放在一个 try 代码块中,如果遇到异常,该异常被 catch 代码块捕获。try 代码块后面还有一个 finally 代码块,它保证不管与客户通信正常结
22、束还是异常结束,最后都会关闭 Socket,断开与这个客户的连接。UnRegistered 资料来源:北软教育 www.softedu.org|专业的java 培训、网络培训、网络安全培训基地 3.3 关闭关闭 ServerSocket ServerSocket 的 close()方法使服务器释放占用的端口,并且断开与所有客户的连接。当一个服务器程序运行结束时,即使没有执行 ServerSocket 的 close()方法,操作系统也会释放这个服务器占用的端口。因此,服务器程序并不一定要在结束之前执行 ServerSocket 的 close()方法。在某些情况下,如果希望及时释放服务器的端口
23、,以便让其他程序能占用该端口,则可以显式调用 ServerSocket 的 close()方法。例如,以下代码用于扫描 165535 之间的端口号。如果 ServerSocket 成功创建,意味着该端口未被其他服务器进程绑定,否者说明该端口已经被其他进程占用:for(int port=1;portjava RandomPort 监听的端口为:3000 C:chapter03classesjava RandomPort 监听的端口为:3004 C:chapter03classesjava RandomPort 监听的端口为:3005 多数服务器会监听固定的端口,这样才便于客户程序访问服务器。匿名
24、端口一般适用于服务器与客户之间的临时通信,通信结束,就断开连接,并且 ServerSocket 占用的临时端口也被释放。FTP(文件传输)协议就使用了匿名端口。如图 3-1 所示,FTP 协议用于在本地文件系统与远程文件系统之间传送文件。图 3-1 FTP 协议用于在本地文件系统与远程文件系统之间传送文件 FTP 使用两个并行的 TCP 连接:一个是控制连接,一个是数据连接。控制连接用于在客户和服务器之间发送控制信息,如用户名和口令、改变远程目录的命令或上传和下载文件的命令。数据连接用于传送文件。TCP 服务器在 21 端口上监听控制连接,如果有客户要求上传或下载文件,就另外建立一个数据连接,
25、通过它来传送文件。数据连接的建立有两种方式。(1)如图 3-2 所示,TCP 服务器在 20 端口上监听数据连接,TCP 客户主动请求建立与该端口的连接。图 3-2 TCP 服务器在 20 端口上监听数据连接(2)如图 3-3 所示,首先由 TCP 客户创建一个监听匿名端口的 ServerSocket,再把这个 ServerSocket 监听的端口号(调用 ServerSocket 的 getLocalPort()方法就能得到端口号)发送给 TCP 服务器,然后由 TCP 服务器主动请求建立与客户端的连接。UnRegistered 资料来源:北软教育 www.softedu.org|专业的ja
- 配套讲稿:
如PPT文件的首页显示word图标,表示该PPT已包含配套word讲稿。双击word图标可打开word文档。
- 特殊限制:
部分文档作品中含有的国旗、国徽等图片,仅作为作品整体效果示例展示,禁止商用。设计者仅对作品中独创性部分享有著作权。
- 关 键 词:
- Java 网络 编程
![提示](https://www.taowenge.com/images/bang_tan.gif)
限制150内