免责声明:
本文内容仅代表作者个人观点和知识水平,不构成任何推荐或建议。因参考本文而造成任何直接或间接损失后果的,作者不承担任何责任。
本文中的信息仅为作者个人学习研究成果的分享,不作为专业意见或建议。文章中如有错误,请斧正。因应用本文内容导致的风险或损失,作者不承担任何法律责任。
以下是正文内容
——————————————————
使用互斥量保证线程安全
线程安全是多线程编程时的计算机程序代码中的一个概念。在拥有共享数据的多条线程并行执行的程序中,线程安全的代码会通过同步机制保证各个线程都可以正常且正确的执行,不会出现数据污染等意外情况。
线程安全就是多个线程同时调用这个函数也不会引起问题。不采用线程安全会出现什么问题呢?想象如果两个线程同时调用printf打印数据,两个线程发出的打印内容就有可能混在一起,造成混乱。
为了避免这个问题,我们采用了互斥量(Mutex)。
每个线程在打印之前获取这个互斥量,获取到了才能打印,打印结束后再释放互斥量。而其他线程只有在互斥量释放后才能获取到。这样就实现了线程排队打印的效果,一个线程打印结束后才能由下一个线程打印,防止内容混在一起。
另外,为了提高打印效率,我们引入了一个缓冲区。每个线程先把要打印的内容格式化到缓冲区,然后再获取互斥量打印出来。如果没有缓冲区,直接使用printf格式化然后打印的话,获取互斥量的时间会很长,降低效率。
下面是threadX操作系统使用互斥锁方式的具体操作:
static TX_MUTEX AppPrintfSemp;
先创建互斥信号量
tx_mutex_create(&safe_printf_semp,"safe_printf_semp",TX_INHERIT);
static void safe_printf(const char *fmt, ...)
{
char buf_str[200 + 1];// 定义一个缓冲区,用于格式化字符串
va_list v_args;// va_list变量,用于访问格式化参数列表
va_start(v_args, fmt);
// 将格式化字符串和参数写入buf_str缓冲区
// vsnprintf函数的作用是格式化字符串并返回格式化后字符串长度
(void)vsnprintf((char*)&buf_str[0],
(size_t) sizeof(buf_str),
(char const *) fmt,
v_args);
va_end(v_args);
/* 互斥操作,用于保护printf的线程安全 */
tx_mutex_get(&safe_printf_semp, TX_WAIT_FOREVER);
printf("%s", buf_str);
tx_mutex_put(&safe_printf_semp);
}