strerror线程安全分析

本文探讨了strerror和errno在多线程环境下的线程安全性问题,并提供了strerror和其线程安全版本strerror_r的具体实现源码分析。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

导读

strerror是否线程安全了? 1

errno是否线程安全? 1

1strerror源码 2

2__strerror_r源码 2

 

strerror是否线程安全了?

答案是NO,但它有个线程安全的版本:strerror_r。借助Linuxman,即可看到详情:

#include <string.h>

char *strerror(int errnum);

int strerror_r(int errnum, char *buf, size_t buflen); /* GNU-specific */

 

那么,在多线程中使用strerror是否安全了?答案是不一定,一定情况下也是非常安全的。

不安全会造成内存违规访问吗?也就是会发生SIGSEGV吗?答案是NO,仍是内存安全的,但是可能会返回错乱的字符串。

 

那么,在多线程程序中,什么情况下使用strerror是绝对安全的了?如果参数errnum是一个已知的errno,则使用strerror是绝对安全的,也就是会返回期待的字符串,而不会出现乱码

对比strerror源码,是因为strerror会在下述直接返回:

  if (__builtin_expect (ret != NULL, 1))

    return ret;

而这走的是_strerror_r中的分支:

return (char *) _(_sys_errlist_internal[errnum]);

errno是否线程安全?

答案是不一定,取决于编译宏:

#  if !defined _LIBC || defined _LIBC_REENTRANT

/* When using threads, errno is a per-thread value.  */

#   define errno (*__errno_location ())

#  endif

 

可以通过下段小代码,来确定默认时是否有定义,如果没有,则需要在编译时加上:

#include <stdio.h>

 

int main()

{

        #ifdef _GNU_SOURCE

                printf("_GNU_SOURCE defined\n");

        #else

                printf("_GNU_SOURCE not defined\n");

        #endif

 

        #ifdef _LIBC_REENTRANT

                printf("_LIBC_REENTRANT defined\n");

        #else

                printf("_LIBC_REENTRANT not defined\n");

        #endif

 

        #ifdef _LIBC

                printf("_LIBC defined\n");

        #else

                printf("_LIBC not defined\n");

        #endif

 

        return 0;

}

 

1strerror源码

// glibc-2.14\string\strerror.c

#include <errno.h>

 

/* Return a string describing the errno code in ERRNUM.

   The storage is good only until the next call to strerror.

   Writing to the storage causes undefined behavior.  */

libc_freeres_ptr (static char *buf);

 

char *

strerror (errnum)

     int errnum;

{

  char *ret = __strerror_r (errnum, NULL, 0);

  int saved_errno;

 

  if (__builtin_expect (ret != NULL, 1))

    return ret;

 

  saved_errno = errno;

  if (buf == NULL)

    buf = malloc (1024);

  __set_errno (saved_errno);

  if (buf == NULL)

    return _("Unknown error");

  return __strerror_r (errnum, buf, 1024);

}

2__strerror_r源码

// glibc-2.14\string\_strerror.c

/* Return a string describing the errno code in ERRNUM.  */

char *

__strerror_r (int errnum, char *buf, size_t buflen)

{

  if (__builtin_expect (errnum < 0 || errnum >= _sys_nerr_internal

|| _sys_errlist_internal[errnum] == NULL, 0))

    {

      /* Buffer we use to print the number in.  For a maximum size for

 `int' of 8 bytes we never need more than 20 digits.  */

      char numbuf[21];

      const char *unk = _("Unknown error ");

      size_t unklen = strlen (unk);

      char *p, *q;

      bool negative = errnum < 0;

 

      numbuf[20] = '\0';

      p = _itoa_word (abs (errnum), &numbuf[20], 10, 0);

 

      /* Now construct the result while taking care for the destination

 buffer size.  */

      q = __mempcpy (buf, unk, MIN (unklen, buflen));

      if (negative && unklen < buflen)

{

  *q++ = '-';

  ++unklen;

}

      if (unklen < buflen)

memcpy (q, p, MIN ((size_t) (&numbuf[21] - p), buflen - unklen));

 

      /* Terminate the string in any case.  */

      if (buflen > 0)

buf[buflen - 1] = '\0';

 

      return buf;

    }

 

  return (char *) _(_sys_errlist_internal[errnum]);

}

weak_alias (__strerror_r, strerror_r)

libc_hidden_def (__strerror_r)

 

 

<think>好的,用户现在问的是关于C语言多线程开发的结构示例。首先,我需要确定用户的需求是什么。他们可能是在学习多线程编程,或者正在项目中遇到多线程相关的问题,想要一个示例来参考结构。 接下来,我需要回忆C语言中多线程开发的主要方法。C语言本身不直接支持多线程,但可以通过POSIX线程(pthread)库来实现。因此,回答应该基于pthread库的使用。 用户可能需要的不仅仅是一个简单的示例,而是结构化的讲解,包括线程创建、同步机制、数据传递等。需要考虑到线程安全、资源竞争等问题,所以应该包括互斥锁、条件变量等同步机制的例子。 另外,用户可能对实际应用场景感兴趣,比如生产者-消费者模型,线程池等。但作为基础的结构示例,可能先从简单的例子入手,再逐步深入。 需要检查示例代码的正确性,确保编译命令和关键函数如pthread_create、pthread_join、mutex的使用正确无误。同时,要提醒用户注意编译时链接pthread库,比如使用gcc时要加-lpthread参数。 还要考虑到常见的错误和注意事项,比如避免死锁,正确初始化互斥锁和条件变量,线程函数的返回值和参数传递方式。可能用户在使用过程中会遇到这些问题,提前在示例中说明可以帮他们避免。 此外,是否需要介绍线程的生命周期,或者分离线程与可连接线程的区别?根据用户的问题,可能不需要太深入,但简要提及可能有助于理解。 最后,总结多线程开发的最佳实践,比如资源管理、错误处理、避免全局变量等,帮助用户建立良好的编程习惯。</think>以下是一个详细的C语言多线程开发结构示例,包含线程创建、同步通信和资源管理的关键实现: ```c #include <stdio.h> #include <stdlib.h> #include <pthread.h> #include <unistd.h> // 共享数据结构 typedef struct { int counter; pthread_mutex_t mutex; pthread_cond_t cond; } SharedData; #define THREAD_NUM 4 #define MAX_TASKS 10 // 线程参数结构 typedef struct { int id; SharedData* data; } ThreadParam; // 线程函数原型 void* worker_thread(void* arg); int main() { pthread_t threads[THREAD_NUM]; SharedData shared_data = { .counter = 0, .mutex = PTHREAD_MUTEX_INITIALIZER, .cond = PTHREAD_COND_INITIALIZER }; // 创建线程池 ThreadParam params[THREAD_NUM]; for (int i = 0; i < THREAD_NUM; ++i) { params[i].id = i + 1; params[i].data = &shared_data; if (pthread_create(&threads[i], NULL, worker_thread, &params[i]) != 0) { perror("Thread creation failed"); exit(EXIT_FAILURE); } } // 主线程产生任务 for (int task = 1; task <= MAX_TASKS; ++task) { pthread_mutex_lock(&shared_data.mutex); while (shared_data.counter != 0) { pthread_cond_wait(&shared_data.cond, &shared_data.mutex); } shared_data.counter = task; printf("[Main] 生成任务 %d\n", task); pthread_cond_broadcast(&shared_data.cond); pthread_mutex_unlock(&shared_data.mutex); sleep(1); // 模拟任务生成间隔 } // 等待线程结束 for (int i = 0; i < THREAD_NUM; ++i) { pthread_join(threads[i], NULL); } // 销毁同步对象 pthread_mutex_destroy(&shared_data.mutex); pthread_cond_destroy(&shared_data.cond); return 0; } // 工作线程实现 void* worker_thread(void* arg) { ThreadParam* param = (ThreadParam*)arg; SharedData* data = param->data; while (1) { pthread_mutex_lock(&data->mutex); // 等待任务到达 while (data->counter == 0) { pthread_cond_wait(&data->cond, &data->mutex); } // 获取任务 int task = data->counter; if (task > MAX_TASKS) { data->counter = -1; // 终止信号 pthread_cond_broadcast(&data->cond); pthread_mutex_unlock(&data->mutex); break; } data->counter = 0; // 重置任务状态 pthread_cond_signal(&data->cond); pthread_mutex_unlock(&data->mutex); // 执行任务 printf("[Thread %d] 处理任务 %d\n", param->id, task); sleep(task % 3 + 1); // 模拟任务处理时间 } printf("[Thread %d] 退出\n", param->id); return NULL; } ``` 🔧 编译命令: ```bash gcc -o thread_demo thread_demo.c -lpthread -Wall -Wextra ``` ### 关键结构解析 1. **共享数据管理** ```c typedef struct { int counter; // 任务计数器 pthread_mutex_t mutex; // 互斥锁 pthread_cond_t cond; // 条件变量 } SharedData; ``` - 使用组合数据结构封装共享资源 - 包含互斥锁和条件变量实现同步 2. **线程通信机制** ```c // 生产者逻辑 pthread_cond_broadcast(&shared_data.cond); // 消费者逻辑 pthread_cond_wait(&data->cond, &data->mutex); ``` - 采用条件变量实现生产者-消费者模式 - 使用broadcast唤醒所有等待线程 3. **资源安全访问** ```c pthread_mutex_lock(&data->mutex); // 临界区操作 pthread_mutex_unlock(&data->mutex); ``` - 严格遵循lock/unlock配对原则 - 使用while循环检查条件避免虚假唤醒 ### 扩展模式建议 1. **线程池优化** ```c // 添加任务队列结构 typedef struct { int* tasks; int front; int rear; int count; int capacity; } TaskQueue; ``` - 实现环形缓冲区提升吞吐量 - 增加任务优先级支持 2. **错误处理增强** ```c #define CHECK_ERR(expr, msg) \ if ((err = expr) != 0) { \ fprintf(stderr, "%s: %s\n", msg, strerror(err)); \ exit(EXIT_FAILURE); \ } ``` - 添加错误码检查宏 - 实现资源回收安全机制 3. **性能监控接口** ```c typedef struct { unsigned long tasks_completed; double avg_latency; pthread_spinlock_t stat_lock; } ThreadStats; ``` - 添加线程统计模块 - 支持运行时性能分析 ### 运行效果示例 ```text [Main] 生成任务 1 [Thread 1] 处理任务 1 [Main] 生成任务 2 [Thread 2] 处理任务 2 [Main] 生成任务 3 [Thread 3] 处理任务 3 ... [Thread 4] 退出 [Thread 1] 退出 ``` ### 注意事项 1. 避免死锁:确保锁的获取顺序一致 2. 资源清理:线程退出前释放已占用的资源 3. 信号安全:不在信号处理函数中使用线程函数 4. CPU亲和性:可通过`pthread_setaffinity_np`优化线程绑定 该示例展示了典型的线程协作模式,可根据实际需求扩展任务分发策略、增加工作线程状态监控或实现动态线程创建等功能。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值