linux线程 | 同步与互斥 | 全解析信号量、环形生产消费者模型

news2024/10/20 23:05:34

        前言: 本节内容讲述linux下的线程的信号量, 我们在之前进程间通信那里学习过一部分信号量, 但是那个是systemV版本的信号量,是以进程间通信的视角谈的。 但是本篇内容会以线程的视角谈一谈信号量。 

        ps:本篇内容建议学习了生产者消费者模型的友友们进行观看哦。

目录

信号量

信号量接口

初始化信号量

环形队列的生产消费模型

创建文件

RingQueue.h

main.cpp

多生产和多消费的循环队列

RingQueue

main.cpp


 

信号量

         我们之前在快速写一个生产者消费者模型的时候, 我们的交易场所设置的一个队列。 当时我们的queue当作整体使用的, 但是queue只有一份, 所以我们加锁保证了这一份资源的安全性 但是, 我们也要认识到, 共享资源可以被看到多份。 就比如我们今天有一个全局的数组, 有三个线程。 假如数组一共有300个元素,这个数组分为左中右三份。 中间的线程只能访问数组的中间的一份。 左边线程访问左边一份, 右边线程访问右边一份。 因为我们线程各自访问不同的区域, 所以多线程可以同时访问这个数组吗?答案是可以的。

        但是如果是四个线程呢? 我们只有三份资源, 但是要给四个线程使用。 所以呢, 我们就只能放进来三个线程来访问这三个资源。 那么我们的共享资源为了保证并发度, 那么就保持了分成了几份资源, 那么就允许多少个线程进来并发执行。 所以, 为了保护我们的临界资源, 就引入了信号量。 而理解信号量的切入点就是:共享资源也可以被看成多份。

        信号量, 也叫做信号灯, 这把计数器用来描述临界资源中资源数目是多少。 就比如电影院, 里面有很多很多的座位。 假如有100个座位。 我们想要抢占其中的一个座位, 就要提前买一下票!所以, 买票的本质就是对电影院座位资源的预定机制。 那么我们可不可以使用一个变量替换掉这个计数器呢? 答案是不可以, 因为普通变量的++或者--其实不是原子的 多线程并发访问时就会出现问题!!!所以, 我们就要使用一个支持pv操作的原子的计数器——信号量.

        问题是, 信号量的本质就是一把计数器, 而计数器的本质是什么呢? 本质就是临界资源的数量。 所以, 一旦我们用p操作, 我们p操作之后,还用判断资源是就绪的吗? 答案是不需要, 因为只要申请成功了, 就一定有你的。 申请不成功的, 就要去信号量下面去等待了, 所以p操作当中, p操作只要成功, 那么就不需要判断资源就没就绪!所以, 这把计数器的本质是什么? 我们说是用来描述资源数目的, 把资源是否就绪放在了临界区之外申请信号量时, 其实就间接的已经在做判断了!!!

  •         我们之前使用的互斥锁, 其实就可以理解为一个二元信号量, 这个信号量只能为零为一。 一个线程拿到锁后, 只要不释放, 其他的线程无法再次拿到锁!!!

信号量接口

初始化信号量

        第一个参数是信号量对象的地址。 第二个参数代表表示的是线程共享还是进程共享。默认为零,是线程共享。 非零表示进程共享。 第三个参数就是设定的信号量的初始值。

        destroy表示的是清空信号量。         

        sem_wait功能是等待信号量, 将信号量的值减一。 也就是P操作

        sem_post的功能是发布信号量,表示资源使用完毕,该归还资源了, 将信号量加一。也就是V操作。

环形队列的生产消费模型

        我们学习环形队列的生产消费模型的目的是为了理解我们的信号量,以及熟悉一下我们信号量的使用。 其实这里的环形队列就类似于我们的循环队列。 上面的head就是放一个数据就向前走一格子,放一个数据就向前走一个格子。 但是和循环队列不同的是当我们放满的时候两个指针还是指向同一个位置, 因为我们有信号量计数器, 所以不利用指针的指向位置判断空和满!

        我们这里以单生产和单消费为例(因为多生产多消费要复杂, 先实现单生产单消费的代码再来考虑多生产和多消费):

 

        生产和消费只要没有访问同一个位置, 我们就能让他们同时是生产和消费。 也就是说, 我们生产和消费必须遵守三个原则:

  •         1、指向同一个位置的时候, 不能同时访问。
  •         2、消费者不能超过生产者。
  •         3、生产者不能超过消费者一个圈。

那么, 当生产者和消费者在这个追逐的过程中, 什么时候才会指向同一个位置呢? 其实就是生产者生产满了的时候。——》

  •         不空和不满的时候, 指向的一定是不同的位置, 两者可以同时访问!!
  •         为空的时候:只能生产者访问!!
  •         为满的时候:只能消费者访问!!
  •         生产者关注什么资源呢?——还有多少剩余空间!
  •         消费者关注什么资源呢?——还有多少剩余数据!

        在定义信号量的时候,我们可以定义两把信号量, 一个叫做SpaceSem,一个叫做DataSem。 最开始的时候, SpaceSem = MAX DataSem = 0;然后对于生产者来说, 要P(SpaceSem), 就代表生产者生产数据让SpaceSem减一,同时V(DataSem)。 对于消费者来说就P(DataSem), 就代表消费者消耗掉一个数据。

        但是, 当我们的队列为空的时候, 一定是生产者先执行, 然后消费者再来消费。 当生产者生产到SpaceSem为零的时候, 生产者就不能再消费了。下面开始实现代码:

创建文件

        先创建好三个文件, 一个makefile, 一个RingQueue.h用来实现环形队列这个交易场所。然后main.cpp用来实现生产者消费者线程以及生产动作和消费动作

RingQueue.h

        首先我们思考一下我们定义的这个生产消费模型里面要有什么——首先一定要有一个环形队列, 这里我们用vector来模拟。 然后我们要规定环形队列的大小, 所以要有一个变量来规定这个大小, 我们这里定义一个cap_变量。 然后要有一个消费者下标, 一个生产者下标来表示生产者和消费者生产或消费的位置。 最后还要定义两把计数器(信号量)让生产者消费者来预定临界资源。所以代码如下:

#include<iostream>
using namespace std;
#include<vector>
#include<pthread.h>
#include<semaphore.h>


const static int defaultcap = 5; //用来初始化cap_


template<class T>
class RingQueue
{
public:

private:
    vector<T> ringqueue_;
    int cap_;   //队列的容量大小

    int c_step_;    //消费者下标
    int p_step_;    //生产者下标

    //注意, 信号量不能保证生产者之间的互斥, 也不能保证消费者之间的互斥。  
    sem_t cdata_sem_;   //消费者关注的数据资源
    sem_t pspace_sem_;  //生产者关注的空间资源



};

        环形生产消费模型里面有什么方法呢? 首先, 一定要有构造和析构。 然后构造就是对vector, 计数器来进行初始化。 析构同理。 所以代码如下:

#include<iostream>
using namespace std;
#include<vector>
#include<pthread.h>
#include<semaphore.h>


const static int defaultcap = 5;


template<class T>
class RingQueue
{
public:
    RingQueue(int cap = defaultcap)
        :cap_(cap)
        , ringqueue_(cap)
        , c_step_(0)
        , p_step_(0)
    {
        sem_init(&cdata_sem_, 0, 0); 
        sem_init(&pspace_sem_, 0, cap);
    }

    ~RingQueue()
    {
        sem_destroy(&cdata_sem_);
        sem_destroy(&pspace_sem_);
    }
private:
    vector<T> ringqueue_;
    int cap_;   //队列的容量大小

    int c_step_;    //消费者下标
    int p_step_;    //生产者下标

    //注意, 信号量不能保证生产者之间的互斥, 也不能保证消费者之间的互斥。  
    sem_t cdata_sem_;   //消费者关注的数据资源
    sem_t pspace_sem_;  //生产者关注的空间资源

};

         然后还有两个方法就是pop和push, push用来向交易场所中加入数据, pop用来从交易场所中拿数据:


    void Push(const T& in)
    {
        //生产数据先要申请信号量空资源
        P(&pspace_sem_);

        ringqueue_[p_step_] = in;
        //维持环形特征
        p_step_++;
        p_step_ %= cap_;

        V(&cdata_sem_);
    }

    //在为空和为满的时候就表现出了局部性的互斥特征!!为空的时候,生产者先运行;为满的时候, 消费者先运行——》这不就是生产和消费具有一定的顺序性, 这是局部性的同步。
    //如果不为空并且不为满,那么我们对应的step两个下标的值一定是不一样的。 两个就叫做并发运行!!!和之前将讲解的不太一样, 
    void Pop(T* out)  //利用指针将数据从队列里面拿出来
    {
        P(&cdata_sem_);


        *out = ringqueue_[c_step_];
        c_step_++;
        c_step_ %= cap_;

        V(&pspace_sem_);
    }

        这里我们需要思考一下, 首先在为空和为满的时候, 是不是这个时候只能消费者消费, 或者生产者进行生产? 而这,不就是表现出了局部性的互斥特征!!然后为空的时候,生产者先运行;为满的时候, 消费者先运行——》这不就是生产和消费具有一定的顺序性, 这是局部性的同步。        

        然后如果不为空并且不为满,那么我们对应的step两个下标的值一定是不一样的。 两个就叫做并发运行!!!        

main.cpp

        主函数比较简单, 分为三个板块。 一个板块是主函数, 用来创建线程, 以及交易场所。 然后第二三个板块用来定义生产者和消费者线程需要执行的代码。 代码如下:
 

#include"RingQueue.h"

#include<unistd.h>
#include<ctime>
#include<iostream>
using namespace std;



void* Productor(void* args)
{
    RingQueue<int>* rq = static_cast<RingQueue<int>*>(args);
    
    while (true)
    {
        //获取数据
        int data = rand() % 10 + 1;

        //生产数据
        rq->Push(data);
        cout << "Productor data done, data is: " << data << endl;
    }

    return nullptr;
}

void* Consumer(void* args)
{
    RingQueue<int>* rq = static_cast<RingQueue<int>*>(args);

    while (true)
    {
        sleep(1);//这里我们的消费者是一秒消费一次。 要知道, 我们的线程跑的是很快的, 所以
//运行后其实生产者一瞬间就能把交易场所打满, 然后就是一秒消费一次, 生产一次!!
        //消费数据
        int data = 0;
        rq->Pop(&data); 

        //处理数据
        cout << "Comsumer data done, data is: " << data << endl; 


    }
    return nullptr;
}


int main()


    //常见循环队列
    RingQueue<int>* rq = new RingQueue<int>();


    //创建线程
    pthread_t c, p;
    
    //运行线程
    pthread_create(&c, nullptr, Productor, rq);
    pthread_create(&p, nullptr, Consumer, rq);

    //等待线程
    pthread_join(c, nullptr);
    pthread_join(p, nullptr);


    //销毁循环队列
    delete rq;

    return 0;
}

运行结果:

多生产和多消费的循环队列

        多生产和多消费要维护它们的互斥关系, 消费者和消费者之间也要维护它们的互斥关系。 如何变成支持多生产和多消费的呢? 答案是加锁!现在我们已经有了生产者和消费者之间的互斥关系。 那么我们只需要再利用锁将生产者之间, 以及消费者之间建立起互斥关系, 就能满足三种关系相互互斥, 就能满足资源的安全。 

RingQueue

#include<iostream>
using namespace std;
#include<vector>
#include<pthread.h>
#include<semaphore.h>


const static int defaultcap = 5;


template<class T>
class RingQueue
{
private:
    void P(sem_t* sem)
    {
        sem_wait(sem);
    }

    void V(sem_t* sem)
    {
        sem_post(sem);
    }

//第一个改动是封装了锁的加锁和解锁

    void Lock(pthread_mutex_t& mutex)
    {
        pthread_mutex_lock(&mutex);
    }

    void Unlock(pthread_mutex_t& mutex)
    {
        pthread_mutex_unlock(&mutex);
    }

public:
    RingQueue(int cap = defaultcap)
        :cap_(cap)
        , ringqueue_(cap)
        , c_step_(0)
        , p_step_(0)
    {
        sem_init(&cdata_sem_, 0, 0); 
        sem_init(&pspace_sem_, 0, cap);
    }

    void Push(const T& in)
    {
        //生产数据先要申请信号量空资源
        P(&pspace_sem_);
        Lock(c_mutex_); //加锁在申请信号量之后比较好

        ringqueue_[p_step_] = in;
        //维持环形特征
        p_step_++;
        p_step_ %= cap_;

        Unlock(c_mutex_);
        V(&cdata_sem_);
    }

    //在为空和为满的时候就表现出了局部性的互斥特征!!为空的时候,生产者先运行;为满的时候, 消费者先运行——》这不就是生产和消费具有一定的顺序性, 这是局部性的同步。
    //如果不为空并且不为满,那么我们对应的step两个下标的值一定是不一样的。 两个就叫做并发运行!!!和之前将讲解的不太一样, 
    void Pop(T* out)  //利用指针将数据从队列里面拿出来
    {
        P(&cdata_sem_);
        Lock(p_mutex_);

        *out = ringqueue_[c_step_];
        c_step_++;
        c_step_ %= cap_;

        Unlock(p_mutex_);
        V(&pspace_sem_);
    }


    ~RingQueue()
    {
        sem_destroy(&cdata_sem_);
        sem_destroy(&pspace_sem_);
    }
private:
    vector<T> ringqueue_;
    int cap_;   //队列的容量大小

    int c_step_;    //消费者下标
    int p_step_;    //生产者下标

    //注意, 信号量不能保证生产者之间的互斥, 也不能保证消费者之间的互斥。  
    sem_t cdata_sem_;   //消费者关注的数据资源
    sem_t pspace_sem_;  //生产者关注的空间资源


    //定义两把锁
    pthread_mutex_t c_mutex_;
    pthread_mutex_t p_mutex_;

};

上面有三个地方改动, 一个是定义了两把锁变量。 第二个就是封装了加锁和解锁的方法。 第三个就是将push和pop里面申请信号量之后的代码加锁变成了临界区。 这里有一个要思考的点就是为什么加锁要在申请信号量之后? 我们可以这样想, 如果我们加锁在信号量之前, 那么我们的所有的生产者之间在执行push的时候就只能串行执行push。 但是如果我们加锁在信号量之后, 我们的生产者之间就能只串行申请信号量或者串行后面的代码。 就可以令申请信号量和执行后面的代码并行起来!!!所以效率就会提高!!!

main.cpp

主函数改动不大, 就是利用了for循环创建线程:

#include"RingQueue.h"

#include<unistd.h>
#include<ctime>
#include<iostream>
using namespace std;

void* Productor(void* args)
{
    RingQueue<int>* rq = static_cast<RingQueue<int>*>(args);
    
    while (true)
    {
        //获取数据
        int data = rand() % 10 + 1;

        //生产数据
        rq->Push(data);
        cout << "Productor data done, data is: " << data << endl;
    }

    return nullptr;
}

void* Consumer(void* args)
{
    RingQueue<int>* rq = static_cast<RingQueue<int>*>(args);

    while (true)
    {
        sleep(1);
        //消费数据
        int data = 0;
        rq->Pop(&data); 

        //处理数据
        cout << "Comsumer data done, data is: " << data << endl; 
    }
    return nullptr;
}

int main()
{
    //多生产和多生产之间要维护它们的互斥关系, 消费者和消费者之间也要维护它们的互斥关系。 如何变成支持多生产和多消费呢?
    //答案是加锁, 现在我们已经有了生产者和消费者之间的互斥关系。 那么我们只需要再利用锁将生产者之间,以及消费者之间建立起
    //互斥关系, 就能满足三种关系相互互斥, 就能满足资源的安全

    //常见循环队列
    RingQueue<int>* rq = new RingQueue<int>();

    //创建线程
    pthread_t c[3], p[5];
    
    //运行线程
    for (int i = 0; i < 3; i++)
    {
        pthread_create(c + i, nullptr, Productor, rq);
        sleep(1);
    }

    for (int i = 0; i < 5; i++)
    {
        pthread_create(p + i, nullptr, Consumer, rq);
        sleep(1);
    }
    
    for (int i = 0; i < 3; i++)
    {
        //等待线程
        pthread_join(c[i], nullptr);
        sleep(1);
    }

    for (int i = 0; i < 5; i++)
    {
        pthread_join(p[i], nullptr);
        sleep(1);
    }

    //销毁循环队列
    delete rq;

    return 0;
}

然后看一下运行结果(注意, 只看运行结果是看不出来的, 所以这里我们使用了一下监视脚本)

我们可以看右边就会看到我们的线程每一秒就会多一个!!这就是我们的多生产和多消费的情况。 

  ——————以上就是本节全部内容哦, 如果对友友们有帮助的话可以关注博主, 方便学习更多知识哦!!!   

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2219619.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

集合collection和泛型

collection可以直接打印内容&#xff0c;而不是地址&#xff0c;内部已经重写了。 List家族&#xff1a; package com.itheima.d6_collection_update_delete;import java.util.ArrayList; import java.util.Iterator; import java.util.List;/**目标&#xff1a;研究集合遍历并…

解决关于HTML+JS + Servlet 实现前后端请求Session不一致的问题

1、前后端不分离情况 在处理session过程中&#xff0c;如果前后端项目在一个容器中&#xff0c;session是可以被获取的。例如如下项目结构&#xff1a; 结构 后端的代码是基本的设置值、获取值、销毁值的内容&#xff1a; 运行结果 由此可见&#xff0c;在前后统一的项目中&a…

Sign Language Dataset: 聋哑人手语数据集(猫脸码客 第209期)

Sign Language Dataset: 聋哑人手语数据集 摘要&#xff1a;手语是聋哑人群体进行沟通交流的重要工具&#xff0c;通过手势、动作及面部表情的组合表达复杂的思想和情感。随着计算机视觉和人工智能技术的发展&#xff0c;聋哑人手语数据集在促进手语识别、翻译和交互系统开发中…

计算机指令系统,打个结~

计算机指令系统是计算机硬件与软件之间的桥梁&#xff0c;它定义了计算机能够执行的各种操作。一个完善的指令系统不仅影响着计算机的性能&#xff0c;还直接决定了计算机能够完成的任务种类和复杂度。本文将从计算机指令的基本概念出发&#xff0c;探讨指令系统的分类、常见指…

第13篇:无线与移动网络安全

目录 引言 13.1 无线网络的安全威胁 13.2 无线局域网的安全协议 13.3 移动通信中的安全机制 13.4 蓝牙和其他无线技术的安全问题 13.5 无线网络安全的最佳实践 13.6 总结 第13篇&#xff1a;无线与移动网络安全 引言 无线和移动网络的发展为我们的生活带来了极大的便利…

cisco网络安全技术第3章测试及考试

测试 使用本地数据库保护设备访问&#xff08;通过使用 AAA 中央服务器来解决&#xff09;有什么缺点&#xff1f; 试题 1选择一项&#xff1a; 必须在每个设备上本地配置用户帐户&#xff0c;是一种不可扩展的身份验证解决方案。 请参见图示。AAA 状态消息的哪一部分可帮助…

Java程序设计:spring boot(2)

目录 1 Spring MVC 零配置创建与部署 1.1 创建Spring MVC Web⼯程 1.2 pom.xml 添加坐标相关配置 1.3 添加源代码 1.4 添加视图 1.5 SpringMVC 配置类添加 1.6 入口文件代码添加 1.7 部署与测试 2 Spring Boot 概念&特点 2.1 框架概念 2.2 框架特点 2.3 Spring…

微知-如何临时设置服务器风扇转速?(ipmitool raw 0x30 0x30 0x02 0xff 0x40)

服务器风扇可以通过PWM输出来控制转速。 设置方式 设置单次PWM ipmitool raw 0x30 0x30 0x02 0xff 0x40如果要持续设置需要类似while循环持续输出&#xff1a; while true; do ipmitool raw 0x30 0x30 0x02 0xff 0x64; done > /dev/null参数说明&#xff1a; 其他参数&a…

Qt(简介)

1. Qt简介 Qt是一个基于C的图形用户界面&#xff08;GUI&#xff09;框架&#xff0c;可以开发可视化人机交互程序&#xff0c;但是这并不是Qt的全部。Qt除了可以绘制漂亮的界面外&#xff0c;还包含很多其他的功能&#xff1a;多线程、数据库、图像处理、音视频处理、网络通信…

CentOS安装NVIDIA驱动、CUDA以及nvidia-container-toolkit

0.提前准备 0.1.更新yum源&#xff08;以阿里为例&#xff09; 0.1.1 备份当前的yum源 mv /etc/yum.repos.d/CentOS-Base.repo /etc/yum.repos.d/CentOS-Base.repo.backup 0.1.2 下载新的CentOS-Base.repo 到/etc/yum.repos.d/ CentOS 5 wget -O /etc/yum.repos.d/CentOS-Base…

【LeetCode每日一题】——523.连续的子数组和

文章目录 一【题目类别】二【题目难度】三【题目编号】四【题目描述】五【题目示例】六【题目提示】七【解题思路】八【时间频度】九【代码实现】十【提交结果】 一【题目类别】 前缀和 二【题目难度】 中等 三【题目编号】 523.连续的子数组和 四【题目描述】 给你一个…

github学生认证(白嫖copilot)-Why are you not on campus?不在校园内

详细申请操作流程可参考如下博文&#xff1a;从0开始的github学生认证并使用copilot教程&#xff08;超详细&#xff01;&#xff09;_copilot学生认证-CSDN博客 在此记录解决“Why are you not on campus?”提示的方法&#xff1a; 当出现这个提示时&#xff0c;说明在选择学…

2024年底蓝奏云最新可用API接口列表 支持优享版 无需手动抓取cookie

Lanzou Pro V1 接口列表 API状态版本路由获取文件与目录✅^1.0.1/v1/getFilesAndDirectories?url{}&page{}获取目录✅^1.0.0/v1/getDirectory?url{}获取文件✅^1.0.1/v1/getFiles?url{}&page{}搜索文件✅^1.0.0/v1/searchFile?url{}&wd{}依Id解析✅^1.0.2/v1/…

从0-1实战演练后台管理系统 (2)从零开始:Pure Admin 环境搭建完全指南,小白也能轻松掌握!

在开始使用Pure Admin之前&#xff0c;我们需要先了解一下Pure Admin是什么? vue-pure-admin (opens new window)是一款开源完全免费且开箱即用的中后台管理系统模版。完全采用 ECMAScript 模块&#xff08;ESM&#xff09;规范来编写和组织代码&#xff0c;使用了最新的 Vue3…

【原创】java+ssm+mysql计算机等级考试网系统设计与实现

个人主页&#xff1a;程序猿小小杨 个人简介&#xff1a;从事开发多年&#xff0c;Java、Php、Python、前端开发均有涉猎 博客内容&#xff1a;Java项目实战、项目演示、技术分享 文末有作者名片&#xff0c;希望和大家一起共同进步&#xff0c;你只管努力&#xff0c;剩下的交…

【ESP32-IDFV5.3.1开发】带SSL的MQTT-demo连接教程

目录 1.VSCODE以及IDF环境配置(略) 2.准备demo 2.1打开VSCODE&#xff0c;主菜单创建示例 找到SSL对应demo&#xff0c;点击创建&#xff0c;并成功创建项目&#xff0c;点击编译&#xff0c;显示编译成功即可以下一步。 确认该demo支持的开发板是你手上的开发板 3.修改demo配…

论文略读:Graph Neural Processes for Spatio-Temporal Extrapolation

2023 KDD 1 背景 时空图数据&#xff0c;无论是交通数据&#xff0c;还是空气质量数据&#xff0c;气候数据。在理想情况下&#xff0c;这些数据应该是细粒度的。 但由于高昂的成本&#xff0c;在某一地区部署和维护足够多的传感器往往是不现实的 ——>因此&#xff0c;许…

Android Studio 的 Gradle 任务列表只显示测试任务

问题现象如下&#xff1a; 问题原因&#xff1a; 这是因为Android Studio 设置中勾选了屏蔽其他gradle任务的选项。 解决方法&#xff1a; File -> Settings -> Experimental 取消勾选Only include test tasks in the Gradle task list generated during Gradle Sync&…

【纯前端excel导出】vue2纯前端导出excel,使用xlsx插件,修改样式、合并单元格

官网&#xff1a; 1、xlsx-js-style xlsx-js-style | xlsx-js-style homepage 2、xlsx SheetJS 中文网 一、使用第三方插件 1、安装 npm install xlsx-js-style 2、引入 import xlsx from xlsx-js-style xlsx插件是基础的导出&#xff0c;不可以修改样式&#xff0c;直接xlsx-s…

Netty通信过程中编解码技术

Netty通信过程中编解码技术 粘包/拆包问题为什么有拆包/粘包MTU最大传输单元和MSS最大分段大小滑动窗口TCP报文如何确保数据包按次序到达且不丢数据Nagle算法 拆包/粘包解决方案 Netty实现自定义通信协议通信协议设计Netty如何实现自定义通信协议 粘包/拆包问题 如何获取一个完…