已解决:多线程环境中,新线程在使用cout函数打印输出到显示器出现数据混乱的情况

  • 错误展示
  • 错误原因
  • 解决办法
    • 1. 在本问题情况下:使用printf函数替代cout:
    • 2. 使用互斥锁使 cout函数线程保持原子状态
  • 什么是原子操作?

错误展示

最近学习多线程的时候,创建了一堆线程,然后每个线程都运行这个方法:

void *start_routine(void *args)
{
    ThreadData *td = static_cast<ThreadData *>(args); // 安全的进行强制类型转化
    int cnt = 10;
    while (cnt)
    {
    	cout << "cnt :" << cnt <<" "<< "&cnt:" << &cnt << endl;
        cnt--;
        sleep(1);
    }
    // delete td;
    return nullptr;
}

出现了下面这种情况,使用cout函数打印输出在显示器上面的内容很混乱。

在这里插入图片描述
我自己认为是cout函数输出内容到stdout文件时,在这个线程还未输出完整的情况下又被别的线程调度然后又继续cout导致了最后刷新缓冲区就成了这样子。

上述出现错误的完整代码:

#include <iostream>
#include <cstdlib>
#include <string>
#include <cassert>
#include <vector>
#include <pthread.h>
#include <unistd.h>

using namespace std;
class ThreadData
{
public:
    int number;
    pthread_t tid;
    char namebuffer[64];
};
void *start_routine(void *args)
{
    ThreadData *td = static_cast<ThreadData *>(args); // 安全的进行强制类型转化
    int cnt = 10;
    while (cnt)
    {
        cout << "cnt :" << cnt <<" "<< "&cnt:" << &cnt << endl;
        cnt--;
        sleep(1);
    }
    return nullptr;
}
int main()
{
    //创建一批线程
    vector<ThreadData *> threads;
#define NUM 10
    for (int i = 0; i < NUM; i++)
    {
        ThreadData *td = new ThreadData();
        td->number = i + 1;
        snprintf(td->namebuffer, sizeof(td->namebuffer), "%s:%d", "thread", i + 1);
        pthread_create(&td->tid, nullptr, start_routine, td);
        threads.push_back(td);
    }

    for (auto &iter : threads)
    {
        cout << "create thread: " << iter->namebuffer << " : " << iter->tid << " suceesss" << endl;
    }

    while(true)
    {
        sleep(2);
        cout<<"mainthred"<<endl;
    }
    return 0;
}

错误原因

错误原因:如果多个线程同时调用cout输出数据,由于cout不是线程安全的,不是原子操作(下面会说什么是原子操作),所以可能会出现数据交错或混乱的情况。这是因为多个线程可能会同时访问和修改cout的输出缓冲区,导致数据交错或丢失。
std::cout的输出操作通常涉及到多个步骤,例如格式化数据、写入缓冲区、输出到终端等,这些步骤可能会在不同的线程中交错执行,从而导致输出混乱。

cout函数的缓冲区策略:在默认情况下,cout函数使用行缓冲区(line-buffered)策略,即每当输出一个换行符时,缓冲区就会被刷新。此外,如果缓冲区已满,也会自动刷新缓冲区。但是,如果程序异常退出或者使用endl或flush等函数强制刷新缓冲区,也会导致缓冲区被刷新。

多线程环境下,cout输出语句可能出现的问题,对于下面的输出语句:

std::cout << "hello" << cnt << std::endl;

多个线程同时执行这个输出语句,那么可能会出现以下情况:

  1. 多个线程同时输出"hello"和cnt的值到输出缓冲区中,导致输出内容被交错输出。
  2. 多个线程同时输出换行符到输出缓冲区中,导致换行符被重复输出或者被覆盖。
  3. 多个线程同时刷新输出缓冲区,导致输出内容被交错输出或者被覆盖。

解决办法

1. 在本问题情况下:使用printf函数替代cout:

printf("cnt : %d &cnt:%p\n",cnt,&cnt);

在这里插入图片描述
printf 函数通常不是原子操作。这意味着两个或多个线程可能会同时尝试打印到控制台,并且输出可能会交错或损坏。但是,printf 函数在用于将输出打印到单个流时是原子操作。这是因为 printf 函数是作为单个系统调用实现的,这意味着在任何其他线程可以访问流之前,它保证由一个线程完整执行。
printf内部会使用同步机制来确保输出的数据不会被其他线程中断。具体来说,printf通常会使用文件锁或互斥锁来保证同一时间只有一个线程在输出数据,从而避免输出混乱或数据交错的问题。

例如:

  • 如果有两个线程都在尝试打印到标准输出流,则 printf 函数将确保一次只能打印一个线程。这将防止输出交错或损坏。
  • 如果有两个线程尝试打印到不同的流,则 printf 函数不会阻止输出交错或损坏。这是因为 printf 函数不会锁定流本身。通常,最好使用互斥锁来保护由多个线程访问的任何共享资源。这将有助于防止争用条件并确保程序的输出正确。

文件锁定是一种允许进程独占访问文件的机制。这意味着一次只有一个进程可以访问该文件。当进程锁定文件时,将阻止其他进程访问该文件,直到释放锁定为止。
互斥锁是一种同步原语,它允许线程锁定资源,以便一次只有一个线程可以访问它。当线程锁定互斥锁时,将阻止其他线程访问资源,直到释放锁。

2. 使用互斥锁使 cout函数线程保持原子状态

使用方法:

#include <iostream>
#include <thread>
#include <mutex>

using namespace std;

mutex mtx;

void print_message(string message) {
  // Lock the mutex
  mtx.lock();

  // Print the message
  cout << message << endl;

  // Unlock the mutex
  mtx.unlock();
}

int main() {
  // Create two threads
  thread t1(print_message, "Hello, world!");
  thread t2(print_message, "Goodbye, world!");

  // Wait for the threads to finish
  t1.join();
  t2.join();

  return 0;
}

代码工作原理的说明:

  • 对象在代码顶部声明。mutex
  • 该函数将字符串作为参数,并将消息打印到控制台。该函数还会在打印消息之前锁定互斥锁,并在打印消息后解锁互斥锁。print_message()
  • main() 函数创建两个线程。每个线程使用不同的消息调用函数。print_message()
  • 函数等待线程完成。join()

确保一次只有一个线程可以访问该对象。这可以防止输出交错或损坏。


什么是原子操作?

不会被线程调度机制打断的操作

原子操作是指在计算机中执行的一种操作,它要么完全成功完成,要么完全不执行,没有中间状态。原子操作通常是一个单个的、不可分割的操作,可以被看作是一种基本操作。原子操作的目的是确保多个并发的执行线程或进程能够正确地协作,而不会产生竞争条件或死锁问题。

原子操作通常用于对共享资源进行访问和修改的情况,例如在多线程编程中,多个线程可能同时访问和修改同一个变量,这时就需要使用原子操作来确保操作的正确性。常见的原子操作包括读取和写入一个共享变量、递增或递减一个计数器、测试和设置一个标志等。


如有错误或者不清楚的地方欢迎私信或者评论指出🚀🚀

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/48786.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

大数据Flink(五十二):Flink中的批和流以及性能比较

文章目录 Flink中的批和流以及性能比较 ​​​​​​​​​​​​​​一、Flink中的批和流

LeetCode 2500. Delete Greatest Value in Each Row【数组,排序】简单

本文属于「征服LeetCode」系列文章之一&#xff0c;这一系列正式开始于2021/08/12。由于LeetCode上部分题目有锁&#xff0c;本系列将至少持续到刷完所有无锁题之日为止&#xff1b;由于LeetCode还在不断地创建新题&#xff0c;本系列的终止日期可能是永远。在这一系列刷题文章…

基于深度学习的裂纹图像分类研究(Matlab代码实现)

&#x1f4a5;&#x1f4a5;&#x1f49e;&#x1f49e;欢迎来到本博客❤️❤️&#x1f4a5;&#x1f4a5; &#x1f3c6;博主优势&#xff1a;&#x1f31e;&#x1f31e;&#x1f31e;博客内容尽量做到思维缜密&#xff0c;逻辑清晰&#xff0c;为了方便读者。 ⛳️座右铭&a…

消息队列(一)-- RabbitMQ入门(1)

初识 RabbitMQ 核心思想&#xff1a;接收并转发消息。可以把它想象成一个邮局。 producer&#xff1a;生产者 queue&#xff1a;队列 consumer&#xff1a;消费者什么是消息队列 MQ&#xff08;Message Queue&#xff09;&#xff1a;本质是队列&#xff0c;FIFO先入先出&…

k8s集群环境的搭建

1.环境规划 1.1 集群类型 Kubernetes集群大致分为两类&#xff1a;一主多从和多主多从。 一主多从&#xff1a;一个Master节点和多台Node节点&#xff0c;搭建简单&#xff0c;但是有单机故障风险&#xff0c;适合用于测试环境。 多主多从&#xff1a;多台Master和多台Node节点…

了解Unity编辑器之组件篇Physics 2D(十二)

一、Area Effector 2D区域施加力&#xff09;&#xff1a;用于控制区域施加力的行为 Use Collider Mask&#xff08;使用碰撞器遮罩&#xff09;&#xff1a;启用后&#xff0c;区域施加力仅会作用于特定的碰撞器。可以使用Collider Mask属性选择要作用的碰撞器。 Collider Ma…

opencv-22 图像几何变换01-缩放-cv2.resize()(图像增强,图像变形,图像拼接)

什么是几何变换&#xff1f; 几何变换是计算机图形学中的一种图像处理技术&#xff0c;用于对图像进行空间上的变换&#xff0c;而不改变图像的内容。这些变换可以通过对图像中的像素位置进行调整来实现。 常见的几何变换包括&#xff1a; 平移&#xff08;Translation&#x…

MySQL-MHA高可用配置及故障切换

MySQL-MHA 一、MHA概述&#xff1a;1.概述&#xff1a;2.MHA的组成&#xff1a;3.MHA的特点&#xff1a;4.MHA的工作原理&#xff1a; 二、搭建MySQL MHA&#xff1a;1.配置主从复制&#xff1a;2.配置MHA&#xff1a;3.manager与node工具使用&#xff1a;4.在 manager 节点上配…

vue3+Luckysheet实现表格的在线预览编辑(electron可用)

前言&#xff1a; 整理中 官方资料&#xff1a; 1、github 项目地址https://github.com/oy-paddy/luckysheet-vue-importAndExport/tree/master/https://github.com/oy-paddy/luckysheet-vue-importAndExport/tree/master/ 2、xlsx vue3 json数据导出excel_vue3导出excel_羊…

vue项目登录页面实现记住用户名和密码

vue项目登录页面实现记住用户名和密码 记录一下实现的逻辑&#xff0c;应该分两步来理解这个逻辑 首次登录&#xff0c;页面没有用户的登录信息&#xff0c;实现逻辑如下&#xff1a; 用户输入用户名和密码登录&#xff0c;用户信息为名为form的响应式对象&#xff0c;v-model…

【Linux下6818开发板(ARM)】硬件空间挂载

(꒪ꇴ꒪ ),hello我是祐言博客主页&#xff1a;C语言基础,Linux基础,软件配置领域博主&#x1f30d;快上&#x1f698;&#xff0c;一起学习&#xff01;送给读者的一句鸡汤&#x1f914;&#xff1a;集中起来的意志可以击穿顽石!作者水平很有限&#xff0c;如果发现错误&#x…

查看8080端口会不会被占用

相关命令 查看8080端口会不会被占用 netstat -ano | findstr 8080 查看 终止占用端口xxx的进程 taskkill /f /pid xxx 具体步骤 第一步&#xff1a;点击起始菜单&#xff08;或是通过winR快捷键&#xff09;&#xff0c;在输入框中输入cmd&#xff0c;点击确定&#x…

MySQL 服务器的调优策略

点击上方↑“追梦 Java”关注&#xff0c;一起追梦&#xff01; 在工作中&#xff0c;我们发现慢查询一般有2个途径&#xff0c;一个是被动的&#xff0c;一个是主动的。被动的是当业务人员反馈某个查询界面响应的时间特别长&#xff0c;你才去处理。主动的是通过通过分析慢查询…

Rabbitmq的安装与使用(Linux版)

目录 Rabbitmq安装 1.在Ubuntu上安装RabbitMQ&#xff1a; 打开终端&#xff0c;运行以下命令以更新软件包列表&#xff1a; 安装RabbitMQ&#xff1a; 安装完成后&#xff0c;RabbitMQ服务会自动启动。你可以使用以下命令来检查RabbitMQ服务状态&#xff1a; 2.在CentOS…

Java集合框架的全面分析和性能增强

Java集合框架的全面分析和性能增强 &#x1f497;摘要&#xff1a;&#x1f497; 1. Java集合框架概述&#x1f497;1.1 Collection接口1.1.1 List接口1.1.2 Set接口1.1.3 Queue接口 &#x1f497;1.2 Map接口 &#x1f497;2. Java集合框架性能优化&#x1f497;2.1 选择合适的…

idea项目依赖全部找不到

目录 1&#xff0c;出错现象2&#xff0c;解决3&#xff0c;其他尝试 1&#xff0c;出错现象 很久没打开的Java项目&#xff0c;打开之后大部分依赖都找不到&#xff0c;出现了所有的含有import语句的文件都会报错和一些注解报红报错&#xff0c;但pom文件中改依赖是确实被引入…

线性模型学习

代码实现 import numpy as np import matplotlib.pyplot as pltx_data [1.0, 2.0, 3.0] y_data [2.0, 4.0, 6.0]def forward(x):return x * wdef loss(x, y):y_pred forward(x)return (y_pred - y) * (y_pred - y)w_list [] mse_list [] for w in np.arange(0.0, 4.1, 0.…

Jenkins插件管理切换国内源地址

一、替换国内插件下载地址 选择系统管理–>插件管理–> Available Plugins 并等待页面完全加载完成、这样做是为了把jenkins官方的插件列表下载到本地、接着修改地址文件、替换为国内插件地址 进入插件文件目录 cd /var/lib/jenkins/updatesdefault.json 为插件源地址…

mysql(五)主从配置

目录 前言 一、MySQL Replication概述 二、MySQL复制类型 三、部署MySQL主从异步复制 总结 前言 为了实现MySQL的读写分离&#xff0c;可以使用MySQL官方提供的工具和技术&#xff0c;如MySQL Replication&#xff08;复制&#xff09;、MySQL Group Replication&#xff08;组…

最全面的msvcp110.dll丢失修复方法分享,快速修复msvcp110.dll文件

今天主要给大家详细的介绍一下msvcp110.dll丢失修复的方法&#xff0c;因为在网上看到很多人因为这个问题而烦恼&#xff0c;其实这个问题不难解决的&#xff0c;下面就给大家分享多种方法&#xff0c;一起来看看吧。 一. 修复msvcp110.dll丢失的方法 重新安装相关程序 首先&…