Linux线程(四) 生产者消费者模型

news2025/1/15 13:07:45

目录

一、什么是生产者消费者模型

基本概念

优点以及应用场景

二、 基于阻塞队列的生产者消费者模型

三、POSIX信号量

四、基于环形队列的生产消费模型


一、什么是生产者消费者模型

        Linux下的生产者消费者模型是一种经典的多线程或多进程编程设计模式,它用于解决资源访问的同步问题,特别是在涉及任务分配、数据处理和资源共享的场景中。

基本概念

生产者:负责生成数据项并将其放入共享的缓冲区(队列)。当缓冲区满时,生产者可能需要等待(阻塞)直到有空间可用。

消费者:从缓冲区中取出数据项进行处理。如果缓冲区为空,消费者可能需要等待(阻塞)直到有新数据产生。

生产者和消费者彼此之间不直接通讯,而通过阻塞队列来进行通讯,所以生产者生产完数据之后不用等待消费者处理,直接扔给阻塞队列,消费者不找生产者要数据,而是直接从阻塞队列里取,阻塞队列就相当于一个缓冲区,平衡了生产者和消费者的处理能力。这个阻塞队列就是用来给生产者和消费者解耦的。

优点以及应用场景

生产者消费者模型作为一种经典的并发设计模式,在软件开发中特别是涉及多线程或多进程协作的场景下,展现出诸多优势。

解耦:生产者和消费者之间通过共享缓冲区(如队列)进行间接通信,减少了直接的依赖关系,使得生产者和消费者的代码可以独立开发和维护,提高了模块的复用性和系统的灵活性。

支持并发:生产者和消费者通常作为独立的执行单元运行,可以并行工作,充分利用多核处理器的计算能力,提升系统整体的吞吐量和响应速度。

平衡资源利用:通过调整缓冲区的大小和管理生产者与消费者的数量,可以有效平衡生产速率和消费速率,防止生产过剩导致资源浪费或者消费过快导致资源饥饿,从而优化系统性能。

应用场景:

生产者消费者模型广泛应用于各种领域,如网络通信中的数据包处理、数据库的异步写入、GUI应用中的事件处理系统、多线程下载和处理等,任何需要解耦数据生产与数据消费过程的场景都可以考虑使用这一模式。 

二、 基于阻塞队列的生产者消费者模型

        在这个模型中,阻塞队列扮演了生产者和消费者之间的中介角色,它负责存储生产者产生的数据,并安全地传递给消费者处理。关键在于,阻塞队列能够自动管理同步问题,确保线程安全,同时提供阻塞机制来平衡生产与消费的速度。

阻塞队列属于仓库这一临界资源,而同一时刻只能有一个线程进入阻塞队列进行操作,所以要用到互斥锁,同时还要思考如果是消费者该如何知道有东西可以买了呢,如果是生产者如何知道仓库的东西不够了需要生产呢,这个时候就需要两个条件变量push_cond和pop_cond

关于条件变量在上篇文章中讲过,可以参考:

Linux线程(三)死锁与线程同步

push_cond

当生产者将阻塞队列放满时,就需要等待消费者消费完来唤醒生产者继续生产。

pop_cond

当消费者把队列消费空时,消费者会等待生产者往阻塞队列加资源后来唤醒消费者继续消费。 

接下来我们来实现一个基于阻塞队列的生产者消费者模型

访问阻塞队列一定会涉及到加锁,我们首先可以设计一个LockGuard(RAII)思想,利用类出作用域自动销毁来实现解锁,防止忘记解锁造成死锁。

LockGuard.hpp

#pragma once
#include <pthread.h>

class Mutex
{
private:
    pthread_mutex_t* _mutex;

public:
    Mutex(pthread_mutex_t* lock)
    :_mutex(lock)
    {
    }
    void Lock()
    {
        pthread_mutex_lock(_mutex);
    }
    void Unlock()
    {
        pthread_mutex_unlock(_mutex);
    }
    ~Mutex()
    {
    }
};

class LockGuard
{
private:
    Mutex _mutex;
public:
    LockGuard(pthread_mutex_t *lock)
    :_mutex(lock)
    {
        _mutex.Lock();
    }
     ~LockGuard()
    {
        _mutex.Unlock();
    }
};

随后我们来实现一个任务类,模仿消费者拿到资源:

Task.hpp:

#pragma once
#include <iostream>
#include <string>

const int defaultvalue = 0;

enum
{
    ok = 0,
    div_zero,
    mod_zero,
    unknow
};

const std::string opers = "+-*/%)(&";

class Task
{
public:
    Task()
    {
    }
    Task(int x, int y, char op)
        : data_x(x), data_y(y), oper(op), result(defaultvalue), code(ok)
    {
    }
    void Run()
    {
        switch (oper)
        {
        case '+':
            result = data_x + data_y;
            break;
        case '-':
            result = data_x - data_y;
            break;
        case '*':
            result = data_x * data_y;
            break;
        case '/':
        {
            if (data_y == 0)
                code = div_zero;
            else
                result = data_x / data_y;
        }
        break;
        case '%':
        {
            if (data_y == 0)
                code = mod_zero;
            else
                result = data_x % data_y;
        }

        break;
        default:
            code = unknow;
            break;
        }
    }
    void operator()()
    {
        Run();
    }
    std::string PrintTask()
    {
        std::string s;
        s = std::to_string(data_x);
        s += oper;
        s += std::to_string(data_y);
        s += "=?";

        return s;
    }
    std::string PrintResult()
    {
        std::string s;
        s = std::to_string(data_x);
        s += oper;
        s += std::to_string(data_y);
        s += "=";
        s += std::to_string(result);
        s += " [";
        s += std::to_string(code);
        s += "]";

        return s;
    }
    ~Task()
    {
    }

private:
    int data_x;
    int data_y;
    char oper; // + - * / %

    int result;
    int code; // 结果码,0: 结果可信 !0: 结果不可信,1,2,3,4
}; 

随后我们来实现一个阻塞队列,要注意一个时刻只能有一个线程访问,所以再push操作和pop操作时要加锁。

block_queue.hpp:

#pragma once
#include<iostream>
#include<queue>
#include<pthread.h>
#include"LockGuard.hpp"

const int defaultcap=5;//默认容量为5
template<class T>
class block_queue
{
private:
    std::queue<T> _q;
    int _capacity;   //_q.size() == _capacity, 满了,不能在生产,_q.size() == 0, 空,不能消费了
    pthread_mutex_t _mutex;
    pthread_cond_t _push_cond;  //给生产者
    pthread_cond_t _pop_cond;  //给消费者
    // int _consumer_water_line;  // _consumer_water_line = _capacity / 3 * 2
    // int _productor_water_line; // _productor_water_line = _capacity / 3
public:
    block_queue(int cap=defaultcap)
    :_capacity(cap)
    {
        pthread_mutex_init(&_mutex,nullptr);
        pthread_cond_init(&_push_cond,nullptr);
        pthread_cond_init(&_pop_cond,nullptr);
    }
     bool IsFull()
    {
        return _q.size() == _capacity;
    }
    bool IsEmpty()
    {
        return _q.size() == 0;
    }
    void Push(const T &in)
    {
        LockGuard lockguard(&_mutex);
        while(IsFull())
        pthread_cond_wait(&_push_cond,&_mutex);

        _q.push(in);
        //通知消费者可以消费了
         // if(_q.size() > _productor_water_line) pthread_cond_signal(&_c_cond);  // 也可以是当资源数量大于指定阈值时再通知
        pthread_cond_signal(&_pop_cond);
        
    }

    void Pop(T *out)//要取出任务
    {
        LockGuard lockgugrd(&_mutex);
        while(IsEmpty())
        pthread_cond_wait(&_pop_cond,&_mutex);

        *out=_q.front();
        _q.pop();
        //通知生产者可以生产了
        // if(_q.size() < _consumer_water_line) pthread_cond_signal(&_p_cond);      //也可以是当资源数量大于指定阈值时再通知
        pthread_cond_signal(&_push_cond);
    }
    ~block_queue()
    {
        pthread_mutex_destroy(&_mutex);
        pthread_cond_destroy(&_push_cond);
        pthread_cond_destroy(&_pop_cond);
    }

};

makefile文件,当然也可手动生成可执行文件,使用这个较为方便

test_block:Main.cc
	g++ -o  $@ $^ -lpthread -std=c++11

.PHONY:clean
clean:
	rm -f test_block

使用Main.cc来测试这个模型:

#include"block_queue.hpp"
#include"Task.hpp"
#include<pthread.h>
#include<ctime>
#include<sys/types.h>
#include<unistd.h>


void *consumer(void *args)
{
    block_queue<Task> *bq=static_cast<block_queue<Task>* >(args);
    while(true)
    {
        sleep(1);
        Task t;

        //取出任务
        bq->Pop(&t);
        t();//运行任务

        std::cout<<"consumer data: "<<t.PrintResult()<<std::endl;
    }
    return nullptr;
}

void *producror(void *args)
{
    block_queue<Task> *bq=static_cast<block_queue<Task>*>(args);

    while(true)
    {
        sleep(1);
        int x=rand()%10;
        usleep(rand()%123);
        int y=rand()%10;
        usleep(rand()%1234);
        char oper=opers[rand()%(opers.size())];
        Task t(x,y,oper);
        std::cout<<"productor data: "<<t.PrintTask()<<std::endl;
        bq->Push(t);
    }
    return nullptr;
}

int main()
{
    srand((uint16_t)time(nullptr) ^ getpid() ^ pthread_self()); // 只是为了形成更随机的数据
    block_queue<Task> *bq=new block_queue<Task>();
    pthread_t c,p;
    pthread_create(&c,nullptr,consumer,bq);
    pthread_create(&p,nullptr,producror,bq);
    pthread_join(c,nullptr);
    pthread_join(p,nullptr);
    return 0;
}

运行结果如下

可以看到生产者每生产一个,消费者就拿到一个。

如果让生产者不休眠

可以看到消费者将阻塞队列填满,消费者取队首元素来执行。

总体流程就是:

生产过程:生产者创建数据项,并尝试将数据放入阻塞队列。如果队列已达到其容量限制,生产者的push()操作将被阻塞,直到队列中有空间可以添加新数据。

消费过程:消费者从阻塞队列中取出数据项进行处理。当队列为空时,消费者的pop()操作也会被阻塞,直到有新的数据被生产者放入队列。

通知与唤醒:一旦队列状态发生变化(例如有数据被放入或移出),阻塞队列会自动唤醒相应等待的线程,实现高效且线程安全的同步。 

三、POSIX信号量

POSIX信号量和SystemV信号量作用相同,都是用于同步操作,达到无冲突的访问共享资源目的。 但POSIX可以用于线程间同步。

在传统的基于阻塞队列的生产者消费者模型中,虽然使用阻塞队列自身可以避免一些复杂的同步问题,但确实存在这样一个情况:当生产者试图向满队列添加数据或消费者试图从空队列中取数据时,它们都需要对整个队列进行加锁,这实际上导致了生产者和消费者之间不必要的锁竞争。

比如生产者消费者模型,生产者只需要关注空间是否足够生产,消费者只需要关注资源是否足够消费,所以开始的时候生产者的信号量就是队列的大小,消费者的信号量就是0,当生成者生产一个资源,生产者信号量-1,消费者+1;当消费者消费一个资源, 生产者信号量+1,消费者-1。

使用POSIX信号量确实可以进一步优化这一模型,使得生产者与生产者之间、消费者与消费者之间存在锁竞争,而生产者和消费者之间不存在直接的锁竞争。这是因为信号量可以用来精确控制对资源的访问权限,而不仅仅是简单地锁定整个资源。

信号量的本质是一个计数器。

这个计数器用于跟踪某个资源(如共享内存区域、打印机等)的可用单位数。信号量机制通过这个计数器来控制多个进程或线程对共享资源的访问,确保资源的合理分配和同步。通过这个计数器的增加和减少,信号量不仅能够控制访问权限,还能协调进程间的同步,是解决并发控制问题的一种有效工具。  

 初始化信号量

#include <semaphore.h>
int sem_init(sem_t *sem, int pshared, unsigned int value);
参数:
pshared:0表示线程间共享,非零表示进程间共享
value:信号量初始值
销毁信号量
int sem_destroy(sem_t *sem);
等待信号量(P操作)
功能:等待信号量,会将信号量的值减1
int sem_wait(sem_t *sem); //P()
发布信号量(V操作)
功能:发布信号量,表示资源使用完毕,可以归还资源了。将信号量值加1。
int sem_post(sem_t *sem);//V()
上一个生产者 - 消费者的例子是基于阻塞队列 , 其空间可以动态分配 , 现在基于固定大小的环形队列重写这个程序

四、基于环形队列的生产消费模型

环形队列采用数组模拟,用模运算来模拟环状特性
生产者只需要关注空间 spaceSem 是否足够生产,消费者只需要关注资源 dataSem 是否足够消费,所以开始的时候生产者的信号量就是队列的大小,消费者的信号量就是0,当生成者生产一个资源,生产者信号量-1,消费者+1;当消费者消费一个资源, 生产者信号量+1,消费者-1。

如图所示

代码示例,基于环形队列的生产消费模型其中的资源依旧使用Task来模拟

ringqueue.hpp

#pragma once

#include <iostream>
#include <vector>
#include <semaphore.h>
#include "LockGuard.hpp"

// 定义默认队列大小
const int defaultSize = 5;

// 泛型环形队列类模板
template <typename T>
class RingQueue
{
private:
    // 信号量P操作,减少信号量计数,若计数<0则阻塞当前线程
    void P(sem_t &sem)
    {
        sem_wait(&sem);
    }

    // 信号量V操作,增加信号量计数,若唤醒等待的线程
    void V(sem_t &sem)
    {
        sem_post(&sem);
    }

public:
    // 构造函数,初始化环形队列
    RingQueue(int size = defaultSize)
        : _ringQueue(size), _size(size), _prodStep(0), _consStep(0)
    {
        // 初始化空间信号量,初始值为队列大小,表示初始时所有空间都是空闲的
        sem_init(&_spaceSem, 0, size);
        // 初始化数据信号量,初始值为0,表示队列初始无数据
        sem_init(&_dataSem, 0, 0);

        // 初始化生产者和消费者的互斥锁,保护各自的操作步骤
        pthread_mutex_init(&_prodMutex, nullptr);
        pthread_mutex_init(&_consMutex, nullptr);
    }

    // 向队列添加元素
    void Push(const T &item)
    {
        // 1. 减少空间信号量,尝试获取生产空间,若无空间则阻塞
        P(_spaceSem);
        {
            // 2. 加生产者锁,确保生产操作的原子性
            LockGuard lockGuard(&_prodMutex);
            // 执行实际的入队操作
            _ringQueue[_prodStep] = item;
            _prodStep++;        // 移动生产指针
            _prodStep %= _size; // 环状处理边界
        }
        V(_dataSem);
    }

    // 从队列移除元素
    void Pop(T *outItem)
    {
        // 1. 减少数据信号量,尝试获取数据,若无数据则阻塞
        P(_dataSem);
        {
            // 2. 加消费者锁,确保消费操作的原子性
            LockGuard lockGuard(&_consMutex);
            // 执行实际的出队操作
            *outItem = _ringQueue[_consStep];
            _consStep++;        // 移动消费指针
            _consStep %= _size; // 环状处理边界
        }//消费者V操作时不冲突,可以解锁 信号量的P操作(wait/减)和V操作(signal/增)都是原子操作。
        V(_spaceSem);
    }

    // 析构函数,释放资源
    ~RingQueue()
    {
        sem_destroy(&_spaceSem);
        sem_destroy(&_dataSem);

        pthread_mutex_destroy(&_prodMutex);
        pthread_mutex_destroy(&_consMutex);
    }

private:
    // 环形队列底层使用std::vector存储
    std::vector<T> _ringQueue;
    int _size; // 队列大小

    // 生产者和消费者的步进索引
    int _prodStep;
    int _consStep;

    // 信号量,管理空间和数据的可用性
    sem_t _spaceSem;
    sem_t _dataSem;

    // 互斥锁,分别保护生产者和消费者的步骤更新
    pthread_mutex_t _prodMutex;
    pthread_mutex_t _consMutex;
};

Main.cc

#include"ringqueue.hpp"
#include "Task.hpp"
#include <unistd.h>
#include <pthread.h>
#include <ctime>

void *Productor(void *args)
{
    // sleep(5);
    RingQueue<Task> *rq = static_cast<RingQueue<Task> *>(args);
    while (true)
    {
        // 数据怎么来的?
        // 1. 有数据,从具体场景中来,从网络中拿数据
        // 生产前,你的任务从哪里来的呢???
        int data1 = rand() % 10; // [1, 10] // 将来深刻理解生产消费,就要从这里入手,TODO
        usleep(rand() % 123);
        int data2 = rand() % 10; // [1, 10] // 将来深刻理解生产消费,就要从这里入手,TODO
        usleep(rand() % 123);
        char oper = opers[rand() % (opers.size())];
        Task t(data1, data2, oper);
        std::cout << "productor task: " << t.PrintTask() << std::endl;

        // rq->push();
        rq->Push(t);

         sleep(1);
    }
}

void *Consumer(void *args)
{
    RingQueue<Task> *rq = static_cast<RingQueue<Task> *>(args);
    while (true)
    {
        // sleep(1);

        Task t;
        rq->Pop(&t);

        t();
        std::cout << "consumer done, data is : " << t.PrintResult() << std::endl;
    }
}

int main()
{

    srand((uint64_t)time(nullptr) ^ pthread_self());
    pthread_t c[3], p[2];

    // 唤醒队列中只能放置整形???
    // RingQueue<int> *rq = new RingQueue<int>();
    RingQueue<Task> *rq = new RingQueue<Task>();

    pthread_create(&p[0], nullptr, Productor, rq);
    pthread_create(&p[1], nullptr, Productor, rq);
    pthread_create(&c[0], nullptr, Consumer, rq);
    pthread_create(&c[1], nullptr, Consumer, rq);
    pthread_create(&c[2], nullptr, Consumer, rq);

    pthread_join(p[0], nullptr);
    pthread_join(p[1], nullptr);
    pthread_join(c[0], nullptr);
    pthread_join(c[1], nullptr);
    pthread_join(c[2], nullptr);

    return 0;
}

运行如图所示 

环形队列中的生产者和消费者通过同步与互斥机制维持着一种动态平衡,确保数据的连续生产和消费,体现了典型的生产者-消费者问题的解决方案。 

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

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

相关文章

2024年3月 电子学会青少年等级考试机器人理论真题六级

202403 青少年等级考试机器人理论真题六级 第 1 题 下列选项中&#xff0c;属于URL的是&#xff1f;&#xff08; &#xff09; A&#xff1a;192.168.1.10 B&#xff1a;www.baidu.com C&#xff1a;http://www.kpcb.org.cn/h-col-147.html D&#xff1a;fe80::7998:ffc8…

【MIT6.S081】Lab7: Multithreading(详细解答版)

实验内容网址:https://xv6.dgs.zone/labs/requirements/lab7.html 本实验的代码分支:https://gitee.com/dragonlalala/xv6-labs-2020/tree/thread2/ Uthread: switching between threads 关键点:线程切换、swtch 思路: 本实验完成的任务为用户级线程系统设计上下文切换机制…

x264 帧类型代价计算原理:slicetype_frame_cost 函数分析

slicetype_frame_cost 函数 函数功能 这个函数的核心是计算编码一系列帧(从 p0 到p1,以 b 为当前帧)的代价 cost,并根据这个代价 cost来辅助帧类型决策。它考虑了运动搜索的结果、帧间和帧内预测的成本,并且可以并行处理以提高效率。该函数在帧类型决策、MBtree 分析、场…

有一个21年的前端vue项目,死活安不上依赖

在公司开发的时候遇到的一个很玄幻的问题,这个项目是21年开发的,现在我是24年中途二开增加新功能 这个项目经过多人之手,现在已经出现了问题------项目依赖安不上,我能启动完全是因为在23年的时候写这个项目的时候将依赖费九牛二虎之力下载好后打成了压缩包发给另外一个安不上依…

【Java学习笔记10 Java Web 应用——JSP

JSP(Java Script Pages)技术是一种网站开发技术&#xff0c;可以让Web开发人员快速、高效的开发出易于维护的动态网页。使用JSP技术开发的Web应用程序具有跨平台性&#xff0c;不需要修改程序&#xff0c;发布后即可在Windows、Linux等不同的操作系统中运行。 10.1 JSP技术概述…

【JavaWeb】前后端分离SpringBoot项目快速排错指南

1 发起业务请求 打开浏览器开发者工具&#xff0c;同时显示网络&#xff08;Internet&#xff09;和控制台&#xff08;console&#xff09; 接着&#xff0c;清空控制台和网络的内容&#xff0c;如下图 然后&#xff0c;点击你的业务按钮&#xff0c;发起请求。 首先看控制台…

nginx 配置域名SSL证书HTTPS服务

下载 上传根目录 /home/wwwroot/xx.com/ssl 从新执行 添加域名命令 选择添加SSL SSL Certificate file: 填写 完整目录 PEM文件地址 SSL Certificate Key file:填写 完整目录 key文件地址

OmniDrive:具有 3D 感知推理和规划功能的自动驾驶整体 LLM-智体框架

24年5月北理工、Nvidia和华中科大的论文“OmniDrive&#xff1a;A Holistic LLM-Agent Framework for Autonomous Driving with 3D Perception Reasoning and Planning”。 多模态大语言模型&#xff08;MLLMs&#xff09;的进展导致了对基于LLM的自动驾驶的兴趣不断增长&…

QT状态机10-QKeyEventTransition和QMouseEventTransition的使用

1、QMouseEventTransition的使用 首先明白 QMouseEventTransition 继承自 QEventTransition类。 关于QEventTransition类的使用,可参考 QT状态机9-QEventTransition和QSignalTransition的使用 回顾 QT状态机9-QEventTransition和QSignalTransition的使用 中的状态切换代码,如…

NSS【web】刷题

[SWPUCTF 2021 新生赛]jicao 类型&#xff1a;PHP、代码审计、RCE 主要知识点&#xff1a;json_decode()函数 json_decode()&#xff1a;对JSON字符串解码&#xff0c;转换为php变量 用法&#xff1a; <?php $json {"ctf":"web","question"…

《Fundamentals of Power Electronics》——阻抗和传递函数的图解构造

通常&#xff0c;我们可以通过观察画出近似的波德图&#xff0c;而不需要大量杂乱的代数和不可避免的相关代数错误。使用这种方法可以对电路的工作原理有很大的了解。在不同频率下&#xff0c;哪些元件主导电路响应变得很清楚&#xff0c;因此合适的近似变得很明显。可以直接得…

JAVA毕业设计141—基于Java+Springboot+Vue的物业管理系统(源代码+数据库)

毕设所有选题&#xff1a; https://blog.csdn.net/2303_76227485/article/details/131104075 基于JavaSpringbootVue的物业管理系统(源代码数据库)141 一、系统介绍 本项目前后端分离&#xff0c;分为管理员、员工、用户三种角色(角色权限可自行分配) 1、用户&#xff1a; …

《看漫画学C++》背后的故事6:死循环

C 中的死循环是指一个循环结构没有终止条件&#xff0c;导致程序永远无法跳出该循环&#xff0c;从而陷入无限循环的状态。这种情况通常是由于逻辑错误或编程错误导致的。 在《看漫画学C》中我们用这样一副漫画描述死循环。 购书链接&#xff1a;https://item.jd.com/144188…

AQS应用--CountDownLatch

一、是什么 顾名思义&#xff0c;Latch是门闩的意思&#xff0c;把到达门闩的线程都阻塞住&#xff0c;CountDown是减少计数的意思。 所以CountDownLatch是当每个线程到达某个状态就将计数减一&#xff0c;计数为0时所有被阻塞线程全部被唤醒。 二、内部实现 CountDownLatch…

[牛客网]——C语言刷题day2

答案&#xff1a;B 解析&#xff1a; char *p[10] 是指针数组,数组里存放了10个指针,在64位系统下指针占8个字节,所以sizeof(p) 10 * 8 80. char (*p1)[10]是数组指针,p1是一个指向存放10个char类型的数组的指针,所以sizeof(p1) 8. 答案&#xff1a;B 解析&#xff1a…

25. K 个一组翻转链表 - 力扣(LeetCode)

基础知识要求&#xff1a; Java&#xff1a;方法、while循环、for循环、if else语句 Python&#xff1a; 方法、while循环、for循环、if else语句 题目&#xff1a; 给你链表的头节点 head &#xff0c;每 k 个节点一组进行翻转&#xff0c;请你返回修改后的链表。 k 是一个…

mysql 8.0 迁移到kingbase v8r6 不兼容之处记录

本文以kingbase 为解释主体 一. 函数 1.无 sysdate()&#xff0c;可以使用 current_timestamp() 替换 SELECT CURRENT_TIMESTAMP() 2.计算前几天 &#xff0c; 不能 执行 now() interval -1 day&#xff0c;可以使用 date_sub( n, unit) 替换 -- 前三天 SELECT DATE_SUB…

6款日常精选手机APP推荐!

AI视频生成&#xff1a;小说文案智能分镜智能识别角色和场景批量Ai绘图自动配音添加音乐一键合成视频https://aitools.jurilu.com/ 1.全能相机软件——无他相机 无他相机App是一款完全免费且功能全面的美颜相机软件。这款相机应用集自拍、美颜、图片编辑、风格化模板、流行贴…

sql实践

1.从excel导入数据 在excel导入数据时要先在数据库中创建对应的数据库表 CREATE TABLE your_table_name (crawl_datetime DATE,url CHAR(255),company_name CHAR(255),company_size CHAR(255),company_type CHAR(255),job_type CHAR(255),job_name CHAR(255),edu CHAR(255),e…

tomcat--安装

官网&#xff1a;Apache Tomcat - Welcome! 官网文档&#xff1a;Apache Tomcat 8 (8.5.100) - Documentation Index 帮助文档&#xff1a;Apache Tomcat Home - Apache Tomcat - Apache Software Foundation FAQ - Apache Tomcat - Apache Software Foundation yum安装 查…