互联网上那么多设备,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(); } } }