1. 套接字介绍
套接字是介于传输层(TCP/UDP)和应用层(HTTP/FTP)之间的一个抽象,被应用程序调用;
在java环境中套接字编程主要使用TCP/IP协议,但是套接字支持的协议族远不止这些;
在java套接字编程中有Socket和ServerSocket两个核心类,ServerSocket位于服务器端监听连接,Socket位于客户端发起连接,服务器端监听到连接后也会产生一个Socket实例来完成与对端Socket的通信,并且服务器端的Socket和客户端的Socket没有任何区别,也就是说进程间通信需要一对套接字完成;
2. API介绍
Socket和ServerSocket是java提供网络编程的门面接口,实际上都是通过SocketImpl及子类完成的;
2.1 重要类关系图
  
 
2.2 重要类介绍
1. InetAddress
InetAddress是java对IP地址的封装,包括IP设备的名称、ip地址,有Inet4Address和Inet6Address两个子类;
没有公开的构造方法,只能通过公开的静态方法实例化对象
方法介绍
1)getByName:实例化一个InetAddress对象,getByName的参数可以是ip地址字符串,也可以是主机名
2)anyLocalAddress:实例化一个表示任何本地地址的网络地址,通常是0.0.0.0
3)getHostAddress一般通过InetAddress的获取主机ip地址串,通过获取主机名
4)getHostName
2. SocketAddress/InetSocketAddress
套接字地址,用来表示一个套接字在网络中的位置;
SocketAddress不依赖任何协议的套接字地址;
InetSocketAddress表示一个IP套接字地址,由套接字地址的 IP地址(InetAddress)、主机名和端口号组成
3. ServerSocket
服务器套接字,在服务器端用于监听网络传入
构造方法
1)ServerSocket():创建一个不绑定任何ip和端口的服务器套接字
创建一个SocksSocketImpl赋值给ServerSocket的SocketImpl,并把自己赋值给SocketImpl的ServerSocket
2)ServerSocket(port, backlog, InetAddress):用指定的端口、连接大小、ip地址创建一个服务器套接字
同无参构造,创建SocksSocketImpl并相互赋值
根据port和InetAddress创建套接字地址InetSocketAddress(InetAddress, port),如果InetAddress为null则通过InetAddress的anyLocalAddress实例化一个IP地址,如果port不合法则抛出异常;
通过调用bind(SocketAddress, backlog)将服务器套接字和套接字地址进行绑定;
backlog说明https://blog.csdn.net/oyueyang1/article/details/80451535
方法介绍
1)bind(SocketAddress, backlog):将服务器套接字和套接字地址进行绑定,tcp accept队列大小为backlog
如果SocketAddress为null则新建一个InetSocketAddress(InetAddress.anyLocalAddress, 0),0表示让系统随机分配端口号
实际上通过SocketImpl的bind(InetAddress, port)绑定的,SocketImpl是在构造函数中创建并绑定到当前类的
有参构造方法在构造过程都进行了bind操作,所以此方法一般与无参构造方法同时使用
2)isBound、getInetAddress、getLocalPort、getLocalSocketAddress查询bind方法的绑定信息
3)accept:阻塞式的监听到此服务器套接字连接,监听到连接后会创建一个Socket套接字,服务端和客户端各有一个Socket,2个套接字可以完成不同进程间通信;
4)implAccept(Socket):为保护方法,当我们自定义ServerSocket需要复写accept方法时,需要新建一个Socket然后通过implAccept加工,accept的阻塞其实是阻塞在implAccept的加工过程
5)getChannel:获取ServerSocketChannel通道
6)close/isClose:关闭套接字/判断套接字是否已关闭
关闭服务器套接字后,阻塞在accept的线程会抛出SocketException;
ServerSocketChannel通道关闭;
注意:没有直接方法判断对端套接字是否关闭
7)getSoTimeout/setSoTimeout: 设置accept的阻塞时间,0时表示无穷大,如果超时抛出SocketTimeOutException;
间接调用SocketImple的getOption(capId)/setOption(capId,Object), capId为SO_TIMEOUT(SocketOption中定义);
8)setSocketFactory(SocketImplFactory):设置套接字实现工厂
构造方法中会创建一个SocketImpl,默认是创建SocksSocketImpl,如果已设置SocketImplFactory则会调用工厂方法创建一个
4. Socket
套接字或客户端套接字,不同进程通信的端点,服务器端会有n多个客户端套接字
构造方法
1)Socket():创建一个不绑定任何套接字地址,不连接任务服务器套接字的套接字
创建一个SocksSocketImpl赋值给Socket的SocketImpl,并把自己赋值给SocketImpl的Socket, 同ServerSocket的无参构造方法
2)Socket(SocketAddress serverAddress, SocketAddress localAddress, boolean stream):创建套接并绑定和连接到指定套接字地址
绑定到本地套接字地址localAddress,连接的服务器套接字地址为serverAddress;
如果localAddress为null则内核随机选择一个本地ip地址和端口进行绑定;
stream表示流套接字(tcp),负责数据包datagram套接字(udp,每个包有大小限制)
3)Socket(InetAddress server, int serverPort, InetAddress local, int localPort):创建套接并绑定和连接到指定套接字地址
将server和serverPort封装成服务器套接字SocketAddress,local和localPort封装成客户端套接字SocketAddress,然后调用Socket(SocketAddress, SocketAddress, stream)创建流式套接字
4)Socket(String host, int port):
将host和port封装成服务器端套接字地址SocketAddress;
本地SocketAddress为null则,本地端口随机,任意一本机ip;
调用Socket(SocketAddress, SocketAddress, stream)创建流式Socket;
方法介绍
1)connect(SocketAddress)/connect(SocketAddress, timeout): 连接到套接字地址指定的服务器套接字
有参数构造方法中都会调用connect方法,所以此方法一般与无参构造方法同时使用
2)bind(SocketAddress) :将套接字和套接字地址进行绑定
如果已绑定则异常;
如果没有绑定在connect方法中会进行绑定,所以如果要调用bind方法需要在connect前调用;
3)getInetAddress()/getPort()/getRemoteSocketAddress(): 获取 对端 套接字的IP地址、端口号、套接字地址,没有连接则返回null
4)getLocalAddress()/getLocalPort()/getLocalSocketAddress():获 此 套接字的IP地址、端口号、套接字地址
5)getChannel():返回此套接字的SocketChannel通道
6)getInputStream():返回套接字的输入流
如果连接正常,inputstream的read会一直阻塞;
关闭输入流会关闭对应的套接字;
当Socket底层的连接中断时,套接字已缓存的字节可以通过read读取,如果已经消耗了所有缓存的字节,read操作会抛出IOException,套接字上没有任何缓存字节,并且套接字没有关闭调用 available返回0
7)getOutputStream():返回套接字的输出流
关闭输出流会关闭对应的套接字
8)setSoTimeout(timeout):Socket关联的InputStream的read操作默认是阻塞的,如果设置超时值,read操作超时引发SocketTimeoutException
实际上通过SocketImpl的setOption(optId)实现, optId为SO_TIMEOUT(SocketOptions中定义)
9)close:关闭套接字
阻塞于当前套接字的IO操作会抛出SocketException;
关闭后不能重新连接和重新绑定(已建立的连接不能用),只能通过新建;
会关闭此套接字的输入输出流;
关闭关联的通道;
10)shutdownInput:丢弃发送到输入流的所有字节,读取内容返回EOF
11)shutdownOutput:禁用输出流,以前发送的字节都会被底层发送,如果关闭后执行写入会抛出IOException
12)isBound/isClose/isConnect/isInputShutdown/isOutputSHutdown:用于判断相应的操作是否结束
13)setSocketImplFactory:设置套接字实现的工厂,构造方法默认会创建一个SocksSocketImpl,如果指定工厂会通过工厂创建
3. 实例
工具类:SocketUtil
 
 
 1 import java.net.InetSocketAddress;
 2 import java.net.SocketAddress;
 3 
 4 /**
 5  * 工具类
 6  */
 7 public class SocketUtil {
 8 
 9     /**
10      * 解析套接字地址中的ip和端口
11      */
12     public static String getSocketAddress(SocketAddress socketAddress) {
13 
14         if (socketAddress instanceof InetSocketAddress) {
15             InetSocketAddress inetSocketAddress = ((InetSocketAddress) socketAddress);
16             int port = inetSocketAddress.getPort();
17             String hostName = inetSocketAddress.getHostName();
18             return String.format("%s:%s", hostName, port);
19         }
20 
21         return null;
22     }
23 }
View Code
服务器类:DemoTcpServer
 
 
 1 import java.io.*;
 2 import java.net.*;
 3 import java.util.*;
 4 import java.util.concurrent.ExecutorService;
 5 import java.util.concurrent.Executors;
 6 
 7 /**
 8  * 群聊服务器:启动服务器套接字,监听并处理连接
 9  */
10 public class DemoTcpServer {
11 
12     static final List<Socket> ONLINE_SOCKET = new ArrayList();
13 
14     public static void main(String[] args) throws IOException {
15 
16         // 创建服务器套接字,并绑定到本机的41440端口监听
17         ServerSocket serverSocket = new ServerSocket(41440, 50, InetAddress.getByName("192.168.1.101"));
18         System.out.println("服务器started...");
19 
20         // 创建线程池用于处理已连接的socket
21         ExecutorService executorService = Executors.newFixedThreadPool(10);
22         while (true) {
23             // 等待socket接入,有连接接入后会创建一个Socket与对端Socket通信,此Socket地址ip为本机、端口随机
24             Socket socket = serverSocket.accept();
25             String socketAddress = SocketUtil.getSocketAddress(socket.getRemoteSocketAddress());
26             System.out.println(socketAddress + " 上线了...");
27             // 加入到在线的socket列表,用于群发消息
28             ONLINE_SOCKET.add(socket);
29             // 单独起一个线程监听消息,可以改成nio的Selectors实现
30             executorService.submit(new SocketHandler(socket));
31         }
32     }
33 }
34 
35 /**
36  * 处理Socket的任务
37  */
38 class SocketHandler implements Runnable {
39 
40     private Socket socket;
41 
42     public SocketHandler(Socket socket) {
43         this.socket = socket;
44     }
45 
46     @Override
47     public void run() {
48         try {
49             String clientFlag = SocketUtil.getSocketAddress(this.socket.getRemoteSocketAddress());
50             byte[] input = new byte[1024];
51             // 循环监听消息
52             while (true) {
53                 // 阻塞式读取消息
54                 int read = this.socket.getInputStream().read(input);
55                 // 服务器记录发送的所有消息
56                 System.out.println(clientFlag + " " + new String(input, 0, read, "utf-8"));
57                 // 群发消息
58                 sendMsg(this.socket, clientFlag, new String(input, 0, read, "utf-8"));
59             }
60         } catch (IOException e) {
61             System.out.println(SocketUtil.getSocketAddress(socket.getRemoteSocketAddress()) + "下线...");
62             DemoTcpServer.ONLINE_SOCKET.remove(socket);
63         }
64     }
65 
66     /**
67      * 群发消息
68      */
69     private void sendMsg(Socket ignoreSocket, String clientFlag, String msg) {
70 
71         Iterator<Socket> iterator = DemoTcpServer.ONLINE_SOCKET.iterator();
72         while (iterator.hasNext()) {
73 
74             Socket currentSocket = iterator.next();
75 
76             // 不需要向此Socket对端发送消息,只有一个客户端时需要注解此判断
77             if (currentSocket.equals(ignoreSocket)) {
78                 continue;
79             }
80 
81             // 发送消息到客户端
82             try {
83                 currentSocket.getOutputStream().write((clientFlag + System.lineSeparator() + msg).getBytes("utf-8"));
84             } catch (Exception e) {
85                 System.out.println(SocketUtil.getSocketAddress(currentSocket.getRemoteSocketAddress()) + "下线...");
86                 iterator.remove();
87             }
88         }
89     }
90 }
View Code
客户端类:
 
 
 1 import java.io.IOException;
 2 import java.net.InetSocketAddress;
 3 import java.net.Socket;
 4 import java.util.Scanner;
 5 
 6 /**
 7  * 客户端
 8  */
 9 public class DemoTcpClient {
10     public static void main(String[] args) throws Exception {
11 
12         //以下3步可以通过有参构造方法一次指定
13         // 创建客户端socket
14         Socket socket = new Socket();
15         // 绑定到本地的ip和端口号
16         socket.bind(new InetSocketAddress("192.168.1.101", 41441));
17         // 连接到远程的套接字服务器
18         socket.connect(new InetSocketAddress("192.168.1.101", 41440), 0);
19         System.out.println("已上线...");
20 
21         // 新启动线程用于回显消息
22         new Thread(() -> {
23             try {
24                 byte[] input = new byte[1024];
25                 while (true) {
26                     int read = socket.getInputStream().read(input);
27                     System.out.println(new String(input, 0, read, "utf-8"));
28                 }
29             } catch (IOException e) {
30                 e.printStackTrace();
31             }
32         }).start();
33 
34         //处理输入
35         Scanner scanner = new Scanner(System.in);
36         while (scanner.hasNext()) {
37             socket.getOutputStream().write(scanner.nextLine().getBytes("utf-8"));
38             socket.getOutputStream().flush();
39         }
40     }
41 }
View Code
参考文献:
 1. https://droidyue.com/blog/2015/03/08/sockets-programming-in-java/
  2. https://my.oschina.net/leejun2005/blog/104955
  3. https://www.cnblogs.com/w-wfy/p/6415840.html
 
  
  
  
 
 
  
 
 
  
 