最新资讯

  • 头歌 Linux之线程同步一

头歌 Linux之线程同步一

2025-05-08 22:37:33 0 阅读

第1关:互斥锁

任务描述


由于同一进程中的多个线程共享全局数据,因此,在多线程编程中如果一个线程对全局变量A进行修改时,而此时又有一个线程正在读取该变量,则有可能会出现数据的不一致性。本关将介绍一种线程同步方式-互斥锁。

本关任务:学会使用互斥锁来实现线程间的同步。

相关知识


在多线程编程中,我们常遇到的问题是当多个线程同时访问共享数据时可能会产生冲突。比如:存在多个线程同时要对一个全局变量进行加一操作,我们知道加一操作需要以下三条指令完成:

从内存中将变量值读取到寄存器中;
将寄存器中的值加一操作;
将寄存器中的值写回内存中;


下图所示:假设有两个线程同时执行以上三条指令,则可能会出现以下情况,线程A将变量值(a=5)读取到寄存器中,并将寄存器中的值加一(a=6),而此时线程B正好将变量的值读取到寄存器中(a=5)。当线程A将寄存器中的值写入内存后,该变量的值完成了加一操作(a=6),而随后线程B也将加一的值写回内存中(a=6),那么最终变量的值只是完成的加一操作,而不是加一再加一。

对于多线程的程序,访问冲突的问题是很普遍的,Linux系统中为了解决线程同步问题引入了互斥锁(mutex)。互斥锁操作主要包括加锁、解锁和测试加锁三个,不论哪种类型的锁,都不可能被两个不同的线程同时得到,而必须等待解锁,因此能够保证同一时刻只运行一个线程执行一个关键部分的代码。

Linux系统中提供了如下几个函数来操作互斥锁:

以上函数我们可以使用man命令来查询该函数的使用方法。具体的查询命令为:man 3 函数名。

初始化互斥锁


使用互斥锁前必须先进行初始化操作。在Linux中初始化互斥锁有两种方式,分别是:

(1) 静态赋值法;

(2)使用初始化函数。

1、静态赋值法


静态赋值法是将宏结构常量直接赋值给互斥锁,常见的宏结构常量有如下几个:

PTHREAD_MUTEX_INITIALIZER: (普通锁)当一个线程加锁后,其余请求锁的线程形成等待队列,解锁后按照优先级获取锁;
PTHREAD_RECURSIVE_MUTEX_INITIALIZER_NP: (嵌套锁)允许一个线程对同一个锁进行多次加锁操作,并通过多次解锁来释放锁;
PTHREAD_ERRORCHECK_MUTEX_INITIALIZER_NP: (检错锁)在同一个线程请求同一个锁的情况下,返回EDEADLK,否则执行的动作与普通锁相同;

例如使用静态赋值法来初始化一个互斥锁变量:

pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER;
2、函数赋值法


Linux系统提供一个pthread_mutex_init库函数来对互斥锁进行初始化。
pthread_mutex_init函数的具体的说明如下:

需要的头文件如下:

#include 

函数格式如下: 

int pthread_mutex_init(pthread_mutex_t *mutex, const pthread_mutex attr_t *mutexattr);

参数说明:

mutex:互斥锁变量;
mutexattr:互斥锁属性,如果为NULL则使用默认属性,其他常见属性见下表;

函数返回值说明:

pthread_mutex_init返回值总为0。

加锁操作


对互斥锁初始化后,就可以给互斥锁进行加锁操作。Linux提供了两个库函数来对加锁,分别是:pthread_mutex_lock和pthread_mutex_trylock,这些函数的具体的说明如下:

需要的头文件如下:

#include 

函数格式如下:

int pthread_mutex_lock(pthread_mutex_t *mutex);
int pthread_mutex_trylock(pthread_mutex_t *mutex);

参数说明:
mutex:要被执行加锁操作的锁变量

函数返回值说明:
调用成功,返回值为0,否则返回一个非零的错误码。

pthread_mutex_lock和pthread_mutex_trylock区别:
用pthread_mutex_lock加锁时,如果mutex已经被锁住,当前尝试加锁的线程就会被阻塞,直到互斥锁被其他线程释放。而pthread_mutex_trylock函数则不同,如果mutex已经被锁住,它将立即返回,返回的错误码为EBUSY,而不是阻塞等待。

解锁操作

有加锁操作就相对应的有解锁操作。Linux提供了一个pthread_mutex_unlock函数来解锁操作,这个函数的具体的说明如下:

需要的头文件如下:

#include 


函数格式如下:

int pthread_mutex_unlock(pthread_mutex_t *mutex);


参数说明:
mutex:要被执行解锁操作的锁变量

函数返回值说明:
调用成功,返回值为0,否则返回一个非零的错误码。

注销锁操作

当一个互斥锁使用完毕后,必须进行清除。Linux提供了一个pthread_mutex_destroy函数来注销一个互斥锁,这个函数的具体的说明如下:

需要的头文件如下:

#include 


函数格式如下:

int pthread_mutex_destroy(pthread_mutex_t *mutex);


参数说明:
mutex:要被执行注销操作的锁变量

函数返回值说明:
调用成功,返回值为0,否则返回一个非零的错误码。

注意:如果使用静态初始化来初始化一个互斥锁,则无需使用pthread_mutex_destroy对其注销。

案例演示1:


编写一个程序,使用静态初始化方法来初始化一个互斥锁,并对一个全局变量进行加锁。详细代码如下所示:

#include 
#include 
int globalNumber = 1;
//初始化一个普通互斥锁number_mutex
pthread_mutex_t number_mutex = PTHREAD_MUTEX_INITIALIZER;
void *addNumer(void *arg)
{
    pthread_mutex_lock(&number_mutex);
    globalNumber++;
    pthread_mutex_unlock(&number_mutex);
    
    return NULL;
}
int main()
{
    pthread_t thread1, thread2;
    pthread_create(&thread1, NULL, addNumer, NULL);
    pthread_create(&thread2, NULL, addNumer, NULL);
    
    pthread_join(thread1, NULL);
    pthread_join(thread2, NULL);
    
    printf("globalNumber = %d
", globalNumber);
    
    return 0;
}

将以上代码保存为mutexThread1.c文件,编译执行。可以看到globalNumber变量变为了3,尽管我们不对globalNumber变量进行加锁,结果可能也是3,但是有可能出现结果为2的情况。

案例演示2:


编写一个程序,使用pthread_mutex_init函数来初始化一个普通的互斥锁,并对一个全局变量进行加锁。详细代码如下所示:

#include 
#include 
int globalNumber = 1;
//初始化一个普通互斥锁number_mutex
pthread_mutex_t number_mutex;
void *addNumer(void *arg)
{
    pthread_mutex_lock(&number_mutex);
    globalNumber++;
    pthread_mutex_unlock(&number_mutex);
    
    return NULL;
}
int main()
{
    pthread_mutex_init(&number_mutex, PTHREAD_MUTEX_TIMED_NP);
    pthread_t thread1, thread2;
    pthread_create(&thread1, NULL, addNumer, NULL);
    pthread_create(&thread2, NULL, addNumer, NULL);
    
    pthread_join(thread1, NULL);
    pthread_join(thread2, NULL);
    
    printf("globalNumber = %d
", globalNumber);
    
    pthread_mutex_destroy(&number_mutex);
    
    return 0;
}

编程要求


本关的编程任务是补全右侧代码片段中Begin至End中间的代码,具体要求如下:

补全ThreadHandler函数中代码,使用互斥锁对position和buffer变量加锁,使其同一时刻只能被一个线程访问。


测试说明


本关的测试需要用户在右侧代码页中补全代码,然后点击评测按钮,平台会自动验证用户是否按照要求去检测结果。

开始你的任务吧,祝你成功!

解答:

#include 
#include 
#include 
 
//全局互斥锁变量
extern pthread_mutex_t mutex;
 
//全局共享变量
extern char *buffer[3];
extern int position;
 
/************************
 * 参数arg: 是线程函数的参数
*************************/
void *ThreadHandler(void *arg)
{
	/********** BEGIN **********/
	 pthread_mutex_lock(&mutex);
	/********** END **********/
	
	buffer[position] = (char *)arg;
	sleep(1);
	position++;
	
	/********** BEGIN **********/
	pthread_mutex_unlock(&mutex);	
	/********** END **********/
	pthread_exit(NULL);
}

第2关:自旋锁

在上一关中,我们介绍了如何使用互斥锁来同步线程,本关将介绍Linux系统中的另一种锁,它就是自旋锁。

本关任务:学会使用自旋锁来实现线程间的同步。

相关知识


在上一关中,我们学习了互斥锁的使用方法。在Linux系统中还有一类锁与互斥锁相似,它就是自旋锁。自旋锁的功能以及使用方法和互斥锁极为相似。自旋锁与互斥锁的主要区别在于,当执行加锁操作时,如果当前锁不可用,对于自旋锁来说,则阻塞后不会让出cpu,会一直忙等待,直到得到锁。而对于互斥锁来说,阻塞后休眠让出cpu。

由于自旋锁不会睡眠,自旋锁一直占用cpu,在未获得锁的情况下,一直运行,所以占用着cpu,如果不能在很短的时间内获得锁,这无疑会使CPU效率降低。因此,自旋锁通常用于驱动程序开发中,适合对短暂处理的资源进行加锁。

注意:在单处理器环境下,我们是不需要自旋锁,尽管调用了自旋锁的函数,里面也不是自旋锁的实现。

Linux系统中提供了如下几个函数来操作自旋锁:

以上函数我们可以使用man命令来查询该函数的使用方法。具体的查询命令为:man 3 函数名。

初始化自旋锁


使用自旋锁前必须先进行初始化操作。初始化自旋锁的库函数是pthread_spin_init。
pthread_spin_init函数的具体的说明如下:

需要的头文件如下:

#include 


函数格式如下:

 int pthread_spin_init(pthread_spinlock_t *lock, int pshared);

参数说明:
lock:自旋锁变量;
pshared:自旋锁属性,常见属性见下表;

函数返回值说明: 调用成功,返回值总为0,否则返回一个非零的错误码。


加锁操作


对自旋锁初始化后,就可以给自旋锁进行加锁操作。Linux提供了两个库函数来对加锁,分别是:pthread_spin_lock和pthread_spin_trylock,这些函数的具体的说明如下:

需要的头文件如下:

#include 


函数格式如下:

int pthread_spin_lock(pthread_spinlock_t *lock);
int pthread_spin_trylock(pthread_spinlock_t *lock);


参数说明:
lock:要被执行加锁操作的锁变量

函数返回值说明:
调用成功,返回值为0,否则,返回一个非零的错误码。

pthread_spin_lock和pthread_spin_trylock区别:
用pthread_spin_lock加锁时,如果lock已经被锁住,当前尝试加锁的线程就会被阻塞,直到自旋锁被其他线程释放。而pthread_spin_trylock函数则不同,如果lock已经被锁住,它将立即返回,返回的错误码为EBUSY,而不是阻塞等待。

解锁操作


有加锁操作就相对应的有解锁操作。Linux提供了一个pthread_spin_unlock函数来解锁操作,这个函数的具体的说明如下:

需要的头文件如下:

#include 


函数格式如下:

int pthread_spin_unlock(pthread_spinlock_t *lock);


参数说明:
lock:要被执行解锁操作的锁变量

函数返回值说明:
调用成功,返回值为0,否则返回一个非零的错误码。

注销锁操作


当一个自旋锁使用完毕后,必须进行清除。Linux提供了一个pthread_spin_destroy函数来注销一个自旋锁,这个函数的具体的说明如下:

需要的头文件如下:

#include 


函数格式如下:

int pthread_spin_destroy(pthread_spinlock_t *lock);


参数说明:
lock:要被执行注销操作的锁变量

函数返回值说明:
调用成功,返回值为0,否则返回一个非零的错误码。

案例演示1:


编写一个程序,使用自旋锁对一个全局变量进行加锁。详细代码如下所示:

#include 
#include 
int globalNumber = 1;
//初始化一个普通自旋锁number_lock
pthread_spinlock_t number_lock;
void *addNumer(void *arg)
{
    pthread_spin_lock(&number_lock);
    globalNumber++;
    pthread_spin_unlock(&number_lock);
    
    return NULL;
}
int main()
{
    pthread_spin_init(&number_lock, PTHREAD_PROCESS_PRIVATE);
    pthread_t thread1, thread2;
    pthread_create(&thread1, NULL, addNumer, NULL);
    pthread_create(&thread2, NULL, addNumer, NULL);
    
    pthread_join(thread1, NULL);
    pthread_join(thread2, NULL);
    
    printf("globalNumber = %d
", globalNumber);
    
    pthread_spin_destroy(&number_lock);
    
    return 0;
}

将以上代码保存为lockThread.c文件,编译执行。可以看到globalNumber变量变为了3,尽管我们不对globalNumber变量进行加锁,结果可能也是3,但是有可能出现结果为2的情况。

编程要求


本关的编程任务是补全右侧代码片段中Begin至End中间的代码,具体要求如下:

补全ThreadHandler函数中代码,使用自旋锁对position和buffer变量加锁,使其同一时刻只能被一个线程访问。


测试说明


本关的测试需要用户在右侧代码页中补全代码,然后点击评测按钮,平台会自动验证用户是否按照要求去检测结果。

开始你的任务吧,祝你成功!

解答:

#include 
#include 
#include 
 
//全局自旋锁变量
extern pthread_spinlock_t lock;
 
//全局共享变量
extern char *buffer[3];
extern int position;
 
/************************
 * 参数arg: 是线程函数的参数
*************************/
void *ThreadHandler(void *arg)
{
	/********** BEGIN **********/
	pthread_spin_lock(&lock);
	/********** END **********/
 
	buffer[position] = (char *)arg;
	sleep(1);
	position++;
 
	/********** BEGIN **********/
	pthread_spin_unlock(&lock);
	/********** END **********/
	pthread_exit(NULL);
}

第3关:条件变量

在以上两关中,我们介绍了如何互斥锁和自旋锁的使用方式,并且使用互斥锁和自旋锁来同步线程,本关我们将学习另一种同步线程的方式。

本关任务:学会使用条件变量来实现线程间的同步。

相关知识


条件变量是利用线程间共享的全局变量进行同步的一种机制。条件变量宏观上类似if语句,符合条件就能执行某段代码,否则,只能等待条件成立。使用条件变量主要包括两个动作,分别为:(1)一个等待使用资源的线程等待"条件变量被设置为真";(2)一个正在使用资源的线程在使用完资源后"设置条件为真";这样就可以保证线程间同步。但是,这样存在一个关键的问题,那就是要保证条件变量能够被正确的修改,因此,条件变量要受到特殊的保护才行。实际上使用互斥锁来扮演着这样的一个保护者的角色。注意:使用的互斥锁必须是普通的互斥锁

与互斥锁不同,条件变量是用来等待而不是用来上锁的。条件变量用来自动阻塞一个线程,直到某特殊情况发生为止。通常条件变量和互斥锁同时使用。

条件的检测是在互斥锁的保护下进行的。线程在改变条件状态之前必须首先锁住互斥量。如果一个条件为假,一个线程自动阻塞,并释放等待状态改变的互斥锁。如果另一个线程改变了条件,它发信号给关联的条件变量,唤醒一个或多个等待它的线程,这些线程将重新锁定互斥锁并重新测试条件是否满足。

Linux 系统中提供了如下几个函数来操作条件变量:

初始化条件变量


使用条件变量前必须先进行初始化操作。在Linux中初始化条件变量有两种方式,分别是:

(1)静态赋值法;

(2)使用初始化函数。

静态赋值法


 静态赋值法是直接将宏结构常量直接赋值给条件变量,将宏结构变量PTHREAD_COND_INITIALIZER赋值给条件变量即可,例如:
pthread_cond_t cond = PTHREAD_COND_INITIALIZER;

函数赋值法


 Linux 系统提供一个pthread_cond_init库函数来对条件变量进行初始化。
pthread_cond_init函数的具体的说明如下:

需要的头文件如下:

#include 


函数格式如下:

int    pthread_cond_init(pthread_cond_t    *cond,   pthread_condattr_t  *cond_attr);


参数说明:

cond:条件变量;
cond_attr:条件变量属性,由于该属性在实际中没有被实现,所以它的值通常是NULL;


函数返回值说明:
调用成功,返回值为0,否则,返回一个非零的错误码。

等待条件成立函数


Linux 提供了两个库函数用来判断等待条件成立,分别是:pthread_cond_wait和pthread_cond_timedwait,这些函数的具体的说明如下:

需要的头文件如下:

#include 


函数格式如下:

int pthread_cond_wait(pthread_cond_t *cond, pthread_mutex_t *mutex);
int   pthread_cond_timedwait(pthread_cond_t   *cond,    pthread_mutex_t *mutex, const struct timespec *abstime);


参数说明:

cond:等待判断的条件变量;
mutex:互斥锁;
abstime:等待的时间;


函数返回值说明:
调用成功,返回值为0,否则,返回一个非零的错误码。

pthread_cond_wait和pthread_cond_timedwait区别:
pthread_cond_timedwait函数将阻塞直到条件变量获取变为真(获得信号)或者经过由abstime指定的时间,也就是说,如果在给定的时间前条件变量没有满足,则返回ETIMEOUT并结束等待。

激活条件变量


当线程被条件变量所阻塞后,需要对其激活。Linux 提供了两个函数来激活条件变量,分别是:pthread_cond_signal和pthread_cond_broadcast,这些函数的具体的说明如下:

需要的头文件如下:

#include 


函数格式如下:

int pthread_cond_signal(pthread_cond_t *cond);
int pthread_cond_broadcast(pthread_cond_t *cond);


参数说明:
 cond:需要被激活的条件变量

函数返回值说明:
 调用成功,返回值为0,否则返回一个非零的错误码。

注意:pthread_cond_signal激活一个等待条件变量成立的线程,当存在多个等待线程时,按照入队顺序激活其中的一个;而pthread_cond_broadcast则是激活所有等待的线程。

注销条件变量


当一个条件变量使用完毕后,必须进行清除。Linux 提供了一个pthread_cond_destroy函数来注销一个条件变量,这个函数的具体的说明如下:

需要的头文件如下:

#include 


函数格式如下:

 int pthread_cond_destroy(pthread_cond_t *cond);


 参数说明:
 cond:需要被注销的条件变量

函数返回值说明:
 调用成功,返回值为0,否则返回一个非零的错误码。

注意:只有在没有线程等待一个条件变量时,该条件变量才能被注销,否则返回EBUSY。

案例演示1:


编写一个程序,使用条件变量对一个全局变量进行加锁。详细代码如下所示:

#include 
#include 
int globalNumber = 1;
//初始化一个普通互斥锁number_lock
pthread_mutex_t number_lock;
//初始化一个条件变量number_cond
pthread_cond_t number_cond;
void *addNumer(void *arg)
{
    pthread_mutex_lock(&number_lock);
    pthread_cond_wait(&number_cond, &number_lock);
    globalNumber++;
    pthread_mutex_unlock(&number_lock);
    
    return NULL;
}
int main()
{
    pthread_cond_init(&number_cond, NULL);
    pthread_t thread1, thread2;
    pthread_create(&thread1, NULL, addNumer, NULL);
    pthread_create(&thread2, NULL, addNumer, NULL);
    
    //激活所有等待的线程
    sleep(1);     //保证有线程处于等待状态后再发激活信号
    pthread_cond_broadcast(&number_cond);
    
    pthread_join(thread1, NULL);
    pthread_join(thread2, NULL);
    
    printf("globalNumber = %d
", globalNumber);
    
    pthread_cond_destroy(&number_cond);
    
    return 0;
}

将以上代码保存为condThread.c文件,编译执行。可以看到globalNumber变量变为了3。注意:我们需要在激活条件变量前要等待所有线程处于等待条件变量状态,这样才能激活等待的线程。

编程要求


本关的编程任务是补全右侧代码片段中Begin至End中间的代码,具体要求如下:

补全ThreadHandler1和ThreadHandler2函数中代码.
ThreadHandler1函数中对position变量进行加一操作(只有一个线程使用,无需加锁),当position变量被加一后,则只通知一个线程执行ThreadHandler2函数完成字符串赋值操作。


测试说明


本关的测试需要用户在右侧代码页中补全代码,然后点击评测按钮,平台会自动验证用户是否按照要求去检测结果。

开始你的任务吧,祝你成功!

解答:

#include 
#include 
#include 
 
//全局互斥锁变量和条件变量
extern pthread_mutex_t mutex;
extern pthread_cond_t cond;
 
//全局共享变量
extern char *buffer[3];
extern int position;
pthread_cond_t has_product = PTHREAD_COND_INITIALIZER;
/************************
 * 参数arg: 是线程函数的参数
*************************/
void *ThreadHandler1(void *arg)
{
	int i;
	for(i = 0; i < 3; i++)
	{
		usleep(500);
		position++;
		//通知ThreadHandler2函数执行赋值操作
		/********** BEGIN **********/
		pthread_cond_signal(&has_product);
		/********** END **********/
	}
	pthread_exit(NULL);
}
 
/************************
 * 参数arg: 是线程函数的参数
*************************/
void *ThreadHandler2(void *arg)
{
	/********** BEGIN **********/
	pthread_mutex_lock(&mutex);
    pthread_cond_wait(&has_product, &mutex);
	/********** END **********/
 
	buffer[position] = (char *)arg;
	
	/********** BEGIN **********/
	pthread_mutex_unlock(&mutex);
	/********** END **********/
	pthread_exit(NULL);
}

第4关:项目实战

任务描述


本关任务:利用互斥锁和条件变量实现一个生产者消费者模型。

相关知识


生产者消费者模型一个著名的同步问题。它描述的是:多个生产者来生产产品,并将这些产品提供给消费者去消费。为使生产者和消费者能够并发执行。在两者之间设置了一个公共区域,生产者进入公共区域生产产品并放入其中。消费者进入公共区域并取走产品进行消费。

生产者消费者模型需要满足如下规则:当一个生产者进入公共区域生产产品时,其他生产者和消费者不能同时进入公共区域生产产品或消费产品。当一个消费者进入公共区域消费产品的时候,其它消费者和生产者不能同时进入该区域消费产品或生产产品。也就是说,任意时刻,最多只允许一个生产者或一个消费者进入公共区域。即生产者和消费者必须互斥的访问公共区域。

通过前3关的学习,我们现在知道如何互斥锁、自旋锁和条件变量来同步线程。那么利用以上3关的知识就可以简单的生产者消费者模型。

实现生产者消费者模型


一个普通的生产者消费者模型需要满足以下标准:

(1)生产者与生产者之间存在竞争即互斥关系;

(2)消费者与消费者之间存在竞争即互斥关系;

(3)生产者与消费者之间存在互斥与同步关系。

案例演示1:


通过互斥锁和条件变量实现生产者消费者模型。我们使用链表来存在数据,生产者向数据区生产随机数,消费者从数据区读取生产好的数据并打印出来。我们设置当生产者与消费者满足特定的生产与消费标准时,则退出程序。

详细的代码设计为:

#include 
#include 
#include 
#include 
struct msg
{
    int data;    //存放生产的数据
    struct msg *next;
};
pthread_cond_t cond;   //条件变量
pthread_mutex_t mutex;   //互斥锁
const int max_number = 10;   //生产者与消费者的最大生产和消费任务
int producer_number = 0;    //生产者已经生产的任务数
int consumer_number = 0;     //消费者已经消费的任务数
struct msg *begin = NULL;    //定义数据区头
struct msg *end = NULL;    //定义数据区尾
/*
 * 生产者线程
 */
void *producer(void *arg)
{
    while(1)
    {
        pthread_mutex_lock(&mutex);
        //判断生产者生产的数据量是否满足需求(一共生产max_number多个数据)
        if(producer_number == max_number)
        {
            pthread_mutex_unlock(&mutex);
            pthread_exit(NULL);
        }
        
        //生产数据
        if(begin == NULL)
            begin = end = malloc(sizeof(struct msg));
        else
            end = end->next =  malloc(sizeof(struct msg));
        end->data = rand()%100;
        producer_number++;
        //激活所有消费者(注意:这里不能用pthread_cond_signal函数来激活)
        //pthread_cond_signal函数只能一次激活一个线程
        pthread_cond_broadcast(&cond);
        pthread_mutex_unlock(&mutex);
    }
}
/*
 * 消费者线程
 */
void *consumer(void *arg)
{
    while(1)
    {
        pthread_mutex_lock(&mutex);
        //判断当前是否有数据可用来消费
        while(begin == NULL)
        {
            //判断消费者消费的数据量是否满足需求(一共消费max_number多个数据)
            if(consumer_number == max_number)
            {
                pthread_mutex_unlock(&mutex);
                pthread_exit(NULL);
            }
            //如果没有数据可消费,则睡眠当前线程
            pthread_cond_wait(&cond, &mutex);
        }
        //消费数据
        printf("Consumer: %d
", begin->data);
        consumer_number++;
        //将消费后的数据释放掉
        struct msg *tmp = begin;
        begin = begin->next;
        free(tmp);
        pthread_mutex_unlock(&mutex);
    }
}
int main(int argc, char *argv[])
{
    //设置随机数种子
    srand(time(NULL));
    //初始化条件变量和互斥锁
    pthread_cond_init(&cond, NULL);
    pthread_mutex_init(&mutex, NULL);
    
    pthread_t thread[5];
    int i;
    
    //创建2个生产者用来生产数据
    for(i = 0; i < 2; i++)
        pthread_create(&thread[i], NULL, producer, NULL);
    //创建3个消费者用来消费数据
    for(i = 2; i < 5; i++)
        pthread_create(&thread[i], NULL, consumer, NULL);
    //等待生产者和消费者结束
    for(i = 0; i < 5; i++)
        pthread_join(thread[i], NULL);
    
    pthread_cond_destroy(&cond);
    pthread_mutex_destroy(&mutex);
    
    return 0;
}

将以上代码保存为ProducerConsumer.c文件中,编译执行。

编程要求


本关的编程任务是补全右侧代码片段中Begin至End中间的代码,具体要求如下:

1、补全Consumer函数,该函数是用来消费由生产者产生的数据。
2、产生者生产的数据存放在一个结构体struct Data中的number变量中。

3、消费的方式是直接将数据打印出来即可(格式为:printf("%d ", Data->number)),并且将该数据从链表中删除(参考案例演示1)。

4、当其中一个消费者遇到消费的数据为-1时,停止消费,并将其它的消费者也停止,退出线程(注意:-1数据不需要打印出来)。
5、提示:当遇到消费数据为-1时,退出线程,并且不删除该数据,那么其它消费线程也会消费到该条数据,并且也会退出,这样就可以实现所有消费者的退出。


测试说明


本关的测试需要用户在右侧代码页中补全代码,然后点击评测按钮,平台会自动验证用户是否按照要求去检测结果。

开始你的任务吧,祝你成功!

解答:

#include 
#include 
#include 
 
struct Data
{
    int number;    //存放生产的数据
    struct Data *next;
};
 
//定义数据区头和尾
extern struct Data *beginData;    
 
const int max_number = 10;   //生产者与消费者的最大生产和消费任务
int consumer_number = 0;     //消费者已经消费的任务数
//全局互斥锁变量和条件变量
extern pthread_mutex_t mutex;
extern pthread_cond_t cond;
 
/************************
 * 参数arg: 是线程函数的参数
*************************/
void *Consumer(void *arg)
{
	while(1)
	{
		/********** BEGIN **********/
		pthread_mutex_lock(&mutex);
		while(beginData == NULL)
        {  
            pthread_cond_wait(&cond, &mutex);
        }
        if(beginData->number==-1){
            pthread_mutex_unlock(&mutex);
            pthread_exit(NULL);
        }      
        //消费数据
        printf("%d
", beginData->number);
        consumer_number++;
        //将消费后的数据释放掉
        struct msg *tmp = beginData;
        beginData = beginData->next;
        free(tmp);
        pthread_mutex_unlock(&mutex);	
		/********** END **********/
	}
	pthread_exit(NULL);
}

本文地址:https://www.vps345.com/9295.html

搜索文章

Tags

PV计算 带宽计算 流量带宽 服务器带宽 上行带宽 上行速率 什么是上行带宽? CC攻击 攻击怎么办 流量攻击 DDOS攻击 服务器被攻击怎么办 源IP 服务器 linux 运维 游戏 云计算 服务器安全 网络安全策略 防御服务器攻击 安全威胁和解决方案 程序员博客保护 数据保护 安全最佳实践 网络工程师 网络管理 软考 2024 2024年上半年 下午真题 答案 阿里云 网络 网络安全 网络协议 llama 算法 opencv 自然语言处理 神经网络 语言模型 ubuntu ssh RTSP xop RTP RTSPServer 推流 视频 Ollama Qwen2.5-coder 离线部署 python MCP deepseek 模型联网 API CherryStudio 进程 操作系统 进程控制 Ubuntu 英语 harmonyos 华为 开发语言 前端 javascript typescript 计算机网络 macos windows conda 数据库 centos oracle 关系型 安全 分布式 chrome edge Deepseek Deepseek-R1 大模型 私有化部署 推理模型 php jvm 虚拟机 mysql adb android 经验分享 tcp/ip ip协议 docker java 容器 ubuntu 18.04 安装教程 nginx 负载均衡 数据库系统 vue.js spring boot tomcat 开发环境 vscode 笔记 mcu c语言 单片机 DNS fastapi mcp mcp-proxy mcp-inspector fastapi-mcp agent sse 人工智能 深度学习 计算机视觉 卷积神经网络 debian PVE word图片自动上传 word一键转存 复制word图片 复制word图文 复制word公式 粘贴word图文 粘贴word公式 USB转串口 CH340 kafka 环境变量 gnu json 学习 高级IO epoll c# 机器学习 ssh漏洞 ssh9.9p2 CVE-2025-23419 ai nlp https ide AI编程 DeepSeek-R1 API接口 集成学习 集成测试 智能路由器 外网访问 内网穿透 端口映射 数据分析 redis Dify pip kylin ollama llm apache 部署 学习方法 asm node.js jellyfin nas react.js 前端面试题 持续部署 openvpn server openvpn配置教程 centos安装openvpn Hyper-V WinRM TrustedHosts 后端 c++ 并查集 leetcode jenkins gitee eureka 科技 个人开发 zotero WebDAV 同步失败 代理模式 onlyoffice java-rabbitmq C# MQTTS 双向认证 emqx gcc centos 7 大数据 spark hive fstab burp suite 抓包 EtherCAT转Modbus ECT转Modbus协议 EtherCAT转485网关 ECT转Modbus串口网关 EtherCAT转485协议 ECT转Modbus网关 pycharm cursor pytorch transformer oceanbase 传统数据库升级 银行 Dell R750XS MacOS录屏软件 物联网 iot 信息与通信 面试 性能优化 jdk intellij-idea 架构 VMware安装mocOS VMware macOS系统安装 华为云 udp unity GPU环境配置 Ubuntu22 CUDA PyTorch Anaconda安装 http 产品经理 agi microsoft vim 媒体 websocket xcode prometheus kubernetes 监控k8s集群 集群内prometheus git elasticsearch rtsp服务器 rtsp server android rtsp服务 安卓rtsp服务器 移动端rtsp服务 大牛直播SDK AI大模型 大模型技术 本地部署大模型 oracle fusion oracle中间件 dubbo Linux PID rust uni-app 聚类 云原生 ip spring cloud compose devops SRS 流媒体 直播 C 进程地址空间 github 缓存 灵办AI golang powerpoint 自动化 dify 计算机外设 电脑 mac 软件需求 数据结构 链表 机器人 HarmonyOS Next DevEco Studio C语言 .netcore bug 虚拟现实 outlook 错误代码2603 无网络连接 2603 多层架构 解耦 运维开发 私有化 本地部署 vmware 卡死 sublime text 编辑器 ssl svn remote-ssh 微信 微信分享 鸿蒙 Image wxopensdk rust腐蚀 java-ee Alexnet 前端框架 开源 milvus 小程序 微信小程序域名配置 微信小程序服务器域名 微信小程序合法域名 小程序配置业务域名 微信小程序需要域名吗 微信小程序添加域名 互信 像素流送api 像素流送UE4 像素流送卡顿 像素流送并发支持 YOLO centos-root /dev/mapper yum clean all df -h / du -sh arm开发 CPU 内存 主板 电源 网卡 vue3 HTML audio 控件组件 vue3 audio音乐播放器 Audio标签自定义样式默认 vue3播放音频文件音效音乐 自定义audio播放器样式 播放暂停调整声音大小下载文件 ddos web安全 filezilla 无法连接服务器 连接被服务器拒绝 vsftpd 331/530 etcd 数据安全 RBAC perf HCIE 数通 ESXi windows 服务器安装 WSL2 上安装 Ubuntu cron crontab日志 嵌入式 linux驱动开发 嵌入式硬件 gateway Clion Nova ResharperC++引擎 Centos7 远程开发 qt QT 5.12.12 QT开发环境 Ubuntu18.04 课程设计 远程连接 rdp 实验 思科 进程优先级 调度队列 进程切换 matplotlib fonts-noto-cjk 驱动开发 硬件工程 嵌入式实习 yum docker-compose docker compose 实时音视频 实时互动 Cline unix yolov5 maxkb ARG grafana Linux 进程信号 程序员创富 bash numpy ui 华为od virtualenv VMware创建虚拟机 策略模式 spring dell服务器 JAVA Java flutter ue5 vr 串口服务器 目标检测 Hive环境搭建 hive3环境 Hive远程模式 指令 ip命令 新增网卡 新增IP 启动网卡 多线程 stm32项目 stm32 vnc Docker Hub docker pull 镜像源 daemon.json 本地环回 bind GaN HEMT 氮化镓 单粒子烧毁 辐射损伤 辐照效应 宝塔面板访问不了 宝塔面板网站访问不了 宝塔面板怎么配置网站能访问 宝塔面板配置ip访问 宝塔面板配置域名访问教程 宝塔面板配置教程 WebUI DeepSeek V3 DeepSeek 创意 社区 k8s rabbitmq audio vue音乐播放器 vue播放音频文件 Audio音频播放器自定义样式 播放暂停进度条音量调节快进快退 自定义audio覆盖默认样式 AI efficientVIT YOLOv8替换主干网络 TOLOv8 Flask FastAPI Waitress Gunicorn uWSGI Uvicorn Agent LLM CrewAI protobuf 序列化和反序列化 安装 iftop 网络流量监控 网络药理学 生信 生物信息学 gromacs 分子动力学模拟 MD 动力学模拟 游戏程序 ios 文件系统 路径解析 axure 富文本编辑器 arm WLAN ros2 moveit 机器人运动 chatgpt llama3 Chatglm 开源大模型 rancher iBMC UltraISO 银河麒麟 kylin v10 麒麟 v10 离线部署dify 软件工程 软件构建 深度优先 图论 并集查找 换根法 树上倍增 嵌入式系统开发 监控k8s 监控kubernetes 虚拟显示器 远程控制 命令 ffmpeg 音视频 电子信息 通信工程 毕业 多线程服务器 Linux网络编程 wireshark 显示过滤器 ICMP Wireshark安装 域名服务 DHCP 符号链接 配置 pygame zabbix LDAP deepseek-r1 大模型本地部署 postgresql 计算生物学 生物信息 基因组 串口驱动 CH341 uart 485 export import save load 迁移镜像 rime shell 磁盘监控 低代码 Ubuntu共享文件夹 共享目录 Linux共享文件夹 温湿度数据上传到服务器 Arduino HTTP cuda CLion IDE maven intellij idea fd 文件描述符 蓝耘科技 元生代平台工作流 ComfyUI rpc bcompare Beyond Compare ansible playbook 剧本 Trae AI 原生集成开发环境 Trae AI nftables 防火墙 银河麒麟服务器操作系统 系统激活 系统开发 binder 车载系统 framework 源码环境 Samba NAS 无人机 sql KingBase 镜像下载 freebsd 5G 3GPP 卫星通信 博客 windwos防火墙 defender防火墙 win防火墙白名单 防火墙白名单效果 防火墙只允许指定应用上网 防火墙允许指定上网其它禁止 kind rc.local 开机自启 systemd 麒麟 软件测试 linux安装配置 IPMI kali 共享文件夹 express p2p 政务 分布式系统 监控运维 Prometheus Grafana 安卓 visual studio code 中兴光猫 换光猫 网络桥接 自己换光猫 nvm 智能手机 Termux 虚拟局域网 ai小智 语音助手 ai小智配网 ai小智教程 智能硬件 esp32语音助手 diy语音助手 selete 爬虫 数据集 AIGC postman mock mock server 模拟服务器 mock服务器 Postman内置变量 Postman随机数据 ESP32 视频编解码 迁移指南 fpga开发 鸿蒙系统 tcpdump 系统架构 微服务 设计模式 环境配置 主从复制 交互 IIS服务器 IIS性能 日志监控 数据挖掘 网络用户购物行为分析可视化平台 大数据毕业设计 1024程序员节 腾讯云 基础环境 oneapi 实战案例 docker搭建pg docker搭建pgsql pg授权 postgresql使用 postgresql搭建 Redis Desktop 测试工具 隐藏文件 隐藏目录 管理器 通配符 华为认证 交换机 hadoop tar docker搭建nacos详解 docker部署nacos docker安装nacos 腾讯云搭建nacos centos7搭建nacos gitlab 工业4.0 pillow 区块链 客户端 live555 rtsp rtp 向日葵 linux上传下载 编程 .net flask docker run 数据卷挂载 交互模式 WSL win11 无法解析服务器的名称或地址 sqlserver 豆瓣 追剧助手 迅雷 html frp rocketmq vSphere vCenter 软件定义数据中心 sddc OD机试真题 华为OD机试真题 服务器能耗统计 MacMini Mac 迷你主机 mini Apple ecmascript KVM linux环境变量 浪潮信息 AI服务器 aws googlecloud 大语言模型 LLMs UEFI Legacy MBR GPT U盘安装操作系统 seatunnel jmeter webrtc open webui 3d 数学建模 rag ragflow ragflow 源码启动 云原生开发 K8S k8s管理系统 Portainer搭建 Portainer使用 Portainer使用详解 Portainer详解 Portainer portainer AI写作 监控 自动化运维 openstack Xen ue4 着色器 虚幻 安防软件 list 模拟实现 EtherNet/IP串口网关 EIP转RS485 EIP转Modbus EtherNet/IP网关协议 EIP转RS485网关 EIP串口服务器 程序人生 目标跟踪 OpenVINO 推理应用 FTP 服务器 django cudnn anaconda 命名管道 客户端与服务端通信 mamba Vmamba Reactor C++ cpu 实时 使用 html5 firefox kamailio sip VoIP 大数据平台 网络结构图 threejs 3D 开发 vue okhttp OpenManus ipython burpsuite 安全工具 mac安全工具 burp安装教程 渗透工具 gpu算力 远程 执行 sshpass 操作 程序 性能分析 云桌面 微软 AD域控 证书服务器 TrueLicense 网易邮箱大师 Kali Linux 黑客 渗透测试 信息收集 YOLOv8 NPU Atlas800 A300I pro asi_bench web3.py Chatbox 生活 微信小程序 miniapp 真机调试 调试 debug 断点 网络API请求调试方法 HiCar CarLife+ CarPlay QT RK3588 jupyter ubuntu20.04 ros ros1 Noetic 20.04 apt 安装 ux firewalld 读写锁 远程工作 Cursor Doris搭建 docker搭建Doris Doris搭建过程 linux搭建Doris Doris搭建详细步骤 Doris部署 mount挂载磁盘 wrong fs type LVM挂载磁盘 Centos7.9 深度求索 私域 知识库 设置代理 实用教程 代码调试 ipdb adobe 大模型部署 腾讯云大模型知识引擎 python2 ubuntu24.04 中间件 iis iperf3 带宽测试 MQTT协议 消息服务器 代码 系统 黑苹果 权限 键盘 源码剖析 rtsp实现步骤 流媒体开发 报错 快捷键 旋转屏幕 自动操作 sqlite3 springcloud IM即时通讯 QQ 企业微信 剪切板对通 HTML FORMAT alias unalias 别名 飞牛nas fnos 本地化部署 远程看看 远程协助 VMware安装Ubuntu Ubuntu安装k8s YOLOv12 僵尸进程 AI-native Docker Desktop linux内核 tcp mq DigitalOcean GPU服务器购买 GPU服务器哪里有 GPU服务器 Invalid Host allowedHosts flash-attention WSL2 Ubuntu22.04 虚拟化 开发人员主页 Linux的权限 安卓模拟器 Google pay Apple pay 压测 ECS camera Arduino GCC aarch64 编译安装 HPC windows日志 NFS ping++ springboot Ubuntu Server Ubuntu 22.04.5 视觉检测 mongodb 话题通信 服务通信 Windsurf WebVM 流水线 脚本式流水线 minicom 串口调试工具 Linux 维护模式 go Linux无人智慧超市 LInux多线程服务器 QT项目 LInux项目 单片机项目 gpt log4j edge浏览器 webstorm file server http server web server Xinference RAGFlow Ubuntu DeepSeek DeepSeek Ubuntu DeepSeek 本地部署 DeepSeek 知识库 DeepSeek 私有化知识库 本地部署 DeepSeek DeepSeek 私有化部署 王者荣耀 系统安全 强制清理 强制删除 mac废纸篓 jar gradle TCP WebServer RAGFLOW 统信UOS bonding 链路聚合 database xml KylinV10 麒麟操作系统 Vmware 自动化编程 openwrt UOS 统信操作系统 selenium mybatis ollama下载加速 cmos 硬件 Erlang OTP gen_server 热代码交换 事务语义 MNN Qwen ROS 自动驾驶 HP Anyware es6 qt6.3 g726 群晖 游戏服务器 Minecraft Helm k8s集群 DOIT 四博智联 lsb_release /etc/issue /proc/version uname -r 查看ubuntu版本 RoboVLM 通用机器人策略 VLA设计哲学 vlm fot robot 视觉语言动作模型 具身智能 c 宝塔面板 Mermaid 可视化图表 自动化生成 ruoyi n8n dity make stable diffusion AI作画 Dell HPE 联想 浪潮 iDRAC R720xd copilot 重启 排查 系统重启 日志 原因 npm 实习 SSH 服务 SSH Server OpenSSH Server 远程桌面 gaussdb Attention NLP 压力测试 测试用例 功能测试 asp.net大文件上传 asp.net大文件上传源码 ASP.NET断点续传 asp.net上传文件夹 asp.net上传大文件 .net core断点续传 .net mvc断点续传 vasp安装 css 内网渗透 靶机渗透 服务器配置 单例模式 聊天室 ci/cd RAG 检索增强生成 文档解析 大模型垂直应用 AutoDL ufw HarmonyOS NEXT 原生鸿蒙 搜索引擎 searxng 开源软件 firewall 模拟器 教程 VPN wireguard uv Zoertier 内网组网 SSH 职场和发展 muduo X11 Xming 金仓数据库 2025 征文 数据库平替用金仓 飞牛 医疗APP开发 app开发 视频平台 录像 视频转发 性能测试 视频流 存储 矩阵 本地部署AI大模型 r语言 数据可视化 硅基流动 ChatBox minio 程序员 AP配网 AK配网 小程序AP配网和AK配网教程 WIFI设备配网小程序UDP开 进程程序替换 execl函数 execv函数 execvp函数 execvpe函数 putenv函数 top Linux top top命令详解 top命令重点 top常用参数 docker命令大全 Typore mariadb elk 显示管理器 lightdm gdm wsl vscode 1.86 网站搭建 serv00 jetty undertow grub 版本升级 扩容 HarmonyOS Tabs组件 TabContent TabBar TabsController 导航页签栏 滚动导航栏 远程登录 telnet Windows 大模型入门 大模型教程 弹性计算 云服务器 裸金属服务器 弹性裸金属服务器 banner lb 协议 scikit-learn 其他 强化学习 CentOS Stream CentOS DIFY Xterminal 网络攻击模型 DeepSeek行业应用 Heroku 网站部署 7z eNSP 网络规划 VLAN 企业网络 kernel Web服务器 多线程下载工具 网络编程 PYTHON 大模型面经 大模型学习 显卡驱动 AnythingLLM AnythingLLM安装 单元测试 游戏机 hugo 网页服务器 web服务器 Nginx Netty 即时通信 NIO 交叉编译 雨云 NPS SWAT 配置文件 服务管理 网络共享 gpt-3 文心一言 kvm llama.cpp 蓝桥杯 毕设 图形化界面 人工智能生成内容 换源 国内源 Debian MQTT 消息队列 金融 匿名管道 半虚拟化 硬件虚拟化 Hypervisor notepad opcua opcda KEPServer安装 容器技术 k8s资源监控 annotations自动化 自动化监控 监控service 监控jvm Linux Vim micropython esp32 mqtt 大模型微调 思科模拟器 Cisco 服务器扩容没有扩容成功 序列化反序列化 nuxt3 IO 算力 计算机 服务器管理 配置教程 服务器安装 网站管理 高效I/O ruby FTP服务器 wsl2 sqlite 游戏引擎 线程 openssl 密码学 tensorflow cfssl trae MS Materials 模拟退火算法 Python 拓扑图 crosstool-ng can 线程池 双系统 GRUB引导 Linux技巧 代码托管服务 clickhouse 社交电子 高效远程协作 TrustViewer体验 跨设备操作便利 智能远程控制 springboot远程调试 java项目远程debug docker远程debug java项目远程调试 springboot远程 rclone AList webdav fnOS Playwright 自动化测试 springsecurity6 oauth2 授权服务器 前后端分离 hibernate ssh远程登录 P2P HDLC 统信 UOS1070e 健康医疗 互联网医院 漏洞 技能大赛 vite 同步 备份 建站 框架搭建 安全威胁分析 智慧农业 开源鸿蒙 团队开发 Linux的基础指令 磁盘清理 图像处理 回显服务器 UDP的API使用 Java Applet URL操作 服务器建立 Socket编程 网络文件读取 分析解读 跨平台 直流充电桩 充电桩 RAID RAID技术 磁盘 微信开放平台 微信公众平台 微信公众号配置 W5500 OLED u8g2 TCP服务器 SEO chfs ubuntu 16.04 nacos etl wps EasyConnect 信号 openEuler MCP server C/S unity3d android studio safari yaml Ultralytics 可视化 Linux权限 权限命令 特殊权限 RustDesk自建服务器 rustdesk服务器 docker rustdesk 网络穿透 火绒安全 c/c++ 串口 历史版本 下载 Nuxt.js excel 多进程 AI代码编辑器 rustdesk 毕昇JDK 小智 致远OA OA服务器 服务器磁盘扩容 rnn 上传视频至服务器代码 vue3批量上传多个视频并预览 如何实现将本地视频上传到网页 element plu视频上传 ant design vue vue3本地上传视频及预览移除 宠物 毕业设计 免费学习 宠物领养 宠物平台 CORS 跨域 matlab nvidia 飞书 Kali MAC SecureCRT OpenSSH 小艺 Pura X dns uniapp 恒源云 SSL证书 边缘计算 Linux24.04 deepin vpn Docker引擎已经停止 Docker无法使用 WSL进度一直是0 镜像加速地址 Cookie 能力提升 面试宝典 技术 IT信息化 Isaac Sim 虚拟仿真 Web应用服务器 xrdp jina 开机自启动 产测工具框架 IMX6ULL 管理框架 XCC Lenovo Ubuntu 24.04.1 轻量级服务器 k8s二次开发 集群管理 移动云 CDN pgpool VSCode 业界资讯 国产数据库 瀚高数据库 数据迁移 下载安装 用户缓冲区 鲲鹏 端口测试 odoo 服务器动作 Server action 田俊楠 Docker Compose VR手套 数据手套 动捕手套 动捕数据手套 支付 微信支付 开放平台 ArkTs ArkUI glibc code-server mosquitto DBeaver composer nfs echarts 信息可视化 网页设计 SenseVoice SSL 域名 skynet 输入法 硬件架构 TRAE av1 电视盒子 机顶盒ROM 魔百盒刷机 AISphereButler Mac内存不够用怎么办 embedding ELF加载 cocoapods wsgiref Web 服务器网关接口 nac 802.1 portal 联网 easyconnect 代理 国产操作系统 虚拟机安装 浏览器自动化 版本 宝塔 Carla 智能驾驶 java-rocketmq Jellyfin keepalived yolov8 LORA sonoma 自动更新 渗透 lio-sam SLAM xshell termius iterm2 ukui 麒麟kylinos openeuler neo4j 数据仓库 数据库开发 数据库架构 Node-Red 编程工具 流编程 Kylin-Server chrome devtools chromedriver 黑客技术 CosyVoice ubuntu24.04.1 URL NLP模型 web ArcTS 登录 ArcUI GridItem 自学笔记 小米 澎湃OS Android VPS arkUI pyqt 华为证书 HarmonyOS认证 华为证书考试 ceph pdf samba 图片增强 增强数据 昇腾 npu open Euler dde PX4 MAVROS 四旋翼无人机 RTMP 应用层 trea idea big data opensearch helm 可信计算技术 安全架构 fork wait waitpid exit fast 服务器主板 AI芯片 curl wget bat su sudo 大模型应用 MI300x 孤岛惊魂4 WebRTC 工作流 workflow sysctl.conf vm.nr_hugepages 工具 大文件分片上传断点续传及进度条 如何批量上传超大文件并显示进度 axios大文件切片上传详细教 node服务器合并切片 vue3大文件上传报错提示错误 vu大文件秒传跨域报错cors IPv4 子网掩码 公网IP 私有IP SSH 密钥生成 SSH 公钥 私钥 生成 聊天服务器 套接字 Socket 智能音箱 智能家居 IPMITOOL BMC 硬件管理 linux 命令 sed 命令 OpenHarmony string模拟实现 深拷贝 浅拷贝 经典的string类问题 三个swap TrinityCore 魔兽世界 redhat 自动化任务管理 命令行 基础入门 prompt easyui langchain asp.net大文件上传下载 dash 正则表达式 文件分享 kotlin iphone 繁忙 服务器繁忙 解决办法 替代网站 汇总推荐 AI推理 图形渲染 云服务 雨云服务器 token sas dba 小游戏 五子棋 vmamba 崖山数据库 YashanDB ocr sdkman k8s部署 MySQL8.0 高可用集群(1主2从) ShenTong 国产化 python3.11 服务器部署ai模型 yum源切换 更换国内yum源 服务器无法访问 ip地址无法访问 无法访问宝塔面板 宝塔面板打不开 IIS .net core Hosting Bundle .NET Framework vs2022 rsyslog 机柜 1U 2U Anolis nginx安装 环境安装 linux插件下载 高效日志打印 串口通信日志 服务器日志 系统状态监控日志 异常记录日志 代码规范 saltstack 服务器数据恢复 数据恢复 存储数据恢复 raid5数据恢复 磁盘阵列数据恢复 visualstudio 银河麒麟操作系统 内核 相机 远程过程调用 Windows环境 deep learning Linux awk awk函数 awk结构 awk内置变量 awk参数 awk脚本 awk详解 mysql离线安装 ubuntu22.04 mysql8.0 三级等保 服务器审计日志备份 源码 VMware Tools vmware tools安装 vmwaretools安装步骤 vmwaretools安装失败 vmware tool安装步骤 vm tools安装步骤 vm tools安装后不能拖 vmware tools安装步骤 宝塔面板无法访问 安装MySQL 混合开发 JDK 无桌面 DeepSeek r1 Open WebUI 考研 在线office 内网环境 bootstrap 免费域名 域名解析 京东云 nextjs react reactjs visual studio conda配置 conda镜像源 eclipse 热榜 流式接口 api webgl React Next.js 开源框架 联想开天P90Z装win10 Reactor反应堆 云耀服务器 less 云电竞 云电脑 todesk 上传视频文件到服务器 uniApp本地上传视频并预览 uniapp移动端h5网页 uniapp微信小程序上传视频 uniapp app端视频上传 uniapp uview组件库 小番茄C盘清理 便捷易用C盘清理工具 小番茄C盘清理的优势尽显何处? 教你深度体验小番茄C盘清理 C盘变红?!不知所措? C盘瘦身后电脑会发生什么变化? 大文件秒传跨域报错cors 办公自动化 pdf教程 网工 chrome 浏览器下载 chrome 下载安装 谷歌浏览器下载 x64 SIGSEGV SSE xmm0 稳定性 看门狗 ssrf 失效的访问控制 宕机切换 服务器宕机 Unity Dedicated Server Host Client 无头主机 xfce 数据管理 数据治理 数据编织 数据虚拟化 arcgis idm MySql lvm 磁盘挂载 磁盘分区 ROS2 邮件APP 免费软件 zip unzip 软链接 硬链接 powerbi ebpf uprobe GameFramework HybridCLR Unity编辑器扩展 自动化工具 环境迁移 树莓派 VNC db 飞牛NAS 飞牛OS MacBook Pro harmonyOS面试题 thingsboard cnn pyautogui XFS xfs文件系统损坏 I_O error 直播推流 es qemu libvirt 网络爬虫 ftp服务 文件上传 bot Docker 相差8小时 UTC 时间 next.js 部署next.js 推荐算法 netty Qualcomm WoS QNN AppBuilder ArkTS 移动端开发 GPU FunASR ASR opengl 电视剧收视率分析与可视化平台 cd 目录切换 状态管理的 UDP 服务器 Arduino RTOS MDK 嵌入式开发工具 论文笔记 gitea 智能电视 risc-v Wi-Fi apt virtualbox DocFlow 信号处理 ubuntu24 vivado24 make命令 makefile文件 k8s集群资源管理 大屏端 执法记录仪 智能安全帽 smarteye 设备 PCI-Express Ark-TS语言 SysBench 基准测试 服务器时间 ecm bpm 开机黑屏 xpath定位元素 镜像 navicat 数码 联机 僵尸毁灭工程 游戏联机 开服 欧标 OCPP 进程间通信 备份SQL Server数据库 数据库备份 傲梅企业备份网络版 Ubuntu 24 常用命令 Ubuntu 24 Ubuntu vi 异常处理 lua 音乐服务器 Navidrome 音流 软件卸载 系统清理 烟花代码 烟花 元旦 vllm 蓝牙 免密 公钥 私钥 企业网络规划 华为eNSP 笔灵AI AI工具 ollama api ollama外网访问 网络建设与运维 网络搭建 神州数码 神州数码云平台 云平台 ECT转485串口服务器 ECT转Modbus485协议 ECT转Modbus串口服务器 音乐库 沙盒 AI员工 dns是什么 如何设置电脑dns dns应该如何设置 AI agent iventoy VmWare OpenEuler 反向代理 PPI String Cytoscape CytoHubba 知识图谱 deekseek Radius 抗锯齿 uni-file-picker 拍摄从相册选择 uni.uploadFile H5上传图片 微信小程序上传图片 scapy 物联网开发 LInux EMQX 通信协议 计算虚拟化 弹性裸金属 Qwen2.5-VL 负载测试 嵌入式Linux IPC react native EMUI 回退 降级 升级 nosql 做raid 装系统 MVS 海康威视相机 游戏开发 磁盘镜像 服务器镜像 服务器实时复制 实时文件备份 多端开发 智慧分发 应用生态 鸿蒙OS 远程服务 dock 加速 Python基础 Python教程 Python技巧 proxy模式 性能调优 安全代理 影刀 #影刀RPA# 7-zip VM搭建win2012 win2012应急响应靶机搭建 攻击者获取服务器权限 上传wakaung病毒 应急响应并溯源 挖矿病毒处置 应急响应综合性靶场 银河麒麟桌面操作系统 Kylin OS 高德地图 鸿蒙接入高德地图 HarmonyOS5.0 xss Claude AD域 win服务器架设 windows server 我的世界服务器搭建 minecraft Apache Beam 批流统一 案例展示 数据分区 容错机制 sublime text3 CPU 使用率 系统监控工具 linux 命令 iTerm2 HTTP 服务器控制 ESP32 DeepSeek 查询数据库服务IP地址 SQL Server ajax 分布式训练 figma netlink libnl3 语音识别 元服务 应用上架 GoogLeNet 授时服务 北斗授时 qt项目 qt项目实战 qt教程 国标28181 视频监控 监控接入 语音广播 流程 SIP SDP word 状态模式 底层实现 多路转接 根服务器 华为机试 USB网络共享 umeditor粘贴word ueditor粘贴word ueditor复制word ueditor上传word图片 ueditor导入word ueditor导入pdf ueditor导入ppt 工具分享 对比 meld DiffMerge 一切皆文件 自定义客户端 SAS 银河麒麟高级服务器 外接硬盘 Kylin VS Code 浏览器开发 AI浏览器 项目部署 mysql安装报错 windows拒绝安装 小智AI服务端 xiaozhi TTS AD 域管理 laravel 服务器部署 本地拉取打包 armbian u-boot junit 冯诺依曼体系 论文阅读 hexo pyicu deepseak 豆包 KIMI 腾讯元宝 CNNs 图像分类 retry 重试机制 内网服务器 内网代理 内网通信 ftp lighttpd安装 Ubuntu配置 Windows安装 服务器优化 CVE-2024-7347 resolv.conf 本地知识库部署 DeepSeek R1 模型 需求分析 规格说明书 备选 网站 调用 示例 vscode1.86 1.86版本 ssh远程连接 ranger MySQL8.0 LLM Web APP Streamlit deployment daemonset statefulset cronjob grep Ardupilot 北亚数据恢复 oracle数据恢复 大版本升 升级Ubuntu系统 kerberos IO模型 大模型训练/推理 推理问题 mindie MacOS HistoryServer Spark YARN jobhistory 程序化交易 量化交易 高频交易 wpf 环境搭建 Maven seleium mcp服务器 client close TCP协议 Headless Linux 计算机科学与技术 Office WINCC 达梦 DM8 钉钉 Logstash 日志采集 端口聚合 windows11 flink 大大通 第三代半导体 碳化硅 System V共享内存 进程通信 ai工具 cmake v10 软件 docker desktop image h.264 Linux环境 项目部署到linux服务器 项目部署过程 超融合 服务网格 istio 软负载 js pppoe radius AI Agent 字节智能运维 华为OD 可以组成网络的服务器 Ubuntu 22.04 算家云 算力租赁 网卡的名称修改 eth0 ens33 cpp-httplib 端口 查看 ss 合成模型 扩散模型 图像生成 web3 NAT转发 NAT Server 空间 查错 语法 单一职责原则 docker部署翻译组件 docker部署deepl docker搭建deepl java对接deepl 翻译组件使用 IMM 鸿蒙开发 移动开发 tidb GLIBC WireGuard 异地组网 捆绑 链接 谷歌浏览器 youtube google gmail 实时内核 nohup后台启动 rpa sequoiaDB 常用命令 文本命令 目录命令 sudo原理 su切换 代理服务器 Linux find grep DevOps 软件交付 数据驱动 应用场景 csrutil mac恢复模式进入方法 恢复模式 Docker快速入门 僵尸世界大战 游戏服务器搭建 prometheus数据采集 prometheus数据模型 prometheus特点 zookeeper qps 高并发 抓包工具 极限编程 swoole 阿里云ECS ldap yashandb regedit 开机启动 AzureDataStudio GIS 遥感 WebGIS iNode Macos MobaXterm 文件传输 qt5 客户端开发 多产物 架构与原理 Xshell 多个客户端访问 IO多路复用 TCP相关API 李心怡 玩机技巧 软件分享 软件图标 端口号 开放端口 访问列表 IDEA 网络文件系统 gunicorn tailscale derp derper 中转 triton 模型分析 充电桩平台 充电桩开源平台 perl ShapeFile GeoJSON 查看显卡进程 fuser ArtTS vue-i18n 国际化多语言 vue2中英文切换详细教程 如何动态加载i18n语言包 把语言json放到服务器调用 前端调用api获取语言配置文件 g++ g++13 docker部署Python whistle 在线预览 xlsx xls文件 在浏览器直接打开解析xls表格 前端实现vue3打开excel 文件地址url或接口文档流二进 H3C nohup 异步执行 运维监控 桌面环境 移动魔百盒 IMX317 MIPI H265 VCU 搭建个人相关服务器 midjourney diskgenius 日志分析 系统取证 DenseNet 信创 信创终端 中科方德 sentinel 玩游戏 佛山戴尔服务器维修 佛山三水服务器维修 deepseek r1 wordpress 无法访问wordpess后台 打开网站页面错乱 linux宝塔面板 wordpress更换服务器 个人博客 增强现实 沉浸式体验 技术实现 案例分析 AR 干货分享 黑客工具 密码爆破 Spring Security 技术共享 我的世界 我的世界联机 beautifulsoup 虚幻引擎 glm4 问题解决 加解密 Yakit yaklang Mac软件 EVE-NG Sealos env 变量 shell脚本免交互 expect linux免交互 线性代数 电商平台 怎么卸载MySQL MySQL怎么卸载干净 MySQL卸载重新安装教程 MySQL5.7卸载 Linux卸载MySQL8.0 如何卸载MySQL教程 MySQL卸载与安装 阻塞队列 生产者消费者模型 服务器崩坏原因 终端工具 远程工具 C++软件实战问题排查经验分享 0xfeeefeee 0xcdcdcdcd 动态库加载失败 程序启动失败 程序运行权限 标准用户权限与管理员权限 hosts ISO镜像作为本地源 fiddler llamafactory 微调 服务器正确解析请求体 电脑桌面出现linux图标 电脑桌面linux图标删除不了 电脑桌面Liunx图标删不掉 linux图标删不掉 全文检索 图搜索算法 零售 信息安全 Unity插件 SVN Server tortoise svn 用户管理 金仓数据库概述 金仓数据库的产品优化提案 蓝桥杯C++组 postgres Dify重启后重新初始化 搜狗输入法 中文输入法 autoware 大模型推理 autodl 子系统 localhost 行情服务器 股票交易 速度慢 切换 股票量化接口 股票API接口 输入系统 智能合约 哈希算法 solidworks安装 电路仿真 multisim 硬件工程师 硬件工程师学习 电路图 电路分析 仪器仪表 pyside6 界面 HAProxy ABAP openjdk Ubuntu20.04 2.35 桌面快捷方式 新盘添加 partedUtil GRE 风扇控制软件 网络原理 Echarts图表 折线图 柱状图 异步动态数据 可视化效果 linuxdeployqt 打包部署程序 appimagetool 弹性服务器 webview termux initramfs Linux内核 Grub 服务器ssl异常解决 配置原理 分布式账本 信任链 共识算法 存储维护 NetApp存储 EMC存储 ardunio BLE Pyppeteer 显示器 动静态库 安全漏洞 内存管理 接口优化 Claude Desktop Claude MCP Windows Cli MCP 显卡驱动持久化 GPU持久化 kubeless 西门子PLC 通讯 IP配置 netplan IPv6 IPv6测试 IPv6测速 IPv6检测 IPv6查询 GeneCards OMIM TTD 分子对接 autodock mgltools PDB PubChem css3 解决方案 终端 UDP 管道 java-zookeeper zerotier yum换源 锁屏不生效 NFC 近场通讯 智能门锁 VGG网络 卷积层 池化层 chromium dpi 打不开xxx软件 无法检查其是否包含恶意软件 arkTs LVM lvresize 磁盘扩容 pvcreate SystemV 数字证书 签署证书 macOS 框架 考试 智能体开发 archlinux kde plasma AimRT mm-wiki搭建 linux搭建mm-wiki mm-wiki搭建与使用 mm-wiki使用 mm-wiki详解 粘包问题 录音麦克风权限判断检测 录音功能 录音文件mp3播放 小程序实现录音及播放功能 RecorderManager 解决录音报错播放没声音问题 带外管理 流量运营 GPU训练 脚本 券商 股票交易接口api 类型 特点 星河版 区块链项目 PyQt PySide6 飞腾处理器 三次握手 ubuntu安装 linux入门小白 Webserver 异步 servlet hdc Python 视频爬取教程 Python 视频爬取 Python 视频教程 NVML nvidia-smi pthread RDP Bug解决 Qt platform OpenCV 支持向量机 文件存储服务器组件 post.io 企业邮箱 搭建邮箱 设备树 动态规划 鸿蒙NEXT macbook 机械臂 rtc 软件开发 logstash 免费 NVM Node Yarn PM2 商用密码产品体系 c/s IPv4/IPv6双栈 双栈技术 网路规划设计 ensp综合实验 IPv4过渡IPv6 IPv4与IPv6 设计规范 物理地址 页表 虚拟地址 跨域请求 massa sui aptos sei mvc element-ui 上传视频并预览视频 vue上传本地视频及进度条功能 vue2选择视频上传到服务器 upload上传视频组件插件 批量上传视频 限制单个上传视频 海康 源代码管理 #STC8 #STM32