当前位置:首页 > 技术知识 > 正文内容

C语言编写多线程,什么时候要使用互斥锁?为什么要使用互斥锁?

maynowei9个月前 (08-03)技术知识148

在多线程编程中,当多个线程同时访问共享资源(如变量、文件等)时,会出现竞态条件(Race Condition)问题,导致程序的行为不可预测。为了避免这种问题,需要使用互斥锁来保护共享资源的访问。

互斥锁是一种线程同步机制,它保证同一时刻只有一个线程可以访问共享资源,其他线程需要等待该线程释放锁才能继续访问。在C语言中,可以使用标准库提供的pthread_mutex_t结构体来实现互斥锁。

在哪些情况下需要使用互斥锁呢?一般来说,只有在多个线程同时访问共享资源时才需要使用互斥锁。以下是一些常见的情况:

  1. 多个线程同时访问同一个全局变量或静态变量。
  2. 多个线程同时访问同一个动态分配的内存块。
  3. 多个线程同时访问同一个文件或网络套接字。
  4. 多个线程同时访问同一个数据结构,如链表或树。

需要注意的是,过多地使用互斥锁可能会导致性能问题。因此,在使用互斥锁时,需要权衡程序的正确性和性能开销,尽可能减少锁的使用次数。

假设有两个线程 A 和 B,它们需要同时访问一个共享资源:一个全局变量 x。线程 A 需要读取 x 的值并将其加 1,线程 B 需要读取 x 的值并将其减 1。

如果不使用互斥锁,可能会出现以下情况:

  1. 线程 A 读取 x 的值为 n,然后将其加 1,得到 n+1。
  2. 在线程 A 执行完加 1 操作之前,线程 B 读取 x 的值为 n,然后将其减 1,得到 n-1。
  3. 线程 A 将 n+1 写回到 x 中,x 的值变成了 n+1。
  4. 线程 B 将 n-1 写回到 x 中,x 的值变成了 n-1,而不是原本的 n。

为了避免这种竞态条件问题,可以使用互斥锁来保护 x 的访问。具体地,在访问 x 的代码前加锁,在访问结束后释放锁。这样可以保证同一时刻只有一个线程可以访问 x,从而避免竞态条件问题。

下面是一个简单的C语言使用互斥锁保护全局变量的例子:

#include <stdio.h>
#include <stdlib.h>
#include <pthread.h>

pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER;  // 定义互斥锁

int global_var = 0;

void *thread_func(void *arg) {
    int thread_num = *(int *)arg;
    int i;
    for (i = 0; i < 100000; i++) {
        pthread_mutex_lock(&mutex);  // 加锁
        global_var++;
        printf("Thread %d: global_var = %d\n", thread_num, global_var);
        pthread_mutex_unlock(&mutex);  // 解锁
    }
    return NULL;
}

int main() {
    pthread_t thread1, thread2;
    int thread_num1 = 1, thread_num2 = 2;

    // 创建两个线程
    if (pthread_create(&thread1, NULL, thread_func, &thread_num1) != 0) {
        perror("pthread_create");
        exit(EXIT_FAILURE);
    }
    if (pthread_create(&thread2, NULL, thread_func, &thread_num2) != 0) {
        perror("pthread_create");
        exit(EXIT_FAILURE);
    }

    // 等待两个线程结束
    if (pthread_join(thread1, NULL) != 0) {
        perror("pthread_join");
        exit(EXIT_FAILURE);
    }
    if (pthread_join(thread2, NULL) != 0) {
        perror("pthread_join");
        exit(EXIT_FAILURE);
    }

    return 0;
}

在上面的代码中,我们定义了一个全局变量 global_var,然后创建了两个线程 thread1 和 thread2,它们会分别对 global_var 进行加 1 操作。在访问 global_var 的代码前,我们使用了 pthread_mutex_lock 函数来加锁,在访问结束后使用 pthread_mutex_unlock 函数来解锁,以保证同一时刻只有一个线程可以访问 global_var。

相关文章

Django 官方推荐的姿势:类视图(django类视图和函数视图哪个好)

作者:HelloGitHub-追梦人物在开发网站的过程中,有一些视图函数虽然处理的对象不同,但是其大致的代码逻辑是一样的。比如一个博客和一个论坛,通常其首页都是展示一系列的文章列表或者帖子列表。对处理...

C++ 原子操作与锁的深度解析:为什么原子操作并非万金油?

大噶好,我是henry,今天来和大家浅浅聊一下为啥C++原子操作并非万能钥匙,原因有三,且听我娓娓道来:一、原子操作的线程安全性C++11 的 std::atomic 确实为单个变量的线程安全操作提供...

掌握C语言多线程:高效并发编程指南

一、多线程基础概念介绍多线程编程是现代软件开发中提高程序性能和响应性的重要技术。在C语言中,pthread(POSIX Threads)库是实现多线程编程的标准工具。本节将通俗易懂地介绍多线程的核心概...

Linux C++实现多线程同步的四种方式(超级详细)

背景问题:在特定的应用场景下,多线程不进行同步会造成什么问题?通过多线程模拟多窗口售票为例:#include <iostream>#include<pthread.h>#inc...

C++26中同步与原子操作新变化(c++ 同步)

引言随着多核处理器和并发编程的普及,C++26进一步增强了对同步与原子操作的支持,为开发者提供了更高效、更安全的工具来应对多线程编程中的数据竞争与同步挑战。自C++11引入原子操作以来,C++标准库在...

c++ 继承简介(c++继承的概念)

24.1 — 继承简介2024 年 6 月 5 日在上一章中,我们讨论了对象组合,即从更简单的类和类型构建复杂类。对象组合非常适合构建与其部分具有“has-a”关系的新对象。但是,对象组合只是 C++...