学习 Linux:如何创建线程 (linux 线程创建线程)

Linux是开源的操作系统,广泛用于服务器和嵌入式设备,它的高度可定制性和稳定性是其吸引开发者的重要因素之一。Linux中的线程是轻量级进程,具有更快的创建和上下文切换速度以及更低的资源消耗。本文将介绍如何使用Linux API创建线程。

创建线程的步骤

创建线程的过程包括以下步骤:

1. 包含头文件

在使用线程相关函数之前,需要包含相关的头文件。

“`c

#include

“`

2. 定义线程入口函数

线程入口函数是在线程启动后所执行的函数,该函数的返回值类型必须是void *,并且参数也必须是void *类型。我们可以在函数内部进行需要线程进行的操作。例如,下面的代码定义了一个简单的线程入口函数,它会输出一条消息。

“`c

void* thread_func(void* arg) {

printf(“Hello, world!\n”);

return NULL;

}

“`

3. 创建线程

使用pthread_create函数创建线程,该函数最多接受四个参数:存储线程ID的指针、线程属性、线程入口函数和传递给线程入口函数的参数指针。如果您不需要使用线程属性,则可以将第二个参数传递为NULL。

“`c

pthread_t tid;

pthread_create(&tid, NULL, thread_func, NULL);

“`

4. 等待线程结束

使用pthread_join函数等待线程结束。该函数会阻塞当前线程,直到指定的线程完成执行,然后该函数才会返回。

“`c

pthread_join(tid, NULL);

“`

完整的代码

下面是一个完整的示例代码,它创建了一个线程并等待直到线程结束执行。

“`c

#include

#include

void* thread_func(void* arg) {

printf(“Hello, world!\n”);

return NULL;

}

int mn() {

pthread_t tid;

pthread_create(&tid, NULL, thread_func, NULL);

pthread_join(tid, NULL);

return 0;

}

“`

编译并运行该程序,可以看到如下输出:

“`

Hello, world!

“`

多线程示例

下面的示例代码展示了如何使用多个线程并行执行任务。

“`c

#include

#include

void* worker(void* arg) {

int id = *((int*)arg);

printf(“Worker %d started.\n”, id);

for (int i = 0; i

printf(“Worker %d finished.\n”, id);

return NULL;

}

int mn() {

const int num_threads = 4;

pthread_t tids[num_threads];

int args[num_threads];

for (int i = 0; i

args[i] = i;

pthread_create(&tids[i], NULL, worker, &args[i]);

}

for (int i = 0; i

pthread_join(tids[i], NULL);

}

return 0;

}

“`

该程序会创建四个线程,每个线程都会执行worker函数。这里,worker函数模拟了一个计算密集型任务,每个线程都会执行一段时间。程序等待所有线程完成执行。

线程同步

在多线程编程中,由于多个线程可以同时访问共享的资源,会引发很多问题,其中最常见的问题是数据竞态。数据竞态是指多个线程在并发执行时,涉及到共享数据的操作中,由于缺乏同步,导致结果不可预知的情况。为了避免这些问题,必须使用同步机制。

Linux提供了多种同步机制,其中最常用的包括互斥锁和条件变量。互斥锁用于控制对共享资源的访问,并保证在任何时候最多只有一个线程访问该资源。条件变量用于在多个线程之间进行通信。线程可以等待条件变量的信号,并在收到信号后执行相应的操作。

互斥锁示例

下面是一个简单的互斥锁示例,它使用互斥锁来保护一个共享的计数器。

“`c

#include

#include

pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER;

int counter = 0;

void* worker(void* arg) {

for (int i = 0; i

pthread_mutex_lock(&mutex);

counter++;

pthread_mutex_unlock(&mutex);

}

return NULL;

}

int mn() {

const int num_threads = 4;

pthread_t tids[num_threads];

for (int i = 0; i

pthread_create(&tids[i], NULL, worker, NULL);

}

for (int i = 0; i

pthread_join(tids[i], NULL);

}

printf(“Counter value: %d\n”, counter);

return 0;

}

“`

该程序创建了四个线程,每个线程会对计数器执行1000000次加1操作。由于该操作涉及到共享的数据,我们使用互斥锁保护计数器,确保每个线程都可以独占计数器。程序会输出计数器的最终值。

条件变量示例

下面是一个简单的条件变量示例,它使用条件变量和互斥锁来实现一个有限缓冲区,生产者线程向缓冲区写入数据,消费者线程从缓冲区读取数据。

“`c

#include

#include

#define BUFFER_SIZE 10

int buffer[BUFFER_SIZE];

int num_items = 0;

pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER;

pthread_cond_t can_produce = PTHREAD_COND_INITIALIZER;

pthread_cond_t can_consume = PTHREAD_COND_INITIALIZER;

void* producer(void* arg) {

for (int i = 0; i

pthread_mutex_lock(&mutex);

while (num_items == BUFFER_SIZE) {

pthread_cond_wt(&can_produce, &mutex);

}

buffer[num_items++] = i;

printf(“Produced item %d\n”, i);

pthread_cond_signal(&can_consume);

pthread_mutex_unlock(&mutex);

}

return NULL;

}

void* consumer(void* arg) {

for (int i = 0; i

pthread_mutex_lock(&mutex);

while (num_items == 0) {

pthread_cond_wt(&can_consume, &mutex);

}

int item = buffer[–num_items];

printf(“Consumed item %d\n”, item);

pthread_cond_signal(&can_produce);

pthread_mutex_unlock(&mutex);

}

return NULL;

}

int mn() {

pthread_t producer_tid, consumer_tid;

pthread_create(&producer_tid, NULL, producer, NULL);

pthread_create(&consumer_tid, NULL, consumer, NULL);

pthread_join(producer_tid, NULL);

pthread_join(consumer_tid, NULL);

return 0;

}

“`

该示例中,生产者线程和消费者线程共享了一个有限大小的缓冲区,生产者线程向缓冲区写入数据,消费者线程从缓冲区读取数据,如果缓冲区已满,则生产者线程等待条件变量can_produce的信号;如果缓冲区为空,则消费者线程等待条件变量can_consume的信号。当生产者写入数据后,会发送信号给消费者,表示缓冲区已经可以消费;同样地,当消费者读取数据后,会发送信号给生产者,表示缓冲区已经可以生产。


数据运维技术 » 学习 Linux:如何创建线程 (linux 线程创建线程)