我是南城余!阿里云开发者平台专家博士证书获得者!
欢迎关注我的博客!一同成长!
一名从事运维开发的worker,记录分享学习。
专注于AI,运维开发,windows Linux 系统领域的分享!
本章节对应知识库
两种架构各有优势,但是无论哪种架构,都离不开网络的支持。网络编程,就是在一定的协议下,实现两台计算机的通信的程序。
》使用IP地址(准确地定位网络上一台或多台主机)
》使用端口号(定位主机上的特定应用)
》规范网络通信协议(可靠、高效地进行数据传输)
作用:用来给网络中地一台计算机设备做唯一地编号
》IP地址分类方式1
IPv4(占用4个字节) IPv6(占用16个字节)
》IP地址分类方式2
公网地址(万维网使用)和私有地址(局域网使用,以192.168开头)
127.0.0.1
》可以唯一识别主机中的进程(应用程序)
》不同进程分配不同的端口号
》范围0~65535
InetAddress类主要表示IP地址
InetAddress 类没有提供公共的构造器,而是提供 了 如下几个 静态方法来获取InetAddress 实例
InetAddress 提供了如下几个常用的方法
- import java.net.InetAddress;
- import java.net.UnknownHostException;
-
- import org.junit.Test;
-
- public class TestInetAddress {
- @Test
- public void test01() throws UnknownHostException{
- InetAddress localHost = InetAddress.getLocalHost();
- System.out.println(localHost);
- }
-
- @Test
- public void test02()throws UnknownHostException{
- InetAddress atguigu = InetAddress.getByName("www.atguigu.com");
- System.out.println(atguigu);
- }
-
- @Test
- public void test03()throws UnknownHostException{
- // byte[] addr = {112,54,108,98};
- byte[] addr = {(byte)192,(byte)168,24,56};
- InetAddress atguigu = InetAddress.getByAddress(addr);
- System.out.println(atguigu);
- }
- }
TCP协议:
可靠的连接(发送数据前,需要三次握手,结束时,四次挥手),进行大数据量的传输,效率低
UDP协议:
不可靠连接(发送前,不需要确认对方是否存在),使用数据报传输(限制在64kb以内),效率高
TCP生活案例:打电话
UDP生活案例:发送短信、发电报
TCP协议中,在发送数据的准备阶段,客户端与服务器之间的三次交互,以保证连接的可靠。
1、客户端会随机一个初始序列号seq=x,设置SYN=1 ,表示这是SYN握手报文。然后就可以把这个 SYN 报文发送给服务端了,表示向服务端发起连接,之后客户端处于同步已发送状态。
2、服务端收到客户端的 SYN 报文后,也随机一个初始序列号(seq=y),设置ack=x+1,表示收到了客户端的x之前的数据,希望客户端下次发送的数据从x+1开始。设置 SYN=1 和 ACK=1。表示这是一个SYN握手和ACK确认应答报文。最后把该报文发给客户端,该报文也不包含应用层数据,之后服务端处于同步已接收状态。
3、客户端收到服务端报文后,还要向服务端回应最后一个应答报文,将ACK置为 1 ,表示这是一个应答报文ack=y+1 ,表示收到了服务器的y之前的数据,希望服务器下次发送的数据从y+1开始。最后把报文发送给服务端,这次报文可以携带数据,之后客户端处于 连接已建立 状态。服务器收到客户端的应答报文后,也进入连接已建立状态。
完成三次握手,连接建立后,客户端和服务器就可以开始进行数据传输了。由于这种面向连接的特性,TCP协议可以保证传输数据的安全,所以应用十分广泛,例如下载文件、浏览网页等。
TCP协议中,在发送数据结束后,释放连接时需要经过四次挥手。
1、客户端打算断开连接,向服务器发送FIN报文(FIN标记位被设置为1,1表示为FIN,0表示不是),FIN报文中会指定一个序列号,之后客户端进入FIN_WAIT_1状态。也就是客户端发出连接释放报文段(FIN报文),指定序列号seq = u,主动关闭TCP连接,等待服务器的确认。
2、服务器收到连接释放报文段(FIN报文)后,就向客户端发送ACK应答报文,以客户端的FIN报文的序列号 seq+1 作为ACK应答报文段的确认序列号ack = seq+1 = u + 1。接着服务器进入CLOSE_WAIT(等待关闭)状态,此时的TCP处于半关闭状态(下面会说什么是半关闭状态),客户端到服务器的连接释放。客户端收到来自服务器的ACK应答报文段后,进入FIN_WAIT_2状态。
3、服务器也打算断开连接,向客户端发送连接释放(FIN)报文段,之后服务器进入LASK_ACK(最后确认)状态,等待客户端的确认。服务器的连接释放(FIN)报文段的FIN=1,ACK=1,序列号seq=m,确认序列号ack=u+1。
4、客户端收到来自服务器的连接释放(FIN)报文段后,会向服务器发送一个ACK应答报文段,以连接释放(FIN)报文段的确认序号 ack 作为ACK应答报文段的序列号 seq,以连接释放(FIN)报文段的序列号 seq+1作为确认序号ack。
之后客户端进入TIME_WAIT(时间等待)状态,服务器收到ACK应答报文段后,服务器就进入CLOSE(关闭)状态,到此服务器的连接已经完成关闭。客户端处于TIME_WAIT状态时,此时的TCP还未释放掉,需要等待2MSL后,客户端才进入CLOSE状态。
- //客户端发送内容给服务端,服务端将内容打印到控制台上。
-
- //客户端 发出方:outputimport java.io.InputStream;
- import java.io.OutputStream;
- import java.net.Socket;
-
- public class Client {
-
- public static void main(String[] args) throws Exception {
- // 1、准备Socket,连接服务器,需要指定服务器的IP地址和端口号
- Socket socket = new Socket("127.0.0.1", 8888);
-
- // 2、获取输出流,用来发送数据给服务器
- OutputStream out = socket.getOutputStream();
- // 发送数据
- out.write("lalala".getBytes());
- //会在流末尾写入一个“流的末尾”标记,对方才能读到-1,否则对方的读取方法会一致阻塞
- socket.shutdownOutput();
-
- //3、获取输入流,用来接收服务器发送给该客户端的数据
- InputStream input = socket.getInputStream();
- // 接收数据
- byte[] data = new byte[1024];
- StringBuilder s = new StringBuilder();
- int len;
- while ((len = input.read(data)) != -1) {
- s.append(new String(data, 0, len));
- }
- System.out.println("服务器返回的消息是:" + s);
-
- //4、关闭socket,不再与服务器通信,即断开与服务器的连接
- //socket关闭,意味着InputStream和OutputStream也关闭了
- socket.close();
- }
- }
-
- //服务器:接受方 input
- import java.io.InputStream;
- import java.io.OutputStream;
- import java.net.InetAddress;
- import java.net.ServerSocket;
- import java.net.Socket;
-
- public class Server {
-
- public static void main(String[] args)throws Exception {
- //1、准备一个ServerSocket对象,并绑定8888端口
- ServerSocket server = new ServerSocket(8888);
- System.out.println("等待连接....");
-
- //2、在8888端口监听客户端的连接,该方法是个阻塞的方法,如果没有客户端连接,将一直等待
- Socket socket = server.accept();
- InetAddress inetAddress = socket.getInetAddress();
- System.out.println(inetAddress.getHostAddress() + "客户端连接成功!!");
-
- //3、获取输入流,用来接收该客户端发送给服务器的数据
- InputStream input = socket.getInputStream();
- //接收数据
- byte[] data = new byte[1024];
- StringBuilder s = new StringBuilder();
- int len;
- while ((len = input.read(data)) != -1) {
- s.append(new String(data, 0, len));
- }
- System.out.println(inetAddress.getHostAddress() + "客户端发送的消息是:" + s);
-
- //4、获取输出流,用来发送数据给该客户端
- OutputStream out = socket.getOutputStream();
- //发送数据
- out.write("欢迎登录".getBytes());
- out.flush();
-
- //5、关闭socket,不再与该客户端通信
- //socket关闭,意味着InputStream和OutputStream也关闭了
- socket.close();
-
- //6、如果不再接收任何客户端通信,可以关闭ServerSocket
- server.close();
- }
- }
作用:一个具体的url对应着互联网上某一资源的地址
URL格式
应用层协议 ip地址+端口号——》对应的域名 资源地址 参数列表