第13章 Java网络编程
13.1 URL类
- URL类是java.net包中的一个重要的类,URL的实例封装着一个统一资源定位符(Uniform Resource Locator),使用URL创建对象的应用程序称作客户端程序。
- 一个URL对象通常包含最基本的三部分信息:协议、地址、资源。
13.1.1 URL的构造方法
URL类通常使用如下的构造方法创建一个URL对象:
public URL(String spec) throws MalformedURLException
public URL(String protocol, String host,String file) throws MalformedURLException
13.1.2 读取URL中的资源
URL对象调用
InputStream openStream()
方法可以返回一个输入流,该输入流指向URL对象所包含的资源。通过该输入流可以将服务器上的资源读入到客户端。
13.2 InetAdress类
13.2.1 地址的表示
Internet上的主机有两种方式表示地址:
- 1.域名
例如 www.tsinghua.edu.cn - 2.IP地址
例如 202.108.35.210
java.net包中的InetAddress类对象含有一个Internet主机地址的域名和IP地址,例如www.sina.com.cn/202.108.37.40
。
13.2.2 获取地址
- 1.获取Internet上主机的地址
可以使用InetAddress类的静态方法getByName(String s);
获得一个InetAddress对象,该对象含有主机地址的域名和IP地址,该对象用如下格式表示它包含的信息:www.sina.com.cn/202.108.37.40
- 2.获取本地机的地址
我们可以使用InetAddress类的静态方法getLocalHost()
获得一个InetAddress对象,该对象含有本地机的域名和IP地址。
13.3 套接字
13.3.1 套接字
IP地址标识Internet上的计算机,端口号标识正在计算机上运行的进程(程序)。端口号被规定为一个16位的0~65535之间的整数。
当两个程序需要通信时,它们可以通过使用Socket类建立套接字对象并连接在一起(端口号与IP地址的组合得出一个网络套接字)。
13.3.2 客户端套接字
客户端的程序使用Socket类建立负责连接到服务器的套接字对象。
建立连接到服务器的套接字对象:
try{ Socket mysocket=new Socket(“http://192.168.0.78”,1880);
}
catch(IOException e)
{}
与mysocket相关的方法
getInputStream()
获得一个输入流getOutputStream()
获得一个输出流- 用
getInputStream()
得到的输入流接到另一个DataInputStream数据流上 - 用
getOutputStream()
得到的输出流接到另一个DataOutputStream数据流上
13.3.3 ServerSocket对象与服务器端套接字
服务器必须建立一个ServerSocket对象,该对象通过将客户端的套接字对象和服务器端的一个套接字对象连接起来,从而达到连接的目的。
建立ServerSocket对象:
try22{ ServerSocket serverForClient =new ServerSocket(2010);
}
catch(IOException e){}
使用方法accept()将客户的套接字和服务器端的套接字连接起来,代码如下所示:
try{ Socket sc= serverForClient .accept();
}
catch(IOException e){}
所谓“接收”客户的套接字连接就是accept()
方法会返回一个和客户端Socket对象相连接的Socket对象。
客户端的套接字获得的输入输出流和服务器端的套接字获得的输出输入流互相连接.
13.3.4 使用多线程技术
为了防止堵塞线程,服务器端收到一个客户的套接字后,就应该启动一个专门为该客户服务的线程。
在Example13_4中,客户使用Socket类不带参数的构造方法Socket()
创建一个套接字对象,该对象需调用
public void connect(SocketAddress endpoint) throws IOException
请求和参数SocketAddress指定地址的套接字建立连接。
为了使用connect方法,可以使用SocketAddress的子类InetSocketAddress创建一个对象,
InetSocketAddress的构造方法是:InetSocketAddress(InetAddress addr, int port)
13.4 UDP数据报
基于UDP的通信和基于TCP的通信不同,基于UDP的信息传递更快,但不提供可靠性保证。
基于UDP通信的基本模式是:
- 1.将数据打包,称为数据包(好比将信件装入信封一样),然后将数据包发往目的地。
- 2.接受别人发来的数据包(好比接收信封一样),然后查看数据包中的内容。
13.4.1 发送数据包
- 1.用DatagramPacket类将数据打包,即用DatagramPacket类创建一个对象,称为数据包。用DatagramPacket的以下两个构造方法创建待发送的数据包:
DatagramPacket(byte data[],int length,InetAddtress address,int port)
DatagramPack(byte data[],int offset,int length,InetAddtress address,int port)
- 2.用DatagramSocket类的不带参数的构造方法:DatagramSocket()创建一个对象,该对象负责发送数据包。例如:
DatagramSocket mail_out=new DatagramSocket();
mail_out.send(data_pack);
13.4.2 接收数据包
-
- 首先用DatagramSocket的另一个构造方法
DatagramSocket(int port)
创建一个对象,其中的参数必须和待接收的数据包的端口号相同。
例如,如果发送方发送的数据包的端口是5666,那么如下创建DatagramSocket对象:DatagramSocket mail_in=new DatagramSocket(5666);
- 首先用DatagramSocket的另一个构造方法
-
- 然后对象mail_in使用方法
receive(DatagramPacket pack)
接受数据包。
- 然后对象mail_in使用方法
-
3.用 DatagramPack类的另外一个构造方法:
DatagramPack(byte data[],int length)
创建一个数据包,用于接收数据包,例如:
byte data[]=new byte[100];
int length=90;
DatagramPacket pack=new DatagramPacket(data,length);
mail_in.receive(pack);
该数据包pack
将接收长度是length
字节的数据放入data
。
13.5 广播数据报
广播数据报涉及到地址和端口。
广播数据报是一种较新的技术,要广播或接收广播的主机都必须加入到同一个D类地址。
13.6 Java 远程调用(RMI)
Java远程调用,RMI(Remote Method Invocation)是一种分布式技术,使用RMI可以让一个虚拟机(JVM)上的应用程序请求调用位于网络上另一处的JVM上的对象方法。习惯上称发出调用请求的虚拟机(JVM)为(本地)客户机,称接受并执行请求的虚拟机(JVM)为(远程)服务器。
13.6.1 远程对象及其代理
- 1.远程对象:
驻留在(远程)服务器上的对象是客户要请求的对象,称作远程对象。 - 2.代理与存根(Stub):
代理的特点是它与远程对象实现了相同的接口。
存根(Stub):一种特殊的字节码,并让这个存根产生的对象为作为远程对象的代理。 - 3.Remote接口:
RMI为了标识一个对象是远程对象,即可以被客户请求的对象,要求远程对象必须实现java.rmi包中的Remote接口,也就是说只有实现该接口的类的实例才被RMI认为是一个远程对象。
13.6.2 RMI的设计细节
- 1.扩展Remote接口
定义一个接口是java.rmi
包中Remote的子接口,即扩展Remote
接口。 - 2.远程对象
创建远程对象的类必须要实现Remote
接口,RMI使用Remote
接口来标识远程对象,但是Remote中没有方法,因此创建远程对象的类需要实现Remote接口的一个子接口。 - 3.存根(Stub)与代理:RMI使用rmic命令生成存根
- 4.启动注册
rmiregistry
: 执行rimregistry
命令 - 5.启动远程对象服务:远程服务器使用java.rmi包中的Naming类调用其类方法
rebind(String name, Remote obj)
绑定一个远程对象到rmiregistry所管理的注册表中,该方法的name参数是URL格式,obj参数是远程对象,将来客户端的代理会通过name 找到远程对象obj。 - 6.运行客户端程序:远程服务器启动远程对象服务后,客户端就可以运行有关程序,访问使用远程对象。