网络编程是指编写运行在多个设备(计算机)的程序,这些设备都通过网络连接起来。java.net
包中 J2SE 的 API 包含有类和接口,它们提供低层次的通信细节。你可以直接使用这些类和接口,来专注于解决问题,而不用关注通信细节。
java.net 包中提供了两种常见的网络协议的支持:TCP;UDP。
Socket通信
套接字使用TCP提供了两台计算机之间的通信机制。 客户端程序创建一个套接字,并尝试连接服务器的套接字。当连接建立时,服务器会创建一个 Socket 对象。客户端和服务器现在可以通过对 Socket 对象的写入和读取来进行通信。java.net.Socket
类代表一个套接字,并且 java.net.ServerSocket
类为服务器程序提供了一种来监听客户端,并与他们建立连接的机制:
1. 服务器实例化一个 ServerSocket 对象,表示通过服务器上的端口通信(建议大于1024的端口)。
2. 服务器调用 ServerSocket 类的 accept() 方法,该方法将一直监听,直到客户端连接到服务器上给定的端口。
3. 服务器正在等待时,一个客户端实例化一个 Socket 对象,指定服务器名称和端口号来请求连接。
4. Socket 类的构造函数试图将客户端连接到指定的服务器和端口号。如果通信被建立,则在客户端创建一个 Socket 对象能够与服务器进行通信。
5. 在服务器端,accept() 方法返回服务器上一个新的 socket 引用,该 socket 连接到客户端的 socket。
6. 连接建立后,通过使用 I/O 流在进行通信,每一个socket都有一个输出流和一个输入流,客户端的输出流连接到服务器端的输入流,而客户端的输入流连接到服务器端的输出流。
Socket类提供了getInputStream()
方法和getOutputStream()
方法,分别返回输入流InputStream
对象和输出流 OutputStream
对象。
- 程序只需向输出流写数据,就能向对方发送数据;
- 程序只需从输入流读数据,就能接收来自对方的数据。
ServerSocket类
- 构造方法
- public ServerSocket(int port) throws IOException
构造方法,创建绑定到特定端口的服务器套接字。 - public ServerSocket(int port, int backlog) throws IOException
构造方法,利用指定的 backlog 创建服务器套接字并将其绑定到指定的本地端口号。 - public ServerSocket(int port, int backlog, InetAddress address) throws IOException
构造方法,使用指定的端口、侦听 backlog 和要绑定到的本地 IP 地址创建服务器。 - public ServerSocket() throws IOException
构造方法,创建非绑定服务器套接字。
- public ServerSocket(int port) throws IOException
- 成员方法
- public Socket accept() throws IOException
侦听并接受到此套接字的连接。 - public int getLocalPort()
返回此套接字在其上侦听的端口。 - public void setSoTimeout(int timeout)
通过指定超时值启用/禁用 SO_TIMEOUT,以毫秒为单位。 - public void bind(SocketAddress host, int backlog)
将 ServerSocket 绑定到特定地址(IP 地址和端口号)。
- public Socket accept() throws IOException
Socket类
- 构造方法
- public Socket(String host, int port) throws UnknownHostException, IOException.
构造方法,创建一个流套接字并将其连接到指定主机上的指定端口号。 - public Socket(InetAddress host, int port) throws IOException
创建一个流套接字并将其连接到指定 IP 地址的指定端口号。 - public Socket(String host, int port, InetAddress localAddress, int localPort) throws IOException.
创建一个套接字并将其连接到指定远程主机上的指定远程端口。 - public Socket(InetAddress host, int port, InetAddress localAddress, int localPort) throws IOException.
创建一个套接字并将其连接到指定远程地址上的指定远程端口。 - public Socket()
通过系统默认类型的 SocketImpl 创建未连接套接字
- public Socket(String host, int port) throws UnknownHostException, IOException.
- 成员方法
- public void connect(SocketAddress host, int timeout) throws IOException
将此套接字连接到服务器,并指定一个超时值。 - public InputStream getInputStream() throws IOException
返回此套接字的输入流。 - public OutputStream getOutputStream() throws IOException
返回此套接字的输出流。 - public InetAddress getInetAddress()
返回套接字连接的地址。 - public int getPort()
返回此套接字连接到的远程端口。 - public int getLocalPort()
返回此套接字绑定到的本地端口。 - public SocketAddress getRemoteSocketAddress()
返回此套接字连接的端点的地址,如果未连接则返回 null。 - public void close() throws IOException
关闭此套接字。
- public void connect(SocketAddress host, int timeout) throws IOException
Socket实现简单多窗口群聊 服务器程序:
import java.io.*;
import java.net.ServerSocket;
import java.net.Socket;
import java.util.Date;
import java.util.Vector;
//Manager是单例,用来管理多个ChatSocket
class ChatManager {
private ChatManager(){}
private static final ChatManager chatManager = new ChatManager();
public static ChatManager getChatManager() {
return chatManager;
}
Vector<ChatSocket> vector = new Vector<ChatSocket>();
public void addToVector(ChatSocket chatSocket) {
vector.add(chatSocket);
}
//向所有客户端推送消息
public void publish(ChatSocket myself, String msg) {
for (int i = 0; i < vector.size(); i++) {
ChatSocket chatSocket = vector.get(i);
if (!chatSocket.equals(myself)) {
chatSocket.echoBack(msg);
}
}
}
}
class ChatSocket extends Thread{
Socket server;
BufferedReader br;
PrintWriter pw;
public ChatSocket(Socket socket) {
this.server = socket;
}
@Override
public void run() {
try {
br = getReader(server);
pw = getWriter(server);
String msg = null;
while ((msg = br.readLine()) != null) {
System.out.println(server.getRemoteSocketAddress() + " : " + msg);
ChatManager.getChatManager().publish(this, msg);
if (msg.equals("bye")) {
System.out.println("客户端" + server.getRemoteSocketAddress() + "已断开连接");
echoBack(server.getRemoteSocketAddress() + "的连接已断开");
break;
}
}
if (server != null) server.close(); //断开连接
} catch (IOException e) {
e.printStackTrace();
}
}
public void echoBack(String msg) {
pw.println("收到信息: " + msg);
}
private PrintWriter getWriter(Socket socket) throws IOException {
OutputStream socketOut = socket.getOutputStream();
return new PrintWriter(socketOut, true);
}
private BufferedReader getReader(Socket socket) throws IOException {
InputStream socketIn = socket.getInputStream();
return new BufferedReader(new InputStreamReader(socketIn));
}
}
class ServerListener extends Thread{
private int port = 8000;
private ServerSocket serverSocket;
@Override
public void run() {
try {
serverSocket = new ServerSocket(port);
System.out.println("服务器启动");
System.out.println(new Date());
System.out.println("等待远程连接...端口号为: " + serverSocket.getLocalPort());
while (true) {
Socket server = serverSocket.accept();
System.out.println("已连接: " + server.getRemoteSocketAddress());
System.out.println(new Date());
ChatSocket chatSocket = new ChatSocket(server);
ChatManager.getChatManager().addToVector(chatSocket);
chatSocket.start();
}
} catch (IOException e) {
e.printStackTrace();
}
}
}
public class SocketServer {
public static void main(String args[]) throws IOException {
new ServerListener().start();
}
}
运行结果:
URL处理
URL(Uniform Resource Locator)中文名为统一资源定位符,有时也被俗称为网页地址。表示为互联网上的资源,如网页或者FTP地址。URL可以分为如下几个部分。
protocol://host:port/path?query#fragment
protocol(协议)可以是 HTTP、HTTPS、FTP 和 File,port 为端口号,path为文件路径及文件名。
URL类
在java.net包中定义了URL类,该类用来处理有关URL的内容。java.net.URL提供了丰富的URL构建方式,并可以通过java.net.URL来获取资源。
构造方法
public URL(String url) throws MalformedURLException
通过给定的URL字符串创建URLpublic URL(String protocol, String host, int port, String file) throws MalformedURLException.
通过给定的参数(协议、主机名、端口号、文件名)创建URL。public URL(String protocol, String host, String file) throws MalformedURLException
使用指定的协议、主机名、文件名创建URL,端口使用协议的默认端口。public URL(URL context, String url) throws MalformedURLException
使用基地址和相对URL创建
成员方法
public URLConnection openConnection() throws IOException
打开一个URL连接,并运行客户端访问资源。public String getPath()
返回URL路径部分。public String getQuery()
返回URL查询部分。public String getAuthority()
获取此 URL 的授权部分。public int getPort()
返回URL端口部分public int getDefaultPort()
返回协议的默认端口号。public String getProtocol()
返回URL的协议public String getHost()
返回URL的主机public String getFile()
返回URL文件名部分public String getRef()
获取此 URL 的锚点(也称为”引用”)。
URLConnections 类
成员方法
public InputStream getInputStream() throws IOException
返回URL的输入流,用于读取资源public OutputStream getOutputStream() throws IOException
返回URL的输出流, 用于写入资源public URL getURL()
返回 URLConnection 对象连接的URLObject getContent()
检索URL链接内容String getContentEncoding()
返回头部 content-encoding 字段值int getContentLength()
返回头部 content-length字段值String getContentType()
返回头部 content-type 字段值int getLastModified()
返回头部 last-modified 字段值long getExpiration()
返回头部 expires 字段值long getIfModifiedSince()
返回对象的 ifModifiedSince 字段值public void setDoInput(boolean input)
URL 连接可用于输入和/或输出。如果打算使用 URL 连接进行输入,则将 DoInput 标志设置为 true;如果不打算使用,则设置为 false。默认值为 truepublic void setDoOutput(boolean output)
URL 连接可用于输入和/或输出。如果打算使用 URL 连接进行输出,则将 DoOutput 标志设置为 true;如果不打算使用,则设置为 false。默认值为 false
示例
import java.net.*;
import java.io.*;
public class Main
{
public static void main(String [] args)
{
try
{
URL url = new URL("http://www.runoob.com");
HttpURLConnection connection = (HttpURLConnection) url.openConnection();
BufferedReader in = new BufferedReader(new InputStreamReader(connection.getInputStream()));
String urlString = "";
String current = null;
while((current = in.readLine()) != null)
{
urlString += current;
}
System.out.println(urlString);
}catch(IOException e)
{
e.printStackTrace();
}
}
}
HTTP通信
待续