当前位置: 首页 > news >正文

C++多线程

  • 虚函数

在C++中, virtual 和 override 关键字用于支持多态,尤其是在涉及类继承和方法重写的情况下。正确地理解和使用这两个关键字对于编写可维护和易于理解的面向对象代码至关重要。

virtual 关键字

  1. 使用场景:在基类中声明虚函数。
  2. 目的:允许派生类重写该函数,实现多态。
  3. 行为:当通过基类的指针或引用调用一个虚函数时,调用的是对象实际类型的函数版本
  4. 示例:
class Base {
public:virtual void func() {std::cout << "Function in Base" << std::endl;}
};

override 关键字

  1. 使用场景:在派生类中重写虚函数。
  2. 目的:明确指示函数意图重写基类的虚函数。
  3. 行为:确保派生类的函数确实重写了基类中的一个虚函数。如果没有匹配的虚函数,编译器会报错。
  1. 示例
class Derived : public Base {
public:void func() override {std::cout << "Function in Derived" << std::endl;}
};

注意点

只在派生类中使用 override: override 应仅用于派生类中重写基类的虚函数。

虚析构函数:如果类中有虚函数,通常应该将析构函数也声明为虚的。

默认情况下,成员函数不是虚的:在C++中,成员函数默认不是虚函数。只有显式地使用 virtual关键字才会成为虚函数。

继承中的虚函数:一旦在基类中声明为虚函数,该函数在所有派生类中自动成为虚函数,无论是否使用 virtual 关键字。

正确使用 virtual 和 override 关键字有助于清晰地表达程序员的意图,并利用编译器检查来避免常见的错误,如签名不匹配导致的非预期的函数重写。


  • 抽象类

抽象类的特点

  1. 包含至少一个纯虚函数
  • 抽象类至少有一个纯虚函数。这是一种特殊的虚函数,在抽象类中没有具体实现,而是留给派生类去实现。
  • 纯虚函数的声明方式是在函数声明的末尾加上 = 0 。
  1. 不能直接实例化
  • 由于抽象类不完整,所以不能直接创建它的对象。就像你不能直接使用“交通工具”的概念去任何地方,你需要一个具体的交通工具。
  1. 用于提供基础结构
  • 抽象类的主要目的是为派生类提供一个共同的基础结构,确保所有派生类都有一致的接口和行为。

  • 纯虚函数-接口

一个类作为接口可以通过以下步骤来实现:

  1. 定义抽象类:创建一个包含纯虚函数的抽象类,这些函数构成了接口的一部分。这些函数在抽象类

中只有声明而没有具体的实现。

  1. 派生类实现接口:派生类继承抽象类,并实现其中的纯虚函数,以具体实现接口定义的方法。
class LiveMove{
public:virtual void eat() = 0;virtual void bite() = 0;virtual void drink() = 0;virtual void la() = 0;
};class Dog : public LiveMove{
public:void eat() override{...};void bite() override{...};void drink() override{...};void la() override{...};
};

  • 线程的创建

  • 通过lambda表达式创建线程

    Lambda 格式:[捕获列表] (参数) 可选修饰 -> 返回类型 { 函数体 }  

    • 多线程场景中,最常用简化格式[&](){ 核心逻辑 }(引用捕获所有,无参无返回);
    • 捕获列表是关键:想修改外部变量用 [&],想只读用 [=],想精准控制用 [&var, num]
    • 线程安全:引用捕获要确保变量生命周期,值捕获要注意拷贝成本。

创建一个lambda表达式,定义变量i,将123传入lambda表达式,输出lambda表示式的内容

int main(int argc, char const *argv[]){std::thread th([](int i){std::cout << "test lambda" << i << std::endl;}, 123);th.join();    return 0;
}

 

在类成员函数里定义一个线程,访问类成员变量

class TestLambda{
public:void start(){std::string hr;std::thread th([this](){std::cout << this->name << std::endl;});th.join();}private:std::string name = "hello";
};int main(int argc, char const *argv[]){   TestLambda test;test.start();return 0;
}

  •  竞争状态、临界区、互斥锁

  1. 临界区:一段对「共享资源(如类成员变量、全局变量、堆内存)」进行读写操作的代码。核心特点:同一时间只能有一个线程执行这段代码,否则会出问题。
  2. 竞争状态:多个线程「同时进入临界区」,对共享资源进行读写操作,导致程序行为不可预测(结果不确定、数据损坏、崩溃)的现象。

示例:

创建了100个线程,并调到后台运行,产生竞争状态,导致运行结果不可预测

void TestThread(){//std::this_thread::sleep_for(std::chrono::milliseconds(10));std::cout << "====================" << std::endl;std::cout << "test 001" << std::endl;std::cout << "test 002" << std::endl;std::cout << "test 003" << std::endl;std::cout << "====================" << std::endl;
}int main(int argc, char const *argv[]){//创建100个进程并调至后台运行,实现并行for(int i = 0; i < 100; ++i){std::thread th(TestThread);th.detach();}return 0;
}

为了解决竞争状态,设置互斥锁,建立临界区

通过定义锁,我们定义获取锁资源到释放锁资源的区间叫做临界区同一时间只能有一个进程访问临界区资源,只有当解锁后,其他被阻塞的线程才能重新获取重复上述操作

static std::mutex mtx;void TestThread(){//获取锁资源,如果获取不到进去阻塞状态mtx.lock();std::cout << "====================" << std::endl;std::cout << "test 001" << std::endl;std::cout << "test 002" << std::endl;std::cout << "test 003" << std::endl;std::cout << "====================" << std::endl;mtx.unlock();
}int main(int argc, char const *argv[]){for(int i = 0; i < 100; ++i){std::thread th(TestThread);th.detach();}return 0;
}

 

  • 常用的mutex函数

lock() 阻塞式获取锁:若锁未被占用则成功获取;若已被占用,线程阻塞直到锁释放。 必须获取锁才能执行临界区(等待可接受)
unlock() 手动释放锁:必须在 lock()/try_lock() 成功后调用,否则行为未定义(崩溃)。 手动管理锁时,临界区结束后释放锁
try_lock() 非阻塞式获取锁:锁未被占用则返回 true(成功);已被占用则返回 false(不阻塞)。 不想阻塞,可先做其他任务(如重试、跳过)
try_lock_for() 超时非阻塞获取:在 rel_time 时间内尝试获取锁,超时未获取则返回 false 允许短暂等待,但不想无限阻塞(如等待 1 秒)
try_lock_until() 绝对时间超时获取:尝试获取锁直到 abs_time 时间点,超时返回 false 需精确控制等待截止时间(如凌晨 3 点前)
  • 处理线程抢不到资源问题

我们总是会遇到,当一个线程解锁后,该进程又一次进入了临界区,使得原本等待解锁的被阻塞线程又得接着等待,使得处理效率变低。

通过每次线程解锁时让其线程睡眠,保证解锁后不会与其他被阻塞线程争夺。

示例:

void ThreadMainMtx(int i){for(;;){mtx.lock();std::cout << i << "[in]\n";std::this_thread::sleep_for(std::chrono::milliseconds(100));mtx.unlock();std::this_thread::sleep_for(std::chrono::seconds(1));//解锁后,睡眠1s,再取争锁}
}int main(int argc, char const *argv[]){for(int i = 0; i < 4; ++i){std::thread th(ThreadMainMtx, i + 1);th.detach();}return 0;
}

 

http://www.jsqmd.com/news/69577/

相关文章:

  • Alientech KESS3 Master: Efficient OBD Protocols Activation for Agri Trucks Buses
  • 内网环境-centos7.6配置chrom和flask项目
  • selenium其他重要的Api
  • 机器学习基础
  • # sg.计算器
  • 洛谷P2860 [USACO06JAN] Redundant Paths G 题解 边双连通分量
  • AI真好玩系列-免费解锁 Google Gemini 的几种方式
  • 智能猫砂盆方案商权威推荐:技术驱动宠物养护新体验 - 星报
  • 网络线序问题了解
  • 洛谷U640024 找割边 题解
  • Python 学习笔记(01)
  • Python Flask service provide data list and retrieve and display in chrome via html and javascript
  • 2025最新PC仿石砖增强剂品牌TOP5评测!绿色建材赋能市政工程,权威榜单发布 - 全局中转站
  • 图文并茂-手把手教宝子们3分钟用 GitHub Pages 搭建免费网站 (保姆级教程)
  • 2025权威聚焦:智能门窗控制器解决方案商综合推荐,引领智慧生活新入口 - 星报
  • 2025最新聚脲防腐防水涂料/厂家TOP5评测!环保科技+工程实证权威榜单发布,功能涂料赋能基建防护新生态 - 全局中转站
  • 2025最新彩砖专用水性色浆服务商/厂家TOP5评测!环保创新+性能实证权威榜单发布,技术赋能重构彩砖涂装生态 - 全局中转站
  • 剪映vip破解版 分享
  • 2025 智能电壁炉解决方案商权威推荐:赋能家居暖意与智慧节能 - 星报
  • 2025年电壁炉解决方案商综合推荐:驱动智能取暖与美学融合的新浪潮 - 星报
  • 2025 最新聚脲地坪服务商 / 厂家 TOP5 评测!环保高性能 + 全场景适配权威榜单发布,技术创新引领地坪材料升级 - 全局中转站
  • 一只菜鸟学深度学习的日记:填充 步幅 下采样
  • 51
  • 2025最新无机水性涂料品牌/厂家TOP5评测!环保性能与工程适配权威榜单发布,功能性涂料技术革新引领行业升级 - 全局中转站
  • 黑马程序员SpringCloud微服务开发与实战-微服务-配置管理
  • git-ssh - yebinghuai-qq
  • Linux中级のNginx~2
  • 2025 最新水性地坪漆厂家 TOP5推荐!水性地坪漆年度品牌榜,环保性能 + 技术创新优质供应商,专业赋能地面涂装新体验 - 全局中转站
  • 数据采集与融合技术实践4
  • 12月9日日记