网络基础

网络模型

网络模型一般是指OSI七层参考模型和TCP/IP四层参考模型。这两个模型在网络中应用最为广泛。

OSI模型,即开放式通信系统互联参考模型(Open System Interconnection),是由ISO(国际标准化组织)制定的,OSI将计算机网络体系结构(architecture)划分为以下七层。

TCP/IP是一组用于实现网络互连的通信协议。Internet网络体系结构以TCP/IP为核心。基于TCP/IP的参考模型将协议分成四个层次。

两种模型有一定的对应关系:

OSI 七层模型通过七个层次化的结构模型使不同的系统不同的网络之间实现可靠的通讯,因此其最主要的功能就是帮助不同类型的主机实现数据传输

完成中继功能的节点通常称为中继系统。在OSI七层模型中,处于不同层的中继系统具有不同的名称。
一个设备工作在哪一层,关键看它工作时利用哪一层的数据头部信息。比如,网桥工作时,是以MAC头部来决定转发端口的,因此显然它是数据链路层的设备。

具体说:
物理层:网卡,网线,集线器,中继器,调制解调器
数据链路层:网桥,交换机
网络层:路由器

集线器是物理层设备,采用广播的形式来传输信息交换机就是用来进行报文交换的机器,多为链路层设备(二层交换机),能够进行地址学习,采用存储转发的形式来交换报文
路由器的一个作用是连通不同的网络,另一个作用是选择信息传送的线路选择通畅快捷的近路,能大大提高通信速度,减轻网络系统通信负荷,节约网络系统资源,提高网络系统畅通率。

交换机和路由器的区别

交换机拥有一条很高带宽的背部总线和内部交换矩阵。交换机的所有的端口都挂接在这条总线上,控制电路收到数据包以后,处理端口会查找内存中的地址对照表以确定目的MAC地址(网卡的硬件地址)的NIC(网卡)挂接在哪个端口上,通过内部交换矩阵迅速将数据包传送到目的端口,目的MAC地址若不存在则广播到所有的端口,接收端口回应后交换机会“学习”新的地址,并把它添加入内部MAC地址表中(可用于MAC欺骗)。

使用交换机也可以把网络“分段”,通过对照MAC地址表,交换机只允许必要的网络流量通过交换机。通过交换机的过滤和转发,可以有效的隔离广播风暴,减少误包和错包的出现,避免共享冲突。

交换机在同一时刻可进行多个端口对之间的数据传输。每一端口都可视为独立的网段,连接在其上的网络设备独自享有全部的带宽,无须同其他设备竞争使用。当节点A向节点D发送数据时,节点B可同时向节点C发送数据,而且这两个传输都享有网络的全部带宽,都有着自己的虚拟连接。假使这里使用的是10Mbps的以太网交换机,那么该交换机这时的总流通量就等于2×10Mbps=20Mbps,而使用10Mbps的共享式HUB时,一个HUB的总流通量也不会超出10Mbps。

总之,交换机是一种基于MAC地址识别,能完成封装转发数据包功能的网络设备。交换机可以“学习”MAC地址,并把其存放在内部地址表中,通过在数据帧的始发者和目标接收者之间建立临时的交换路径,使数据帧直接由源地址到达目的地址。

从过滤网络流量的角度来看,路由器的作用与交换机和网桥非常相似。但是与工作在网络物理层,从物理上划分网段的交换机不同,路由器使用专门的软件协议从逻辑上对整个网络进行划分。例如,一台支持IP协议的路由器可以把网络划分成多个子网段,只有指向特殊IP地址的网络流量才可以通过路由器。对于每一个接收到的数据包,路由器都会重新计算其校验值,并写入新的物理地址。因此,使用路由器转发和过滤数据的速度往往要比只查看数据包物理地址的交换机慢。但是,对于那些结构复杂的网络,使用路由器可以提高网络的整体效率。路由器的另外一个明显优势就是可以自动过滤网络广播。
集线器与路由器在功能上有什么不同?

首先说HUB,也就是集线器。它的作用可以简单的理解为将一些机器连接起来组成一个局域网。而交换机(又名交换式集线器)作用与集线器大体相同。但是两者在性能上有区别:集线器采用的是共享带宽的工作方式,而交换机是独享带宽。这样在机器很多或数据量很大时,两者将会有比较明显的。而路由器与以上两者有明显区别,它的作用在于连接不同的网段并且找到网络中数据传输最合适的路径。路由器是产生于交换机之后,就像交换机产生于集线器之后,所以路由器与交换机也有一定联系,不是完全独立的两种设备。路由器主要克服了交换机不能路由转发数据包的不足。

总的来说,路由器与交换机的主要区别体现在以下几个方面:

  1. 工作层次不同
    最初的交换机是工作在数据链路层,而路由器一开始就设计工作在网络层。由于交换机工作在数据链路层,所以它的工作原理比较简单,而路由器工作在网络层,可以得到更多的协议信息,路由器可以做出更加智能的转发决策。

  2. 数据转发所依据的对象不同
    交换机是利用物理地址或者说MAC地址来确定转发数据的目的地址。而路由器则是利用IP地址来确定数据转发的地址。IP地址是在软件中实现的,描述的是设备所在的网络。MAC地址通常是硬件自带的,由网卡生产商来分配的,而且已经固化到了网卡中去,一般来说是不可更改的。而IP地址则通常由网络管理员或系统自动分配。

  3. 传统的交换机只能分割冲突域,不能分割广播域;而路由器可以分割广播域。由交换机连接的网段仍属于同一个广播域,广播数据包会在交换机连接的所有网段上传播,在某些情况下会导致通信拥挤和安全漏洞。连接到路由器上的网段会被分配成不同的广播域,广播数据不会穿过路由器。虽然第三层以上交换机具有VLAN功能,也可以分割广播域,但是各子广播域之间是不能通信交流的,它们之间的交流仍然需要路由器。

  4. 路由器提供了防火墙的服务
    路由器仅仅转发特定地址的数据包,不传送不支持路由协议的数据包和未知目标网络数据包,从而可以防止广播风暴。

网关工作在第四层传输层及其以上。

物理层

在OSI参考模型中,物理层(Physical Layer)是参考模型的最低层,也是OSI模型的第一层。
物理层的主要功能是:利用传输介质为数据链路层提供物理连接,实现比特流的透明传输。
物理层的作用是实现相邻计算机节点之间比特流的透明传送,尽可能屏蔽掉具体传输介质和物理设备的差异。使其上面的数据链路层不必考虑网络的具体传输介质是什么。“透明传送比特流”表示经实际电路传送后的比特流没有发生变化,对传送的比特流来说,这个电路好像是看不见的。

数据链路层

数据链路层(Data Link Layer)是OSI模型的第二层,负责建立和管理节点间的链路。该层的主要功能是:通过各种控制协议,将有差错的物理信道变为无差错的、能可靠传输数据帧的数据链路。在计算机网络中由于各种干扰的存在,物理链路是不可靠的。因此,这一层的主要功能是在物理层提供的比特流的基础上,通过差错控制、流量控制方法,使有差错的物理线路变为无差错的数据链路,即提供可靠的通过物理介质传输数据的方法。
该层通常又被分为介质访问控制(MAC)和逻辑链路控制(LLC)两个子层。
MAC子层的主要任务是解决共享型网络中多用户对信道竞争的问题,完成网络介质的访问控制;
LLC子层的主要任务是建立和维护网络连接,执行差错校验、流量控制和链路控制。
数据链路层的具体工作是接收来自物理层的位流形式的数据,并封装成帧,传送到上一层;同样,也将来自上层的数据帧,拆装为位流形式的数据转发到物理层;并且,还负责处理接收端发回的确认帧的信息,以便提供可靠的数据传输。

网络层

网络层(Network Layer)是OSI模型的第三层,它是OSI参考模型中最复杂的一层,也是通信子网的最高一层。它在下两层的基础上向资源子网提供服务。其主要任务是:通过路由选择算法,为报文或分组通过通信子网选择最适当的路径。该层控制数据链路层与传输层之间的信息转发,建立、维持和终止网络的连接。具体地说,数据链路层的数据在这一层被转换为数据包,然后通过路径选择、分段组合、顺序、进/出路由等控制,将信息从一个网络设备传送到另一个网络设备。
一般地,数据链路层是解决同一网络内节点之间的通信,而网络层主要解决不同子网间的通信。例如在广域网之间通信时,必然会遇到路由(即两节点间可能有多条路径)选择问题。
在实现网络层功能时,需要解决的主要问题如下:
寻址:数据链路层中使用的物理地址(如MAC地址)仅解决网络内部的寻址问题。在不同子网之间通信时,为了识别和找到网络中的设备,每一子网中的设备都会被分配一个唯一的地址。由于各子网使用的物理技术可能不同,因此这个地址应当是逻辑地址(如IP地址)。
交换:规定不同的信息交换方式。常见的交换技术有:线路交换技术和存储转发技术,后者又包括报文交换技术和分组交换技术。
路由算法:当源节点和目的节点之间存在多条路径时,本层可以根据路由算法,通过网络为数据分组选择最佳路径,并将信息从最合适的路径由发送端传送到接收端。
连接服务:与数据链路层流量控制不同的是,前者控制的是网络相邻节点间的流量,后者控制的是从源节点到目的节点间的流量。其目的在于防止阻塞,并进行差错检测。

传输层

OSI下3层的主要任务是数据通信,上3层的任务是数据处理。而传输层(Transport Layer)是OSI模型的第4层。因此该层是通信子网和资源子网的接口和桥梁,起到承上启下的作用。
该层的主要任务是:向用户提供可靠的端到端的差错和流量控制,保证报文的正确传输。传输层的作用是向高层屏蔽下层数据通信的细节,即向用户透明地传送报文。该层常见的协议:TCP/IP中的TCP协议、Novell网络中的SPX协议和微软的NetBIOS/NetBEUI协议。
传输层提供会话层和网络层之间的传输服务,这种服务从会话层获得数据,并在必要时,对数据进行分割。然后,传输层将数据传递到网络层,并确保数据能正确无误地传送到网络层。因此,传输层负责提供两节点之间数据的可靠传送,当两节点的联系确定之后,传输层则负责监督工作。综上,传输层的主要功能如下:
传输连接管理:提供建立、维护和拆除传输连接的功能。传输层在网络层的基础上为高层提供“面向连接”和“面向无接连”的两种服务。
处理传输差错:提供可靠的“面向连接”和不太可靠的“面向无连接”的数据传输服务、差错控制和流量控制。在提供“面向连接”服务时,通过这一层传输的数据将由目标设备确认,如果在指定的时间内未收到确认信息,数据将被重发。
监控服务质量。

会话层

会话层(Session Layer)是OSI模型的第5层,是用户应用程序和网络之间的接口,主要任务是:向两个实体的表示层提供建立和使用连接的方法。将不同实体之间的表示层的连接称为会话。因此会话层的任务就是组织和协调两个会话进程之间的通信,并对数据交换进行管理。
用户可以按照半双工、单工和全双工的方式建立会话。当建立会话时,用户必须提供他们想要连接的远程地址。而这些地址与MAC(介质访问控制子层)地址或网络层的逻辑地址不同,它们是为用户专门设计的,更便于用户记忆。域名(DN)就是一种网络上使用的远程地址例如:www.3721.com就是一个域名。会话层的具体功能如下:
会话管理:允许用户在两个实体设备之间建立、维持和终止会话,并支持它们之间的数据交换。例如提供单方向会话或双向同时会话,并管理会话中的发送顺序,以及会话所占用时间的长短。
会话流量控制:提供会话流量控制和交叉会话功能。
寻址:使用远程地址建立会话连接。l
出错控制:从逻辑上讲会话层主要负责数据交换的建立、保持和终止,但实际的工作却是接收来自传输层的数据,并负责纠正错误。会话控制和远程过程调用均属于这一层的功能。但应注意,此层检查的错误不是通信介质的错误,而是磁盘空间、打印机缺纸等类型的高级错误。
表示层
表示层(Presentation Layer)是OSI模型的第六层,它对来自应用层的命令和数据进行解释,对各种语法赋予相应的含义,并按照一定的格式传送给会话层。其主要功能是“处理用户信息的表示问题,如编码、数据格式转换和加密解密”等。表示层的具体功能如下:
数据格式处理:协商和建立数据交换的格式,解决各应用程序之间在数据格式表示上的差异。
数据的编码:处理字符集和数字的转换。例如由于用户程序中的数据类型(整型或实型、有符号或无符号等)、用户标识等都可以有不同的表示方式,因此,在设备之间需要具有在不同字符集或格式之间转换的功能。
压缩和解压缩:为了减少数据的传输量,这一层还负责数据的压缩与恢复。
数据的加密和解密:可以提高网络的安全性。

应用层

应用层(Application Layer)是OSI参考模型的最高层,它是计算机用户,以及各种应用程序和网络之间的接口,其功能是直接向用户提供服务,完成用户希望在网络上完成的各种工作。它在其他6层工作的基础上,负责完成网络中应用程序与网络操作系统之间的联系,建立与结束使用者之间的联系,并完成网络用户提出的各种网络服务及应用所需的监督、管理和服务等各种协议。此外,该层还负责协调各个应用程序间的工作。
应用层为用户提供的服务和协议有:文件服务、目录服务、文件传输服务(FTP)、远程登录服务(Telnet)、电子邮件服务(E-mail)、打印服务、安全服务、网络管理服务、数据库服务等。上述的各种网络服务由该层的不同应用协议和程序完成,不同的网络操作系统之间在功能、界面、实现技术、对硬件的支持、安全可靠性以及具有的各种应用程序接口等各个方面的差异是很大的。应用层的主要功能如下:
用户接口:应用层是用户与网络,以及应用程序与网络间的直接接口,使得用户能够与网络进行交互式联系。
实现各种服务:该层具有的各种应用程序可以完成和实现用户请求的各种服务。
OSI7层模型的小结
由于OSI是一个理想的模型,因此一般网络系统只涉及其中的几层,很少有系统能够具有所有的7层,并完全遵循它的规定。
在7层模型中,每一层都提供一个特殊的网络功能。从网络功能的角度观察:下面4层(物理层、数据链路层、网络层和传输层)主要提供数据传输和交换功能,即以节点到节点之间的通信为主;第4层作为上下两部分的桥梁,是整个网络体系结构中最关键的部分;而上3层(会话层、表示层和应用层)则以提供用户与应用程序之间的信息和数据处理功能为主。简言之,下4层主要完成通信子网的功能,上3层主要完成资源子网的功能。

以下是TCP/IP分层模型
┌────——────┐┌─┬─┬─-┬─┬─-┬─┬─-┬─┬─-┬─┬─-┐
  │        ││D│F│W│F│H│G│T│I│S│U│ │
  │        ││N│I│H│T│T│O│E│R│M│S│其│
  │第四层,应用层 ││S│N│O│P│T│P│L│C│T│E│ │
  │        ││ │G│I│ │P│H│N│ │P│N│ │
  │        ││ │E│S│ │ │E│E│ │ │E│它│
  │        ││ │R│ │ │ │R│T│ │ │T│ │
  └───────——─┘└─┴─┴─-┴─┴─-┴─┴─-┴─┴─-┴─┴-─┘
  ┌───────—–─┐┌─────────——-┬──——–─────────┐
  │第三层,传输层 ││   TCP   │    UDP    │
  └───────—–─┘└────────——-─┴──────────——–─┘
  ┌───────—–─┐┌───—-──┬───—─┬────────——-──┐
  │        ││     │ICMP│          │
  │第二层,网间层 ││     └──—──┘          │
  │        ││       IP            │
  └────────—–┘└────────────────────————-─-┘
  ┌────────—–┐┌─────────——-┬──────——–─────┐
  │第一层,网络接口││ARP/RARP │    其它     │
  └────────——┘└─────────——┴─────——–──────┘
       TCP/IP四层参考模型

  TCP/IP协议被组织成四个概念层,其中有三层对应于ISO参考模型中的相应层。ICP/IP协议族并不包含物理层和数据链路层,因此它不能独立完成整个计算机网络系统的功能,必须与许多其他的协议协同工作。
  TCP/IP分层模型的四个协议层分别完成以下的功能:
  第一层:网络接口层
  包括用于协作IP数据在已有网络介质上传输的协议。实际上TCP/IP标准并不定义与ISO数据链路层和物理层相对应的功能。相反,它定义像地址解析协议(Address Resolution Protocol,ARP)这样的协议,提供TCP/IP协议的数据结构和实际物理硬件之间的接口。
  第二层:网间层
  对应于OSI七层参考模型的网络层。本层包含IP协议、RIP协议(Routing Information Protocol,路由信息协议),负责数据的包装、寻址和路由。同时还包含网间控制报文协议(Internet Control Message Protocol,ICMP)用来提供网络诊断信息。
  第三层:传输层
  对应于OSI七层参考模型的传输层,它提供两种端到端的通信服务。其中TCP协议(Transmission Control Protocol)提供可靠的数据流运输服务,UDP协议(Use Datagram Protocol)提供不可靠的用户数据报服务。
  第四层:应用层
  对应于OSI七层参考模型的应用层和表达层。因特网的应用层协议包括Finger、Whois、FTP(文件传输协议)、Gopher、HTTP(超文本传输协议)、Telent(远程终端协议)、SMTP(简单邮件传送协议)、IRC(因特网中继会话)、NNTP(网络新闻传输协议)等

IP地址和端口

IP地址用于唯一地表示网络中的一个通讯实体。IP地址是IP协议提供的一种统一的地址格式,IP地址分为五类:

  • A类用于大型网络(能容纳网络126个,主机1677214台)
  • B类用于中型网络(能容纳网络16384个,主机65534台)
  • C类用于小型网络(能容纳网络2097152个,主机254台)
  • D类用于组播(多目的地址的发送)
  • E类用于实验

另外,全零(0.0.0.0.)地址指任意网络。全1的IP地址(255.255.255.255)是当前子网的广播地址。
IP地址采用层次结构,按照逻辑结构划分为两个部分:网络号和主机号。网络号用于识别一个逻辑网络,而主机号用于识别网络中的一台主机的一个连接。因此,IP地址的编址方式携带了明显的位置消息。
一个完整的IP地址由4个字节32位数字组成,为了方便用户理解和记忆,采用点分十进制标记法,字节之间使用符号“.”隔开。
采用32位形式的IP地址如下:00001010 00000000 00000000 00000001
采用点分十进制标记法如下:10.0.0.1

A类地址

  1. A类IP地址。由1个字节的网络地址和3个字节的主机地址,网络地址的最高位必须是“0”。
    如:0XXXXXXX.XXXXXXXX.XXXXXXXX.XXXXXXXX(X代表0或1)
  2. A类IP地址范围:1.0.0.1—126.255.255.254
  3. A类IP地址中的私有地址和保留地址:
    ① 10.X.X.X是私有地址(所谓的私有地址就是在互联网上不使用,而被用在局域网络中的地址)。
    范围(10.0.0.1—10.255.255.254)
    ② 127.X.X.X是保留地址,用做循环测试用的。

B类地址

  1. B类IP地址。由2个字节的网络地址和2个字节的主机地址,网络地址的最高位必须是“10”。
    如:10XXXXXX.XXXXXXXX.XXXXXXXX.XXXXXXXX(X代表0或1)
  2. B类IP地址范围:128.0.0.1—191.255.255.254。
  3. B类IP地址的私有地址和保留地址
    ① 172.16.0.0—172.31.255.254是私有地址
    ② 169.254.X.X是保留地址。如果你的IP地址是自动获取IP地址,而你在网络上又没有找到可用的DHCP服务器。就会得到其中一个IP。191.255.255.255是广播地址,不能分配。

C类地址

  1. C类IP地址。由3个字节的网络地址和1个字节的主机地址,网络地址的最高位必须是“110”。
    如:110XXXXX.XXXXXXXX.XXXXXXXX.XXXXXXXX(X代表0或1)
  2. C类IP地址范围:192.0.0.1—223.255.255.254。
  3. C类地址中的私有地址:
    192.168.X.X是私有地址。(192.168.0.1—192.168.255.255)这些私有地址是用于局域网的。

D类地址

  1. D类地址不分网络地址和主机地址,它的第1个字节的前四位固定为1110。
    如:1110XXXX.XXXXXXXX.XXXXXXXX.XXXXXXXX(X代表0或1)
  2. D类地址范围:224.0.0.1—239.255.255.254

E类地址

  1. E类地址不分网络地址和主机地址,它的第1个字节的前四位固定为 1111。
    如:1111XXXX.XXXXXXXX.XXXXXXXX.XXXXXXXX(X代表0或1)
  2. E类地址范围:240.0.0.1—255.255.255.254是一个32位的二进制地址,为了便于记忆,将它们分为4组,每组8位,由小数点分开,用四个字节来表示,而且,用点分开的每个字节的数值范围是0~255,如202.116.0.1,这种书写方法叫做点数表示法。

一个通信实体同时有多个通信程序提供服务只有IP还不行,还需要端口。端口是一个16位整数,是一种抽象的软件结构,是应用程序与网络交流的门户。不同的应用程序处理不同的端口上的数据,同一台机器上不能有两个程序使用同一个端口。端口号可以是0~65535,通常分为三类:

  • 公认端口(Well Known Ports):0~1023,它们紧密绑定一些特定服务;
  • 注册端口(Registered Ports):1024~49151,它们松散地绑定一些服务,应用程序通常使用这些端口;
  • 动态和或私有端口(Dynamic and/or Private Ports ):49152~65535,这些端口是应用程序使用的动态端口,应用程序一般不会主动使用这些端口。

Java的基本网络支持

Java在java.net包下提供了一些网络支持的类和接口。

InetAddress类

InetAddress类代表一个IP地址, 没有构造器,依靠静态方法获取InetAddress实例:

  • static InetAddress[] getAllByName(String host) :在给定主机名的情况下,根据系统上配置的名称服务返回其 IP 地址所组成的数组。因为有些域名是有多个IP地址的。;
  • static InetAddress getByAddress(byte[] addr) :在给定原始 IP 地址的情况下,返回 InetAddress 对象;
  • static InetAddress getByAddress(String host, byte[] addr) :根据提供的主机名和 IP 地址创建 InetAddress;
  • static InetAddress getByName(String host) :在给定主机名的情况下确定主机的 IP 地址;
  • static InetAddress getLocalHost() :返回本地主机IP对应的InetAddress对象。实现尽最大努力试图到达主机,但防火墙和服务器配置可能阻塞请求,使其在某些特定的端口可以访问时处于不可到达状态。如果可以获得权限,则典型实现将使用 ICMP ECHO REQUEST;否则它将试图在目标主机的端口 7 (Echo) 上建立 TCP 连接。

其他方法摘要:

  • byte[] getAddress() : 返回此 InetAddress 对象的原始 IP 地址;
  • String getCanonicalHostName() :获取此 IP 地址的完全限定域名;
  • String getHostAddress() :返回 IP 地址字符串(以文本表现形式);
  • String getHostName() :获取此 IP 地址的主机名;
  • boolean isReachable(int timeout) :测试是否可以达到该地址。

InetAddress类还有两个子类:Inet4Address和Inet6Address,分别代表IPv4地址和IPv6地址。

InetAddress示例:

public class InetAddressTest
{public static void main(String[] args)throws Exception{// 根据主机名来获取对应的InetAddress实例InetAddress ip = InetAddress.getByName("www.hao123.com");// 判断是否可达System.out.println("hao123是否可达:" + ip.isReachable(2000));// 获取该InetAddress实例的IP字符串System.out.println(ip.getHostAddress());// 根据原始IP地址来获取对应的InetAddress实例InetAddress local = InetAddress.getByAddress(new byte[]{127,0,0,1});System.out.println("本机是否可达:" + local.isReachable(5000));// 获取该InetAddress实例对应的全限定域名System.out.println(local.getCanonicalHostName());}
}

InetSocketAddress
此类代表了IP 地址和端口号。它还可以是一个对主机名+端口号,在此情况下,将尝试解析主机名。如果解析失败,则该地址将被视为未解析地址,但是其在某些情形下仍然可以使用,比如通过代理连接。它提供不可变对象,供套接字用于绑定、连接或用作返回值。 通配符是一个特殊的本地 IP 地址。它通常表示“任何”,只能用于 bind 操作。

构造方法摘要:

  • InetSocketAddress(InetAddress addr, int port)
    根据 IP 地址和端口号创建套接字地址。
  • InetSocketAddress(int port)
    创建套接字地址,其中 IP 地址为通配符地址,端口号为指定值。
  • InetSocketAddress(String hostname, int port)
    根据主机名和端口号创建套接字地址。

常用方法方法摘要:

  • static InetSocketAddress createUnresolved(String host, int port):根据主机名和端口号创建未解析的套接字地址;
  • boolean equals(Object obj):将此对象与指定对象比较;
  • InetAddress getAddress():获取 InetAddress;
  • String getHostName():获取 hostname;
  • int getPort():获取端口号;
  • boolean isUnresolved():检查是否已解析地址。

URLDecoder和URLEncoder

在Google搜索“中国”,在搜索结果的页面的地址栏末尾显式“%E4%B8%AD%E5%9B%BD”,这种字符串称为application/x-www-form-urlencoded字符串。URL中含有非西欧字符串时会将其转换为application/x-www-form-urlencoded字符串。Java中提供了URLDecoder类和URLEncoder类来完成这种相互转换:

  • URLDecoder类提供了static String decode(String s, String enc) 静态方法:使用指定的编码机制对 application/x-www-form-urlencoded 字符串解码;
  • URLEncoder类提供了static String encode(String s, String enc) 静态方法: 使用指定的编码机制将字符串转换为 application/x-www-form-urlencoded 格式。

示例:

public class URLDecoderTest
{public static void main(String[] args)throws Exception{// 将application/x-www-form-urlencoded字符串// 转换成普通字符串,输出中国String keyWord = URLDecoder.decode("%E4%B8%AD%E5%9B%BD", "utf-8");System.out.println(keyWord);// 将普通字符串转换成// application/x-www-form-urlencoded字符串String urlStr = URLEncoder.encode("中国" , "GBK");System.out.println(urlStr);}
} 

URL 、URLConnection、 URLPermission

URL(Uniform Resource Locator)即统一资源定位器,它是指向互联网“资源”的指针。资源可以是简单的文件或目录,也可以是对更为复杂的对象的引用,例如对数据库或搜索引擎的查询。有关 URL 的类型和格式的更多信息,可从以下位置找到:
http://www.socs.uts.edu.au/MosaicDocs-old/url-primer.html 。
URL主要有协议名,主机名,端口号,资源等组成。格式为:
protocol://host:port/resourceName
例如:http://www.socs.uts.edu.au/MosaicDocs-old/url-primer.html,它的协议是http,中间是主机路径,后面为文件名,没有端口号(默认80)。
URL类就是这样一个处理同意资源定位器的类。
构造方法摘要:

  • URL(String spec) :根据 String 表示形式创建 URL 对象;
  • URL(URL context, String spec) :通过在指定的上下文中对给定的 spec 进行解析创建 URL;
  • URL(URL context, String spec, URLStreamHandler handler):通过在指定的上下文中用指定的处理程序对给定的 spec 进行解析来创建 URL;
  • URL(String protocol, String host, String file) :根据指定的 protocol 名称、host 名称和 file 名称创建 URL;
  • URL(String protocol, String host, int port, String file):根据指定 protocol、host、port 号和 file 创建 URL 对象;
  • URL(String protocol, String host, int port, String file, URLStreamHandler handler) :根据指定的 protocol、host、port 号、file 和 handler 创建 URL 对象。

常用方法摘要:

  • String getFile() :获取此 URL 的文件名;
  • String getHost() :获取此 URL 的主机名(如果适用);
  • String getPath() :获取此 URL 的路径部分;
  • int getPort() :获取此 URL 的端口号;
  • String getProtocol() :获取此 URL 的协议名称;
  • String getQuery() :获取此 URL 的查询部分;
  • String getRef() :获取此 URL 的锚点(也称为“引用”);
  • String getUserInfo() :获取此 URL 的 userInfo 部分;
  • protected void set(String protocol, String host, int port, String file, String ref) :设置 URL 的字段;
  • protected void set(String protocol, String host, int port, String authority, String userInfo, String path, - String query, String ref) :设置 URL 的指定的 8 个字段;
  • URLConnection openConnection() :返回一个 URLConnection 对象,它表示到 URL 所引用的远程对象的连接;
  • URLConnection openConnection(Proxy proxy):与openConnection()类似,所不同是连接通过指定的代理建立;不支持代理方式的协议处理程序将忽略该代理参数并建立正常的连接;
  • InputStream openStream() :打开到此 URL 的连接并返回一个用于从该连接读入的 InputStream。

URLConnection类是一个抽象类,它是所有应用程序和URL通信连接类的超类,此类的实例可用于读取和写入此 URL 引用的资源。通常,创建一个到 URL 的连接需要几个步骤:

  • 通过在 URL 上调用 openConnection 方法创建URLConnection连接对象;
  • 设置URLCOnnection参数和一般请求属性;
  • 如果只发送GET方式请求,则使用 connect 方法建立到远程对象的连接即可。如果需要发送POST方式的请求,则需要获取URLConnection实例对应的输出流来发送请求参数;
  • 远程对象变为可用。可以访问远程对象的头字段或通过输入流读取远程对象的数据。

该类提供以下字段:

  • protected boolean allowUserInteraction:如果为true,则在允许用户交互(例如弹出一个验证对话框)的上下文中对此 URL 进行检查;
  • protected boolean connected :如果为 false,则此连接对象尚未创建到指定 URL 的通信连接;
  • protected boolean doInput :此变量由 setDoInput 方法设置;
  • protected boolean doOutput :此变量由 setDoOutput 方法设置;
  • protected long ifModifiedSince :有些协议支持跳过对象获取,除非该对象在某个特定时间点之后又进行了修改;
  • protected URL url :URL 表示此连接要在互联网上打开的远程对象;
  • protected boolean useCaches :如果为 true,则只要有条件就允许协议使用缓存。

在建立到远程对象的连接之前,可使用以下方法修改或设置以上参数:

void setConnectTimeout(int timeout) :设置一个指定的超时值(以毫秒为单位),该值将在打开到此 URLConnection 引用的资源的通信链接时使用;
- static void setDefaultAllowUserInteraction(boolean defaultallowuserinteraction) :将未来的所有 URLConnection 对象的 allowUserInteraction 字段的默认值设置为指定的值;
- void setAllowUserInteraction(boolean allowuserinteraction) :设置此 URLConnection 的 allowUserInteraction 字段的值;
- void setDefaultUseCaches(boolean defaultusecaches) :将 useCaches 字段的默认值设置为指定的值;
- void setDoInput(boolean doinput) :将此 URLConnection 的 doInput 字段的值设置为指定的值;
- void setDoOutput(boolean dooutput) :将此 URLConnection 的 doOutput 字段的值设置为指定的值;
- void setIfModifiedSince(long ifmodifiedsince) :将此 URLConnection 的 ifModifiedSince 字段的值设置为指定的值;
- void setUseCaches(boolean usecaches) :将此 URLConnection 的 useCaches 字段的值设置为指定的值。

使用以下方法修改一般请求属性:

  • void setRequestProperty(String key, String value) :设置一般请求属性。如果已存在具有该关键字的属性,则用新值改写其值。使用以逗号分隔的列表语法,这样可实现将多个属性添加到一个属性中。
  • void addRequestProperty(String key, Stringvalue):添加由键值对指定的一般请求属性。此方法不会改写与相同键关联的现有值。

上面每个 set 方法都有一个用于获取参数值或一般请求属性值的对应get方法。适用的具体参数和一般请求属性取决于协议。

建立到远程对象的连接后,以下方法用于访问头字段和内容:

  • Object getContent() :获取此 URL 连接的内容;
  • Object getContent(Class[] classes) :获取此 URL 连接的内容;
  • String getHeaderField(int n) :返回第 n 个头字段的值;
  • String getHeaderField(String name) :返回指定的头字段的值;
  • OutputStream getOutputStream() :返回写入到此连接的输出流;
  • InputStream getInputStream() :返回从此打开的连接读取的输入流。

某些头字段需要经常访问。以下方法提供对这些字段的便捷访问:

  • String getContentEncoding() :获取content-encoding响应头字段的值;
  • int getContentLength() :获取content-length响应头字段的值;
  • String getContentType() :获取content-type响应头字段的值;
  • long getDate():获取date响应头字段的值;
  • long getExpiration():获取expires(有效期)响应头字段的值;
  • long getLastModified():获取last-modified响应头字段的值。

getContent 方法使用 getContentType 方法以确定远程对象类型;子类重写 getContentType 方法很容易。

HttpURLConnection是URLConnection的子类,代表与URL之间的HTTP连接。该类提供了许多HTTP状态码的字段。
构造方法摘要:

  • protected HttpURLConnection(URL u) :HttpURLConnection 的构造方法。

方法摘要:

  • abstract void disconnect() :指示近期服务器不太可能有其他请求;
  • abstract boolean usingProxy() :指示连接是否通过代理。
  • static boolean getFollowRedirects() :返回指示是否应该自动执行 HTTP 重定向 (3xx) 的 boolean 值;
  • InputStream getErrorStream() :如果连接失败但服务器仍然发送了有用数据,则返回错误流;
  • String getHeaderField(int n) :返回 nth 头字段的值;
  • long getHeaderFieldDate(String name, long Default) :返回解析为日期的指定字段的值;
  • String getHeaderFieldKey(int n) : 返回 nth 头字段的键;
  • boolean getInstanceFollowRedirects() :返回此 HttpURLConnection 的 instanceFollowRedirects 字段的值;
  • Permission getPermission() :返回一个权限对象,其代表建立此对象表示的连接所需的权限;
  • String getRequestMethod() :获取请求方法;
  • int getResponseCode() :从 HTTP 响应消息获取状态码;
  • String getResponseMessage() :获取与来自服务器的响应代码一起返回的 HTTP 响应消息(如果有);
  • void setChunkedStreamingMode(int chunklen) :此方法用于在预先不知道内容长度时启用没有进行内部缓冲的 HTTP 请求正文的流;
  • void setFixedLengthStreamingMode(int contentLength) : 此方法用于在预先已知内容长度时启用没有进行内部缓冲的 HTTP 请求正文的流;
  • static void setFollowRedirects(boolean set) : 设置此类是否应该自动执行 HTTP 重定向(响应代码为 3xx 的请求);
  • void setInstanceFollowRedirects(boolean followRedirects) :设置此 HttpURLConnection 实例是否应该自动执行 HTTP 重定向(响应代码为 3xx 的请求);
  • void setRequestMethod(String method) :设置 URL 请求的方法, GET POST HEAD OPTIONS PUT DELETE TRACE 以上方法之一是合法的,具体取决于协议的限制;

Java8新增了一个URLPermission工具类,用于管理HttpURLConnection的权限问题,如果在HttpURLConnection安装了安全管理器,通过该对象打开链接时就需要先取得权限。

多线程下载工具类示例:

public class DownUtil
{// 定义下载资源的路径private String path;// 指定所下载的文件的保存位置private String targetFile;// 定义需要使用多少线程下载资源private int threadNum;// 定义下载的线程对象private DownThread[] threads;// 定义下载的文件的总大小private int fileSize;public DownUtil(String path, String targetFile, int threadNum){this.path = path;this.threadNum = threadNum;// 初始化threads数组threads = new DownThread[threadNum];this.targetFile = targetFile;}public void download() throws Exception{URL url = new URL(path);HttpURLConnection conn = (HttpURLConnection) url.openConnection();conn.setConnectTimeout(5 * 1000);conn.setRequestMethod("GET");conn.setRequestProperty("Accept","image/gif, image/jpeg, image/pjpeg, image/pjpeg, "+ "application/x-shockwave-flash, application/xaml+xml, "+ "application/vnd.ms-xpsdocument, application/x-ms-xbap, "+ "application/x-ms-application, application/vnd.ms-excel, "+ "application/vnd.ms-powerpoint, application/msword, */*");conn.setRequestProperty("Accept-Language", "zh-CN");conn.setRequestProperty("Charset", "UTF-8");conn.setRequestProperty("Connection", "Keep-Alive");// 得到文件大小fileSize = conn.getContentLength();conn.disconnect();int currentPartSize = fileSize / threadNum + 1;RandomAccessFile file = new RandomAccessFile(targetFile, "rw");// 设置本地文件的大小file.setLength(fileSize);file.close();for (int i = 0; i < threadNum; i++){// 计算每条线程的下载的开始位置int startPos = i * currentPartSize;// 每个线程使用一个RandomAccessFile进行下载RandomAccessFile currentPart = new RandomAccessFile(targetFile,"rw");// 定位该线程的下载位置currentPart.seek(startPos);// 创建下载线程threads[i] = new DownThread(startPos, currentPartSize,currentPart);// 启动下载线程threads[i].start();}}// 获取下载的完成百分比public double getCompleteRate(){// 统计多条线程已经下载的总大小int sumSize = 0;for (int i = 0; i < threadNum; i++){sumSize += threads[i].length;}// 返回已经完成的百分比return sumSize * 1.0 / fileSize;}private class DownThread extends Thread{// 当前线程的下载位置private int startPos;// 定义当前线程负责下载的文件大小private int currentPartSize;// 当前线程需要下载的文件块private RandomAccessFile currentPart;// 定义已经该线程已下载的字节数public int length;public DownThread(int startPos, int currentPartSize,RandomAccessFile currentPart){this.startPos = startPos;this.currentPartSize = currentPartSize;this.currentPart = currentPart;}@Overridepublic void run(){try{URL url = new URL(path);HttpURLConnection conn = (HttpURLConnection)url.openConnection();conn.setConnectTimeout(5 * 1000);conn.setRequestMethod("GET");conn.setRequestProperty("Accept","image/gif, image/jpeg, image/pjpeg, image/pjpeg, "+ "application/x-shockwave-flash, application/xaml+xml, "+ "application/vnd.ms-xpsdocument, application/x-ms-xbap, "+ "application/x-ms-application, application/vnd.ms-excel, "+ "application/vnd.ms-powerpoint, application/msword, */*");conn.setRequestProperty("Accept-Language", "zh-CN");conn.setRequestProperty("Charset", "UTF-8");InputStream inStream = conn.getInputStream();// 跳过startPos个字节,表明该线程只下载自己负责哪部分文件。inStream.skip(this.startPos);byte[] buffer = new byte[1024];int hasRead = 0;// 读取网络数据,并写入本地文件while (length < currentPartSize&& (hasRead = inStream.read(buffer)) != -1){currentPart.write(buffer, 0, hasRead);// 累计该线程下载的总大小length += hasRead;}currentPart.close();inStream.close();}catch (Exception e){e.printStackTrace();}}}
}

上面多线程下载的步骤是:

  1. 创建URL对象;
  2. 通过URL对象openConnection方法来获取对应的额URLConnection对象,然后获得其所指向资源的大小(通过getContentLength()方法);
  3. 在本地磁盘上创建一个与网络资源大小相等的空文件;
  4. 计算每个线程应该下载网络资源的哪个部分;
  5. 依次创建、启动多个线程来下载网络资源的多个部分。

执行多线程下载示例:

public class MultiThreadDown
{public static void main(String[] args) throws Exception{// 初始化DownUtil对象final DownUtil downUtil = new DownUtil("https://img-blog.csdn.net/20161225130602309?watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvcm9zeV9kYXdu/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/SouthEast", "collection.png", 4);// 开始下载downUtil.download();new Thread(() -> {while(downUtil.getCompleteRate() < 1){// 每隔0.1秒查询一次任务的完成进度,// GUI程序中可根据该进度来绘制进度条System.out.println("已完成:"+ downUtil.getCompleteRate());try{Thread.sleep(1000);}catch (Exception ex){}}}).start();}
}

如果既要使用输入流读取URLConnection响应的内容,又要使用输出流发送请求参数,一定要先使用输出流,再使用输入流。

发送GET请求时只需要请求参数放在URL字符串之后,以?隔开,程序直接调用URLConnection对象的connect()方法即可;如果程序需要发送POST请求,则需要先设置doIn和doOut两个请求头字段的值,再使用URLConnection对应的输出流来发送请求参数。不管是事发送GET请求,还是POST请求,程序获取URLConnection响应的方式一样——如果程序确定远程响应是字符流,则可以使用字符流来读取,如果程序无法确定远程响应是字符流,则使用字节流即可。

GET和POST请求示例:

public class GetPostTest
{/*** 向指定URL发送GET方法的请求* @param url 发送请求的URL* @param param 请求参数,格式满足name1=value1&name2=value2的形式。* @return URL所代表远程资源的响应*/public static String sendGet(String url , String param){String result = "";String urlName = url + "?" + param;try{URL realUrl = new URL(urlName);// 打开和URL之间的连接URLConnection conn = realUrl.openConnection();// 设置通用的请求属性conn.setRequestProperty("accept", "*/*");conn.setRequestProperty("connection", "Keep-Alive");conn.setRequestProperty("user-agent", "Mozilla/4.0 (compatible; MSIE 6.0; Windows NT 5.1; SV1)");// 建立实际的连接conn.connect();// 获取所有响应头字段Map<String, List<String>> map = conn.getHeaderFields();// 遍历所有的响应头字段for (String key : map.keySet()){System.out.println(key + "--->" + map.get(key));}try(// 定义BufferedReader输入流来读取URL的响应BufferedReader in = new BufferedReader(new InputStreamReader(conn.getInputStream() , "utf-8"))){String line;while ((line = in.readLine())!= null){result += "\n" + line;}}}catch(Exception e){System.out.println("发送GET请求出现异常!" + e);e.printStackTrace();}return result;}/*** 向指定URL发送POST方法的请求* @param url 发送请求的URL* @param param 请求参数,格式应该满足name1=value1&name2=value2的形式。* @return URL所代表远程资源的响应*/public static String sendPost(String url , String param){String result = "";try{URL realUrl = new URL(url);// 打开和URL之间的连接URLConnection conn = realUrl.openConnection();// 设置通用的请求属性conn.setRequestProperty("accept", "*/*");conn.setRequestProperty("connection", "Keep-Alive");conn.setRequestProperty("user-agent","Mozilla/4.0 (compatible; MSIE 6.0; Windows NT 5.1; SV1)");// 发送POST请求必须设置如下两行conn.setDoOutput(true);conn.setDoInput(true);try(// 获取URLConnection对象对应的输出流PrintWriter out = new PrintWriter(conn.getOutputStream())){// 发送请求参数out.print(param);// flush输出流的缓冲out.flush();}try(// 定义BufferedReader输入流来读取URL的响应BufferedReader in = new BufferedReader(new InputStreamReader(conn.getInputStream() , "utf-8"))){String line;while ((line = in.readLine())!= null){result += "\n" + line;}}}catch(Exception e){System.out.println("发送POST请求出现异常!" + e);e.printStackTrace();}return result;}// 提供主方法,测试发送GET请求和POST请求public static void main(String args[]){// 发送GET请求String s = GetPostTest.sendGet("http://blog.csdn.net/", null);System.out.println(s);// 发送POST请求String s1 = GetPostTest.sendPost("http://passport.csdn.net/account/login", "ref=toolbar");System.out.println(s1);}
}

基于TCP协议的网络传输

TCP/IP协议

TCP/IP(Transmission Control Protocol/Internet Protocol),即传输控制协议/因特网互联协议,又名网络通讯协议,是Internet最基本的协议、Internet国际互联网络的基础,由网络层的IP协议和传输层的TCP协议组成。

IP协议的责任就是把数据以数据包的形式从源地址传送到目的地址,根据数据包包头中包括的目的地址将数据包传送到目的地址。它还提供对数据大小的重新组装功能,以适应不同网络对包大小的要求。但是IP协议不负责保证传送可靠性,流控制,包顺序和其它对于主机到主机协议来说很普通的服务。

TCP协议用于在通信设备之间建立用于发送和接受数据的链路。TCP协议采用的重发机制(一但传输出现问题就发出信号,要求重新传输,直到所有数据安全正确地传输到目的地)保证了数据传输的可靠性。

虽然以上两种协议功能不同,但是它们是在同一时期作为同一个协议设计的,功能上互补,访问Internet的计算机必须都安装这两个协议,实际上将两者统称为TCP/IP协议。

使用ServerSocket/Socket进行通信

ServerSocket类
在两个通信端没有建立虚拟链路之前,必须有一个通信实体首先主动监听来自另一端的请求。Java提供了ServerSocket类来接受其他通信实体的连接请求。ServerSocket对象使用accept()方法用于监听来自客户端的Socket连接,如果收到一个客户端Socket的连接请求,该方法将返回一个与客户端Socket对应的Socket对象。如果没有连接,它将一直处于等待状态。通常情况下,服务器不应只接受一个客户端请求,而应该通过循环调用accept()不断接受来自客户端的所有请求。

构造器摘要:

  • ServerSocket():创建非绑定ServerSocket对象;
  • ServerSocket(int port):创建绑定到特定端口(推荐1024以上的端口号)的ServerSocket对象,没有使用IP时默认绑定到本机默认的IP;
  • ServerSocket(int port, int backlog):增加一个用来改变连接队列长度backlog参数;
  • ServerSocket(int port, int backlog, InetAddress bindAddr):在机器存在多个IP地址的情况下,允许通过localAddr参数来指定将ServerSocket绑定到指定的IP地址。

方法摘要:

  • Socket accept():如果接收到来自客户端的一个连接请求,则返回一个与客户端对应的Socket,否则该方法将会被阻塞;
  • void bind(SocketAddress endpoint):将 ServerSocket 绑定到特定地址(IP 地址和端口号);
  • void bind(SocketAddress endpoint, int backlog):将 ServerSocket 绑定到特定地址(IP 地址和端口号);
  • void close():当使用完ServerSocket之后,应该使用该方法来关闭此ServerSocket对象;
  • ServerSocketChannel getChannel():返回与此套接字关联的唯一 ServerSocketChannel 对象(如果有);
  • InetAddress getInetAddress():返回此服务器套接字的本地地址;
  • int getLocalPort():返回此套接字在其上侦听的端口;
  • SocketAddress getLocalSocketAddress():返回此套接字绑定的端点的地址,如果尚未绑定则返回 null;
  • int getReceiveBufferSize():获取此 ServerSocket 的 SO_RCVBUF 选项的值,该值是将用于从此 ServerSocket 接受的套接字的建议缓冲区大小;
  • boolean getReuseAddress():测试是否启用 SO_REUSEADDR;
  • int getSoTimeout():获取 SO_TIMEOUT 的设置;
  • protected void implAccept(Socket s):ServerSocket 的子类使用此方法重写 accept() 以返回它们自己的套接字子类;
  • boolean isBound():返回 ServerSocket 的绑定状态;
  • boolean isClosed():返回 ServerSocket 的关闭状态;
  • void setPerformancePreferences(int connectionTime, int latency, int bandwidth):设置此 ServerSocket 的性能首选项;
  • void setReceiveBufferSize(int size):为从此 ServerSocket 接受的套接字的 SO_RCVBUF 选项设置默认建议值;
  • void setReuseAddress(boolean on):启用/禁用 SO_REUSEADDR 套接字选项;
  • static void setSocketFactory(SocketImplFactory fac):为应用程序设置服务器套接字实现工厂;
  • void setSoTimeout(int timeout):通过指定超时值启用/禁用 SO_TIMEOUT,以毫秒为单位。

Socket类
客户端通常使用Socket对象连接到指定的服务器。

构造器摘要:

  • Socket(InetAddress/String address, int port):创建一个连接到指定 IP 地址的指定端口号的Socket,由于没有指定本地IP地址和本地端口,默认使用本机主机的默认IP地址,默认使用系统分配的端口,适用于本机主机只有一个IP地址的情况;
  • Socket(InetAddress/String address, int port, InetAddress localAddr, int localPort)::创建一个连接到指定 IP 地址的指定端口号的Socket,并指定本地IP地址和本地端口,适用于本地主机有多个IP地址的情况;

方法摘要:

  • void bind(SocketAddress bindpoint) :将套接字绑定到本地地址;
  • void close():关闭此套接字;
  • void connect(SocketAddress endpoint):将此套接字连接到服务器,获取连接之前会一直阻塞;
  • void connect(SocketAddress endpoint, int timeout):将此套接字连接到服务器,并指定一个超时值;
  • void setSoTimeout(int timeout):启用/禁用带有指定超时值的 SO_TIMEOUT,以毫秒为单位;
  • boolean isBound():返回套接字的绑定状态;
  • boolean isClosed():返回套接字的关闭状态;
  • boolean isConnected():返回套接字的连接状态;
  • SocketChannel getChannel():返回与此数据报套接字关联的唯一 SocketChannel 对象(如果有);
  • InetAddress getInetAddress():返回套接字连接的地址;
  • boolean getKeepAlive():测试是否启用 SO_KEEPALIVE;
  • int getPort():返回此套接字连接到的远程端口;
  • InetAddress getLocalAddress():获取套接字绑定的本地地址;
  • int getLocalPort():返回此套接字绑定到的本地端口;
  • SocketAddress getLocalSocketAddress():返回此套接字绑定的端点的地址,如果尚未绑定则返回 null;
  • InputStream getInputStream():返回此套接字的输入流,利用该输入流从该套接字读取数据;
  • OutputStream getOutputStream():返回此套接字的输出流,利用该输出力向该套接字输出数据;

实际运行中可能不想一直阻塞,可以通过connect方法或setSoTimeout方法设置连接的超时时长,超时未连接成功则抛出SocketTimeoutException异常。

示例:

public class Server
{public static void main(String[] args)  {try{// 创建一个ServerSocket,用于监听客户端Socket的连接请求ServerSocket ss = new ServerSocket(30000);// 采用循环不断接受来自客户端的请求while (true){// 每当接受到客户端Socket的请求,服务器端也对应产生一个SocketSocket s = ss.accept();// 将Socket对应的输出流包装成PrintStreamPrintStream ps = new PrintStream(s.getOutputStream());// 进行普通IO操作ps.println("您好,您收到了服务器的新年祝福!");}}catch (IOException e) {e.printStacktrace();}finally {// 关闭输出流,关闭Socketps.close();s.close();}}
}
public class Client
{public static void main(String[] args){try { //Socket socket = new Socket("127.0.0.1" , 30000); //设置超时为10s//socket.setSoTimeout(10000);//或者使用connect()方法设置超时Socket socket = new Socket;socket.connect(new InetAddress(host, port), 10000);// 将Socket对应的输入流包装成BufferedReaderScaner scan = new Scaner(socket.getInputStream()));// 进行普通IO操作String line = scan.readLine();System.out.println("来自服务器的数据:" + line);}catch (IOException e){e.printStacktrace();}catch (SocketTimeoutException e) {e.printStacktrace();}finally {  // 关闭输入流、socketbr.close();socket.close();}}
}

先运行Server类,再运行Client类,程序会输出:“来自服务器的数据:您好,您收到了服务器的新年祝福!”。

由于连接和流的操作会阻塞,应该为客户端和服务端的每次通信都另外开辟一条线程。

一个命令行界面的C/S聊天室示例,该聊天室支持群聊和私聊:
下面是在server包下的类:

public interface MyProtocol
{// 定义协议字符串的长度int PROTOCOL_LEN = 2;// 下面是一些协议字符串,服务器和客户端交换的信息都应该在前、后添加这种特殊字符串。String MSG_ROUND = "§γ";String USER_ROUND = "∏∑";String LOGIN_SUCCESS = "1";String NAME_REP = "-1";String PRIVATE_ROUND = "★【";String SPLIT_SIGN = "※";
}
// 通过组合HashMap对象来实现MyMap,MyMap要求value也不可重复
public class MyMap<K,V>
{// 创建一个线程安全的HashMappublic Map<K ,V> map = Collections.synchronizedMap(new HashMap<K,V>());// 根据value来删除指定项public synchronized void removeByValue(Object value){for (Object key : map.keySet()){if (map.get(key) == value){map.remove(key);break;}}}// 获取所有value组成的Set集合public synchronized Set<V> valueSet(){Set<V> result = new HashSet<V>();// 将map中所有value添加到result集合中map.forEach((key , value) -> result.add(value));return result;}// 根据value查找key。public synchronized K getKeyByValue(V val){// 遍历所有key组成的集合for (K key : map.keySet()){// 如果指定key对应的value与被搜索的value相同,则返回对应的keyif (map.get(key) == val || map.get(key).equals(val)){return key;}}return null;}// 实现put()方法,该方法不允许value重复public synchronized V put(K key,V value){// 遍历所有value组成的集合for (V val : valueSet() ){// 如果某个value与试图放入集合的value相同// 则抛出一个RuntimeException异常if (val.equals(value)&& val.hashCode()== value.hashCode()){throw new RuntimeException("MyMap实例中不允许有重复value!");}}return map.put(key , value);}
}
public class Server
{private static final int SERVER_PORT = 30000;// 使用MyMap对象来保存每个客户名字和对应输出流之间的对应关系。public static MyMap<String , PrintStream> clients= new MyMap<>();public void init(){try(// 建立监听的ServerSocketServerSocket ss = new ServerSocket(SERVER_PORT)){// 采用死循环来不断接受来自客户端的请求while(true){Socket socket = ss.accept();new ServerThread(socket).start();}}// 如果抛出异常catch (IOException ex){System.out.println("服务器启动失败,是否端口"+ SERVER_PORT + "已被占用?");}}public static void main(String[] args){Server server = new Server();server.init();}
}
public class ServerThread extends Thread
{private Socket socket;BufferedReader br = null;PrintStream ps = null;// 定义一个构造器,用于接收一个Socket来创建ServerThread线程public ServerThread(Socket socket){this.socket = socket;}public void run(){try{// 获取该Socket对应的输入流br = new BufferedReader(new InputStreamReader(socket.getInputStream()));// 获取该Socket对应的输出流ps = new PrintStream(socket.getOutputStream());String line = null;while((line = br.readLine())!= null){// 如果读到的行以MyProtocol.USER_ROUND开始,并以其结束,// 可以确定读到的是用户登录的用户名if (line.startsWith(MyProtocol.USER_ROUND)&& line.endsWith(MyProtocol.USER_ROUND)){// 得到真实消息String userName = getRealMsg(line);// 如果用户名重复if (Server.clients.map.containsKey(userName)){System.out.println("重复");ps.println(MyProtocol.NAME_REP);}else{System.out.println("成功");ps.println(MyProtocol.LOGIN_SUCCESS);Server.clients.put(userName , ps);}}// 如果读到的行以MyProtocol.PRIVATE_ROUND开始,并以其结束,// 可以确定是私聊信息,私聊信息只向特定的输出流发送else if (line.startsWith(MyProtocol.PRIVATE_ROUND)&& line.endsWith(MyProtocol.PRIVATE_ROUND)){// 得到真实消息String userAndMsg = getRealMsg(line);// 以SPLIT_SIGN分割字符串,前半是私聊用户,后半是聊天信息String user = userAndMsg.split(MyProtocol.SPLIT_SIGN)[0];String msg = userAndMsg.split(MyProtocol.SPLIT_SIGN)[1];// 获取私聊用户对应的输出流,并发送私聊信息Server.clients.map.get(user).println(Server.clients.getKeyByValue(ps) + "悄悄地对你说:" + msg);}// 公聊要向每个Socket发送else{// 得到真实消息String msg = getRealMsg(line);// 遍历clients中的每个输出流for (PrintStream clientPs : Server.clients.valueSet()){clientPs.println(Server.clients.getKeyByValue(ps)+ "说:" + msg);}}}}// 捕捉到异常后,表明该Socket对应的客户端已经出现了问题// 所以程序将其对应的输出流从Map中删除catch (IOException e){Server.clients.removeByValue(ps);System.out.println(Server.clients.map.size());// 关闭网络、IO资源try{if (br != null){br.close();}if (ps != null){ps.close();}if (socket != null){socket.close();}}catch (IOException ex){ex.printStackTrace();}}}// 将读到的内容去掉前后的协议字符,恢复成真实数据private String getRealMsg(String line){return line.substring(MyProtocol.PROTOCOL_LEN, line.length() - MyProtocol.PROTOCOL_LEN);}
}

下面是在client包下的类:

public interface MyProtocol
{// 定义协议字符串的长度int PROTOCOL_LEN = 2;// 下面是一些协议字符串,服务器和客户端交换的信息都应该在前、后添加这种特殊字符串。String MSG_ROUND = "§γ";String USER_ROUND = "∏∑";String LOGIN_SUCCESS = "1";String NAME_REP = "-1";String PRIVATE_ROUND = "★【";String SPLIT_SIGN = "※";
}
public class Client
{private static final int SERVER_PORT = 30000;private Socket socket;private PrintStream ps;private BufferedReader brServer;private BufferedReader keyIn;public void init(){try{// 初始化代表键盘的输入流keyIn = new BufferedReader(new InputStreamReader(System.in));// 连接到服务器socket = new Socket("127.0.0.1", SERVER_PORT);// 获取该Socket对应的输入流和输出流ps = new PrintStream(socket.getOutputStream());brServer = new BufferedReader(new InputStreamReader(socket.getInputStream()));String tip = "";// 采用循环不断地弹出对话框要求输入用户名while(true){String userName = JOptionPane.showInputDialog(tip+ "输入用户名"); // 将用户输入的用户名的前后增加协议字符串后发送ps.println(MyProtocol.USER_ROUND + userName+ MyProtocol.USER_ROUND);// 读取服务器的响应String result = brServer.readLine();// 如果用户重复,开始下次循环if (result.equals(MyProtocol.NAME_REP)){tip = "用户名重复!请重新";continue;}// 如果服务器返回登录成功,结束循环if (result.equals(MyProtocol.LOGIN_SUCCESS)){break;}}}// 捕捉到异常,关闭网络资源,并退出该程序catch (UnknownHostException ex){System.out.println("找不到远程服务器,请确定服务器已经启动!");closeRs();System.exit(1);}catch (IOException ex){System.out.println("网络异常!请重新登录!");closeRs();System.exit(1);}// 以该Socket对应的输入流启动ClientThread线程new ClientThread(brServer).start();}// 定义一个读取键盘输出,并向网络发送的方法private void readAndSend(){try{// 不断读取键盘输入String line = null;while((line = keyIn.readLine()) != null){// 如果发送的信息中有冒号,且以//开头,则认为想发送私聊信息if (line.indexOf(":") > 0 && line.startsWith("//")){line = line.substring(2);ps.println(MyProtocol.PRIVATE_ROUND +line.split(":")[0] + MyProtocol.SPLIT_SIGN+ line.split(":")[1] + MyProtocol.PRIVATE_ROUND);}else{ps.println(MyProtocol.MSG_ROUND + line+ MyProtocol.MSG_ROUND);}}}// 捕捉到异常,关闭网络资源,并退出该程序catch (IOException ex){System.out.println("网络通信异常!请重新登录!");closeRs();System.exit(1);}}// 关闭Socket、输入流、输出流的方法private void closeRs(){try{if (keyIn != null){ps.close();}if (brServer != null){ps.close();}if (ps != null){ps.close();}if (socket != null){keyIn.close();}}catch (IOException ex){ex.printStackTrace();}}public static void main(String[] args){Client client = new Client();client.init();client.readAndSend();}
}
public class ClientThread extends Thread
{// 该客户端线程负责处理的输入流BufferedReader br = null;// 使用一个网络输入流来创建客户端线程public ClientThread(BufferedReader br){this.br = br;}public void run(){try{String line = null;// 不断从输入流中读取数据,并将这些数据打印输出while((line = br.readLine())!= null){System.out.println(line);/*本例仅打印了从服务器端读到的内容。实际上,此处的情况可以更复杂:如果希望客户端能看到聊天室的用户列表,则可以让服务器在每次有用户登录、用户退出时,将所有用户列表信息都向客户端发送一遍。为了区分服务器发送的是聊天信息,还是用户列表,服务器也应该在要发送的信息前、后都添加一定的协议字符串,客户端此处则根据协议字符串的不同而进行不同的处理!更复杂的情况:如果两端进行游戏,则还有可能发送游戏信息,例如两端进行五子棋游戏,则还需要发送下棋坐标信息等,服务器同样在这些下棋坐标信息前、后添加协议字符串后再发送,客户端就可以根据该信息知道对手的下棋坐标。*/}}catch (IOException ex){ex.printStackTrace();}// 使用finally块来关闭该线程对应的输入流finally{try{if (br != null){br.close();}}catch (IOException ex){ex.printStackTrace();}}}
}

先运行Server类,再多次运行Client类启动多个客户端,并输入用户名。

半关闭的Socket

在IO中表示输出已经结束,则通过关闭输出流来实现,但是在socket中是行不通的,因为关闭socket,会导致无法再从该socket中读取数据了。为了解决这种问题,java提供了半关闭以及判断半关闭的方法:

  • shutdownInput():关闭该Socket的输入流,程序还可以通过该Socket的输出流输出数据;
  • shutdownOutput():关闭该Socket的输出流,程序还可以通过该Socket的输入流读取数据;
  • boolean isInputShutdown():返回该Socket是否为半读状态 (read-half);
  • boolean isOutputShutdown():返回该Socket是否为半写状态 (write-half)。

如果我们对同一个Socket实例先后调用shutdownInput和shutdownOutput方法,该Socket实例依然没有被关闭,只是该Socket既不能输出数据,也不能读取数据。
示例:

public class Server
{public static void main(String[] args)throws Exception{ServerSocket ss = new ServerSocket(30000);Socket socket = ss.accept();PrintStream ps = new PrintStream(socket.getOutputStream());ps.println("服务器的第一行数据");ps.println("服务器的第二行数据");// 关闭socket的输出流,表明输出数据已经结束socket.shutdownOutput();// 下面语句将输出false,表明socket还未关闭。System.out.println(socket.isClosed());Scanner scan = new Scanner(socket.getInputStream());while (scan.hasNextLine()){System.out.println(scan.nextLine());}scan.close();socket.close();ss.close();}
}
public class Client
{public static void main(String[] args)throws Exception{Socket s = new Socket("localhost" , 30000);Scanner scan = new Scanner(s.getInputStream());while (scan.hasNextLine()){System.out.println(scan.nextLine());}PrintStream ps = new PrintStream(s.getOutputStream());ps.println("客户端的第一行数据");ps.println("客户端的第二行数据");ps.close();scan.close();s.close();}
}

当调用了shutdownOutput()或shutdownInput()方法关闭了输出流或输入流之后,该Socket就无法再次打开输出流或输入流,因此这种做法不适合保持持久通信状态的交互式应用,只适用于一站式的通信协议,例如HTTP协议:客户端连接到服务器之后,开始发送完数据,发送完之后无须再次发送数据,只需要读取服务器响应的数据即可,当读取完成后,该Socket也该关闭了。

使用NIO实现非阻塞Socket通信

从JDK1.4开始,Java提供了NIO API实现非阻塞式Socket通信。之前介绍的都是阻塞式通信,当服务器需要同时处理大量客户端时,需要创建很多线程,这样会导致性能下降,而使用NIO API则可以让服务器使用有限的几个线程处理所有连接到服务器的所有客户端。

NIO提供了以下几个类来支持非阻塞式Socket通信:
Selector
该抽象类代表SelectableChannel 对象的多路复用器。所有希望采用非阻塞方式通信的Channel都应该注册到Selector对象上。可以通过该类的open()静态方法使用系统默认的Selector来创建该类的实例,也可通过调用自定义的Selector的 openSelector() 方法来创建选择器。通过Selector对象的 close() 方法关闭选择器之前,它一直保持打开状态。

通过 SelectionKey 对象来表示SelectableChannel到Selector的注册关系。选择器维护了三种选择键集:

  • 所有注册在该Selector上的SelectionKey组成的集合:此集合由该Selector对象的keys()方法返回;
  • 所有需要进行IO处理而被选择的SelectionKey集合:此集合由该Selector对象selectedKeys()方法返回;
  • 所有被取消注册关系的Channel对应的SelectionKey组成的集合:在下一次调用select()方法时,这些SelectionKey会被彻底删除,程序通常无须访问该集合。

在新创建的Selector中,这三个集合都是空集合。

方法摘要:

  • abstract void close():关闭此选择器;
  • abstract boolean isOpen():告知此选择器是否已打开;
  • abstract Set keys():所有注册在该Selector上的SelectionKey组成的集合;
  • static Selector open():使用系统默认的Selector来创建该类的实例;
  • abstract SelectorProvider provider():返回创建此通道的提供者;
  • abstract int select():监控所有注册的Channel,返回所有需要进行IO操作的Channel对应的SelectionKey的数目;
  • abstract int select(long timeout):同上,增加了超时时长;
  • abstract int selectNow():执行一个立即返回的select()操作,相对于无参数的select()方法而言,该方法不会阻塞线程;
  • abstract Set selectedKeys():监控所有注册的Channel,返回所有需要进行IO操作的Channel对应的SelectionKey组成的集合;
  • abstract Selector wakeup():使一个尚未返回的select()方法立即返回。

SelectableChannel
该抽象类代表一个支持非阻塞IO操作的Channel,它可以被注册到Selector上,这种注册关系用SelectionKey示例表示。SelectionKey对象支持阻塞和非阻塞两种模式,但所有的Channel默认都是阻塞模式,在结合使用基于Selector的多路复用时,非阻塞模式是最有用的,向选择器注册某个通道前,必须将该通道置于非阻塞模式(通过configureBlocking方法设置),并且在注销之前可能无法返回到阻塞模式。

不能直接注销已注册Channel,必须先取消该Channel对应的SelectionKey。可通过调用SelectionKey对象的cancel()方法显式地取消该SelectionKey。无论是通过调用Channel的close方法,还是中断阻塞于该通道上I/O操作中的线程来关闭该Channel,都会隐式地取消该通道的所有SelectionKey对象。如果已注册的某个Selector本身已关闭,则立即将注销该Channel在该Selector上的注册。 一个Channel至多只能在任意特定Selector上注册一次。

方法摘要:

  • abstract boolean isBlocking():判断该Channel上的每个I/O操作在完成前是否被阻塞。
  • abstract SelectableChannel configureBlocking(boolean block):设置该Channel的是否为阻塞模式;
  • abstract Object blockingLock():获取其 configureBlocking 和 register 方法实现同步的对象。
  • abstract boolean isRegistered():判断该Channel当前是否已向一个或多个选择器注册。
  • SelectionKey register(Selector sel, int ops):向给定的Selector注册该Channel,返回对应的SelectionKey。
  • abstract SelectionKey register(Selector sel, int ops, Object att):同上;
  • abstract SelectionKey keyFor(Selector sel):获取表示该Channel向给定Selector注册的SelectionKey。不存在就返回null;
  • abstract SelectorProvider provider():返回创建该Channel的提供者。
  • abstract int validOps():返回一个整数值,表示该Channel所支持的操作。SelectionKey中用静态常量定义了4中IO操作:OP_READ(1)、OP_WRITE(4)、OP_CONNECT(8)、OP_ACCEPT(16)。 这4个值中的任意2个、3个、4个进行按位或的计算结果都相等,而且任意2个、3个、4个相加的结果都不相等,所以可以根据validOps()方法的返回值来确定该SelectionKey支持的操作。

ServerSocketChannel:该抽象类支持非阻塞操作,对应于java.net.ServerSocket类,只支持OP_ACCEPT操作。该抽象类也有一个accept()方法,类似于ServerSocket的accept()方法。

SocketChannel:类似的,该抽象类也支持非阻塞操作,对应于java.net.Socket类,支持OP_CONNECT、OP_READ和OP_WRITE操作。这个类还实现了ByteChannel接口、ScatteringByteChannel接口和GatheringByteChannel接口,所以它可以读写ByteBuffer对象。

服务器上的所有Channel(包括ServerSocketChannel和SocketChannel)都需要向Selector注册,该Selector负责监视这些Socket的IO状态,当存在需要进行IO操作的Channel时,该Selector的select()方法将会返回所有这些Channel对应的SelectionKey的数目,可以通过selectedKeys()方法返回这些SelectionKey的集合。程序通过不断调用Selector的select()方法即可知道当前有哪些Channel需要进行IO操作。如果该Selector上没有需要进行IO操作的Channel,那么select()方法竟会被阻塞。

使用NIO实现多人聊天室示例:

public class NServer
{// 用于检测所有Channel状态的Selectorprivate Selector selector = null;static final int PORT = 30000;// 定义实现编码、解码的字符集对象private Charset charset = Charset.forName("UTF-8");public void init()throws IOException{selector = Selector.open();// ServerSocketChannel不能直接监听某个端口// 也不能使用已有ServerSocket的getChannel()方法来获取它// 必须先通过open静态方法来创建一个ServerSocketChannel实例,在使用bind()方法绑定一个端口ServerSocketChannel server = ServerSocketChannel.open();InetSocketAddress isa = new InetSocketAddress("127.0.0.1", PORT);// 将该ServerSocketChannel绑定到指定IP地址server.bind(isa);// 设置ServerSocket以非阻塞方式工作server.configureBlocking(false);// 将server注册到指定Selector对象server.register(selector, SelectionKey.OP_ACCEPT);while (selector.select() > 0){// 依次处理selector上的每个已选择的SelectionKeyfor (SelectionKey sk : selector.selectedKeys()){// 从selector上的已选择Key集中删除正在处理的SelectionKeyselector.selectedKeys().remove(sk);      // ①// 如果sk对应的Channel包含客户端的连接请求if (sk.isAcceptable())        // ②{// 调用accept方法接受连接,产生服务器端的SocketChannelSocketChannel sc = server.accept();// 设置采用非阻塞模式sc.configureBlocking(false);// 将该SocketChannel也注册到selectorsc.register(selector, SelectionKey.OP_READ);// 将sk对应的Channel设置成准备接受其他请求sk.interestOps(SelectionKey.OP_ACCEPT);}// 如果sk对应的Channel有数据需要读取if (sk.isReadable())     // ③{// 获取该SelectionKey对应的Channel,该Channel中有可读的数据SocketChannel sc = (SocketChannel)sk.channel();// 定义准备执行读取数据的ByteBufferByteBuffer buff = ByteBuffer.allocate(1024);String content = "";// 开始读取数据try{while(sc.read(buff) > 0){buff.flip();content += charset.decode(buff);}// 打印从该sk对应的Channel里读取到的数据System.out.println("读取的数据:" + content);// 将sk对应的Channel设置成准备下一次读取sk.interestOps(SelectionKey.OP_READ);}// 如果捕捉到该sk对应的Channel出现了异常,即表明该Channel// 对应的Client出现了问题,所以从Selector中取消sk的注册catch (IOException ex){// 从Selector中删除指定的SelectionKeysk.cancel();if (sk.channel() != null){sk.channel().close();}}// 如果content的长度大于0,即聊天信息不为空if (content.length() > 0){// 遍历该selector里注册的所有SelectionKeyfor (SelectionKey key : selector.keys()){// 获取该key对应的ChannelChannel targetChannel = key.channel();// 如果该channel是SocketChannel对象if (targetChannel instanceof SocketChannel){// 将读到的内容写入该Channel中SocketChannel dest = (SocketChannel)targetChannel;dest.write(charset.encode(content));}}}}}}}public static void main(String[] args)throws IOException{new NServer().init();}
}
public class NClient
{// 定义检测SocketChannel的Selector对象private Selector selector = null;static final int PORT = 30000;// 定义处理编码和解码的字符集private Charset charset = Charset.forName("UTF-8");// 客户端SocketChannelprivate SocketChannel sc = null;public void init()throws IOException{selector = Selector.open();InetSocketAddress isa = new InetSocketAddress("127.0.0.1", PORT);// 调用open静态方法创建连接到指定主机的SocketChannelsc = SocketChannel.open(isa);// 设置该sc以非阻塞方式工作sc.configureBlocking(false);// 将SocketChannel对象注册到指定Selectorsc.register(selector, SelectionKey.OP_READ);// 启动读取服务器端数据的线程new ClientThread().start();// 创建键盘输入流Scanner scan = new Scanner(System.in);while (scan.hasNextLine()){// 读取键盘输入String line = scan.nextLine();// 将键盘输入的内容输出到SocketChannel中sc.write(charset.encode(line));}}// 定义读取服务器数据的线程private class ClientThread extends Thread{public void run(){try{while (selector.select() > 0){// 遍历每个有可用IO操作Channel对应的SelectionKeyfor (SelectionKey sk : selector.selectedKeys()){// 删除正在处理的SelectionKeyselector.selectedKeys().remove(sk);// 如果该SelectionKey对应的Channel中有可读的数据if (sk.isReadable()){// 使用NIO读取Channel中的数据SocketChannel sc = (SocketChannel)sk.channel();ByteBuffer buff = ByteBuffer.allocate(1024);String content = "";while(sc.read(buff) > 0){sc.read(buff);buff.flip();content += charset.decode(buff);}// 打印输出读取的内容System.out.println("聊天信息:" + content);// 为下一次读取作准备sk.interestOps(SelectionKey.OP_READ);}}}}catch (IOException ex){ex.printStackTrace();}}}public static void main(String[] args)throws IOException{new NClient().init();}
}

使用JDK7的AIO实现异步非阻塞通信

Java7提供的NIO.2提供了更高效的异步Channel支持,基于这种异步Channel的IO机制称为异步IO(Asynchronous IO)。

对于IO操作可以分成两步:

  1. 发出IO请求。如果发出的IO请求阻塞线程就是阻塞IO,如果发出的IO请求不阻塞线程就是非阻塞IO;
  2. 完成IO操作。如果实际IO操作由操作系统完成,再将结果返回给应用程序,这就是异步IO;如果实际IO操作需要应用程序本身执行,会阻塞线程,那就是同步IO。

Java对BIO、NIO、AIO的支持:

  • Java BIO : 同步并阻塞,服务器实现模式为一个连接一个线程,即客户端有连接请求时服务器端就需要启动一个线程进行处理,如果这个连接不做任何事情会造成不必要的线程开销,当然可以通过线程池机制改善;
  • Java NIO : 同步非阻塞,服务器实现模式为一个请求一个线程,即客户端发送的连接请求都会注册到多路复用器上,多路复用器轮询到连接有I/O请求时才启动一个线程进行处理;
  • Java AIO(NIO.2) : 异步非阻塞,服务器实现模式为一个有效请求一个线程,客户端的I/O请求都是由OS先完成了再通知服务器应用去启动线程进行处理。

BIO、NIO、AIO适用场景分析:

  • BIO方式适用于连接数目比较小且固定的架构,这种方式对服务器资源要求比较高,并发局限于应用中,JDK1.4以前的唯一选择,但程序直观简单易理解。
  • NIO方式适用于连接数目多且连接比较短(轻操作)的架构,比如聊天服务器,并发局限于应用中,编程比较复杂,JDK1.4开始支持。
  • AIO方式使用于连接数目多且连接比较长(重操作)的架构,比如相册服务器,充分调用OS参与并发操作,编程比较复杂,JDK7开始支持。

NIO.2提供了一系列以Asynchronous开头的Channel接口和类:

AsynchronousServerSocketChannel

该抽象类代表一个异步通道,用于面向流的监听套接字,可以创建TCP服务端,绑定地址,监听端口等。

方法摘要:

  • static AsynchronousServerSocketChannel open():创建一个默认的未监听的AsynchronousServerSocketChannel对象;
  • static AsynchronousServerSocketChannel open(AsynchronousChannelGroup group):使用指定的AsynchronousChannelGroup来创建一个AsynchronousServerSocketChannel;
  • AsynchronousServerSocketChannel bind(SocketAddress local):将该Channel的套接字绑定到一个本地地址来监听;
  • abstract AsynchronousServerSocketChannel bind(SocketAddress local, int backlog):同上,指定了backlog参数;
  • abstract Future<AsynchronousSocketChannel> accept():创建一个接受来自客户端的连接。由于异步操作由操作系统完成,程序无法获知accept方法何时返回结果。如果要获取连接成功后返回AsynchronousSocketChannel对象,则应该调用该方法返回值的get()方法——但是get()方法会阻塞该线程;
  • abstract <A> void accept(A attachment, CompletionHandler<AsynchronousSocketChannel,? super A> handler):通过指定的CompletionHandler参数来创建一个连接。连接成功或失败都会触发CompletionHandler对象中的相应方法。其中连接成功后应该返回AsynchronousSocketChannel对象;
  • AsynchronousChannelProvider provider():返回该Channel的提供者;
  • abstract <T> AsynchronousServerSocketChannel setOption(SocketOption<T> name, T value):设置套接字选项的值。可以为:SO_RCVBUF(该套接字接受缓冲区的大小)和SO_REUSEADDR(重用地址);

通过open方法新创建的AsynchronousServerSocketChannel是打开的,但是并没有绑定,可以通过bind方法绑定一个本地地址来监听连接。一旦绑定,通过accept方法来初始化来自客户端的连接。

Interface CompletionHandler<V,A>
该接口用来消费异步IO操作的结果,V 代表IO操作返回的结果类型,A代表发起IO操作时传入的对象类型。该接口定义了以下两个方法:

  • void completed(V result, A attachment):当IO操作成功完成后调用此方法;
  • void failed(Throwable exc, A attachment):当IO操作失败时调用该方法。

以上方法的实现应该以一种及时的方式完成,以便于调用的线程分发给其他ompletionHandler。

AsynchronousChannelGroup

该抽象类代表一个异步通道组,它可以实现资源共享。它封装了这样一种结构,用来控制由绑定到该AsynchronousChannelGroup的异步通道初始化的IO操作。AsynchronousChannelGroup有一个相关的线程池来提交IO操作,并分发给CompletionHandler来消费异步操作的执行结果。终止AsynchronousChannelGroup会导致其相关线程池的shutdown。通过调用其 withFixedThreadPool 或 withCachedThreadPool 静态方法来创建一个AsynchronousChannelGroup对象。JVM中有一个自动创建的系统范围的默认AsynchronousChannelGroup,创建时未指定AsynchronousChannelGroup的异步Channel会绑定到这个默认AsynchronousChannelGroup。这个默认AsynchronousChannelGroup可以通过以下系统属性来配置:

  • java.nio.channels.DefaultThreadPool.threadFactory:该属性值是一个具体的ThreadFactory类的全名。该类通过系统类加载器加载和实例化。该工厂类的 newThread 方法用来创建这个默认AsynchronousChannelGroup相关的线程池中的每个线程。如果加载和实例化过程该属性值失败会在创建该默认组时抛出一个未指明的错误;
  • java.nio.channels.DefaultThreadPool.initialSize:该属性值为该默认组的initialSize参数(参见withCachedThreadPool方法)。该属性值是初始大小参数的字符串表示,如果该值不能解析为Integer类型会在创建该默认组时抛出一个未指明的错误。

如果ThreadFactory类没有配置,那么该线程池中的线程是守护线程。完成Channel上IO操作的初始化工作的CompletionHandler肯定会如预期地被线程池中的线程调用。

方法摘要:

  • static AsynchronousChannelGroup withCachedThreadPool(ExecutorService executor, int - initialSize):通过指定的线程池来创建一个AsynchronousChannelGroup,该线程组用来创建需要的新线程;
  • static AsynchronousChannelGroup withFixedThreadPool(int nThreads, ThreadFactory threadFactory):通过指定的线程池来创建一个AsynchronousChannelGroup,该线程组用来创建需要的新线程;
  • static AsynchronousChannelGroup withThreadPool(ExecutorService executor):通过指定的线程池来创建一个AsynchronousChannelGroup;-
  • AsynchronousChannelProvider provider():返回该异步通道组的提供者;
  • abstract void shutdown():关闭该异步通道组;
  • abstract boolean isShutdown():判断该异步通道组是否关闭;
  • abstract void shutdownNow():关闭该异步通道组并关闭该组中所有已打开的通道;
  • abstract boolean awaitTermination(long timeout, TimeUnit unit):等待该组的终止;
  • abstract boolean isTerminated():判断该异步通道组是否终止;
AsynchronousSocketChannel

该抽象类代表一个异步通道,用于面向流的监听套接字。该类对象只能通过以下两种方式:

  • 通过该类的open静态方法创建未绑定的AsynchronousSocketChannel;
  • 或者通过AsynchronousServerSocketChannel的accept方法返回。

不能通过任何已存在的Socket创建该类对象。

一个新创建的AsynchronousSocketChannel通过调用其connect方法来连接,一旦连接,则一直保持连接到其被关闭。
方法摘要:

  • static AsynchronousSocketChannel open():创建一个异步套接字通道;
  • static AsynchronousSocketChannel open(AsynchronousChannelGroup group):通过指定的AsynchronousChannelGroup来创建AsynchronousSocketChannel对象;
  • abstract AsynchronousSocketChannel bind(SocketAddress local):绑定一个本地地址;
  • abstract Future<Void> connect(SocketAddress remote):连接这个Channel;
  • abstract <A> void connect(SocketAddress remote, A attachment, CompletionHandler<Void,? super A> handler):同上,增加了CompletionHandler参数;
  • abstract SocketAddress getRemoteAddress():返回该通道的套接字所连接的远程地址。可用于判断该Channel是否任然处于连接状态;
  • AsynchronousChannelProvider provider():返回此Channel的提供者;
  • abstract Future<Integer> read(ByteBuffer dst):将该Channel中的字节序列读取到给定的缓冲区中;
  • abstract <A> void read(ByteBuffer[] dsts, int offset, int length, long timeout, TimeUnit unit, A attachment, CompletionHandler<Long,? super A> handler):将该Channel中的字节序列读取到给定的缓冲区中;
  • <A> void read(ByteBuffer dst, A attachment, CompletionHandler<Integer,? super A> handler):将该Channel中的字节序列读取到给定的缓冲区中;
  • abstract <A> void read(ByteBuffer dst, long timeout, TimeUnit unit, A attachment, - CompletionHandler<Integer,? super A> handler):将该Channel中的字节序列读取到给定的缓冲区中;
  • abstract Future<Integer> write(ByteBuffer src):将一段字节序列从给定缓冲区写入该通道中;
  • abstract <A> void write(ByteBuffer[] srcs, int offset, int length, long timeout, TimeUnit unit, A attachment, CompletionHandler<Long,? super A> handler):将一段字节序列从给定缓冲区写入该通道中;
  • <A> void write(ByteBuffer src, A attachment, CompletionHandler<Integer,? super A> handler):将一段字节序列从给定缓冲区写入该通道中;
  • abstract <A> void write(ByteBuffer src, long timeout, TimeUnit unit, A attachment, CompletionHandler<Integer,? super A> handler):将一段字节序列从给定缓冲区写入该通道中;
  • abstract AsynchronousSocketChannel shutdownInput():在不管该Channel情况下关闭输入流;
  • abstract AsynchronousSocketChannel shutdownOutput():在不管该Channel情况下关闭输出流;
  • abstract AsynchronousSocketChannel setOption(SocketOption<T> name, T value):设置套接字选项的值:

    • SO_SNDBUF:套接字发送缓冲区的大小;
    • SO_RCVBUF:套接字接受缓冲区的大小;
    • SO_KEEPALIVE:保持活动连接;
    • SO_REUSEADDR:重用地址;
    • TCP_NODELAY:禁用纳格算法。

AsynchronousSocketChannel用法一般分以下几步:

  1. 调用open静态方法创建AsynchronousSocketChannel对象;
  2. 调用AsynchronousSocketChannel对象的connect方法连接到指定的IP地址、指定端口的服务器;
  3. 调用AsynchronousSocketChannel对象的read、write方法进行读写。

使用线程池管理异步Channel,并使用CompletionHandler监听异步IO操作的多人聊天器示例:

public class AIOServer
{static final int PORT = 30000;final static String UTF_8 = "utf-8";static List<AsynchronousSocketChannel> channelList= new ArrayList<>();public void startListen() throws InterruptedException,Exception{// 创建一个线程池ExecutorService executor = Executors.newFixedThreadPool(20);// 以指定线程池来创建一个AsynchronousChannelGroupAsynchronousChannelGroup channelGroup = AsynchronousChannelGroup.withThreadPool(executor);// 以指定线程池来创建一个AsynchronousServerSocketChannelAsynchronousServerSocketChannel serverChannel= AsynchronousServerSocketChannel.open(channelGroup)// 指定监听本机的PORT端口.bind(new InetSocketAddress(PORT));// 使用CompletionHandler接受来自客户端的连接请求serverChannel.accept(null, new AcceptHandler(serverChannel));  // ①Thread.sleep(5000);}public static void main(String[] args)throws Exception{AIOServer server = new AIOServer();server.startListen();}
}
// 实现自己的CompletionHandler类
class AcceptHandler implementsCompletionHandler<AsynchronousSocketChannel, Object>
{private AsynchronousServerSocketChannel serverChannel;public AcceptHandler(AsynchronousServerSocketChannel sc){this.serverChannel = sc;}// 定义一个ByteBuffer准备读取数据ByteBuffer buff = ByteBuffer.allocate(1024);// 当实际IO操作完成时候触发该方法@Overridepublic void completed(final AsynchronousSocketChannel sc, Object attachment){// 记录新连接的进来的ChannelAIOServer.channelList.add(sc);// 准备接受客户端的下一次连接serverChannel.accept(null , this);sc.read(buff , null, new CompletionHandler<Integer,Object>() {@Overridepublic void completed(Integer result, Object attachment){buff.flip();// 将buff中内容转换为字符串String content = StandardCharsets.UTF_8.decode(buff).toString();// 遍历每个Channel,将收到的信息写入各Channel中for(AsynchronousSocketChannel c : AIOServer.channelList){try{c.write(ByteBuffer.wrap(content.getBytes(AIOServer.UTF_8))).get();}catch (Exception ex){ex.printStackTrace();}}buff.clear();// 读取下一次数据sc.read(buff , null , this);}@Overridepublic void failed(Throwable ex, Object attachment){System.out.println("读取数据失败: " + ex);// 从该Channel读取数据失败,就将该Channel删除AIOServer.channelList.remove(sc);}});}@Overridepublic void failed(Throwable ex, Object attachment){System.out.println("连接失败: " + ex);}
}
public class AIOClient
{final static String UTF_8 = "utf-8";final static int PORT = 30000;// 与服务器端通信的异步ChannelAsynchronousSocketChannel clientChannel;JFrame mainWin = new JFrame("多人聊天");JTextArea jta = new JTextArea(16 , 48);JTextField jtf = new JTextField(40);JButton sendBn = new JButton("发送");public void init(){mainWin.setLayout(new BorderLayout());jta.setEditable(false);mainWin.add(new JScrollPane(jta), BorderLayout.CENTER);JPanel jp = new JPanel();jp.add(jtf);jp.add(sendBn);// 发送消息的Action,Action是ActionListener的子接口Action sendAction = new AbstractAction(){public void actionPerformed(ActionEvent e){String content = jtf.getText();if (content.trim().length() > 0){try{// 将content内容写入Channel中clientChannel.write(ByteBuffer.wrap(content.trim().getBytes(UTF_8))).get(); }catch (Exception ex){ex.printStackTrace();}}// 清空输入框jtf.setText("");}};sendBn.addActionListener(sendAction);// 将Ctrl+Enter键和"send"关联jtf.getInputMap().put(KeyStroke.getKeyStroke('\n', java.awt.event.InputEvent.CTRL_MASK) , "send");// 将"send"和sendAction关联jtf.getActionMap().put("send", sendAction);mainWin.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);mainWin.add(jp , BorderLayout.SOUTH);mainWin.pack();mainWin.setVisible(true);}public void connect()throws Exception{// 定义一个ByteBuffer准备读取数据final ByteBuffer buff = ByteBuffer.allocate(1024);// 创建一个线程池ExecutorService executor = Executors.newFixedThreadPool(80);// 以指定线程池来创建一个AsynchronousChannelGroupAsynchronousChannelGroup channelGroup =AsynchronousChannelGroup.withThreadPool(executor);// 以channelGroup作为组管理器来创建AsynchronousSocketChannelclientChannel = AsynchronousSocketChannel.open(channelGroup);// 让AsynchronousSocketChannel连接到指定IP、指定端口clientChannel.connect(new InetSocketAddress("127.0.0.1", PORT)).get();jta.append("---与服务器连接成功---\n");buff.clear();clientChannel.read(buff, null, new CompletionHandler<Integer,Object>(){@Overridepublic void completed(Integer result, Object attachment){buff.flip();// 将buff中内容转换为字符串String content = StandardCharsets.UTF_8.decode(buff).toString();// 显示从服务器端读取的数据jta.append("某人说:" + content + "\n");buff.clear();clientChannel.read(buff , null , this);}@Overridepublic void failed(Throwable ex, Object attachment){System.out.println("读取数据失败: " + ex);}});}public static void main(String[] args)throws Exception{AIOClient client = new AIOClient();client.init();client.connect();}
}

UDP协议网络编程

UDP协议

UDP(User Datagram Protocol)协议,即用户数据报协议,是OSI参考模型中一种无连接的传输层协议,主要支持那些需要在计算机之间传输数据的不可靠网络连接。UDP的主要作用是完成网络数据和数据报之间的转换——在信息发送端,UDP协议将网络数据流封装为数据报,然后将数据报发送出去;在信息接收端,UDP协议将数据报转换成实际数据内容。

UDP协议与TCP协议的比较:

  • TCP传输的效率不如UDP高。使用UDP时,每个数据报中都给出了完整的地址信息,因此无需要建立发送方和接收方的连接,而且也没有超时重发等机制,故而传输速度很快。对于TCP协议,由于它是一个面向连接的协议,在socket之间进行数据传输之前必然要建立连接,所以在TCP中多了一个连接建立的时间。
  • UDP的吞吐量和可靠性不如TCP。使用UDP传输数据时是有大小限制的,每个被传输的数据报必须限定在64KB之内。而TCP没有这方面的限制。UDP是一个不可靠的协议,发送方所发送的数据报并不一定以相同的次序到达接收方。而TCP是一个可靠的协议,它确保接收方完全正确地获取发送方所发送的全部数据。
  • 由于TCP在网络通信上有极强的生命力,主要用于远程连接(Telnet)和文件传输(FTP)、邮件发送(SMTP、POP3)、HTTP连接等;相比之下UDP操作简单,而且仅需要较少的监护,主要用于并不需要保证严格传输可靠性的场景,比如网络游戏、视频会议系统,并不要求音频视频数据绝对的正确,只要保证连贯性就可以了。

DatagramSocket

此类表示用来发送和接收UDP协议数据报的Socket(在通讯实例的两端个建立一个Socket,但这两个Socket之间并没有虚拟链路)。
构造方法摘要:
- DatagramSocket():创建一个DatagramSocket对象,并将其绑定到本地主机上的默认IP地址和随机一个可用的端口;
- DatagramSocket(int port):创建一个DatagramSocket对象,并将其绑定到本地主机上的默认IP地址和指定端口;
- DatagramSocket(SocketAddress bindaddr):创建一个DatagramSocket对象,并将其绑定到本地主机上指定的地址;
- DatagramSocket(int port, InetAddress laddr):创建一个DatagramSocket对象,并将其绑定到本地主机上的指定的IP地址和指定端口;
- protected DatagramSocket(DatagramSocketImpl impl):创建带有指定 DatagramSocketImpl 的未绑定数据报套接字。

方法摘要:

  • void bind(SocketAddress addr):将此 DatagramSocket 绑定到特定的地址和端口。
  • boolean isBound():返回套接字的绑定状态。
  • void connect(InetAddress address, int port):将套接字连接到此套接字的远程地址。
  • void connect(SocketAddress addr):将此套接字连接到远程套接字地址(IP 地址 + 端口号)。
  • boolean isConnected():返回套接字的连接状态;
  • void receive(DatagramPacket p):从此套接字接收数据报包。
  • void send(DatagramPacket p):从此套接字发送数据报包。
  • void disconnect():断开套接字的连接。
  • void close():关闭此数据报套接字。
  • boolean isClosed():返回是否关闭了套接字。
  • DatagramChannel getChannel():返回与此数据报套接字关联的唯一 DatagramChannel 对象(如果有)。
  • InetAddress getInetAddress():返回此套接字连接的地址。
  • InetAddress getLocalAddress():获取套接字绑定的本地地址。
  • int getLocalPort():返回此套接字绑定的本地主机上的端口号。
  • SocketAddress getLocalSocketAddress():返回此套接字绑定的端点的地址,如果尚未绑定则返回 null。
  • int getPort():返回此套接字的端口。
  • int getReceiveBufferSize():获取此 DatagramSocket 的 SO_RCVBUF 选项的值,该值是平台在 DatagramSocket 上输入时使用的缓冲区大小。
  • SocketAddress getRemoteSocketAddress():返回此套接字连接的端点的地址,如果未连接则返回 null。
  • boolean getReuseAddress():检测是否启用了 SO_REUSEADDR。
  • int getSendBufferSize():获取此 DatagramSocket 的 SO_SNDBUF 选项的值,该值是平台在 DatagramSocket 上输出时使用的缓冲区大小。
  • int getSoTimeout():获取 SO_TIMEOUT 的设置。
  • int getTrafficClass():为从此 DatagramSocket 上发送的包获取 IP 数据报头中的流量类别或服务类型。
  • void setBroadcast(boolean on):启用/禁用 SO_BROADCAST。
    -static void setDatagramSocketImplFactory(DatagramSocketImplFactory fac):为应用程序设置数据报套接字实现工厂。
  • void setReceiveBufferSize(int size):将此 DatagramSocket 的 SO_RCVBUF 选项设置为指定的值。
  • void setReuseAddress(boolean on):启用/禁用 SO_REUSEADDR 套接字选项。
  • void setSendBufferSize(int size):将此 DatagramSocket 的 SO_SNDBUF 选项设置为指定的值。
  • void setSoTimeout(int timeout):启用/禁用带有指定超时值的 SO_TIMEOUT,以毫秒为单位。
  • void setTrafficClass(int tc):为从此 DatagramSocket 上发送的数据报在 IP 数据报头中设置流量类别 (traffic class) 或服务类型八位组 (type-of-service octet)。

使用DatagramSocket发送数据时,DatagramSocket并不知道将该数据报发送到哪里,而是由DatagramPacket自身决定数据报的目的地。

DatagramPacket

此类表示数据报包。

构造器摘要:

  • DatagramPacket(byte[] buf, int length):构造 DatagramPacket,用来接收长度为 length 的数据包;
  • DatagramPacket(byte[] buf, int length, InetAddress address, int port):构造数据报包,用来将长度为 length 的包发送到指定IP地址的主机上的指定端口号;
  • DatagramPacket(byte[] buf, int offset, int length):构造 DatagramPacket,用来接收长度为 length 的包,在缓冲区中指定了偏移量;
  • DatagramPacket(byte[] buf, int offset, int length, InetAddress address, int port):构造数据报包,用来将长度为 length 偏移量为 offset 的包发送到指定IP地址的主机上的指定端口号;
  • DatagramPacket(byte[] buf, int offset, int length, SocketAddress address):构造数据报包,用来将长度为 length 偏移量为 offset 的包发送到指定IP地址的主机上的指定端口号;
  • DatagramPacket(byte[] buf, int length, SocketAddress address):构造数据报包,用来将长度为 length 的包发送到指定IP地址的主机上的指定端口号。

方法摘要:

  • InetAddress getAddress():返回某台机器的 IP 地址,此数据报将要发往该机器或者是从该机器接收到的;
  • byte[] getData():返回数据缓冲区;
  • int getLength():返回将要发送或接收到的数据的长度;
  • int getOffset():返回将要发送或接收到的数据的偏移量;
  • int getPort():返回某台远程主机的端口号,此数据报将要发往该主机或者是从该主机接收到的;
  • SocketAddress getSocketAddress():获取要将此包发送到的或发出此数据报的远程主机的 SocketAddress(通常为 IP 地址 + 端口号);
  • void setAddress(InetAddress iaddr):设置要将此数据报发往的那台机器的 IP 地址;
  • void setData(byte[] buf):为此包设置数据缓冲区;
  • void setData(byte[] buf, int offset, int length):为此包设置数据缓冲区;
  • void setLength(int length):为此包设置长度;
  • void setPort(int iport):设置要将此数据报发往的远程主机上的端口号;
  • void setSocketAddress(SocketAddress address):设置要将此数据报发往的远程主机的 SocketAddress(通常为 IP 地址 + 端口号)。

在C/S程序中使用UDP协议时,实际上没有没明确的客户端和服务器端之分,因为两者功能相似,一般把固定IP地址、固定端口号的DatagramSocket对象所在的程序称为服务器,因为该DatagramSocket可以主动接收客户端数据。

在接收数据之前,应该使用DatagramPacket的不包含地址的构造器创建DatagramPacket对象,给出接收数据的字节数组及其长度。然后调用DatagramSocket的receive方法等待数据报的到来,receive将一直等待(阻塞)到接收到一个数据报为止。

在发送数据之前,应该使用DatagramPacket的包含完整地址的构造器创建DatagramPacket对象,此时字节数组里存放了想发送的数据。然后通过调用DatagramSocket的send方法来发送该DatagramPacket对象到指定地址。

当另一端接收到一个DatagramPacket对象后,程序可以调用DatagramPacket对象的一系列getXxx方法来获取发送者的信息。

使用UDP协议实现的C/S通信示例:

public class UdpServer
{public static final int PORT = 30000;// 定义每个数据报的最大大小为4Kprivate static final int DATA_LEN = 4096;// 定义接收网络数据的字节数组byte[] inBuff = new byte[DATA_LEN];// 以指定字节数组创建准备接受数据的DatagramPacket对象private DatagramPacket inPacket =new DatagramPacket(inBuff , inBuff.length);// 定义一个用于发送的DatagramPacket对象private DatagramPacket outPacket;// 定义一个字符串数组,服务器发送该数组的的元素String[] books = new String[]{"疯狂Java讲义","轻量级Java EE企业应用实战","疯狂Android讲义","疯狂Ajax讲义"};public void init()throws IOException{try(// 创建DatagramSocket对象DatagramSocket socket = new DatagramSocket(PORT)){// 采用循环接受数据for (int i = 0; i < 1000 ; i++ ){// 读取Socket中的数据,读到的数据放入inPacket封装的数组里。socket.receive(inPacket);// 判断inPacket.getData()和inBuff是否是同一个数组System.out.println(inBuff == inPacket.getData());// 将接收到的内容转成字符串后输出System.out.println(new String(inBuff, 0 , inPacket.getLength()));// 从字符串数组中取出一个元素作为发送的数据byte[] sendData = books[i % 4].getBytes();// 以指定字节数组作为发送数据、以刚接受到的DatagramPacket的// 源SocketAddress作为目标SocketAddress创建DatagramPacket。outPacket = new DatagramPacket(sendData, sendData.length , inPacket.getSocketAddress());// 发送数据socket.send(outPacket);}}}public static void main(String[] args)throws IOException{new UdpServer().init();}
}
public class UdpClient
{// 定义发送数据报的目的地public static final int DEST_PORT = 30000;public static final String DEST_IP = "127.0.0.1";// 定义每个数据报的最大大小为4Kprivate static final int DATA_LEN = 4096;// 定义接收网络数据的字节数组byte[] inBuff = new byte[DATA_LEN];// 以指定字节数组创建准备接受数据的DatagramPacket对象private DatagramPacket inPacket =new DatagramPacket(inBuff , inBuff.length);// 定义一个用于发送的DatagramPacket对象private DatagramPacket outPacket = null;public void init()throws IOException{try(// 创建一个客户端DatagramSocket,使用随机端口DatagramSocket socket = new DatagramSocket()){// 初始化发送用的DatagramSocket,它包含一个长度为0的字节数组outPacket = new DatagramPacket(new byte[0] , 0, InetAddress.getByName(DEST_IP) , DEST_PORT);// 创建键盘输入流Scanner scan = new Scanner(System.in);// 不断读取键盘输入while(scan.hasNextLine()){// 将键盘输入的一行字符串转换字节数组byte[] buff = scan.nextLine().getBytes();// 设置发送用的DatagramPacket里的字节数据outPacket.setData(buff);// 发送数据报socket.send(outPacket);// 读取Socket中的数据,读到的数据放在inPacket所封装的字节数组里。socket.receive(inPacket);System.out.println(new String(inBuff , 0, inPacket.getLength()));}}}public static void main(String[] args)throws IOException{new UdpClient().init();}
}

MulticastSocket

此类代表一个多播套接字,继承自DatagramSocket类,可以将数据报以广播的方式发送到多个客户端。该类通过 D 类 IP 地址和标准 UDP 端口号指定。D 类 IP 地址在 224.0.0.0~239.255.255.255 的范围内,但地址 224.0.0.0 被保留,不应使用。

若要使用多点广播,则需要让一个数据报标有一组目标主机地址,当数据报发出后,整个组的所有主机都能收到该数据报。IP多点广播实现了将单一信息发送到多个接受者的广播,其思想是设置一组特殊网络地址作为多点广播地址,当客户端需要发送、接受广播消息时,加入到这个多点广播地址(一组地址)即可。

具体的实现策略就是定义一个多点广播地址,使得每个MulticastSocket都加入到这个地址中。从而每次使用MulticastSocket发送数据报(包含的广播地址)时,所有加入了这个多点广播地址的MulticastSocket对象都可以收到信息。当然,MulticastSocket也可以接收加入到该组地址的其他MulticastSocket发送的消息。

构造方法摘要:

  • MulticastSocket():使用本机默认地址、随机端口创建创建多播套接字;
  • MulticastSocket(int port):使用本机默认地址、指定端口创建多播套接字;
  • MulticastSocket(SocketAddress bindaddr):使用本机指定IP地址、指定端口的套接字地址创建多播套接字。

如果创建仅用于发送数据报的MulticastSocket对象,则使用默认地址,随机端口即可。但如果创建接受用的MulticastSocket对象,则该MulticastSocket对象必须具有指定端口,否则发送方无法确定发送数据报的目标端口。

方法摘要:

  • void joinGroup(InetAddress mcastaddr):将该MulticastSocket加入指定的多点广播地址;
  • void joinGroup(SocketAddress mcastaddr, NetworkInterface netIf):将该MulticastSocket加入指定接口上的多点广播地址。netIf为指定要接收多播数据报包的本地接口,也可以为 null,表示由 setInterface(InetAddress) 或 setNetworkInterface(NetworkInterface) 设置的接口;
  • void leaveGroup(InetAddress mcastaddr):让该MulticastSocket离开指定的多点广播地址;
  • void leaveGroup(SocketAddress mcastaddr, NetworkInterface netIf):让该MulticastSocket离开指定接口上的多点广播地址;
  • void setInterface(InetAddress inf):某些系统中肯能有多个网络接口,需要在一个指定的网络接口上监听。设置该MulticastSocket使用指定的网络接口;
  • InetAddress getInterface():获取用于多播数据包的网络接口的地址;
  • void setNetworkInterface(NetworkInterface netIf):指定在此套接字上发送的输出多播数据报的网络接口;
  • NetworkInterface getNetworkInterface():获取多播网络接口集合;
  • void setTimeToLive(int ttl):该ttl参数用于设置数据报最多可以跨过多少个网络。当ttl为0时,指定数据报应停留在本机地址;当ttl为1时,指定数据报发送到本地局域网;当ttl为32时,指定数据报只能发送到本站点的网络上;当ttl为64时,意味着数据报应保留在本地区;当ttl为128时,意味着数据报应保留在本大洲;当ttl为255时,意味着数据报可以发送到所有地方。该值默认为1;
  • int getTimeToLive():获取在套接字上发出的多播数据包的默认生存时间;
  • void setLoopbackMode(boolean disable):启用/禁用多播数据报的本地回送;
  • boolean getLoopbackMode():设置该MulticastSocket发送的数据报会被回送到自身。

一个基于广播的多人聊天室示例:

// 让该类实现Runnable接口,该类的实例可作为线程的target
public class MulticastSocketTest implements Runnable
{// 使用常量作为本程序的多点广播IP地址private static final String BROADCAST_IP= "230.0.0.1";// 使用常量作为本程序的多点广播目的的端口public static final int BROADCAST_PORT = 30000;// 定义每个数据报的最大大小为4Kprivate static final int DATA_LEN = 4096;//定义本程序的MulticastSocket实例private MulticastSocket socket = null;private InetAddress broadcastAddress = null;private Scanner scan = null;// 定义接收网络数据的字节数组byte[] inBuff = new byte[DATA_LEN];// 以指定字节数组创建准备接受数据的DatagramPacket对象private DatagramPacket inPacket= new DatagramPacket(inBuff , inBuff.length);// 定义一个用于发送的DatagramPacket对象private DatagramPacket outPacket = null;public void init()throws IOException{try(// 创建键盘输入流Scanner scan = new Scanner(System.in)){// 创建用于发送、接收数据的MulticastSocket对象// 由于该MulticastSocket对象需要接收数据,所以有指定端口socket = new MulticastSocket(BROADCAST_PORT);broadcastAddress = InetAddress.getByName(BROADCAST_IP);// 将该socket加入指定的多点广播地址socket.joinGroup(broadcastAddress);// 设置本MulticastSocket发送的数据报会被回送到自身socket.setLoopbackMode(false);// 初始化发送用的DatagramSocket,它包含一个长度为0的字节数组outPacket = new DatagramPacket(new byte[0], 0 , broadcastAddress , BROADCAST_PORT);// 启动以本实例的run()方法作为线程体的线程new Thread(this).start();// 不断读取键盘输入while(scan.hasNextLine()){// 将键盘输入的一行字符串转换字节数组byte[] buff = scan.nextLine().getBytes();// 设置发送用的DatagramPacket里的字节数据outPacket.setData(buff);// 发送数据报socket.send(outPacket);}}finally{socket.close();}}public void run(){try{while(true){// 读取Socket中的数据,读到的数据放在inPacket所封装的字节数组里。socket.receive(inPacket);// 打印输出从socket中读取的内容System.out.println("聊天信息:" + new String(inBuff, 0 , inPacket.getLength()));}}// 捕捉异常catch (IOException ex){ex.printStackTrace();try{if (socket != null){// 让该Socket离开该多点IP广播地址socket.leaveGroup(broadcastAddress);// 关闭该Socket对象socket.close();}System.exit(1);}catch (IOException e){e.printStackTrace();}}}public static void main(String[] args)throws IOException{new MulticastSocketTest().init();}
}

使用代理服务器

从Java5开始,java.net包下提供了Proxy和ProxySelector两个类来提供代理服务。

Proxy

该类表示代理设置。Proxy 是不可变对象。该类仅有一个构造器:

  • Proxy(Proxy.Type type, SocketAddress sa) :sa参数指定代理服务器的地址,Proxy.Type是该类的内部枚举类,该参数表示代理服务器的三种类型:

    • Proxy.Type.DIRECT:表示直接连接,不使用代理;
    • Proxy.Type.HTTP:表示支持高级协议代理,如 HTTP 或 FTP;
    • Proxy.Type.SOCKS:表示 SOCKS(V4 或 V5)代理。

创建了Proxy对象后,程序就可以在使用URLConnection的openConnection(Proxy proxy)方法打开连接时,或者在使用Socket构造器Socket(Proxy proxy)时传入一个Proxy对象,作为本次连接所用的代理服务器。
在URLConnection中使用代理服务器:

public class ProxyTest
{// 下面是代理服务器的地址和端口,// 换成实际有效的代理服务器的地址和端口final String PROXY_ADDR = "129.82.12.188";final int PROXY_PORT = 3124;// 定义需要访问的网站地址String urlStr = "http://www.crazyit.org";public void init()throws IOException , MalformedURLException{URL url = new URL(urlStr);// 创建一个代理服务器对象Proxy proxy = new Proxy(Proxy.Type.HTTP, new InetSocketAddress(PROXY_ADDR , PROXY_PORT));// 使用指定的代理服务器打开连接URLConnection conn = url.openConnection(proxy);// 设置超时时长。conn.setConnectTimeout(5000);try(// 通过代理服务器读取数据的ScannerScanner scan = new Scanner(conn.getInputStream(), "utf-8");PrintStream ps = new PrintStream("index.htm")){while (scan.hasNextLine()){String line = scan.nextLine();// 在控制台输出网页资源内容System.out.println(line);// 将网页资源内容输出到指定输出流ps.println(line);}}}public static void main(String[] args)throws IOException , MalformedURLException{new ProxyTest().init();}
}

ProxySelector

该抽象类代表一个代理服务器选择器,可以继承该类实现自己的代理服务器选择器。

方法摘要:

  • abstract void connectFailed(URI uri, SocketAddress sa, IOException ioe):连接代理服务器失败时回调该方法;
  • abstract List select(URI uri):根据业务需要返回代理服务器列表,如果该方法返回的集合中只包含一个Proxy,那么该Proxy将会作为默认代理服务器;
  • static void setDefault(ProxySelector ps):设置(或取消设置)系统级代理选择器;
  • static ProxySelector getDefault():获取系统级代理选择器。

实现自定义代理服务器选择器后,调用该类的setDefault(ProxySelector ps)静态方法来注册该代理选择器即可。
示例:

ublic class ProxySelectorTest
{// 下面是代理服务器的地址和端口,// 随便一个代理服务器的地址和端口final String PROXY_ADDR = "139.82.12.188";final int PROXY_PORT = 3124;// 定义需要访问的网站地址String urlStr = "http://www.crazyit.org";public void init()throws IOException , MalformedURLException{// 注册默认的代理选择器ProxySelector.setDefault(new ProxySelector(){@Overridepublic void connectFailed(URI uri, SocketAddress sa, IOException ioe){System.out.println("无法连接到指定代理服务器!");}// 根据"业务需要"返回特定的对应的代理服务器@Overridepublic List<Proxy> select(URI uri){// 本程序总是返回某个固定的代理服务器。List<Proxy> result = new ArrayList<>();result.add(new Proxy(Proxy.Type.HTTP, new InetSocketAddress(PROXY_ADDR , PROXY_PORT)));return result;}});URL url = new URL(urlStr);// 没有指定代理服务器、直接打开连接URLConnection conn = url.openConnection(); // 设置超时时长。conn.setConnectTimeout(3000);try(// 通过代理服务器读取数据的ScannerScanner scan = new Scanner(conn.getInputStream());PrintStream ps = new PrintStream("index.htm")){while (scan.hasNextLine()){String line = scan.nextLine();// 在控制台输出网页资源内容System.out.println(line);// 将网页资源内容输出到指定输出流ps.println(line);}}}public static void main(String[] args)throws IOException , MalformedURLException{new ProxySelectorTest().init();}
}

DefaultProxySelector
此类是ProxySelector的默认实现类,只是未公开,应该尽量避免直接使用该类。系统已经将该类注册为默认的代理选择器。可以通过ProxySelector类的getDefault()方法来获取该类的实例。该类对其父类抽象方法的实现为:
- connectFailed方法:如果连接失败,DefaultProxySelector将尝试不使用代理服务器,而是直接连接;
- select方法:DefaultProxySelector会根据系统属性来决定使用哪个代理服务器。ProxySelector会检测系统属性与URL之间的匹配,然后使用相应的属性值作为代理服务器;

常用的代理服务器属性名:

  • http.proxyHost:设置HTTP访问所使用的代理服务器的主机地址。该属性名的前缀可以改为https、ftp等;
  • http.proxyPort:设置HTTP访问所使用的代理服务器端口。该属性名的前缀可以改为https、ftp等;
  • http.nonProxyHosts:设置HTTP访问中不需要使用代理服务器的主机,支持使用*通配符;支持指定多个地址,多个地址之间使用竖线(|)分隔。

通过改变系统属性来改变默认的代理服务器:

public class DefaultProxySelectorTest
{// 定义需要访问的网站地址static String urlStr = "http://www.crazyit.org";public static void main(String[] args) throws Exception{// 获取系统的默认属性Properties props = System.getProperties();// 通过系统属性设置HTTP访问所用的代理服务器的主机地址、端口props.setProperty("http.proxyHost", "192.168.10.96");props.setProperty("http.proxyPort", "8080");// 通过系统属性设置HTTP访问无需使用代理服务器的主机// 可以使用*通配符,多个地址用|分隔props.setProperty("http.nonProxyHosts", "localhost|192.168.10.*");// 通过系统属性设置HTTPS访问所用的代理服务器的主机地址、端口props.setProperty("https.proxyHost", "192.168.10.96");props.setProperty("https.proxyPort", "443");/* DefaultProxySelector不支持https.nonProxyHosts属性,DefaultProxySelector直接按http.nonProxyHosts的设置规则处理 */// 通过系统属性设置FTP访问所用的代理服务器的主机地址、端口props.setProperty("ftp.proxyHost", "192.168.10.96");props.setProperty("ftp.proxyPort", "2121");// 通过系统属性设置FTP访问无需使用代理服务器的主机props.setProperty("ftp.nonProxyHosts", "localhost|192.168.10.*");// 通过系统属性设置设置SOCKS代理服务器的主机地址、端口props.setProperty("socks.ProxyHost", "192.168.10.96");props.setProperty("socks.ProxyPort", "1080");// 获取系统默认的代理选择器ProxySelector selector = ProxySelector.getDefault(); System.out.println("系统默认的代理选择器:" + selector);// 根据URI动态决定所使用的代理服务器System.out.println("系统为ftp://www.crazyit.org选择的代理服务器为:"+ ProxySelector.getDefault().select(new URI("ftp://www.crazyit.org"))); URL url = new URL(urlStr);// 直接打开连接,默认的代理选择器会使用http.proxyHost、// http.proxyPort系统属性设置的代理服务器,// 如果无法连接代理服务器,默认的代理选择器会尝试直接连接URLConnection conn = url.openConnection();   // ③// 设置超时时长。conn.setConnectTimeout(3000);try(Scanner scan = new Scanner(conn.getInputStream() , "utf-8")){// 读取远程主机的内容while(scan.hasNextLine()){System.out.println(scan.nextLine());}}}
}

JavaSE学习笔记之网络编程相关推荐

  1. 黑马程序员_java自学学习笔记(八)----网络编程

    黑马程序员_java自学学习笔记(八)----网络编程 android培训. java培训.期待与您交流! 网络编程对于很多的初学者来说,都是很向往的一种编程技能,但是很多的初学者却因为很长一段时间无 ...

  2. Netty学习笔记二网络编程

    Netty学习笔记二 二. 网络编程 1. 阻塞模式 阻塞主要表现为: 连接时阻塞 读取数据时阻塞 缺点: 阻塞单线程在没有连接时会阻塞等待连接的到达,连接到了以后,要进行读取数据,如果没有数据,还要 ...

  3. Qt学习笔记之网络编程

    Qt网络模块提供了允许您编写TCP / IP客户端和服务器的类.它提供了代表低级网络概念的低级类(例如QTcpSocket,QTcpServer和QUdpSocket),以及高级类(例如QNetwor ...

  4. JavaSE学习16(网络编程)

    网络编程 1.网络编程入门 1.1 网络编程概述 计算机网络:将地理位置不同的具有独立功能的多台计算机及其外部设备,通过通信线路连接起来,在网络操作系统,网络管理软件以及网络通信协议的管理和协调下,实 ...

  5. TCP/IP网络编程 学习笔记_1 --网络编程入门

    前言:这个系列网络编程教程实例代码是在Xcode上运行的,MacOSX,是一个基于UNIX核心的系统,所以基于Linux的网络编程代码一般可以直接在Xcode上运行,如果要移植到Windows其实就只 ...

  6. Java基础学习笔记之网络编程

    Java基础之网络编程 1.网络编程概述 什么是网络编程 指的是在多个设备(计算机)执行,其中的设备使用一个网络中的所有连接到对方编写程序 网络编程的目的 与其他计算机进行通信 网络编程的问题 1.如 ...

  7. 【Java学习笔记】 网络编程04 优化字符串拼接:JSON

    学习时间 0731 优化拼接字符串 String : 是复合类型 ,相当于char的数组 是final类,也就是不支持继承 public final class String {private fin ...

  8. C# 学习笔记 20.网络编程

    1.基础知识 OSI(open system interconnection)模型把网络通信分成7层:物理层,数据链路层,网络层,传输层,会话层,表示层,和应用层. 网络开发人员一般分成5层   应用 ...

  9. JavaSE学习笔记(持续更新)

    这里写目录标题 JavaSE学习笔记(持续更新) Java跨平台原理与核心机制 1.跨平台原理: 2.两种核心机制: JDK11的安装流程 Java程序开发的三个步骤(无编辑器版) Eclipse安装 ...

最新文章

  1. 打开文件和关闭文件的含义
  2. 浅析2007年商业智能领域上演的收购大戏
  3. option:contains后面加变量_Python基础篇 -2:初识Python中的变量
  4. MFC之进度条CProgressCtrl
  5. C#创建带参数的线程
  6. php 函数有命名空间吗_一篇弄懂PHP命名空间及use的使用
  7. 1202此服务器的证书无效,ios - iOS:URLRequest错误域= NSURLErrorDomain代码= -1202“此服务器的证书无效 - 堆栈内存溢出...
  8. EasyMock学习笔记
  9. php iconv 无效,关于nodejs iconv的编码无效 说明
  10. Android 学习
  11. 根据Dockerfile构建镜像
  12. jprofiler_监控远程linux服务器的JVM进程(转 非常棒)
  13. antd的Tree控件实现点击展开功能
  14. 一道SQL题考你数据库的使用能力
  15. python爬取斗鱼礼物数据_Python---20行代码爬取斗鱼平台房间数据(上)
  16. 蓝桥杯c语言试题 历届真题 天干地支【第十一届】【决赛】【A组】
  17. 基于樽海鞘群算法的线性规划求解matlab程序
  18. CS61A Proj 3
  19. Go : const rotates恒定旋转(附完整源码)
  20. MATLAB Signal Rrocessing(8)滤波器分类

热门文章

  1. 推荐系统遇上深度学习(八十七)-[阿里]基于搜索的用户终身行为序列建模
  2. 优课联盟 实境英语Test for Unit 2
  3. 陕西二本计算机软件工程,高考分数不理想,仍然想报考软件工程专业,这四所二本大学不错...
  4. pr 调整图层缩放移动无效
  5. 理解“万事万物皆对象”
  6. 航测新旗舰|大疆M300+赛尔102S
  7. dva是什么游戏_守望先锋DVA上分攻略 DVA使用技巧详解
  8. 家居家装行业人群洞察白皮书.pdf
  9. 详细!!计划任务设置详细步骤
  10. 重装 UOS Deepin V20 后必安装软件(个人习惯)