IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 网络协议 -> TCP和UDP在Java中的应用 -> 正文阅读

[网络协议]TCP和UDP在Java中的应用

网络的七层在这里插入图片描述

网络层:ip协议(IPV4,IPV6)
传输层:TCP协议和UDP协议
应用层: HTTP协议

TCP和UDP的区别

维度TCPUDP
可靠性可靠不可靠
速度稍慢更快
通信方式点对点点对点、一对多、多对多
通信内容字节码数据报
应用场景传输文件聊天、视频、语音

三次握手和四次挥手

TCP(传输控制协议 Transfer Control Protocol)

建立TCP连接需要通过三次握手机制在服务端和客户端之间建立连接

三次握手(Three-way Handshake)其实就是指建立一个TCP连接时,需要客户端和服务器总共发送3个包。进行三次握手的主要作用就是为了确认双方的接收能力和发送能力是否正常、指定自己的初始化序列号为后面的可靠性传送做准备。实质上其实就是连接服务器指定端口,建立TCP连接,并同步连接双方的序列号和确认号,交换TCP窗口大小信息。

在这里插入图片描述
三次握手:

  1. 客户端发送信号量SYN=1 和序列号seq=J 给服务端,客户端进入SYN_SENT(信号发送状态),等待服务器回答
  2. 服务器收到客户端信号后,发送信号量SYN=1、ACK=1,应答值ack=J+1,新的序列号seq=K给客户端,服务器进入SYN_RCVD(收到信号)状态
  3. 客户端收到服务器的应答消息,发送ACK=1,和应答值ack=K+1给服务器,客户端和服务器都进入ESTABLISHED(连接建立)状态

在socket编程中,客户端执行connect()时,将触发三次握手。

客户端和服务端断开连接需要通过四次挥手

四次挥手:

  1. 客户端发送信号量FIN M给服务器端,进入FIN_WAIT1状态
  2. 服务端收到后,返回ack M+1给客户端,进入CLOSE_WAIT状态
  3. 服务端在发送FIN N信号给客户端
  4. 客户端收到后,返回应答信号ACK=1 ack=N+1给服务端,连接关闭

在socket编程中,任何一方执行close()操作即可产生挥手操作。

在这里插入图片描述

Socket编程

Socket(套接字)基于TCP/IP协议的网络编程机制

服务端

ServerSocket类 ,用于对本机的某个端口进行监听

  • 创建
    ServerSocket(int port)port是端口号
  • 方法
    close() 关闭连接
    Socker accept() 接收客户端的Socket,会阻塞线程直到有客户端连接

客户端

Socket类,代表客户端向服务端发送连接,进行网络通知

  • 创建

    Socket(String ip,int port) 通过ip和端口连接服务端

  • 方法

    disconnect() 关闭

    OutputStream getOutputStream() 获得输出流

    InputStream getInputStream() 获得输入流

    注意:IO流关闭后,socket会自动关闭
    在这里插入图片描述

/**
 * 服务端
 */
public class Server {

    public static final int PORT = 8888;

    public void start() {
        System.out.println("服务器启动了!");
        //创建服务端Socket
        try(ServerSocket serverSocket = new ServerSocket(PORT)){
            //接收客户端连接
            while(true) {
                Socket socket = serverSocket.accept();
                System.out.println("客户端连接了!" + socket.getInetAddress());
                //获得输入流,得到客户端发送的消息
                ObjectInputStream in = new ObjectInputStream(socket.getInputStream());
                String str = in.readUTF();
                System.out.println("接收消息:" + str);
                //获得输出流
                ObjectOutputStream out = new ObjectOutputStream(socket.getOutputStream());
                out.writeUTF("服务器给你问好了!!");
                out.flush();
            }
        } catch (IOException e) {
            e.printStackTrace();
        }
    }

    public static void main(String[] args) {
        new Server().start();
    }
}
/**
 * 客户端
 */
public class Client {

    /**
     * 发送消息
     */
    public void send(String ip,int port,String msg){
        //创建连接
        try(Socket socket = new Socket(ip,port)){
            //获得输出流
            ObjectOutputStream out = new ObjectOutputStream(socket.getOutputStream());
            out.writeUTF(msg);
            out.flush();
            //获得输入流
            ObjectInputStream in = new ObjectInputStream(socket.getInputStream());
            String s = in.readUTF();
            System.out.println("服务器回应:" + s);
        } catch (UnknownHostException e) {
            e.printStackTrace();
        } catch (IOException e) {
            e.printStackTrace();
        }
    }

    public static void main(String[] args) {
        new Client().send("192.168.1.113",9999,"Hello,今天学点什么??");
    }
}

UDP编程

数据报协议,类似广播,属于不可靠的协议,无连接

服务端

DatagramSocket(int port) 指定端口获得发送的数据报

客户端

DatagramSocket() 作为客户端使用

主要方法

receive(DatagramPacket) 接收数据包

send(DatagramPacket) 发送数据报

DatagramPacket 包

DatagramPacket(String ip,int port) 发送前指定ip和端口

public class UDPServer {

    public static final int PORT = 8888;

    public void start(){
        System.out.println("启动服务");
        //创建UDP服务端
        try {
            DatagramSocket server = new DatagramSocket(PORT);
            //接收数据报
            while(true) {
                byte[] buff = new byte[1024];
                DatagramPacket packet = new DatagramPacket(buff,0,buff.length);
                //将接收的数据放入数据报
                server.receive(packet);
                String msg = new String(packet.getData(),0,packet.getLength(),"UTF-8");
                System.out.println("接收到:" + msg);
            }
        } catch (SocketException e) {
            e.printStackTrace();
        } catch (IOException e) {
            e.printStackTrace();
        }
    }

    public static void main(String[] args) {
        new UDPServer().start();
    }

}

public class UDPClient {

    public void send(String ip,int port,String msg){
        //创建客户端
        try {
            DatagramSocket client = new DatagramSocket();
            //创建数据报
            byte[] buff = msg.getBytes();
            DatagramPacket packet = new DatagramPacket(buff,0,buff.length, InetAddress.getByName(ip),port);
            //发送数据报
            client.send(packet);
        } catch (SocketException | UnknownHostException e) {
            e.printStackTrace();
        } catch (IOException e) {
            e.printStackTrace();
        }
    }

    public static void main(String[] args) {
        new UDPClient().send("127.0.0.1",8888,"Hello UDP!!");
    }
}

例子

1、用TCP连接做一个简易版的聊天室

/**
 * 输出流
 */
public class HandleWriteRunnable implements Runnable {
    private Socket socket;

    public HandleWriteRunnable(Socket socket) {
        this.socket = socket;
    }

    @Override
    public void run() {
        try {
            ObjectOutputStream outputStream = new ObjectOutputStream(socket.getOutputStream());
            Scanner scanner = new Scanner(System.in);
            while (true) {
                System.out.println("输入你想说的话");
                String msg = scanner.next();
                outputStream.writeUTF(msg);
                outputStream.flush();
            }
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
}


/**
 * 输入流
 */
public class HandleReadRunnable implements Runnable {
    private Socket socket;

    public HandleReadRunnable(Socket socket) {
        this.socket = socket;
    }

    @Override
    public void run() {
        try {
            ObjectInputStream inputStream = new ObjectInputStream(socket.getInputStream());
            while (true) {
                String msg = inputStream.readUTF();
                System.out.println(socket.getInetAddress() + msg);
            }
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
}



/**
 * 客户端
 */
public class Client {
    public void send(String ip, int port) throws IOException {
        ExecutorService executorService = Executors.newFixedThreadPool(5);
        //创建联接
        Socket socket = new Socket(ip, port);
        executorService.execute(new HandleReadRunnable(socket));
        executorService.execute(new HandleWriteRunnable(socket));

    }

    public static void main(String[] args) {
        try {
            new Client().send("192.168.1.141",5000);
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
}


/**
 * 服务端
 */
public class Server {
    private final int PROT = 5000;

    public void start(){
        ExecutorService executorService = Executors.newFixedThreadPool(5);
        System.out.println("服务器启动了");
        //创建服务器连接
        try( ServerSocket serverSocket = new ServerSocket(PROT)) {
            Scanner scanner = new Scanner(System.in);
            //接收客户端连接
           while (true){
               Socket socket = serverSocket.accept();
               System.out.println("客户端连接了"+socket.getInetAddress());
                //获得输入流,得到客户端发送的消息
               executorService.execute(new HandleWriteRunnable(socket));
               executorService.execute(new HandleReadRunnable(socket));
           }
        } catch (IOException e) {
            e.printStackTrace();
        }
    }

    public static void main(String[] args) {
        new Server().start();
    }
}

2、写一个死锁案例(两个线程都持有对方需要的锁,又有需要对方持有的锁)

/**
 * 写一个死锁案例(两个线程都持有对方需要的锁,又有需要对方持有的锁)
 */

public class Demo1 {
    Object lock1 = new Object();
    Object lock2 = new Object();

    public static void main(String[] args) throws InterruptedException {
        new Demo1().test1();
    }

    public void test1() throws InterruptedException {
        Thread a = new Thread(()->{
            synchronized (lock1){
                for (int i = 0; i <1000 ; i++) {
                    try {
                        Thread.sleep(2000);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }
                synchronized (lock2){
                    System.out.println("aaaaaaa");
                }
            }
        });

        Thread b = new Thread(()->{
            synchronized (lock2){
                for (int i = 0; i <1000 ; i++) {
                    try {
                        Thread.sleep(2000);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }
                synchronized (lock1){
                    System.out.println("bbbbbbb");
                }
            }
        });
        a.start();
        b.start();
    }
}

3、写一个线程交替输出案例(A线程循环输出A,B线程循环输出B,出现ABABABAB… )

package com.hopu.Demo;

/**
 * 写一个线程交替输出案例(A线程循环输出A,B线程循环输出B,出现ABABABAB..... )
 */
public class Demo2 {
    int num;//用于交替输出
    public static final Object lock = new Object();

    public static void main(String[] args) {
        Demo2 demo2 = new Demo2();
        new Thread(() -> {
            try {
                demo2.test1(0);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }, "A").start();
        new Thread(() -> {
            try {
                demo2.test1(1);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }, "B").start();
    }

    public void test1(int number) throws InterruptedException {
        for (int i = 0; i < 10; i++) {
            synchronized (lock) {
                while (num % 2 != number) {
                    lock.wait();
                }
                num++;
                System.out.println(Thread.currentThread().getName());
                lock.notifyAll();
            }
        }
    }
}

  网络协议 最新文章
使用Easyswoole 搭建简单的Websoket服务
常见的数据通信方式有哪些?
Openssl 1024bit RSA算法---公私钥获取和处
HTTPS协议的密钥交换流程
《小白WEB安全入门》03. 漏洞篇
HttpRunner4.x 安装与使用
2021-07-04
手写RPC学习笔记
K8S高可用版本部署
mySQL计算IP地址范围
上一篇文章      下一篇文章      查看所有文章
加:2021-12-15 18:40:09  更:2021-12-15 18:40:34 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年10日历 -2024/10/5 7:11:30-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码