java 网络编程
概念
网络编程是指编写运行在多个设备的程序,这些设备通过网络连接起来。
java.net
包中的api类和接口,提供了低层次的通信细节,可以直接调用。其中提供了两种常见的网络协议的支持:
- TCP通信:这是一种面向连接的、可靠的、基于字节流的传输层通信协议,TCP层是位于IP层上、应用层下的中间层。通常用于互联网协议,被称为 TCP/IP 。
- UDP通信:一个无连接的协议。提供了应用程序之间要发送数据的数据报。由于缺乏可靠性,所有应用程序通常必须容许一些丢失、错误、重复的数据包。一般用作广播通信。
Socket编程
使用的很广泛的网络概念。
套接字使用TCP提供了两台计算机之间的通信机制。客户端创建一个套接字,并尝试连接服务器的套接字。当连接建立时,服务器会创建一个Socket对象。客户端和服务器现在可以通过Socket对象的写入和读取来进行通信。
java.net.Socket
类代表一个套接字,并且java.net.ServerSocket
类为服务器程序提供了一种来监听客户端,并与他们建立连接的机制。
步骤
- 服务器实例化一个 ServerSocket对象,表示通过服务器上的端口通信
- 服务器调用 ServerSocket类的 accept() 方法,该方法将一致等待,知道客户端连接到服务器上给定的端口。
- 服务器等待时,一个客户端实例化一个Socket 对象,指定服务器名称和端口来请求连接。
- Socket类的构造函数试图将客户端连接到指定的服务器和端口号,如果通信被确立,则在客户端创建一个Socket对象能够与服务器进行通信。
- 在服务器端,accept()方法返回服务器上一个新的 Socket 引用,该socket连接到客户端的socket。
连接建立后,通过 I/O 流进行通信,每个Socket都有一个输出流和一个输入流,客户端的输出流连接到服务器端的输入流,而客户端的输入流连接到服务器端的输出流。
TCP 是一个双向的通信协议,因此数据可以通过两个数据流在同一时间发送。
ServerSocket 类的方法列表
服务器应用程序通过使用 java.net.ServerSocket 类以获取一个端口,并侦听客户端的请求。
构造方法 | 方法描述 |
---|---|
ServerSocket(int port) throws IOException | 创建绑定到特定端口的服务器套接字 |
ServerSocket(int port , int backlog) throws IOException | 利用指定的 backlog 创建服务器套接字并将其绑定到指定的本地端口号。 |
ServerSocket(int port,int backlog,InetAddress address) throws IOException | 使用指定的端口,侦听 backlog 和要绑定到的本地IP 地址穿件服务器 |
ServerSocket() throws IOException | 创建非绑定服务器套接字 |
如果未抛出异常,则程序成功绑定到指定端口并侦听客户端请求。
常用方法 | 方法描述 |
---|---|
int getLocalPort() | 返回此套接字在其上侦听的端口 |
Socket accept() throws IOException | 侦听并接受到此套接字的连接 |
void setSoTimeout(int timeout) | 通过指定超时值启用/禁用 SO_TIMEOUT,以毫秒为单位 |
public void bind(SocketAddress host,int backlog) | 将ServerSocket 绑定到特定地址(ip地址和端口号) |
Socket 类的方法
java.net.Socket
类代表客户端和服务器都是用来互相沟通的套接字。客户端要获取一个Socket对象通过实例化,而服务器获取一个Socket对象则通过accept() 方法的返回值。
构造方法 | 方法描述 |
---|---|
Socket(String host,int port) | 创建一个套接字并将其连接到指定主机上的指定端口 |
Socket(InetAddress host,int port) | 创建一个流套接字并将其连接到指定IP地址的指定端口号 |
Socket(String host,int port,InetAddress localAddress,int localPort) | 创建一个套接字并将其连接到指定的远程主机上的指定远程端口 |
Socket(InetAddress host,int port,InetAddress localAddress,int localPort) | 创建一个套接字并将其连接到指定远程地址上的指定远程端口 |
Socket() | 通过系统默认类型的 SocketImpl 创建未连接套接字 |
当Socket构造方法返回,并没有简单的实例化一个Socket对象,它实际上会尝试连接到指定的服务器和端口。
方法 | 描述 |
---|---|
void connect(SocketAddress host,int timeout) | 将此套接字连接到服务器,并指定超时值 |
InetAddress getInetAddress() | 返回套接字连接的地址 |
int getPort() | 返回此套接字连接到的远程端口 |
int getLocalPort() | 返回此套接字绑定的本地端口 |
SocketAddress getRemoteSocketAddress() | 返回此套接字连接的端点地址,如果未连接则返回 null |
InputStream getInputStream() | 返回此套接字的输入流 |
OutputStream getOutputStream | 返回此套接字的输出流 |
void close() | 关闭此套接字 |
因为客户端和服务器端都有一个Socket对象,所以无论客户端还是服务器端都能调用这些方法。
InetAddress 类的方法
这个类表示互联网协议(IP)地址。
常用方法 | 描述 |
---|---|
static InetAddress getByAddress(byte[] addr) | 在给定原始IP地址的情况下,返回 InetAddress对象。 |
static InetAddress getByAddress(String host,byte[] addr) | 根据提供的主机名和IP地址创建 InetAddress |
static InetAddress getByName(String host) | 在给定主机名的情况下确定主机的IP地址 |
String getHostAddress() | 返回IP地址字符串 |
String getHostName() | 获取此IP的主机名 |
static InetAddress getLocalHost() | 返回本地主机 |
String toString() | 将此IP地址转换为 String |
Socket客户端实例
import java.net.*;
import java.io.*;
public class GreetingClient{
public static void main(String [] args){
String serverName = args[0];
int port = Integer.parseInt(args[1]);
try{
System.out.println("连接到主机:" + serverName + " ,端口号:" + port);
Socket client = new Socket(serverName, port);
System.out.println("远程主机地址:" + client.getRemoteSocketAddress());
OutputStream outToServer = client.getOutputStream();
DataOutputStream out = new DataOutputStream(outToServer);
out.writeUTF("Hello from " + client.getLocalSocketAddress());
InputStream inFromServer = client.getInputStream();
DataInputStream in = new DataInputStream(inFromServer);
System.out.println("服务器响应: " + in.readUTF());
client.close();
}catch(IOException e){
e.printStackTrace();
}
}
}
服务端实例
import java.net.*;
import java.io.*;
public class GreetingServer extends Thread{
private ServerSocket serverSocket;
public GreetingServer(int port) throws IOException{
serverSocket = new ServerSocket(port);
serverSocket.setSoTimeout(10000);
}
public void run(){
while(true){
try{
System.out.println("等待远程连接,端口号为:" + serverSocket.getLocalPort() + "...");
Socket server = serverSocket.accept();
System.out.println("远程主机地址:" + server.getRemoteSocketAddress());
DataInputStream in = new DataInputStream(server.getInputStream());
System.out.println(in.readUTF());
DataOutputStream out = new DataOutputStream(server.getOutputStream());
out.writeUTF("谢谢连接我:" + server.getLocalSocketAddress() + "\nGoodbye!");
server.close();
}catch(SocketTimeoutException s){
System.out.println("Socket timed out!");
break;
}catch(IOException e){
e.printStackTrace();
break;
}
}
}
public static void main(String [] args){
int port = Integer.parseInt(args[0]);
try{
Thread t = new GreetingServer(port);
t.run();
}catch(IOException e){
e.printStackTrace();
}
}
}
URL处理
url俗称网页地址,中文名为统一资源定位符。表示为互联网上的资源。
protocol://host:port/path?query#fragment
协议:// 主机:端口号/文件路径?请求参数#定位位置
protocol(协议),可以是HTTP、HTTPS、FTP和File。
port 为端口号
path 为文件路径即文件名
query 为请求参数
fragment 为定位位置
构建
URL url = new URL("http://www.baidu.com/index.html?user=root#j2se")
访问方法
方法 | 描述 |
---|---|
String getPath() | 返回URL 路径部分 |
String getQuery() | 返回查询部分 |
String getAuthority() | 获取此URL的授权部分 |
int getPort() | 返回URL端口部分 |
int getDefaultPort() | 返回协议的默认端口 |
String getProtocol() | 返回URL的协议 |
String getHost() | 返回URL的主机 |
String getFile() | 返回URL文件名部分 |
String getRef() | 返回此URL的锚点(也称 引用) |
URLConnection openConnection() throws IOException | 打开一个URL连接,并运行客户端访问资源 |
关于 URLConnections 类
openConnection() 方法返回一个 java.net.URLConnection
例如:
- 如果你连接HTTP协议的URL,该方法返回 HTTPURLConnection 对象。
- 如果你连接的URL为一个JAR文件,openConnection() 方法将返回 JarURLConnection 对象。
URLConnection方法列表
方法 | 描述 |
---|---|
Object getContent() | 检索URL链接内容 |
Object getContent(Class[] classes) | 检索URL链接内容 |
String getContentEncoding() | 返回头部 content-encoding 字段值 |
int getContentLength() | 返回头部 content-length 值 |
String getContentType() | 返回头部 content-type 字段值 |
int getLastModified() | 返回头部last-modified 字段值 |
long getExpiration() | 返回头部 expires 字段值 |
long getIfModifiedSince() | 返回对象的 ifModifiedSince 字段值 |
void setDoInput(boolean input) | url 连接可用于输入或输出。如果打算用 url 连接进行输入,则将DoInput 标志设置为 true;如果不打算使用,则设置为false。默认为 true |
void setDoOutput(boolean output) | 如果打算进行输出,则设为true,否则为false,默认为false |
InputStream getInputStream() throws IOException | 返回 URL的输入流,用于读取资源 |
OutputStream getOutputStream() throws IOException | 返回url的输出流,用于写入资源 |
URL getURL() | 返回URLConnection对象连接的URL |
URL url = new URL("http://www.baidu.com");
URLConnection urlConnection = url.openConnection();