【Linux —— 线程控制】

news2024/9/22 1:27:32

Linux —— 线程控制

  • 进程和线程
    • 进程
    • 线程
    • 进程和线程的关系
    • 进程的优缺点
    • Linux进程VS线程
  • Linux线程控制
    • POSIX线程库
    • 创建线程
    • 线程等待
    • 线程终止
    • 线程分离

进程和线程

进程

进程是一个正在执行的程序实例,拥有独立地址空间数据段代码段堆栈以及系统资源(如文件描述符、信号处理器等)。
 每个进程在内存中都有自己的地址空间,相互之间不能直接访问对方的内存。进程之间的通信通常需要通过进程间通信(IPC)机制,如管道、消息队列、共享内存等。
 由于进程拥有独立的地址空间,进程之间的上下文切换需要保存和恢复大量的状态信息,因此开销较大。

线程

线程是进程中的一个执行单元,多个线程可以共享进程的地址空间和资源一个进程可以包含多个线程,这些线程可以并发执行。

 线程之间共享进程的所有资源(包括代码段、数据段、文件描述符等),但每个线程有自己的栈和寄存器。


在Linux系统中,看到的PCB都是轻量级进程(线程的本质是轻量级进程)

进程和线程的关系

  • 进程是承担分配系统资源的基本实体
  • 线程是CPU调度和分配的基本单位,是进程里面的执行流

在Linux中没有真正意义上的线程,线程是用进程模拟的,数据结构也是用的task_struct

进程的优缺点

进程的优点:

  1. 提高并发性和效率:
     线程允许程序并发执行多个任务,在多核处理器上,多个线程可以同时在不同的 CPU 核心上执行,从而提高程序的性能。
    线程之间可以共享数据和资源,减少了进程间通信的开销。
  2. 资源共享:
     同一进程中的线程共享相同的内存地址空间和系统资源(如文件描述符、全局变量等),这使得线程之间的通信比进程间通信更高效、更简单。
  3. 上下文切换开销低:
     线程的上下文切换(在不同线程之间切换执行)比进程的上下文切换更快,因为线程共享相同的地址空间,不需要频繁地切换内存页表和其他资源。
  4. 响应性强:
     在 GUI 应用程序中,线程可以用来执行耗时的任务(如 I/O 操作、复杂计算)而不会阻塞用户界面主线程,从而提高程序的响应性。

进程的缺点:

  1. 编程复杂性增加:
     多线程编程容易引发并发问题,如竞态条件、死锁、资源竞争等。这些问题难以调试和测试,增加了开发的复杂性。
  2. 资源共享带来的风险:
     虽然线程共享进程的资源,但这也带来了潜在的风险,如一个线程的崩溃可能导致整个进程崩溃,或者一个线程的不当操作(如对共享数据的错误修改)可能影响其他线程的正常运行。
  3. 调试困难:
     由于线程之间的并发执行,程序的执行顺序可能不可预测,这使得调试多线程程序变得更加困难。
  4. 缺乏可扩展性:
     在某些情况下,使用线程可能不如使用进程那样具有可扩展性,特别是在需要隔离和安全的环境中,进程提供了更好的隔离性和独立性。

Linux进程VS线程

进程是资源分配的基本单位,线程是调度的基本单位。
线程共享进程的数据,但是也拥有自己的一部分数据:

  • 线程ID, 一组寄存器,栈, errno,信号屏蔽字, 调度优先级.

进程和线程的关系如下图:
在这里插入图片描述

Linux线程控制

POSIX线程库

POSIX线程库(Pthreads) 是POSIX标准的一部分,提供了一组用于多线程编程的API,使能够在Unix-like系统上创建、管理和同步线程。它包括线程的创建与管理、线程同步(如互斥锁、条件变量)、线程属性设置,以及线程本地存储等功能。Pthreads广泛用于编写可移植的多线程应用程序。

注意:

  • 与线程有关的函数构成了一个完整的系列,绝大多数函数的名字都是以“pthread_”打头的
  • 要使用这些函数库,要通过引入头文<pthread.h>
  • 链接这些线程函数库时要使用编译器命令的“-lpthread”选项

创建线程

pthread_create

int pthread_create(pthread_t *thread, const pthread_attr_t *attr, 
                   void *(*start_routine) (void *), void *arg);

参数说明:

  • pthread_t *thread : 这是一个指向线程标识符的指针。pthread_create 会将新线程的 ID 保存到这个指针指向的变量中,以便之后可以使用该 ID 对线程进行操作(如 pthread_join 等)。
  • const pthread_attr_t *attr: 指向线程属性对象的指针,用于指定新线程的属性。如果传入 NULL,则使用默认属性。
  • void *(*start_routine)(void *): 这是线程执行的函数的指针,新线程启动后会执行这个函数。函数需要接受一个 void* 类型的参数,并返回一个 void* 类型的值。
  • void *arg: 传递给 start_routine 的参数。 可以将需要传递给线程的参数通过这个指针传递。

返回值:

  • 成功: 返回 0,表示线程创建成功。
  • 失败: 返回错误码(非零),表示创建线程时发生错误。

示例:

#include <iostream>
#include <pthread.h>
#include <unistd.h>

void * routine(void* arg)
{
    while(1)
    {
        std::cout << "I am new thread" << ((char*)arg) << std::endl;
        sleep(1);
    }
}

int main ()
{
    pthread_t tid;      //声明变量
    pthread_create(&tid,nullptr,routine,(void*)"thread 1");    //创建线程
    //主线程
    while(1)
    {
        std::cout << "I am main thread" << std::endl;
        sleep(1);
    }

    return 0;
}

在这里插入图片描述

输出错乱现象是由于多个线程在标准输出(std::cout)上竞争所导致的。std::cout
是一个共享资源,当主线程和新创建的线程几乎同时向 std::cout
输出内容时,由于输出操作没有被同步控制,它们的输出可能会相互交错,从而导致输出混乱。


问题一: main和new线程,谁先运行?
答:不确定。

pthread_t 是什么?


pthread_t 是 POSIX 线程库(Pthreads)中用于标识线程的类型。它是一个平台相关的类型,在不同的系统中可能被定义为不同的数据类型,但通常是一个无符号整数或指针类型。 pthread_t 用来唯一标识一个线程,使得线程可以通过这个标识符进行管理和操作。

通过查看进程发现只有一个进程
在这里插入图片描述
使用 ps -aL 可以看到两个线程
在这里插入图片描述
在这里插入图片描述


问题四:全面看待函数传参


答:因为arg是一个void* 的参数,所以我们可以传入各种类型最后进行强转即可。

#include <iostream>
#include <pthread.h>
#include <unistd.h>

void *routine(void *arg)
{
    int n = 10;
    int m = *(int *)arg;
    while (n--)
    {
        std::cout << m <<  "  thread running ... cnt : " << n << std::endl;
        sleep(1);
    }
    return nullptr;
}

int main()
{
    pthread_t tid; // 声明变量
    int n = 10;
    pthread_create(&tid, nullptr, routine, (void *)&n); // 创建线程
    std::cout << "main thread wait begin... " << std::endl;

    return 0;
}

对于int型的变量n,强转void* ,在新进程中再次强转int,拿到arg的值。
在这里插入图片描述
也可以传入类对象

#include <iostream>
#include <pthread.h>
#include <unistd.h>

class ThreadDate
{
    public:
    std::string _name;
    int _num;
};

void *routine(void *arg)
{
    int n = 10;
    ThreadDate *td = static_cast<ThreadDate*>(arg);     // (ThreadDate*)arg
    while (n--)
    {
        std::cout << td->_name <<  " running ... cnt : " << n  << " _num : " << td->_num << std::endl;
        sleep(1);
    }
    return nullptr;
}

int main()
{
    pthread_t tid; // 声明变量
    ThreadDate td;
    td._name = "thread -1";
    td._num = 5;

    pthread_create(&tid, nullptr, routine, (void *)&td); // 创建线程
    std::cout << "tid -> " << PrintToHex(tid) << std::endl;

    std::cout << "main thread wait begin... " << std::endl;
    return 0;
}

在这里插入图片描述
因为可以传入类对象,所以我们可以通过arg来给线程传入多个参数,甚至方法了。

但是对于上面的类对象的声明不太推荐,因为在main中直接声明相当于在其栈上开辟空间,这回污染别的线程的栈,所以可以通过使用new在堆上开辟,再把地址传给新线程。
在这里插入图片描述


问题六:如何创建多个线程:


使用for循环创建

const int num = 10;

void *threadRun(void *args)
{
    std::string name = static_cast<const char *>(args);
    std::cout << name << " is running... " << std::endl;

    sleep(2);
    return args;
}

int main()
{
    std::vector<pthread_t> tids;
    for (int i = 0; i < num; i++)
    {
        // 1 创建线程id
        pthread_t tid;

        // 2 线程的名字
        char *name = new char[128];
        snprintf(name, 128, "thread-%d", i + 1);
        pthread_create(&tid, nullptr, threadRun, (void *)name);

        // 3 保存所有线程的id信息
        tids.emplace_back(tid);
        // sleep(1);
    }

    for (auto tid : tids)
    {
        const char *name = nullptr;
        pthread_join(tid, (void **)&name);
        std::cout << name << " quit ... " << std::endl;
    }

    return 0;
}

在这里插入图片描述

线程等待

pthread_join

int pthread_join(pthread_t thread, void **retval);

参数:

  • pthread_t thread: 这是被等待线程的标识符
  • void **retval: 这是一个指向指针的指针,用于存储被等待线程的返回值。如果不需要获取返回值,可以传入 NULL。

返回值

  • 成功: 返回 0 表示线程成功等待并获取到其退出状态。
  • 失败: 返回错误码(非零),表示等待线程时发生错误。

问题五:如何全面看待线程函数返回:


  1. 线程的返回只考虑正确的返回,不考虑异常,因为异常了,整个进程就崩溃了,包括主线程。
  2. 我们可以传递任意类型,但都必须穿地址才能实现。
  1. 出异常就直接终止进程了。
void *routine(void *arg)
{
    sleep(3);
    int m = 10 / 0;
}

int main()
{
    pthread_t tid; // 声明变量
    
    pthread_create(&tid, nullptr, routine, (void*)"thread 1"); // 创建线程
 

    std::cout << "main thread wait begin... " << std::endl;
    sleep(5);
    return 0;
}

在这里插入图片描述
当出现了 / 0 异常的时候,进程直接本奔溃。

  1. 可以传递任意类型
  • 首先肯定是可以返回数字的。

在这里插入图片描述
在这里插入图片描述
因为我们当前是64位系统,void *占有8个字节,但是int只占有4个字节,所以出现loses precision错误。
在这里插入图片描述

这样修改即可
在这里插入图片描述

  • 其次,可以返回类对象。
class ThreadDate
{
public:
    int Execute()
    {
        return x + y;
    }

public:
    int x;
    int y;
    std::string name;
};

class ThreadResult
{
public:
    std::string Print()
    {
        return std::to_string(x) + "+" + std::to_string(y) + "=" + std::to_string(ret);
    }

public:
    int x;
    int y;
    int ret;
};

void *Routine(void *arg)
{

    ThreadDate *td = static_cast<ThreadDate *>(arg); // (ThreadDate*)arg
    ThreadResult *result = new ThreadResult();
    result->ret = td->Execute();
    result->x = td->x;
    result->y = td->y;
    std::cout << "running ..." << std::endl;
    std::cout << "--------------------------" << std::endl;

    delete td;

    return (void *)result;
}

std::string PrintToHex(pthread_t &tid)
{
    char buffer[128];
    snprintf(buffer, sizeof(buffer), "0x%lx", tid);
    return buffer;
}

int main()
{
    pthread_t tid; // 声明变量
    ThreadDate *td = new ThreadDate();
    td->name = "Thread-1";
    td->x = 10;
    td->y = 20;

    pthread_create(&tid, nullptr, Routine, (void *)td); // 创建线程
    // 主线程
    // 问题二:我们希望主线程最后退出,则需要用join来等待新线程,否则主线程退了新线程没退导致类似于僵尸进程的情况。

    // 问题三:tid是什么样子的呢?
    std::cout << "tid -> " << PrintToHex(tid) << std::endl;

    std::cout << "main thread wait begin... " << std::endl;
    // sleep(5);
    ThreadResult *ret_ = nullptr;
    int ret = pthread_join(tid, (void **)&ret_);

    if (ret == 0)
    {
        std::cout << "main thread wait success  get result -> " << ret_->Print() << std::endl;
    }

    return 0;
}

在这里插入图片描述


pthread_join 的作用

  • 同步线程: pthread_join 的主要作用是同步线程,确保主线程或其他线程等待目标线程执行完毕后再继续执行。
  • 回收资源: 当一个线程结束时,它的资源不会立即被系统回收,只有当其他线程调用 pthread_join 来等待它结束时,系统才会回收它的资源。因此,pthread_join 也起到了一定的资源管理作用。

注意:

  • 只能对已经创建的线程调用 pthread_join
  • 如果多次调用 pthread_join 对同一个线程进行等待,可能会导致未定义行为。
  • 如果某个线程不希望被等待或需要自动释放资源,可以使用 pthread_detach 函数将其设置为“分离”状态。
#include <iostream>
#include <pthread.h>
#include <unistd.h>

void * routine(void* arg)
{
    int n = 10;
    while(n--)
    {
        std::cout << "new thread running ... cnt : " << n << std::endl;
        sleep(1);
    }
    return nullptr;
}

int main ()
{
    pthread_t tid;      //声明变量
    pthread_create(&tid,nullptr,routine,(void*)"thread 1");    //创建线程
    //主线程
    //我们希望主线程最后退出,则需要用join来等待新线程,
    //否则主线程退了新线程没退导致类似于僵尸进程的情况。

    std::cout << "main thread wait begin... " << std::endl;
    int ret = pthread_join(tid,nullptr);

    if(ret == 0)
    {
        std::cout << "main thread wait success " << std::endl;
    }
    
    return 0;
}

在这里插入图片描述

在这里插入图片描述
可以看到主线程一直在阻塞式的等待新线程,等待新线程循环完后主线程等待完成,主线程退出。

问题二:如何保证主线程最后退出


答:join来保证。

问题三:tid 是什么样子的呢?


答:实际上就是一个虚拟地址。先通过打印来验证。

在这里插入图片描述

线程终止

线程终止一共三种方法:

  1. 新线程调用return
  2. 新线程调用pthread_exit终止自己
  3. 一个线程调用pthread_cancel终止另外一个线程

1.return退出
在这里插入图片描述

2.pthread_exit函数

void pthread_exit(void *retval);

参数说明:

  • retval:这是一个指向 void 的指针,允许线程在退出时返回一个值给任何通过 pthread_join 等待这个线程结束的线程。这个返回值可以作为 pthread_join 函数的返回值获得。

返回值:

  • 该函数函数本身不返回任何值,因为它直接导致线程退出。任何从 pthread_exit 调用的函数或后续代码都不会被执行。
const int num = 10;

void *threadRun(void *args)
{
    std::string name = static_cast<const char *>(args);
    std::cout << name << " is running... " << std::endl;

    sleep(2);
    pthread_exit((void *)111);
}

int main()
{
    std::vector<pthread_t> tids;
    for (int i = 0; i < num; i++)
    {
        // 1 创建线程id
        pthread_t tid;

        // 2 线程的名字
        char *name = new char[128];
        snprintf(name, 128, "thread-%d", i + 1);
        pthread_create(&tid, nullptr, threadRun, (void *)name);

        // 3 保存所有线程的id信息
        tids.emplace_back(tid);
        // sleep(1);
    }

    for (auto tid : tids)
    {
        void *num = nullptr;
        pthread_join(tid, &num);
        std::cout << " quit ... " << (uint64_t)num << std::endl;
    }

    return 0;
}

在这里插入图片描述


3. pthread_cancel函数

int pthread_cancel(pthread_t thread);

参数说明:

  • thread:这是一个 pthread_t 类型的参数,表示要取消的线程的标识符。线程标识符是在调用 pthread_create 函数创建线程时返回的。

返回值:

  • pthread_cancel 函数的返回值是一个整数,用于指示函数调用的成功或失败状态:
    • 0:表示成功发送了取消请求给目标线程。但请注意,这并不意味着目标线程已经实际停止执行;它只是表示取消请求已被成功接收。
    • 非0值:表示函数调用失败,无法向目标线程发送取消请求。具体的错误码可以用来进一步诊断问题。

示例:

#include <iostream>
#include <pthread.h>
#include <unistd.h>
#include <vector>

const int num = 10;

void *threadRun(void *args)
{
    std::string name = static_cast<const char *>(args);
    while (1)
    {
        std::cout << name << " is running... " << std::endl;
        sleep(1);
    }

    sleep(2);
    pthread_exit((void *)111);
}

int main()
{
    std::vector<pthread_t> tids;
    for (int i = 0; i < num; i++)
    {
        // 1 创建线程id
        pthread_t tid;

        // 2 线程的名字
        char *name = new char[128];
        snprintf(name, 128, "thread-%d", i + 1);
        pthread_create(&tid, nullptr, threadRun, (void *)name);

        // 3 保存所有线程的id信息
        tids.emplace_back(tid);
        // sleep(1);
    }

    sleep(3);

    for (auto tid : tids)
    {
        pthread_cancel(tid);
        void *result = nullptr;
        pthread_join(tid, &result);
        std::cout << (long long int)result << " quit ... " <<  std::endl;
    }


    return 0;
}

在这里插入图片描述
当自己取消自己的时候,返回的是-1.

问题七:新线程如何终止


  1. 线程函数 return
  2. pthread_exit
  3. main thread call pthread_cancel, 新线程退出结果是-1

线程分离

pthread_cancel:

int pthread_cancel(pthread_t thread);

参数说明:

  • thread:被分离线程的ID。

返回值:

  • 线程分离成功返回0,失败返回错误码。

示例:

使用detach来分离,join异常返回
在这里插入图片描述

在这里插入图片描述


问题8: 可以不可以不join线程,让他执行完就退出呢??可以!


  • 一个线程被创建,默认是joinable的,必须要被join的
  • 如果一个线程被分离,线程的工作状态分离状态,不须要/不能被join的. 依旧属于进程内部,但是不需要被等待了

使用pthread_detach对新创建的线程进行分离,然后当新线程的任务处理完成后,自动完成对其回收,不再需要等待。

#include <iostream>
#include <pthread.h>
#include <unistd.h>
#include <vector>

const int num = 10;

void *threadRun(void *args)
{
    int n = 5;		//新线程循环5次后直接退出。
    std::string name = static_cast<const char *>(args);
    while (n--)
    {
        std::cout << name << " is running... " << std::endl;
        sleep(1);
    }

    sleep(2);
    pthread_exit((void *)111);
}

std::string PrintToHex(pthread_t &tid)
{
    char buffer[128];
    snprintf(buffer, sizeof(buffer), "0x%lx", tid);
    return buffer;
}
int main()
{
    std::vector<pthread_t> tids;
    for (int i = 0; i < num; i++)
    {
        // 1 创建线程id
        pthread_t tid;

        // 2 线程的名字
        char *name = new char[128];
        snprintf(name, 128, "thread-%d", i + 1);
        pthread_create(&tid, nullptr, threadRun, (void *)name);

        // 3 保存所有线程的id信息
        tids.emplace_back(tid);
        // sleep(1);
    }

    sleep(5);		//主线程直接对新线程进行分离
    for (auto tid : tids)
    {
        pthread_detach(tid);
        std::cout << "detach tid: " << PrintToHex(tid) << std::endl;
    }
    std::cout << "Detach finish... " << std::endl;

    while (1)
    {
        std::cout << "I am main thread , i am running ... " << std::endl;
        sleep(1);
    }

	//不再进行join等待回收
    // for (auto tid : tids)
    // {
    //     pthread_cancel(tid);
    //     void *result = nullptr;
    //     int n = pthread_join(tid, &result);
    //     std::cout << (long long int)result << " quit ... n: " << n <<  std::endl;
    // }

    return 0;
}

在这里插入图片描述
在这里插入图片描述

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

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

相关文章

大型语言模型(LLM)——直接偏好优化完整指南

概述 将大型语言模型 (LLM) 与人类价值观和偏好相结合是一项挑战。传统方法&#xff0c;例如 [从人类反馈中强化学习](&#xff08;RLHF&#xff09;通过整合人类输入来完善模型输出&#xff0c;为这一领域的研究铺平了道路。然而&#xff0c;RLHF 可能非常复杂且资源密集&…

CVE-2024-27198 和 CVE-2024-27199:JetBrains TeamCity 服务器的漏洞利用及其防护措施

引言 JetBrains TeamCity 作为一个广泛使用的持续集成和部署工具&#xff0c;其安全性备受关注。然而&#xff0c;最近披露的CVE-2024-27198和CVE-2024-27199两个漏洞揭示了该平台存在的重大安全隐患。这些漏洞允许攻击者通过绕过身份验证机制&#xff0c;创建未经授权的管理员…

Java代码基础算法练习-乘阶求和-2024.08.18

对应的源代码可以在我的 Gitee 仓库中找到&#xff0c;欢迎star~ [Gitee 仓库](https://gitee.com/yukongji/java-basic-algorithm) 任务描述&#xff1a; 求Sn1!2!3!4!5!…n!之值&#xff0c;其中n是一个数字(n<10)。 解决思路&#xff1a; 输入: 读取用户输入的 n 值。检查…

Java Sream中自定义Collector实现复杂数据收集方法

❃博主首页 &#xff1a; 「码到三十五」 &#xff0c;同名公众号 :「码到三十五」&#xff0c;wx号 : 「liwu0213」 ☠博主专栏 &#xff1a; <mysql高手> <elasticsearch高手> <源码解读> <java核心> <面试攻关> ♝博主的话 &#xff1a…

雷达气象学(10)——双偏振雷达及其变量

文章目录 10.1 双偏振雷达的优势10.2 双偏振变量——反映降水粒子特性的变量10.3 差分反射率 Z D R Z_{DR} ZDR​10.3.1 差分反射率的定义10.3.2 雨滴的差分反射率10.3.3 冰、雪和冰雹的差分反射率10.3.4 层云降水的差分反射率10.3.5 对流降水的差分反射率 10.4 相关系数 ρ …

Chrome快捷键提高效率

浏览效率提高快捷建 快速切换标签页 Ctrl 数字&#xff08;1或者2&#xff09;&#xff0c;标签页数字从左到右为顺序&#xff0c;1开始。快速切换标签页。 Ctrl1 到 Ctrl8 切换到标签栏中指定位置编号所对应的标签页 Ctrl9切换到最后一个标签页 CtrlTab 或 CtrlPgDown 切…

Elasticsearch中磁盘水位线的深度解析

❃博主首页 &#xff1a; 「码到三十五」 &#xff0c;同名公众号 :「码到三十五」&#xff0c;wx号 : 「liwu0213」 ☠博主专栏 &#xff1a; <mysql高手> <elasticsearch高手> <源码解读> <java核心> <面试攻关> ♝博主的话 &#xff1a…

Clobbering DOM attributes to bypass HTML filters

目录 寻找注入点 代码分析 payload构造 注入结果 寻找注入点 DOM破坏肯定是出现在js文件中&#xff0c;我们首先来看源码 /resources/labheader/js/labHeader.js这个源码没什么问题我们重点关注在下面两个源码上 /resources/js/loadCommentsWithHtmlJanitor.js这个源码中重…

从关键新闻和最新技术看AI行业发展(第二十九期2024.7.29-8.11) |【WeThinkIn老实人报】

写在前面 【WeThinkIn老实人报】旨在整理&挖掘AI行业的关键新闻和最新技术&#xff0c;同时Rocky会对这些关键信息进行解读&#xff0c;力求让读者们能从容跟随AI科技潮流。也欢迎大家提出宝贵的优化建议&#xff0c;一起交流学习&#x1f4aa; 欢迎大家关注Rocky的公众号&…

docker 安装mino服务,启动报错: Fatal glibc error: CPU does not support x86-64-v2

背景 docker 安装mino服务&#xff0c;启动报错&#xff1a; Fatal glibc error: CPU does not support x86-64-v2 原因 Docker 镜像中的 glibc 版本要求 CPU 支持 x86-64-v2 指令集&#xff0c;而你的硬件不支持。 解决办法 降低minio对应的镜像版本 经过验证&#xff1a;qu…

sanic + webSocket:股票实时行情推送服务实现

&#x1f49d;&#x1f49d;&#x1f49d;欢迎莅临我的博客&#xff0c;很高兴能够在这里和您见面&#xff01;希望您在这里可以感受到一份轻松愉快的氛围&#xff0c;不仅可以获得有趣的内容和知识&#xff0c;也可以畅所欲言、分享您的想法和见解。 推荐&#xff1a;「storm…

Linux中的内核编程

Linux内核是操作系统的核心组件&#xff0c;负责管理系统的资源、提供硬件抽象和执行系统调用。内核编程是一项涉及操作系统核心的高级任务&#xff0c;它允许开发人员直接与系统内核进行交互&#xff0c;实现更高效、更特定的功能。本文将深入探讨Linux中的内核编程&#xff0…

2021年上半年网络工程师考试上午真题

2021年上半年网络工程师考试上午真题 网络工程师历年真题含答案与解析 第 1 题 以下关于RISC和CISC计算机的叙述中&#xff0c;正确的是&#xff08; &#xff09;。 (A) RISC不采用流水线技术&#xff0c;CISC采用流水线技术(B) RISC使用复杂的指令&#xff0c;CISC使用简…

事件驱动架构的事件版本管理

有一种办法&#xff1a;发送会议邀请给所有团队&#xff0c;经过101次会议后&#xff0c;发布维护横幅&#xff0c;所有人同时点击发布按钮。或... 可用适配器&#xff0c;但微调。没错&#xff01;就像软件开发中90%问题一样&#xff0c;有种模式帮助你找到聪明解决方案。 1…

C Primer Plus(中文版)第13章编程练习,仅供参考

第十三章编程练习 对于文件的操作是程序开发过程中必不可少的。首先&#xff0c;来看一下第一题&#xff0c;对13.1程序进行修改&#xff0c;输入文件名&#xff0c;而不是命令行参数。完整程序代码以及运行结果如下&#xff1a; #include<stdio.h> #include<stdlib…

【数据结构篇】~单链表(附源码)

【数据结构篇】~链表 链表前言链表的实现1.头文件2.源文件 链表前言 链表是一种物理存储结构上非连续、非顺序的存储结构&#xff0c;数据元素的逻辑顺序是通过链表中的指针链接次序实现的。 1、链式机构在逻辑上是连续的&#xff0c;在物理结构上不一定连续​ 2、结点一般是从…

Java二十三种设计模式-命令模式(18/23)

命令模式&#xff1a;将请求封装为对象的策略 概要 本文全面探讨了命令模式&#xff0c;从基础概念到实现细节&#xff0c;再到使用场景、优缺点分析&#xff0c;以及与其他设计模式的比较&#xff0c;并提供了最佳实践和替代方案&#xff0c;旨在帮助读者深入理解命令模式并…

【xr-frame】微信小程序xr-frame典型案例

微信小程序xr-frame典型案例 在之前的工作中&#xff0c;我大量使用XR-Frame框架进行AR开发&#xff0c;并积累了一些案例和业务代码。其中包括2D图像识别、手部动作识别、Gltf模型加载、动态模型加载、模型动画等内容。小程序部分使用TypeScript编写&#xff0c;而XR-Frame组…

利用puppeteer将html网页生成图片

1.什么是puppeteer&#xff1f; Puppeteer是一个Node库&#xff0c;它提供了一个高级API来通过DevTools协议控制Chromium或Chrome。 可以使用Puppeteer来自动化完成浏览器的操作&#xff0c;官方给出的一些使用场景如下&#xff1a; 生成页面PDF抓取 SPA&#xff08;单页应用…

3.Windows Login Unlocker-忘记电脑密码也可以解决

想要解锁Windows系统的开机密码&#xff0c;但官网的传统方法只适合Windows本地账户&#xff0c;对微软账户或PIN码()束手无策&#xff1f;别担心&#xff0c;小编之前推荐过的「Windows Login Unlocker」软件能为您排忧解难。这款出色的工具不仅能够轻松绕过各种Windows密码&a…