【Linux】线程安全——同步和互斥

需要云服务器等云产品来学习Linux的同学可以移步/–>腾讯云<–/官网,轻量型云服务器低至112元/年,优惠多多。(联系我有折扣哦)

文章目录

  • 引入
  • 1. Linux线程互斥
    • 1.1 互斥的相关概念
    • 1.2 互斥量mutex
    • 1.3 mutex的使用
    • 1.4 mutex的理解与实现
    • 1.5 mutex的封装——C++版本
    • 1.6 可重入与线程安全
      • 1.6.1 常见的线程不安全情况
      • 1.6.2 常见的线程安全情况
      • 1.6.3 常见的不可重入的情况
      • 1.6.4 常见的可重入情况
      • 1.6.5 可重入与线程安全的联系
      • 1.6.6 可重入与线程安全的区别
    • 1.7 常见的锁的概念
  • 2. Linux线程同步
    • 2.1 线程同步的相关概念和理解
    • 2.2 条件变量
      • 2.2.1 条件变量的概念
      • 2.2.2 条件变量的使用
      • 2.2.3 条件变量的理解
    • **由于条件变量本身并不具有互斥功能,所以我们在进行等待的时候必须配合互斥锁使用**

引入

上节中我们学习了多线程的概念和控制,但是实际上多线程的程序会有很多问题的出现。

首先我们看一个多线程程序的例子,一个模拟抢票的程序

#include <iostream>
#include <unistd.h>
#include "Thread.hpp"

int ticket = 10000;

void *getTicket(void *arg) // 执行抢票的逻辑
{
    while (true)
    {
        if (ticket > 0) // 当票量大于0的时候才能抢
        {
            usleep(1245); // 模拟抢票前执行的操作
            std::cout << static_cast<const char *>(arg) << "正在抢票" << ticket-- << std::endl; // 抢票
        }
        else // 如果票量小于0就不抢了
        {
            break;
        }
    }
    return nullptr;
}

int main()
{
    // 创建4个线程用于抢票
    Thread *thread1 = new Thread(getTicket, (void *)"thread-1");
    Thread *thread2 = new Thread(getTicket, (void *)"thread-2");
    Thread *thread3 = new Thread(getTicket, (void *)"thread-3");
    Thread *thread4 = new Thread(getTicket, (void *)"thread-4");

    thread1->join();
    thread2->join();
    thread3->join();
    thread4->join();

    return 0;
}

image-20240130152957444

这里可以看出多线程的程序出现了一些问题,那么怎么解决呢?

1. Linux线程互斥

1.1 互斥的相关概念

  • 临界资源:在计算机中存在着很多共享资源,这些资源是会被很多线程、进程共享的,为了保证这些资源的安全,所以需要被保护起来,这些被保护起来的共享资源就是临界资源

  • 临界区:临界资源被访问总是存在访问这些临界资源的代码,这些访问临界资源的代码叫做临界区

  • 互斥:任何时刻,互斥保证有且只有一个执行流进入临界区,访问临界资源,通常对临界资源起保护作用

  • 原子性:不会被任何调度机制打断的操作,该操作只有两态,要么完成,要么未完成,不会有中间状态

1.2 互斥量mutex

现在我们来分析引入中的问题产生的原因:

image-20240130155511793

我们看上面的抢票逻辑的代码,在第11行的地方访问了共享资源ticket

线程1在执行了第11行的代码之后,线程可不可以被切换?当然可以

线程1在此时被切换之后,(我们把情况推向极端,此时ticket就是1),然后线程2被切换过来执行

此时ticket还是1然后执行11行代码进入if的代码块执行完本次循环之后再切回线程1

此时线程1可以直接执行14行代码,将ticket–,此时就出现了抢到-1的情况

这就是上述问题产生的原因,那么如何解决呢?

我们在上面说到了原子性,对于一件事,要么全做,要么不做,这里如果让11行到吗到14行代码变成原子性的,也就能够让这个问题得到解决

使用互斥量mutex就能完成这个任务


我们知道,一般来说,一条汇编指令就是原子的,但是像++i或者i++都不是原子的,有可能会有数据一致性问题

这是因为这种操作对应了三条汇编指令:

  • load:将共享变量i从内存加载到寄存器中
  • updata:更新寄存器里面i的值
  • store:将新的值从寄存器协会i的内存地址

那如果要解决以上问题,就需要做到三点

  • 代码必须要有互斥行为:代码进入临界区执行的时候,不允许其他线程进入该临界区
  • 如果多个线程通知要求执行临界区代码,并且临界区没有其他进程在执行,那么只能允许一个线程进入该临界区
  • 如果线程不再临界区中执行,那么该线程不能组织其他线程进入临界区
image-20240131161805208

1.3 mutex的使用

mutex也是pthread库提供的。形象的来说,我们也可以把它叫做锁。所以也就有了互斥锁这种说法

1. 互斥锁的类型

image-20240131152814709

2. 定义初始化与销毁

image-20240131153440920

头文件:
#include <pthread.h>
函数原型:
int pthread_mutex_destory(pthread_mutex_t *mutex); // 销毁
int pthread_mutex_init(pthread_mutex_t *restrict mutex, const pthread_mutexattr_t *restrict attr); // 初始化
pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER; // 全局变量的初始化方式
参数解释:
	mutex:需要操作的锁
	attr:是一个pthread_mutexattr_t类型的联合体,这里我们不关心,设为NULL即可
函数描述:
	pthread_mutex_init:初始化一个互斥锁
	pthread_mutex_destory:销毁一个互斥锁
    锁有初始化就要有销毁
返回值:
	成功返回0,失败返回错误码

注意:mutex的定义可以是全局的也可以是局部的,如果是全局的话,可以直接使用宏来初始化PTHREAD_MUTEX_INITIALIZER,同时也就不需要销毁了,使用init函数初始化的就一定要使用destory销毁


小tips:restrict关键字表示mutexattr指针不会被函数以外的方式访问或修改,可以提高编译器对互斥锁初始化的优化

3. 互斥锁的加锁和解锁

image-20240131162735759

头文件:
	#include <pthread.h>
函数原型:
    int pthread_mutex_lock(pthread_mutex_t *mutex);
    int pthread_mutex_trylock(pthread_mutex_t *mutex);
    int pthread_mutex_unlock(pthread_mutex_t *mutex);
参数解释:
	mutex:要操作的互斥锁
函数描述:
    pthread_mutex_lock:加锁。如果互斥量处于未锁状态,该函数会将互斥量锁定,同时返回成功;发起函数调用时,其他线程已经锁定互斥量,或者存在其他线程同时申请互斥量,但没有竞争到互斥量,那么pthread_lock调用会陷入阻塞(执行流被挂起),等待互斥量解锁。
    pthread_mutex_trylock:尝试给mutex加锁,如果此时互斥量未锁状态,或者竞争到互斥量,就加锁然后返回0,如果没有竞争到或者已经被锁定,就执行其他内容,不会被挂起
    pthread_mutex_unlock:给传入的mutex解锁
返回值:
	lock和unlock调用成功返回0否则返回错误码,trylock如果获取到这个互斥锁,就返回0,否则返回错误码

使用互斥量改进抢票代码:

#include <iostream>
#include <unistd.h>
#include "Thread.hpp"

int ticket = 10000;

pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER;

void *getTicket(void *arg)
{
    while (true)
    {
        pthread_mutex_lock(&mutex); // 给mutex加锁
        if (ticket > 0)
        {
            usleep(1245);
            std::cout << static_cast<const char *>(arg) << "正在抢票" << ticket-- << std::endl;
            pthread_mutex_unlock(&mutex); // 给mutex解锁
        }
        else
        {
            pthread_mutex_unlock(&mutex); // 给mutex解锁
            break;
        }
    }
    return nullptr;
}

int main()
{
    Thread *thread1 = new Thread(getTicket, (void *)"thread-1");
    Thread *thread2 = new Thread(getTicket, (void *)"thread-2");
    Thread *thread3 = new Thread(getTicket, (void *)"thread-3");
    Thread *thread4 = new Thread(getTicket, (void *)"thread-4");

    thread1->join();
    thread2->join();
    thread3->join();
    thread4->join();

    return 0;
}

image-20240131190211262

1.4 mutex的理解与实现

1. 如何看待锁

经过了前面的使用,我们发现**锁本身就要被多个线程共同看见,所以锁本身就是一个共享资源!!**我们知道锁是用来保护共享资源的,那锁的安全由谁来保护?

实际上我们对锁的操作pthread_mutex_lockpthread_mutex_unlock本身就是原子的,所以加锁解锁的过程天生就是安全的。谁持有锁谁就进入临界区,否则就会在阻塞等待

image-20240201190427555

2. mutex加锁解锁的汇编实现

事实上,为了实现互斥锁的操作,大多数体系结构都提供了swap或exchange指令,该指令的作用就是把寄存器和内存单元的数据进行交换,由于只有一条指令,所以保证了原子性,即使是多处理器平台,访问内存的总线周期也有先后,一个处理器上的交换指令执行期间另一个处理器的交换指令也只能等待总线周期

lock:
	movb $0, %al
	xchgb %al, mutex
	if(%al > 0){
		return 0;
	} else
		挂起等待;
	goto lock;
unlock:
	movb $1, mutex
	唤醒等待Mutex的线程;
	return 0;

汇编代码的分析:

加锁过程

  • 首先将寄存器al的内容清0,本质上是将该线程的上下文中对应的al的内容清零,(因为每个线程都有一个对应的al寄存器内容)

  • 然后使用原子的指令xchgb将al寄存器和内存中mutex的值进行交换,如果当前锁没有被申请,那么可以理解成mutex中的值为1,如果被申请,那么就是0。

  • 交换后检查al寄存器中的值,如果大于0,那么就说明当前线程竞争到了锁资源,此时就可以进入临界区访问对应的临界资源;否则申请锁失败需要被挂起等待,直到锁被释放后再次竞争申请锁。

解锁过程

  • 首先把内存中mutex的值置1,使得下一次使用xchgb指令能够交换到1进寄存器,表示获取锁
  • 唤醒正在等待mutex的线程,让其竞争这个锁
  • 在申请锁时本质上就是哪一个线程先执行了交换指令,那么该线程就申请锁成功,因为此时该线程的al寄存器中的值就是1了。而交换指令就只是一条汇编指令,一个线程要么执行了交换指令,要么没有执行交换指令,所以申请锁的过程是原子的。
  • 在线程释放锁时没有将当前线程al寄存器中的值清0,这不会造成影响,因为每次线程在申请锁时都会先将自己al寄存器中的值清0,再执行交换指令。
  • CPU内的寄存器不是被所有的线程共享的,每个线程都有自己的一组寄存器数据,但内存中的数据是各个线程共享的。申请锁实际就是,把内存中的mutex通过交换指令,原子性的交换到自己的al寄存器中。

1.5 mutex的封装——C++版本

当然,使用pthread版本的锁是C语言风格的,而且较为复杂,有可能会写出逻辑异常的代码,在C++的开发中,我们希望使用的还是一个C++式的锁,也就是面向对象的结构,那么接下来我们可以手动封装出一个C++版本的mutex。顺手实现成RAII风格的,帮助我们自动加锁和解锁

/*Mutex.hpp*/
#pragma once

#include <pthread.h>

class Mutex
{
public:
    Mutex(pthread_mutex_t *lock_p = nullptr) : _lock_p(lock_p) {} // 构造函数

    void lock() // 加锁
    {
        if (_lock_p)
            pthread_mutex_lock(_lock_p);
    }
    void unlock() // 解锁
    {
        if (_lock_p)
            pthread_mutex_lock(_lock_p);
    }

    ~Mutex() {} // 析构函数

private:
    pthread_mutex_t *_lock_p;
};

class LockGuard // RAII风格的锁的实现
{
public:
    LockGuard(pthread_mutex_t *mutex) : _mutex(mutex) // 构造函数
    {
        _mutex.lock(); // 在构造函数中加锁
    }
    ~LockGuard() // 析构函数
    {
        _mutex.unlock(); // 在析构函数中解锁
    }

private:
    Mutex _mutex;
};

改写抢票程序:

#include <iostream>
#include <unistd.h>
#include "Thread.hpp"
#include "Mutex.hpp"

int ticket = 10000;
pthread_mutex_t lock = PTHREAD_MUTEX_INITIALIZER;

void *getTicket(void *arg)
{
    while (true)
    {
         LockGuard lockguard(&lock); // 这里在栈区创建Mutex对象并加锁,在这个代码块结束之后解锁,可以手动设置代码块,以达到在指定位置解锁的功能
        {
            if (ticket > 0)
            {
                std::cout << static_cast<const char *>(arg) << "正在抢票" << ticket-- << std::endl;
            }
            else
            {
                break;
            }
        }
        usleep(1245);
    }
    return nullptr;
}

int main()
{
    Thread *thread1 = new Thread(getTicket, (void *)"thread-1");
    Thread *thread2 = new Thread(getTicket, (void *)"thread-2");
    Thread *thread3 = new Thread(getTicket, (void *)"thread-3");
    Thread *thread4 = new Thread(getTicket, (void *)"thread-4");

    thread1->join();
    thread2->join();
    thread3->join();
    thread4->join();

    return 0;
}

1.6 可重入与线程安全

1.6.1 常见的线程不安全情况

  • 不保护共享变量的函数
  • 函数状态随着被调用,状态发生变化的函数
  • 返回指向静态变量指针的函数
  • 调用线程不安全函数的函数

1.6.2 常见的线程安全情况

  • 每个线程对全局变量或者静态变量只有读取的权限,而没有写入的权限,一般来说这些线程是安全的
  • 类或者接口对于线程来说都是原子操作
  • 多个线程之间的切换不会导致该接口的执行结果存在二义性

1.6.3 常见的不可重入的情况

  • 调用了malloc/free函数,因为malloc函数是用全局链表来管理堆的
  • 调用了标准I/O库函数,标准I/O库的很多实现都以不可重入的方式使用全局数据结构
  • 可重入函数体内使用了静态的数据结构

1.6.4 常见的可重入情况

  • 不使用全局变量或静态变量 不使用用malloc或者new开辟出的空间
  • 不调用不可重入函数
  • 不返回静态或全局数据,所有数据都有函数的调用者提供
  • 使用本地数据,或者通过制作全局数据的本地拷贝来保护全局数据

1.6.5 可重入与线程安全的联系

  • 函数是可重入的,那就是线程安全的
  • 函数是不可重入的,那就不能由多个线程使用,有可能引发线程安全问题
  • 如果一个函数中有全局变量,那么这个函数既不是线程安全也不是可重入的。

1.6.6 可重入与线程安全的区别

  • 可重入函数是线程安全函数的一种
  • 线程安全不一定是可重入的,而可重入函数则一定是线程安全的
  • 如果将对临界资源的访问加上锁,则这个函数是线程安全的,但如果这个重入函数若锁还未释放则会产生
    死锁,因此是不可重入的。

1.7 常见的锁的概念

  • 死锁:一组执行流(不管进程还是线程)持有自己锁资源的同时,还想要申请对方的锁,锁是不可抢占的(除非自己主动归还),会导致多个执行流互相等待对方的资源,而导致代码无法推进。

一把锁可以造成死锁,比如说在抢票的时候,如果在申请锁之后再加申请一次锁导致死锁。

为什么会有死锁?一定是你用了锁,锁保证临界资源的安全

多线程访问我们可能出现数据不一致的问题 多线程共享全局资源全局资源

解决问题的同时带来了新的问题:死锁,任何技术都有自己的边界,在解决问题的同时一定可能会引入新的问题

死锁四个必要条件:

  • 互斥:一个共享资源每次被一个执行流使用

  • 请求与保持:一个执行流因请求资源而阻塞,对已有资源保持不放

  • 不剥夺:一个执行流获得的资源在未使用完之前,不能强行剥夺

  • 环路等待条件:执行流间形成环路问题,循环等待资源

避免死锁的方法:

  1. 破坏死锁的四个必要条件
  2. 加锁顺序一致
  3. 避免锁未释放的场景
  4. 资源一次性分配

也有一些算法可以避免死锁(了解):死锁检测算法、银行家算法

但是,在实际开发过程中,我们要尽可能少的使用锁

2. Linux线程同步

2.1 线程同步的相关概念和理解

举一个例子:现在学校里面有一个VIP自习室,里面只有一个座位,统一时间容纳一个人自习,大家都想去这个自习室学习,其中某个人起床特别早,抢到了这个自习室的名额,然后就一直在里面学习,其他人想要使用这个自习室,就在外面排队,等这个人出来之后才能进去。中午这个人想出去吃饭,但是刚出门又犹豫了,好不容易抢到的自习室,一定要多学一会,此时这个人刚出门,所以离自习室最近,所以肯定能抢到这个自习室的名额。这个往复很多次,进行了很多次的资源争夺,但是每次都是这个人能抢到,其他人抢不到。这种情况显然是不符合逻辑的,所以需要采用一些措施来避免这种情况。

在回到上文,我们可以发现一个现象,当我们给抢票代码加锁之后,每次抢票的都是同一个线程,这样就造成了其他线程的饥饿现象。为了解决这个问题:我们在数据安全的情况下让这些线程按照一定的顺序进行访问,这就是线程同步

首先我们明确几个概念:

  • 饥饿状态:由于一直得不到锁资源而无法访问公共资源的线程的状态。虽然并没有错,但是不合理
  • 竞态条件:因为时序问题,而导致程序异常,我们称之为竞态条件
  • 同步:在保证数据安全的前提下,让线程能够按照某种特定的顺序访问临界资源,从而有效避免饥饿问题,叫做同步

2.2 条件变量

2.2.1 条件变量的概念

当一个线程互斥地访问某个变量时,它可能发现在其他线程改变状态之前,它什么也做不了(因为被阻塞)

例如一个线程访问队列时,发现队列为空,它只能等待,直到其他线程将一个节点添加到队列中。这种情况就需要用到条件变量

  • 条件变量是利用线程间共享的全局变量进行同步的一种机制,条件变量是用来描述某种资源是否就绪的一种数据化描述。

使用条件变量主要包括两个动作:

  • 一个线程等待条件变量的条件成立而被挂起。
  • 另一个线程使条件成立后唤醒等待的线程。

条件变量通常需要配合互斥锁一起使用。

2.2.2 条件变量的使用

1. 初始化和销毁

image-20240202191824614

头文件:
#include <pthread.h>
函数原型:
int pthread_cond_init(pthread_cond_t *restrict cond, const pthread_condattr *restrict attr);//初始化
int pthread_cond_destory(pthread_cond_t cond);
参数解释:
	cond:需要初始化的条件变量,类型是pthread_t
	attr:这里我们不关心,设为NULL即可
函数描述:
	初始化和销毁条件变量
返回值:
	如果调用成功,函数返回0,否则返回错误码

当然,和mutex一样,如果创建全局的条件变量,那么可以直接使用PTHREAD_COND_INITIALIZER初始化

2. 等待条件满足

image-20240202194928919

头文件:
#include <pthread.h>
函数原型:
int pthread_cond_wait(pthread_cond_t *restrict cond, pthread_mutex_t mutex);
参数解释:
	cond:在指定条件变量下等待
	mutex:传入一个锁,这里我们后面在解释
返回值:
	成功返回0,失败返回错误码

3. 唤醒等待

image-20240202200109606

头文件:
#include <pthread.h>
函数原型:
int pthread_cond_broadcast(pthread_cond_t *cond);
int pthread_cond_signal(pthread_cond_t *cond);
参数解释:
	cond:要操作的条件变量
函数描述:
	pthread_cond_broadcast:唤醒等待队列中首个线程
	pthread_cond_signal:唤醒等待队列中所有线程
返回值:
	成功返回0,失败返回错误码

一个小例子:

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

pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER;
pthread_cond_t cond = PTHREAD_COND_INITIALIZER;

void *start_routine(void *args)
{
    while (true)
    {
        pthread_mutex_lock(&mutex);       // 假设这里是临界区,使用mutex保护
        pthread_cond_wait(&cond, &mutex); // 判断条件变量条件
        std::cout << static_cast<const char *>(args) << "is running" << std::endl;

        pthread_mutex_unlock(&mutex);
    }
}

int main()
{
    pthread_t t1, t2, t3;

    pthread_create(&t1, nullptr, start_routine, (void *)"thread-1");
    pthread_create(&t2, nullptr, start_routine, (void *)"thread-2");
    pthread_create(&t3, nullptr, start_routine, (void *)"thread-3");

    while (true)
    {
        getchar();
        pthread_cond_signal(&cond);
        std::cout << "main thread is waked up a thread" << std::endl;
    }

    pthread_join(t1, nullptr);
    pthread_join(t2, nullptr);
    pthread_join(t3, nullptr);

    return 0;
}

image-20240202202339926

2.2.3 条件变量的理解

举个例子:某公司进行招聘:应聘者要面试,大家不能同时进入房间进行面试,但是没有由于没有组织,上一个人面试完之后,面试官打开门准备面试下一个,一群人在外面等待面试,但是有人抢不过别人,人太多了,面试官记不住谁面试过了,所以有可能一个人面试完之后又去面试了,造成其他人饥饿问题,这时候效率很低

后来HR重新进行管理:设立一个等待区,先排队去等待区进行等待面试,现在每个人都进行排队,都有机会面试了,而这个等待区就是条件变量,如果一个人想面试,先得去排队等待区等待,未来所有应聘者都要去条件变量等

条件不满足的时候,线程必须去某些定义好的条件变量上进行等待

所以我们可以知道,条件变量应该是一个结构体(struct cond)里面包含状态,队列,而我们定义好的条件变量包含一个队列,不满足条件的线程就链接在这个队列上进行等待。

由于条件变量本身并不具有互斥功能,所以我们在进行等待的时候必须配合互斥锁使用

本节完。。。

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

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

相关文章

Windows11 用 HyperV 安装 Ubuntu-16.04 虚拟机

Windows11 用 HyperV 安装 Ubuntu-16.04 虚拟机 1. 确保已经开启HyperV2. 准备Ubuntu16.04镜像&#xff08;推荐64位的&#xff09;3. HyperV ->快速创建 -> 更改安装源 选刚刚下载的镜像&#xff08;.iso&#xff09;文件就好 -> 创建虚拟机[^1] 前提&#xff1a;VMw…

<网络安全>《15 移动安全管理系统》

1 概念 移动安全管理系统&#xff0c;MSM&#xff0c;Mobile security management,提供大而全的功能解决方案&#xff0c;覆盖了企业移动信息化中所涉及到安全沙箱、数据落地保护、威胁防护、设备管理、应用管理、文档管理、身份认证等各个维度。移动安全管理系统将设备管理和…

基于SpringBoot Vue单位考勤管理系统

大家好✌&#xff01;我是Dwzun。很高兴你能来阅读我&#xff0c;我会陆续更新Java后端、前端、数据库、项目案例等相关知识点总结&#xff0c;还为大家分享优质的实战项目&#xff0c;本人在Java项目开发领域有多年的经验&#xff0c;陆续会更新更多优质的Java实战项目&#x…

MacOS安装JDK+Maven+Idea插件+nvm等

Java安装环境(MacOS)JDKMavenIdea插件nvm等 背景&#xff1a;新机安装开发环境发现需要找很多文章&#xff0c;&#xff0c;&#xff0c;&#xff0c;这里一篇文章安装所有环境 文章目录 Java安装环境(MacOS)JDKMavenIdea插件nvm等一、安装JDK①&#xff1a;下载②&#xff1a;…

opencv0014 索贝尔(sobel)算子

前面学习的滤波器主要是用来模糊图像&#xff0c;今天一起来了解关于边缘识别的滤波吧&#xff01;嘿嘿 边缘 边缘是像素值发生跃迁的位置&#xff0c;是图像的显著特征之一&#xff0c;在图像特征提取&#xff0c;对象检测&#xff0c;模式识别等方面都有重要的作用。 人眼如…

【牛B得一塌糊涂】窗口归一化技术,改进医学图像的分布外泛化能力

窗口归一化技术&#xff0c;改进医学图像的分布外泛化能力 提出背景WIN、WIN-WIN、无参数归一化、特征级别数据增强如何提升分布外的泛化&#xff1f; 总结子问题1: 医学图像中的局部特征表示不足子问题2: 训练数据与新场景数据分布不一致子问题3: 模型在分布外数据上泛化能力不…

docker 容器指定主机网段

docker 容器指定主机网段。 直接连接到物理网络&#xff1a;使用macvlan技术可以让Docker容器直接连接到物理网络&#xff0c;而不需要通过NAT或端口映射的方式来访问它们。可以提高网络性能和稳定性&#xff0c;同时也可以使容器更加透明和易于管理。 1、查询网卡的名称&…

C++初阶之类与对象(上)详细解析

个人主页&#xff1a;点我进入主页 专栏分类&#xff1a;C语言初阶 C语言进阶 数据结构初阶 Linux C初阶 欢迎大家点赞&#xff0c;评论&#xff0c;收藏。 一起努力&#xff0c;一起奔赴大厂 目录 一.前言 二.类的定义和使用 2.1类的引入 2.2类的定义和访问限定…

ubuntu22.04安装部署02:禁用显卡更新

一、查看可用显卡驱动 ubuntu-drivers devices 二、查看显卡信息 # -i表示不区分大小写 lspci | grep -i nvidia nvidia-smi 三、查看已安装显卡驱动 cat /proc/driver/nvidia/version 四、锁定显卡升级 使用cuda自带额显卡驱动&#xff0c;居然无法&#xff0c;找到如何锁…

构建LLM辅助生物威胁制造预警系统 人类越发展获取的超能力越大,破坏力越大,威胁越大。我们需要什么样的预警系统?既克服威胁又具有超能力 安全基础

https://openai.com/research/building-an-early-warning-system-for-llm-aided-biological-threat-creation 人类越发展获取的超能力越大&#xff0c;破坏力就越大&#xff0c;威胁越大。 人工智能就是为了赋予人人都能有超能力&#xff0c;而一旦被恶意或无意使用又威胁到人…

KNIME 节点之战(Game of Nodes)锦标赛

“Hark! I summon thee to a contest of nodes. Art thou endowed with the courage for the encounter?” “听着&#xff01;我在此邀请你加入一场节点之战。你有勇气面对吗&#xff1f;” 官方链接 活动概要与参赛守则 诚邀您加入 KNIME 节点之战 —— 首届全球工作流挑战大…

Megatron-LM源码系列(七):Distributed-Optimizer分布式优化器实现Part2

1. 使用入口 DistributedOptimizer类定义在megatron/optimizer/distrib_optimizer.py文件中。创建的入口是在megatron/optimizer/__init__.py文件中的get_megatron_optimizer函数中。根据传入的args.use_distributed_optimizer参数来判断是用DistributedOptimizer还是Float16O…

【C++初阶】--入门基础(二)

目录 一.C输出与输入 二.缺省参数 1.概念 2.缺省参数分类 (1) 全缺省参数 (2)半缺省参数 三.函数重载 1.概念 2.C支持函数重载的原理--名字修饰 四.引用 1.概念 2.语法 3.引用的特性 (1)引用在定义时必须初始化 (2)引用时不能改变指向 (3)一个变量…

区间时间检索

前端 <el-col :md"6" v-if"advanced"><el-form-item :label"$t(inRecord.column.createTime)"><el-date-pickerstyle"width: 100%;"v-model"daterangeCreateTime"value-format"yyyy-MM-dd"type&qu…

装饰你的APP:使用Lottie-Android创建动画效果

装饰你的APP&#xff1a;使用Lottie-Android创建动画效果 1. Lottie-Android简介 Lottie-Android是一个强大的开源库&#xff0c;由Airbnb开发&#xff0c;旨在帮助开发者轻松地在Android应用中添加高质量的动画效果。它基于Adobe After Effects软件中的Bodymovin插件&#x…

【项目简记】逆向工程裸机内核镜像

本教程将是裸机逆向工程系列的一部分。 自从拆解了几部安卓手机后&#xff0c;我对嵌入式系统的兴趣越来越大。 虽然手机本身并不是嵌入式系统&#xff0c;但我知道手机最终会取代计算机&#xff1b;因此&#xff0c;我想学习更多关于它们的知识。 就在那时&#xff0c;我开始…

Linux 系统开始配置

文章目录 备份源为root 设置密码安装基本工具切换root 用户删除snap从 Ubuntu 移除 Snap 后使用 deb 文件安装软件商店和 Firefox在 Ubuntu 系统恢复到 Snap 软件包总结 删除 vim安装neovim在线安装neovim压缩安装neovim安装lazyvim安装剪切板 安装qt配置 Qt 环境不在sudoers文…

SAP 消息号 FAGL_CLOSING_ACT011

在S4当中&#xff0c;月末外币评估的时候&#xff0c;会出现如下报错&#xff1a; 解决方法是&#xff1a; “创建错误更正和暂记会计核算运行标识的编号范围、 在ECS中创建凭证编号范围” 给以上2个事务&#xff0c;添加号码范围即可。

关于破解IDEA后启动闪退的问题

问题描述&#xff1a;2023.1启动不了&#xff0c;双击桌面图标&#xff0c;没有响应。 解决办法&#xff1a; 打开C:\Users\c\AppData\Roaming\JetBrains\IntelliJIdea2023.1\idea64.exe.vmoptions 这个文件。 内容如下所示&#xff1a; 删除红框的数据以后&#xff0c;再登录…

使用 IDEA 开发一个简单易用的 SDK

目录 一、什么是 SDK 二、为什么要开发 SDK 三、开发 SDK 的详细步骤 四、导入 SDK 进行测试 附&#xff1a;ConfigurationProperties 注解的介绍及使用 一、什么是 SDK 1. 定义&#xff1a;软件开发工具包 Software Development Kit 2. 用于开发特定软件或应用程序的工…