互联网上那么多设备,java 是如何与其他设备通信的呢?这次的内容是网络通信的基础,有了它咱们才能上网页、玩游戏、视频聊天。
Socket 客户端套接字
Socket 客户端套接字,用于连接互联网提供服务的设备。
Socket 构造方法
| 构造方法 | 说明 |
|---|---|
| Socket() | 通过系统默认类型的 SocketImpl 创建未连接套接字 |
| Socket(String host, int port) | 创建一个流套接字并将其连接到指定主机上的指定端口号 |
常用方法
| 方法名称 | 说明 |
|---|---|
| getOutputStream() | 返回此套接字的输出流 |
| getInputStream() | 返回此套接字的输入流 |
下面示例模拟了一个 HTTP 请求
import java.io.IOException;import java.io.InputStream;import java.io.OutputStream;import java.net.Socket;import java.net.UnknownHostException;import java.util.Scanner;public class TestSocket { public static void main(String[] args) {
//创建套接字
try(Socket s=new Socket("www.baidu.com", 80);){ //创建向服务器发送数据的输出流
OutputStream os=s.getOutputStream();
StringBuffer sb=new StringBuffer(); //HTTP协议 请求报文
sb.append("GET / HTTP/1.1\r\n");
sb.append("Host: www.baidu.com:80\r\n");
sb.append("Connection: Keep-Alive\r\n"); //这里一定要一个回车换行,表示消息头完,不然服务器会等待
sb.append("\r\n"); //发送
os.write(sb.toString().getBytes()); //获取服务器相应内容
InputStream is=s.getInputStream(); //通过输入流创建扫描器,并指定编码为utf-8防止中文乱码
Scanner scanner=new Scanner(is,"utf-8");
while(scanner.hasNextLine()) {
String line=scanner.nextLine();
System.out.println(line);
}
} catch (UnknownHostException e) {
e.printStackTrace();
System.out.println("未知主机");
} catch (IOException e) {
e.printStackTrace();
System.out.println("IO异常");
}
}
}ServerSocket
ServerSocket:实现服务器套接字,服务器套接字等待请求通过网络传入。它基于该请求执行某些操作,然后可能向请求者返回结果。
import java.io.IOException;import java.io.InputStream;import java.io.OutputStream;import java.io.PrintWriter;import java.net.ServerSocket;import java.net.Socket;import java.util.Scanner;public class TestServerSocket { public static void main(String[] args) { //创建 ServerSocket 监听1666端口
try(ServerSocket server=new ServerSocket(1666)){ //阻塞方法,当有客户端连入,获取客户端Socket
try(Socket client=server.accept()){ //获取客户端发送的数据
InputStream is=client.getInputStream(); //获取向客户端发送数据的流
OutputStream os=client.getOutputStream(); //通过输入流创建扫描器
try(Scanner scanner=new Scanner(is)){
PrintWriter pw=new PrintWriter(os,true/*自动刷新*/); //向客户端发送消息
pw.println("Hello,enter bye to exit."); boolean done=false; //客户端有输入数据并且没有发送 bye
while(!done&&scanner.hasNextLine()) { //接收客户端发送的数据
String line=scanner.nextLine(); //将客户端发送的数据发回客户端
pw.println("Echo:"+line); //如果客户端输入bye 结束通信
if(line.trim().equalsIgnoreCase("bye")) {done=true;}
}
}
}
} catch (IOException e) {
e.printStackTrace();
}
}
}测试方式:
在DOS 中输入命令:telnet 127.0.0.1 1666
telnet 不是内部或外部命令的读者,需要在 Windows 功能中启用 Telnet 客户端。
上面的代码如果有多个客户端连入就不行了,如果希望服务能被多个客户端连接,可以使用线程。
多线程服务器
import java.io.IOException;import java.io.InputStream;import java.io.OutputStream;import java.io.PrintWriter;import java.net.ServerSocket;import java.net.Socket;import java.util.Scanner;public class TestMultiServerSocket { public static void main(String[] args) { //创建 ServerSocket 监听 1666端口
try(ServerSocket server=new ServerSocket(1666)){ while(true) { //accept() 是一个阻塞方法
Socket client=server.accept();
InputStream is=client.getInputStream();
OutputStream os=client.getOutputStream(); //开启新的线程处理,传入当前客户端
Thread t=new Thread(new ThreadEchoHandler(client));
t.start();
}
} catch (IOException e) {
e.printStackTrace();
}
}
}class ThreadEchoHandler implements Runnable{ private Socket socket=null; public ThreadEchoHandler(Socket socket) { this.socket=socket;
} @Override
public void run() { try {
InputStream is=socket.getInputStream();
OutputStream os=socket.getOutputStream(); try(Scanner scanner=new Scanner(is)){
PrintWriter pw=new PrintWriter(os,true);
pw.println("Hello,enter bye to exit."); boolean done=false; while(!done&&scanner.hasNextLine()) {
String line=scanner.nextLine();
pw.println("Echo:"+line); if(line.trim().equalsIgnoreCase("bye")) {done=true;}
}
}
socket.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}URLConnection
抽象类 URLConnection 是所有类的超类,它代表应用程序和 URL 之间的通信链接。此类的实例可用于读取和写入此 URL 引用的资源。
Socket 可以默认任意类型的网络通信,URLConnection 更适合 HTTP 请求,使用 URLConnection 进行HTTP操作更方便,模拟请求报文,获取响应报文和内容。
URLConnection 常用方法
| 方法 | 说明 |
|---|---|
| connect() | 打开到此 URL 引用的资源的通信链接(如果尚未建立这样的连接) |
| getContentEncoding() | 返回 content-encoding 头字段的值 |
| getContentType() | 返回 content-type 头字段的值 |
| getHeaderFields() | 返回头字段的不可修改的 Map |
| getInputStream() | 返回从此打开的连接读取的输入流 |
| setRequestProperty(String key, String value) | 设置一般请求属性 |
获取 URLConnection 需要先创建 URL 对象:URL url=new URL(host);
使用 URLConnection 获取网页的内容
import java.io.IOException;import java.io.InputStream;import java.net.HttpURLConnection;import java.net.MalformedURLException;import java.net.URL;import java.net.URLConnection;import java.util.List;import java.util.Map;import java.util.Map.Entry;import java.util.Scanner;public class TestURLConnection { public static void main(String[] args) {
try { //创建URL对象
URL url=new URL("http://www.baidu.com"); //创建 URLConnection对象
URLConnection connection=url.openConnection(); //设置请求属性
//connection.setRequestProperty("", "");
//连接
connection.connect(); //获取输入流
InputStream is=connection.getInputStream(); //通过输入流构建一个扫描器
Scanner scanner=new Scanner(is,"utf-8"); while(scanner.hasNextLine()) {
String line=scanner.nextLine();
System.out.println(line);
}
System.out.println("===响应头===");
Map<String,List<String>> headers=connection.getHeaderFields(); for (Entry<String, List<String>> entry: headers.entrySet()) {
String key=entry.getKey();
System.out.print(key+":"); for (String string : entry.getValue()) {
System.out.print(string);
}
System.out.println();
}
} catch (MalformedURLException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
}
}
随时随地看视频