C++笔记之获取线程ID以及线程ID的用处

news2024/10/6 16:22:33

C++笔记之获取线程ID以及线程ID的用处

code review!
在这里插入图片描述

文章目录

  • C++笔记之获取线程ID以及线程ID的用处
    • 一.获取ID
    • 二.线程ID的用处
      • 2.1.线程池管理
      • 2.2.动态资源分配
      • 2.3.使用线程同步机制实现互斥访问共享资源
      • 2.4.使用线程 ID 辅助线程同步
      • 2.5.任务分发:线程ID可以用于将任务分发给不同的线程。例如,一个任务队列可以分发任务给一组线程,并使用线程ID来跟踪任务的状态和进度。
      • 2.6.线程间通信:线程可以使用线程ID来识别接收消息的线程。这可用于实现多线程间的消息传递或共享数据。

一.获取ID

std::this_thread::get_id() 是 C++ 标准库中的一个函数,用于获取当前线程的唯一标识符。这个标识符通常是一个对象,它可以与其他线程的标识符进行比较,以确定它们是否代表同一线程。

以下是 std::this_thread::get_id() 的基本用法:

#include <iostream>
#include <thread>

int main() {
    // 获取当前线程的标识符
    std::thread::id threadId = std::this_thread::get_id();

    // 将标识符打印到标准输出
    std::cout << "Thread ID: " << threadId << std::endl;

    return 0;
}

在上面的示例中,std::this_thread::get_id() 被用来获取当前线程的标识符,并将其打印到标准输出。这个标识符通常是一个唯一的值,可以用来区分不同的线程。

请注意,std::this_thread::get_id() 返回的是一个 std::thread::id 类型的对象,可以使用 ==!= 运算符来比较两个线程的标识符,以确定它们是否相同。这对于多线程编程中的线程管理和同步非常有用。

上面的例子获取的是主线程的 ID。在 main 函数中调用 std::this_thread::get_id() 会返回主线程的唯一标识符。在多线程应用程序中,每个线程都有自己的唯一标识符,包括主线程。你可以在任何线程中使用 std::this_thread::get_id() 来获取该线程的标识符,不仅仅是主线程。

如果你在多线程程序中创建了其他线程,你可以在这些线程中使用 std::this_thread::get_id() 来获取它们各自的标识符,以便在需要时进行线程识别和管理。每个线程都有自己的标识符,这有助于区分和跟踪线程的行为。

二.线程ID的用处

2.1.线程池管理

在线程池中,线程 ID 可以帮助你识别特定工作者线程。例如,你可以将任务分配给特定的线程,以便更精确地控制资源分配和任务调度。

#include <iostream>
#include <thread>
#include <vector>
#include <functional>
#include <future>

void worker(int id) {
    std::cout << "Worker " << id << " is executing." << std::endl;
    // 执行任务
}

int main() {
    int numThreads = 4;
    std::vector<std::thread> threads;
    
    for (int i = 0; i < numThreads; ++i) {
        threads.push_back(std::thread(worker, i));
    }

    // 等待所有工作者线程完成
    for (auto& thread : threads) {
        thread.join();
    }

    std::cout << "All workers finished." << std::endl;

    return 0;
}

在这个示例中,线程 ID 有助于标识线程池中的不同工作者线程。

2.2.动态资源分配

线程 ID 可以用于动态分配资源给不同的线程。例如,你可以为特定线程分配不同的计算资源或内存区域,以提高性能或实现隔离。

#include <iostream>
#include <thread>

void worker(int id) {
    // 执行需要大量内存的计算任务
    // 分配特定的内存区域
    std::cout << "Worker " << id << " is executing." << std::endl;
    // 释放内存区域
}

int main() {
    int numThreads = 4;
    std::vector<std::thread> threads;

    for (int i = 0; i < numThreads; ++i) {
        threads.push_back(std::thread(worker, i));
    }

    for (auto& thread : threads) {
        thread.join();
    }

    std::cout << "All workers finished." << std::endl;

    return 0;
}

对于线程 ID 用于动态分配资源的示例,考虑以下情况:你希望为不同的线程分配不同的计算资源以优化性能。在这种情况下,你可以使用线程 ID 来识别和区分不同的线程,并为它们分配不同的资源。以下是一个示例:

#include <iostream>
#include <thread>
#include <vector>
#include <mutex>
#include <chrono>

// 模拟不同线程需要不同计算资源的任务
void performTask(int id) {
    std::cout << "Thread " << id << " is performing a task." << std::endl;
    std::this_thread::sleep_for(std::chrono::seconds(2));
    std::cout << "Thread " << id << " completed the task." << std::endl;
}

int main() {
    int numThreads = 4;
    std::vector<std::thread> threads;

    for (int i = 0; i < numThreads; ++i) {
        // 根据线程 ID 分配不同的计算资源
        if (i % 2 == 0) {
            threads.push_back(std::thread(performTask, i));
        } else {
            // 为奇数线程分配更多的计算资源
            threads.push_back(std::thread([i] {
                // 分配更多的计算资源
                performTask(i);
            }));
        }
    }

    for (auto& thread : threads) {
        thread.join();
    }

    std::cout << "All threads finished." << std::endl;

    return 0;
}

在这个示例中,有四个线程执行任务,但奇数线程(线程1和线程3)被分配更多的计算资源。通过线程 ID(i)的奇偶性来确定分配不同计算资源的策略。

请注意,这个示例是一个简化的演示,真实的资源分配通常更复杂。线程 ID 可以用于更复杂的分配策略,例如在多核处理器上优化计算资源分配,或在不同的线程之间实现资源隔离。

2.3.使用线程同步机制实现互斥访问共享资源

线程同步是多线程编程中的一个关键概念,它用于确保多个线程能够安全地协同工作,避免数据竞争和并发问题。线程 ID 可以在线程同步中发挥重要作用,以下是一个示例说明线程同步的用途:

示例:使用线程同步机制实现互斥访问共享资源

在多线程环境中,多个线程可能会同时访问共享资源,如果不进行同步,会导致数据竞争和不确定的行为。为了解决这个问题,我们可以使用互斥锁(std::mutex)来保护共享资源,同时使用线程 ID 来标识哪个线程拥有锁。

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

std::mutex mtx;  // 用于保护共享资源的互斥锁

void worker(int id) {
    // 一些工作

    // 使用互斥锁来保护共享资源
    mtx.lock();
    std::cout << "Worker " << id << " is accessing the shared resource." << std::endl;
    // 模拟工作
    std::this_thread::sleep_for(std::chrono::seconds(1));
    std::cout << "Worker " << id << " finished accessing the shared resource." << std::endl;
    mtx.unlock();
}

int main() {
    int numThreads = 4;
    std::vector<std::thread> threads;

    for (int i = 0; i < numThreads; ++i) {
        threads.push_back(std::thread(worker, i));
    }

    for (auto& thread : threads) {
        thread.join();
    }

    std::cout << "All workers finished." << std::endl;

    return 0;
}

在上面的示例中,多个工作者线程(Worker 0、Worker 1、Worker 2、Worker 3)同时访问一个共享资源。互斥锁 mtx 用于保护共享资源,确保一次只有一个线程可以访问。线程 ID 用于标识哪个线程当前拥有锁并在访问共享资源时进行输出。

线程同步是确保多线程程序安全运行的关键部分,使用线程 ID 和互斥锁可以帮助你实现正确的线程同步。这有助于防止并发问题,如竞态条件和数据竞争,从而确保多线程程序的可靠性。

2.4.使用线程 ID 辅助线程同步

线程 ID 并不是直接用于线程同步的工具,而是用于标识不同的线程。然而,线程同步机制(如互斥锁、条件变量等)通常需要用到线程 ID 来实现更复杂的同步逻辑。下面是一个示例,演示如何使用线程 ID 来辅助线程同步。

示例:使用线程 ID 辅助线程同步

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

std::mutex mtx; // 互斥锁
std::condition_variable cv; // 条件变量
int sharedData = 0;

void worker(int id) {
    std::unique_lock<std::mutex> lock(mtx);
    
    // 等待主线程发送信号
    cv.wait(lock, [id] { return id == 1; });

    // 执行工作
    std::cout << "Worker " << id << " is accessing shared data: " << sharedData << std::endl;
    sharedData += id;
    std::this_thread::sleep_for(std::chrono::seconds(1));
    
    // 释放锁
    lock.unlock();
}

int main() {
    std::thread t1(worker, 1);
    std::thread t2(worker, 2);

    // 等待一段时间
    std::this_thread::sleep_for(std::chrono::seconds(2));

    // 向线程1发送信号
    {
        std::unique_lock<std::mutex> lock(mtx);
        std::cout << "Main thread is sending a signal to Worker 1." << std::endl;
        cv.notify_all();
    }

    t1.join();
    t2.join();

    std::cout << "All workers finished." << std::endl;

    return 0;
}

在这个示例中,有两个工作者线程(Worker 1 和 Worker 2)。线程 1首先被阻塞在条件变量上等待一个特定信号,然后主线程向线程 1 发送信号,线程 1被唤醒后可以开始执行工作。线程 2只是简单地等待。

线程同步机制包括互斥锁 mtx 和条件变量 cv。线程 ID(id)用于确定哪个线程应该在条件变量上等待信号。线程同步的核心思想是确保线程在正确的时间点执行,并且不会出现竞争条件。

这个示例使用线程 ID 辅助线程同步,但实际上线程同步可能涉及更复杂的逻辑和多个线程之间的交互,线程 ID 通常是用于确定特定线程的条件是否满足,从而执行或等待。

2.5.任务分发:线程ID可以用于将任务分发给不同的线程。例如,一个任务队列可以分发任务给一组线程,并使用线程ID来跟踪任务的状态和进度。

线程ID可以用于将任务分发给不同的线程。例如,一个任务队列可以分发任务给一组线程,并使用线程ID来跟踪任务的状态和进度。

# Python 示例
from threading import Thread

def worker(task_id):
    # 执行任务
    print(f"线程 {task_id} 正在执行任务")

# 创建多个线程并分发任务
threads = []
for i in range(5):
    thread = Thread(target=worker, args=(i,))
    threads.append(thread)
    thread.start()

# 等待所有线程完成
for thread in threads:
    thread.join()

2.6.线程间通信:线程可以使用线程ID来识别接收消息的线程。这可用于实现多线程间的消息传递或共享数据。

// C++ 示例
#include <iostream>
#include <thread>
#include <mutex>

std::mutex mtx;
void sendMessage(int senderID, int receiverID, const std::string& message) {
    std::lock_guard<std::mutex> lock(mtx);
    std::cout << "线程 " << senderID << " 向线程 " << receiverID << " 发送消息: " << message << std::endl;
}

int main() {
    std::thread thread1(sendMessage, 1, 2, "Hello from Thread 1!");
    std::thread thread2(sendMessage, 2, 1, "Hi from Thread 2!");
    
    thread1.join();
    thread2.join();

    return 0;
}

这些示例演示了如何使用线程ID来实现线程同步、任务分发和线程间通信。线程ID用于唯一标识线程,并允许线程之间进行通信和协作。请注意,具体的线程ID分配和使用方式可能因编程语言和操作系统而异。

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【Java每日一题】——第三十题:班级管理程序设计(2023.10.14)

&#x1f383;个人专栏&#xff1a; &#x1f42c; 算法设计与分析&#xff1a;算法设计与分析_IT闫的博客-CSDN博客 &#x1f433;Java基础&#xff1a;Java基础_IT闫的博客-CSDN博客 &#x1f40b;c语言&#xff1a;c语言_IT闫的博客-CSDN博客 &#x1f41f;MySQL&#xff1a…

10-网络篇-DHCP获取的参数详解

一个设备接入路由器局域网时&#xff0c;是通过DHCP获取网络信息&#xff0c;从而完成网络配置的获取。如下图所示为windows系统通过DHCP所要获取的网络配置&#xff1a;IP、子网掩码、网关、DNS服务器。任何设备要上网前&#xff0c;都需要知道这几个参数&#xff0c;下面对这…

ODrive移植keil(七)—— 插值算法和偏置校准

目录 一、角度读取1.1、硬件接线1.2、程序演示1.3、代码说明 二、锁相环和插值算法2.1、锁相环2.2、插值2.3、角度补偿 三、偏置校准3.1、硬件接线3.2、官方代码操作3.3、移植后的代码操作3.4、代码说明3.5、SimpleFOC的偏置校准对比 ODrive、VESC和SimpleFOC 教程链接汇总&…

Ubuntu的Python从2.x升级到3.x

我的Ubuntu系统默认是2.7,我想升级为3.5 升级python3.5 下载python sudo apt-get install python3查看 刚才下载的Python程序被安装在usr/local/lib/python3.5 中 cd usr/local/lib备份一下 sudo cp /usr/bin/python /usr/bin/python_bak删除python的旧关联 sudo rm -rf py…

SpringBean的初始化流程

当我们启动Spring容器后&#xff0c;会先通过AbstractApplicationContext#refresh方法&#xff0c;调用BeanFactoryPostProcess方法&#xff0c;可以在bean初始化前&#xff0c;修改context中的BeanDefinition&#xff0c;但是因为此时Bean还没有初始化&#xff0c;所以并不会修…

valarray 包含对象成员的类(cpp14章)

C代码重用 1.公有继承可以实现 2.包含、私有继承、保护继承用于实现has-a关系&#xff0c;即新的类将包含另一个类的对象。 &#xff08;使用这样类成员&#xff1a;本身是另外一个类对象称为包含 &#xff08;组合或层次化&#xff09;。&#xff09; 3.函数模板、类模…

使用匿名函数在Golang中的好处

发挥Golang中无名代码块的潜力 匿名函数&#xff0c;也被称为lambda函数或闭包&#xff0c;是Golang中的一个强大功能&#xff0c;提供了许多好处。这些无名代码块为开发人员在设计和构建其代码时提供了更大的灵活性和模块化。在本节中&#xff0c;我们将探讨使用匿名函数可以…

访问控制列表ACL讲解——想偷偷访问数据,我ACL可不同意

作者&#xff1a;Insist-- 个人主页&#xff1a;insist--个人主页 梦想从未散场&#xff0c;传奇永不落幕&#xff0c;博主会持续更新优质网络知识、Python知识、Linux知识以及各种小技巧&#xff0c;愿你我共同在CSDN进步 目录 一、ACL的基本概念 1. ACL是什么 2. 为什么需…

Jenkins+Gitlab+Docker(Dockerfile)部署

Docker部署运行 ​ 上一篇内容中使用Jenkins(运行服务器)Gitlab(代码存储库)Webhook(网络钩子)的方式部署运行我们的项目。需要我们在服务器上做好很多相关的环境配置及依赖。 ​ 那么假如有这样一个场景&#xff1a;需要把不同技术栈的项目部署到同一台服务器上运行。比如PH…

DocCMS keyword SQL注入

漏洞描述 DocCMS keyword参数存在 SQL注入漏洞&#xff0c;攻击者通过漏洞可以获取数据库信息 漏洞复现 访问url&#xff1a; 漏洞证明&#xff1a; 文笔生疏&#xff0c;措辞浅薄&#xff0c;望各位大佬不吝赐教&#xff0c;万分感谢。 免责声明&#xff1a;由于传播或利…

400电话的技术实现要点

摘要&#xff1a;本文将介绍400电话的技术实现要点。首先&#xff0c;我们将讨论400电话的基本原理和技术架构。然后&#xff0c;我们将深入探讨400电话的关键技术&#xff0c;包括呼叫路由、语音导航、呼叫转接等。最后&#xff0c;我们将讨论如何保障400电话的稳定性和安全性…

JUnit5 【最实用最简洁】

JUnit5 文章目录 JUnit5一、JUnit 的相关技术二、参数化三、给测试用例指定顺序四、断言五、测试套件 安装依赖&#xff1a;在Maven库中安装 为什么学了 Selenium 还要学 JUnit&#xff1f; 1、JUnit5 是单元测试框架&#xff0c;拿着一个技术写自动化测试用例&#xff08;Sele…

云上攻防-云原生篇Docker安全系统内核版本漏洞CDK自动利用容器逃逸

文章目录 云原生-Docker安全-容器逃逸&内核漏洞云原生-Docker安全-容器逃逸&版本漏洞-CVE-2019-5736 runC容器逃逸-CVE-2020-15257 containerd逃逸 云原生-Docker安全-容器逃逸&CDK自动化 云原生-Docker安全-容器逃逸&内核漏洞 细节部分在权限提升章节会详解&…

【肌电信号】OpenSignals使用方法 --- 肌电信号采集及导入matlab

一、 多通道采集教学 1. 数据线连接 将PLUX设备通过USB或蓝牙与电脑连接&#xff0c;注意确认在几号通道接线。 2.实时数据采集可视化 进行设置。需要在软件中选择你的PLUX设备&#xff0c;并配置相关的参数&#xff0c;如采样率、分辨率、信号类型等 3 支持数据回放和…

zabbix监控实战2

4、zabbix添加监控项 nginx监控 在server上安装nginx 添加模板 浏览图形 mysql监控 zabbix自带mysql模板&#xff0c;所以可以在server1上直接做 创建数据库连接用户 percona数据库模板 清理掉mysql的模块链接 安装并配置好percona的数据库模板 测试脚本 删除tmp下的缓存文…

短链接系统如何设计

shigen坚持日更的博客写手&#xff0c;擅长Java、python、vue、shell等编程语言和各种应用程序、脚本的开发。坚持记录和分享从业两年以来的技术积累和思考&#xff0c;不断沉淀和成长。 今天给大家带来的文章是&#xff1a;《短链接系统如何设计》。在开始之前&#xff0c;先让…

SpringBoot讲义

SpringBoot 文档更新日志 版本更新日期操作描述v1.02021/11/14A基础篇 前言 ​ 很荣幸有机会能以这样的形式和互联网上的各位小伙伴一起学习交流技术课程&#xff0c;这次给大家带来的是Spring家族中比较重要的一门技术课程——SpringBoot。一句话介绍这个技术&#xff0c;…

【基于Kmeans、Kmeans++和二分K均值算法的图像分割】数据挖掘实验三

文章目录 Ⅰ、项目任务要求任务描述&#xff1a;主要任务要求&#xff1a; II、实现过程数据集描述实现描述具体实现过程 III、完整代码代码①代码② Ⅰ、项目任务要求 任务描述&#xff1a; 图像分割是图像处理和计算机视觉中重要的一环&#xff0c;在实际生活中得到了广泛的…

【C++】:关键字+命名空间+输入输出+缺省参数+函数重载+引用

【本节目标】 C关键字命名空间C输入&输出缺省参数函数重载引用 C是在C的基础之上&#xff0c;容纳进去了面向对象编程思想&#xff0c;并增加了许多有用的库&#xff0c;以及编程范式等 熟悉C语言之后&#xff0c;对C学习有一定的帮助&#xff0c;本章节主要目标&#xff…

2023年天猫淘宝双11红包活动入口是什么时候开始至什么时间结束?

​2023年淘宝天猫双11超级红包活动领取时间是从2023年10月24日20:00开始至11月11日23:59结束&#xff0c;淘宝天猫双十一活动时间内每天都可以领取1超级红包最高可得23888元。 2023年天猫淘宝双十一红包使用时间分为2个阶段&#xff1a;第一阶段是从2023年10月31日20:00开始至…