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

Linux 线程控制

2. Linux 线程控制

首先,**内核中有没有很明确的线程的概念**,而有**轻量级进程的概念**。当我们想写多线程代码时,可以使用**POSIX线程库**,这是一个
处于应用层位置的库,几乎所有的Linux发行版都默认带这个库,使用时需要引入头文件`pthread.h`

2.1 创建线程

2.1.1 基础创建

功能:创建一个新的线程
原型:
int pthread_create(pthread_t *thread, const pthread_attr_t *attr, void *(*start_routine)(void*), void *arg);
参数:
thread:输出型参数,返回线程ID
attr:设置线程的属性,attr为NULL表示使用默认属性
start_routine:是个函数地址,线程启动后要执行的函数
arg:传给线程启动函数的参数
返回值:成功返回0;失败返回错误码
  • 传统的一些函数是,成功返回0,失败返回-1,并且对全局变量errno赋值以指示错误
  • pthreads函数出错时不会设置全局变量errno(而大部分其他POSIX函数会这样做)。而是将错误代码通过返回值返回
  • pthreads同样也提供了线程内的errno变量,以支持其它使用errno的代码。对于pthreads函数的错误,建议通过返回值来判定,因为读取返回值要比读取线程内的errno变量的开销更小

由于是第三方库,所以编译时要加上-lpthread

myThread : myThread.ccg++ -o $@ $^ -std=c++11 -lpthread.PHONY : clean
clean:rm -f myThread

image-20240914203245933

可以看到,这不同的线程拥有同一个进程的pid

当我们想观察线程状态时,可以使用ps -aL命令,LWP的时ligtht weight process的缩写,PID等于LWP的线程是主线程。当我们使用kill -9命令时,杀掉任意一个线程,整个进程都会被kill掉

image-20240914203630015

下面这个可以持续观察线程状态

while :; do ps -aL | head -1 && ps -aL | grep 'myThread' | grep -v ps; echo "##############################################"; sleep 1; done

2.1.2 让一个函数被重入,让多个执行流同时调用

// 可以被多个执行流执行
void Show(const string& s) 
{cout << s << " say: hello!" << endl;
}void* ThreadRoutine(void* args) 
{while(true) {// cout << "new thread, pid is " << getpid() << endl;Show("[new thread]");sleep(1); }
}int main()
{pthread_t tid;pthread_create(&tid, nullptr, ThreadRoutine, nullptr);while(true) {// cout << "main thread, pid is " << getpid() << endl;Show("[main thread]");sleep(1); }return 0;
}

image-20240914204817506

虽然打印出现了错行的情况,但不影响现象的产生


2.1.3 线程之间进行通信很容易

int g_val = 0;void* ThreadRoutine(void* args) 
{while(true) {// cout << "new thread, pid is " << getpid() << endl;// Show("[new thread]");printf("I am new thread, g_val: %d, &g_val: %p\n", g_val, &g_val);sleep(1); }
}int main()
{pthread_t tid;pthread_create(&tid, nullptr, ThreadRoutine, nullptr);while(true) {// cout << "main thread, pid is " << getpid() << endl;// Show("[main thread]");g_val++;printf("I am main thread, g_val: %d, &g_val: %p\n", g_val, &g_val);sleep(1); }return 0;
}

image-20240914205310595


2.1.4 线程任意一个出现异常,进程都会退出

void* ThreadRoutine(void* args) 
{while(true) {cout << "new thread, pid is " << getpid() << endl;sleep(5);int *p =nullptr;*p = 1;}
}int main()
{pthread_t tid;pthread_create(&tid, nullptr, ThreadRoutine, nullptr);while(true) {cout << "main thread, pid is " << getpid() << endl;sleep(1); }return 0;
}

image-20240914205647889


2.1.5 观察类型是pthread_t 的 tid

定义 pthread_t 类型是 无符号长整型

typedef unsigned long int pthread_t;

可以打印查看

int main()
{pthread_t tid;pthread_create(&tid, nullptr, ThreadRoutine, nullptr);while(true) {printf("I am main thread, creat new thread tid is %lu\n");sleep(1); }return 0;
}

image-20240914210917302

可以看到,打印出来的tid的值与LWP的值完全不一样,这是因为tid是给用户使用的的,而LWP是给OS使用的。
实际上,tid充当的是地址,在2.4中会介绍

2.1.6 给子线程传递参数

void* ThreadRoutine(void* args) 
{const char* name = (const char*)args;while(true) {printf("%s is running\n", name);sleep(1); }
}int main()
{pthread_t tid;pthread_create(&tid, nullptr, ThreadRoutine, (void*)"new thread");while(true) {printf("main thread is running\n");sleep(1); }return 0;
}

image-20240914211629840

2.2 线程等待

2.2.1 为什么需要线程等待?

  • 已经退出的线程,其空间没有被释放,仍然在进程的地址空间内。
  • 创建新的线程不会复用刚才退出线程的地址空间。
  • 如果需要,可以获取子线程的返回值

2.2.2 基础使用

功能:等待线程结束
原型
int pthread_join(pthread_t thread, void **value_ptr);
参数
thread:线程ID
value_ptr:它指向一个指针,后者指向线程的返回值
返回值:成功返回0;失败返回错误码
  1. 如果thread线程通过return返回,value_ ptr所指向的单元里存放的是thread线程函数的返回值。
  2. 如果thread线程被别的线程调用pthread_ cancel异常终掉,value_ ptr所指向的单元里存放的是常数PTHREAD_ CANCELED,其值为-1。
  3. 如果thread线程是自己调用pthread_exit终止的,value_ptr所指向的单元存放的是传给pthread_exit的参数。
  4. 如果对thread线程的终止状态不感兴趣,可以传NULL给value_ ptr参数。
void* ThreadRoutine(void* args) 
{const char* name = (const char*)args;int cnt = 5;while(cnt--) {printf("%s is running\n", name);sleep(1); }return nullptr;     // 线程走到这里后,会默认退出
}int main()
{pthread_t tid;pthread_create(&tid, nullptr, ThreadRoutine, (void*)"new thread");pthread_join(tid, nullptr);     // 主线程等待的时候,默认是阻塞等待cout << "main thread quit" << endl;return 0;
}

image-20240914213150340

2.2.3 获取子线程的返回值

image-20240914214809868

void* ThreadRoutine(void* args) 
{const char* name = (const char*)args;int cnt = 5;while(cnt--) {printf("%s is running\n", name);sleep(1); }return (void*)1;     // 线程走到这里后,会默认退出
}int main()
{pthread_t tid;pthread_create(&tid, nullptr, ThreadRoutine, (void*)"new thread");void* ret;pthread_join(tid, &ret);     // 主线程等待的时候,默认是阻塞等待printf("main thread quit, ret: %d\n", (long long)ret);  // 由于void* 是8字节,所以这里要用8字节的long long 来进行强转return 0;
}

image-20240914220539475


如果想要多返回些信息,可以使用类

struct Request
{Request(int s, int e, string n) : _start(s), _end(e), _name(n) {}// 累加 [start, end]int Run(){int tmp = 0;for(int i = _start; i<=_end; ++i) {tmp += i;}return tmp;}int _start;int _end;string _name;
};struct Response
{Response(int r, int e) : _res(r), _exitCode(e) {}int _res;int _exitCode;
};void* GetSum(void* args)
{Request* rq = static_cast<Request*>(args);Response* rs = new Response(0, 0);// 计算rs->_res = rq->Run();delete rq;return rs;
}int main()
{pthread_t tid;Request* rq = new Request(1, 100, "new Thread");pthread_create(&tid, nullptr, GetSum, rq);void* ret;pthread_join(tid, &ret);Response* rs = static_cast<Response*>(ret);printf("res: %d, exitCode: %d\n", rs->_res, rs->_exitCode);delete rs;return 0;
}

image-20240914225356363

以这个为例子,如果一个计算任务很大,比如1-100000,就可以拆分,让不同的线程执行不同的范围,最后主线程再将子线程的结果进行汇总

2.3 线程终止

2.3.1 直接使用return

上面的所有例子都是使用这个方法,当线程走到return后,会默认退出。注意:exit()是进程退出函数,不是线程,当线程函数使用它时会使整个进程退出

2.3.2 pthread_exit

功能:线程终止
原型
void pthread_exit(void *value_ptr);
参数
value_ptr:value_ptr不要指向一个局部变量。
返回值:无返回值,跟进程一样,线程结束的时候无法返回到它的调用者(自身)
void* ThreadRoutine(void* args) 
{const char* name = (const char*)args;int cnt = 5;while(cnt--) {printf("%s is running\n", name);sleep(1); }pthread_exit((void*) 123);
}int main()
{pthread_t tid;pthread_create(&tid, nullptr, ThreadRoutine, (void*)"new thread");void* ret;pthread_join(tid, &ret);     // 主线程等待的时候,默认是阻塞等待printf("main thread quit, ret: %d\n", (long long)ret);  // 由于void* 是8字节,所以这里要用8字节的long long 来进行强转return 0;
}

image-20240914221801170

2.3.3 pthread_cancel

功能:取消一个执行中的线程
原型
int pthread_cancel(pthread_t thread);
参数
thread:线程ID
返回值:成功返回0;失败返回错误码
void* ThreadRoutine(void* args) 
{const char* name = (const char*)args;while(true) {printf("%s is running\n", name);sleep(1); } return nullptr;
}int main()
{pthread_t tid;pthread_create(&tid, nullptr, ThreadRoutine, (void*)"new thread");// 主进程取消掉子进程sleep(1);pthread_cancel(tid);void* ret;pthread_join(tid, &ret);     // 主线程等待的时候,默认是阻塞等待printf("main thread quit, ret: %d\n", (long long)ret);  // 由于void* 是8字节,所以这里要用8字节的long long 来进行强转return 0;
}

image-20240914222346956

可以看到ret为-1,那是因为PTHREAD_ CANCELED,其值为-1。见2.2.2

2.4 线程ID及进程地址空间布局

  • pthread_ create函数会产生一个线程ID,存放在第一个参数指向的地址中。该线程ID和前面说的线程ID不是一回事,(pid和LWP)
  • 前面讲的线程ID属于进程调度的范畴。因为线程是轻量级进程,是操作系统调度器的最小单位,所以需要一个数值来唯一表示该线程
  • pthread_ create函数第一个参数指向一个虚拟内存单元,该内存单元的地址即为新创建线程的线程ID,属于NPTL线程库的范畴。线程库的后续操作,就是根据该线程ID来操作线程的
  • 线程库NPTL提供了pthread_ self函数,可以获得线程自身的ID:
功能:pthread_self - 获取调用线程的ID原型:pthread_t pthread_self(void);
描述:pthread_self()函数返回调用线程的ID。与在创建this的pthread_create(3)调用中的*thread中返回的值线程是一样的。
返回值:这个函数总是成功,返回调用线程的ID。
// 将tid以地址的格式打出来,即16进制
string ToHex(pthread_t tid)
{char hex[64];snprintf(hex, sizeof(hex), "%p", tid);return hex;
}void* ThreadRoutine(void* args)
{while(true) {// 打印出进程id,将它转成16进程// cout << "thread id: " << ToHex(pthread_self()) << endl;printf("thread id: %s\n", ToHex(pthread_self()).c_str());sleep(1);}
}int main()
{pthread_t tid;pthread_create(&tid, nullptr, ThreadRoutine, nullptr);while(true) {printf("child tid: %s\n", ToHex(tid).c_str());sleep(1);}pthread_join(tid, nullptr);return 0;
}

image-20240915094954153

pthread_t到底是什么类型呢?取决于实现。对于Linux目前实现的NPTL实现而言,pthread_t类型的线程ID,本质就是一个进程地址空间上的一个地址

image-20240915100849785

可以看到,除了主线程,所有其他线程的独立栈,都在共享区,具体来讲是在pthread库中,tid指向的用户tcb中,这样,线程在调度运行的时候就不会互相干扰了。

2.5 其它问题

2.5.1 创建多个线程

#include <iostream>
#include <vector>
#include <string>
#include <unistd.h>
#include <pthread.h>
#define NUM 3
using namespace std;struct ThreadData
{   ThreadData(string name) : _threadName(name) {}string _threadName;
};void* ThreadRoutine(void* args)
{ThreadData* td = static_cast<ThreadData*>(args);int cnt = 5;while(cnt--) {printf("%s, tid: %p, pid: %d\n", td->_threadName.c_str(), pthread_self(), getpid());sleep(1);}delete td;return nullptr;
}int main()
{// 创建多个线程vector<pthread_t> tids;for (size_t i = 0; i < NUM; i++) {pthread_t tid;ThreadData* td = new ThreadData("Thread-" + to_string(i+1));// 给子线程传递数据,堆空间是共享的pthread_create(&tid, nullptr, ThreadRoutine, td);tids.push_back(tid);}// 等待线程for(const auto& t : tids) {pthread_join(t, nullptr);}return 0;
}

image-20240918115709458

2.5.2 每一个线程自己独立的栈结构

修改ThreadRoutine()函数

void* ThreadRoutine(void* args)
{ThreadData* td = static_cast<ThreadData*>(args);int cnt = 5, x = 0;while(cnt--) {printf("%s, tid: %p, pid: %d, x: %d, &x: %p\n", td->_threadName.c_str(), pthread_self(), getpid(), x, &x);x++;sleep(1);}delete td;return nullptr;
}

image-20240918195740301

可以看到,每一个线程都独享一个x,即独立的栈空间。注意这里是独立,并不是私有,其它线程想访问还是可以的,比如主线程想访问Thread-2的x值

int *p = nullptr;void* ThreadRoutine(void* args)
{ThreadData* td = static_cast<ThreadData*>(args);int cnt = 5, x = 0;while(cnt--) {printf("%s, tid: %p, pid: %d, x: %d, &x: %p\n", td->_threadName.c_str(), pthread_self(), getpid(), x, &x);x++;// 获取该线程的xif(td->_threadName == "Thread-2")   p = &x;sleep(1);}delete td;return nullptr;
}int main()
{// 创建多个线程vector<pthread_t> tids;for (size_t i = 0; i < NUM; i++) {pthread_t tid;ThreadData* td = new ThreadData("Thread-" + to_string(i+1));// 给子线程传递数据,堆空间是共享的pthread_create(&tid, nullptr, ThreadRoutine, td);tids.push_back(tid);}sleep(2);printf("Main Thread get x: %d, &x: %p\n", *p, p);// ...
}

image-20240918203622785

2.5.3 全局变量

默认情况下,全局变量被所有线程共享

// 共享资源
int gVal = 100;void* ThreadRoutine(void* args)
{ThreadData* td = static_cast<ThreadData*>(args);int cnt = 5;while(cnt--) {gVal++;printf("%s, tid: %p, pid: %d, gVal: %d, &gVal: %p\n", td->_threadName.c_str(), pthread_self(), getpid(), gVal, &gVal);sleep(1);}delete td;return nullptr;
}

image-20240918205249938

如果想要让每个进程独享该变量,可以在变量前加上__thread,让编译器把该变量放到tcb的线程局部存储单元中,见2.4。注意这里只能初始化内置类型,自定义类型是不可以的

__thread int gVal = 100;

image-20240918210206504

介绍一个__thread的用法:减少系统调用

__thread unsigned long int self = 0;
__thread int tPid = 0;
void* ThreadRoutine(void* args)
{ThreadData* td = static_cast<ThreadData*>(args);self = pthread_self();tPid = getpid();int cnt = 5;while(cnt--) {printf("%s, tid: %p, pid: %d\n", td->_threadName.c_str(), self, tPid);sleep(1);}delete td;return nullptr;
}

像上面这样,就减少了系统调用的次数。可以将这样的变量理解为线程级别的全局变量,不同线程之间变量互不干扰

2.6 分离线程

  • 默认情况下,新创建的线程是joinable的,线程退出后,需要对其进行pthread_join操作,否则无法释放资源,从而造成系统泄漏。
  • 如果不关心线程的返回值,join是一种负担,这个时候,我们可以告诉系统,当线程退出时,自动释放线程资源。
int pthread_detach(pthread_t thread);

可以是线程组内其他线程对目标线程进行分离,也可以是线程自己分离:

pthread_detach(pthread_self());

joinable和分离是冲突的,一个线程不能既是joinable又是分离的

__thread unsigned long int self = 0;
__thread int tPid = 0;
void* ThreadRoutine(void* args)
{ThreadData* td = static_cast<ThreadData*>(args);self = pthread_self();tPid = getpid();// 自己分离pthread_detach(self);int cnt = 5;while(cnt--) {printf("%s, tid: %p, pid: %d\n", td->_threadName.c_str(), self, tPid);sleep(1);}delete td;return nullptr;
}int main()
{// 创建多个线程vector<pthread_t> tids;for (size_t i = 0; i < NUM; i++) {pthread_t tid;ThreadData* td = new ThreadData("Thread-" + to_string(i+1));// 给子线程传递数据,堆空间是共享的pthread_create(&tid, nullptr, ThreadRoutine, td);tids.push_back(tid);}sleep(2);// 等待线程for(const auto& t : tids) {int ret = pthread_join(t, nullptr);printf("ret: %d, who: %p, why: %s\n", ret, t, strerror(ret));}return 0;
}

image-20240918213816179

相关文章:

Linux 线程控制

2. Linux 线程控制 首先&#xff0c;**内核中有没有很明确的线程的概念**&#xff0c;而有**轻量级进程的概念**。当我们想写多线程代码时&#xff0c;可以使用**POSIX线程库**&#xff0c;这是一个 处于应用层位置的库&#xff0c;几乎所有的Linux发行版都默认带这个库&#x…...

内网通3.4.3045广告码、积分码

内网通3.4.3045广告码、积分码 https://download.csdn.net/download/weixin_42120669/89772091...

MATLAB给一段数据加宽频噪声的方法(随机噪声+带通滤波器)

文章目录 引言方法概述完整代码:结果分析结论参考文献引言 在信号处理领域,添加噪声是模拟实际环境中信号传输时常见的操作。宽频噪声可以用于测试系统的鲁棒性和信号处理算法的有效性。本文将介绍如何使用 M A T L A B MATLAB MATLAB给一段数据添加宽频噪声,具体方法是结合…...

网安标委发布敏感个人信息识别指南

9月14日全国网络安全标准化技术委员会秘书处发布《网络安全标准实践指南——敏感个人信息识别指南》 敏感个人信息识别规则&#xff1a; 一旦遭到泄露或者非法使用&#xff0c;容易导致自然人的人格尊严受到侵害、自然人的人身安全受到危害、自然人财产安全受到危害。 注意&am…...

音视频入门基础:AAC专题(5)——FFmpeg源码中,判断某文件是否为AAC裸流文件的实现

音视频入门基础&#xff1a;AAC专题系列文章&#xff1a; 音视频入门基础&#xff1a;AAC专题&#xff08;1&#xff09;——AAC官方文档下载 音视频入门基础&#xff1a;AAC专题&#xff08;2&#xff09;——使用FFmpeg命令生成AAC裸流文件 音视频入门基础&#xff1a;AAC…...

几何 | 数学专项

日期内容2024.9.19创建 { d > 0 , 递增数列 d < 0 , 递减数列 d 0 &#xff0c;常数列 \begin{cases} d>0,递增数列\\ d<0,递减数列\\ d0&#xff0c;常数列 \end{cases} ⎩ ⎨ ⎧​d>0,递增数列d<0,递减数列d0&#xff0c;常数列​ 【2010.13】 【1.历年真…...

学习CubeIDE——定时器开发

在b站上学习洋桃电子关于HAL库开发&#xff0c;发现使用CubeIDE是真的简单又方便。 实验现象&#xff1a;使用定时器来产生中断&#xff0c;中断程序是LED灯翻转 在我看来&#xff0c;定时器&#xff0c;是一个从0开始增1&#xff08;常规&#xff09;&#xff0c;增加到一定…...

【Elasticsearch】-图片向量化存储

需要结合深度学习模型 1、pom依赖 注意结尾的webp-imageio 包&#xff0c;用于解决ImageIO.read读取部分图片返回为null的问题 <dependency><groupId>org.openpnp</groupId><artifactId>opencv</artifactId><version>4.7.0-0</versio…...

二叉树(一)高度与深度

高度&#xff1a;从最底层往上数&#xff08;后序遍历&#xff0c;左右根&#xff09;&#xff0c;更简单&#xff08;递归&#xff09; 深度&#xff1a;从上往下数直到有叶子&#xff08;前序遍历&#xff0c;根左右&#xff09;&#xff0c;较复杂 高度是最大深度 一、求…...

梧桐数据库(WuTongDB):MySQL 优化器简介

MySQL 优化器是数据库管理系统中的一个重要组件&#xff0c;用于生成并选择最优的查询执行计划&#xff0c;以提高 SQL 查询的执行效率。它采用了基于代价的优化方法&#xff08;Cost-Based Optimizer, CBO&#xff09;&#xff0c;通过评估不同查询执行方案的代价&#xff0c;…...

交通运输部力推高速公路监测,做好结构安全预警,保护人民安全

在快速发展的交通网络中&#xff0c;高速公路作为经济命脉与生命通道&#xff0c;其结构安全直接关系到每一位行路者的生命财产安全。为此&#xff0c;广东省交通运输厅正式发布《关于积极申报高速公路监测预警应用示范揭榜的通知》&#xff0c;旨在通过技术创新与应用示范&…...

基于PHP+MySQL组合开发的在线客服源码系统 聊天记录实时保存 带完整的安装代码包以及搭建部署教程

系统概述 随着互联网技术的飞速发展&#xff0c;企业与客户之间的沟通方式日益多样化&#xff0c;在线客服系统作为连接企业与客户的桥梁&#xff0c;其重要性不言而喻。然而&#xff0c;市场上现有的在线客服系统往往存在成本高、定制性差、维护复杂等问题。针对这些痛点&…...

NEXT.js 创建postgres数据库-关联github项目-连接数据库-在项目初始化数据库的数据

github创建项目仓库创建Vercel账号选择hobby连接github仓库install - deploy创建postgres数据库&#xff08;等待deploy完成&#xff09; Continue to DashboardStorage&#xff08;头部nav哪里&#xff09;create Postgresconnect连接完后&#xff0c;切换到.env.local&#x…...

Matlab如何配置小波工具(Wavelet Toolbox)

1、发现问题 因为实验要使用小波工具函数&#xff0c;运行时报错如下&#xff1a; 查看对应文件夹发现没有小波工具&#xff08;也可在控制台输入ver&#xff09;&#xff0c;检查是否有该工具&#xff0c;输入后回车返回如下&#xff1a; 2、下载工具包 没有这个工具就要去下…...

FTP、SFTP安装,整合Springboot教程

文章目录 前言一、FTP、SFTP是什么&#xff1f;1.FTP2.SFTP 二、安装FTP1.安装vsftp服务2.启动服务并设置开机自启动3.开放防火墙和SELinux4.创建用户和FTP目录4.修改vsftpd.conf文件5.启动FTP服务6.问题 二、安装SFTP1、 创建用户2、配置ssh和权限3、建立目录并赋予权限4、启动…...

24年蓝桥杯及攻防世界赛题-MISC-3

21 reverseMe 复制图片&#xff0c;在线ocr识别&#xff0c;https://ocr.wdku.net/&#xff0c;都不费眼睛。 22 misc_pic_again ┌──(holyeyes㉿kali2023)-[~/Misc/tool-misc/zsteg] └─$ zsteg misc_pic_again.png imagedata … text: “$$KaTeX parse error: Undefined…...

阿里云容器服务Kubernetes部署新服务

这里部署的是前端项目 1.登录控制台-选择集群 2.选择无状态-命名空间-使用镜像创建 3.填写相关信息 应用基本信息&#xff1a; 容器配置&#xff1a; 高级配置&#xff1a; 创建成功后就可以通过30006端口访问项目了...

记录生产环境,通过域名访问的图片展示不全,通过ip+端口的方式访问图片是完整的

原因&#xff1a;部署nginx的服务器硬盘满了 排查发现nginx日志文件占用了大量硬盘 解决方案&#xff1a; 删除该文件&#xff0c;重启nginx服务&#xff0c;问题解决。...

网络安全实训八(y0usef靶机渗透实例)

1 信息收集 1.1 扫描靶机IP 1.2 收集靶机的端口开放情况 1.3 探测靶机网站的目录 1.4 发现可疑网站 1.5 打开可疑网站 2 渗透 2.1 使用BP获取请求 2.2 使用工具403bypasser.py探测可疑网页 2.3 显示可以添加头信息X-Forwarded-For:localhost来访问 2.4 添加之后转发&#xff…...

QT信号槽原理是什么,如何去使用它?

QT的信号槽&#xff08;Signals and Slots&#xff09;机制是QT框架的核心特性之一&#xff0c;它提供了一种对象间通信的方式&#xff0c;使得QT的部件可以在不知道彼此详细实现的情况下相互通信。这种机制在图形用户界面编程中尤为重要&#xff0c;因为它有助于降低对象间的耦…...

【Axure高保真原型】引导弹窗

今天和大家中分享引导弹窗的原型模板&#xff0c;载入页面后&#xff0c;会显示引导弹窗&#xff0c;适用于引导用户使用页面&#xff0c;点击完成后&#xff0c;会显示下一个引导弹窗&#xff0c;直至最后一个引导弹窗完成后进入首页。具体效果可以点击下方视频观看或打开下方…...

51c自动驾驶~合集58

我自己的原文哦~ https://blog.51cto.com/whaosoft/13967107 #CCA-Attention 全局池化局部保留&#xff0c;CCA-Attention为LLM长文本建模带来突破性进展 琶洲实验室、华南理工大学联合推出关键上下文感知注意力机制&#xff08;CCA-Attention&#xff09;&#xff0c;…...

突破不可导策略的训练难题:零阶优化与强化学习的深度嵌合

强化学习&#xff08;Reinforcement Learning, RL&#xff09;是工业领域智能控制的重要方法。它的基本原理是将最优控制问题建模为马尔可夫决策过程&#xff0c;然后使用强化学习的Actor-Critic机制&#xff08;中文译作“知行互动”机制&#xff09;&#xff0c;逐步迭代求解…...

java 实现excel文件转pdf | 无水印 | 无限制

文章目录 目录 文章目录 前言 1.项目远程仓库配置 2.pom文件引入相关依赖 3.代码破解 二、Excel转PDF 1.代码实现 2.Aspose.License.xml 授权文件 总结 前言 java处理excel转pdf一直没找到什么好用的免费jar包工具,自己手写的难度,恐怕高级程序员花费一年的事件,也…...

TRS收益互换:跨境资本流动的金融创新工具与系统化解决方案

一、TRS收益互换的本质与业务逻辑 &#xff08;一&#xff09;概念解析 TRS&#xff08;Total Return Swap&#xff09;收益互换是一种金融衍生工具&#xff0c;指交易双方约定在未来一定期限内&#xff0c;基于特定资产或指数的表现进行现金流交换的协议。其核心特征包括&am…...

管理学院权限管理系统开发总结

文章目录 &#x1f393; 管理学院权限管理系统开发总结 - 现代化Web应用实践之路&#x1f4dd; 项目概述&#x1f3d7;️ 技术架构设计后端技术栈前端技术栈 &#x1f4a1; 核心功能特性1. 用户管理模块2. 权限管理系统3. 统计报表功能4. 用户体验优化 &#x1f5c4;️ 数据库设…...

Selenium常用函数介绍

目录 一&#xff0c;元素定位 1.1 cssSeector 1.2 xpath 二&#xff0c;操作测试对象 三&#xff0c;窗口 3.1 案例 3.2 窗口切换 3.3 窗口大小 3.4 屏幕截图 3.5 关闭窗口 四&#xff0c;弹窗 五&#xff0c;等待 六&#xff0c;导航 七&#xff0c;文件上传 …...

libfmt: 现代C++的格式化工具库介绍与酷炫功能

libfmt: 现代C的格式化工具库介绍与酷炫功能 libfmt 是一个开源的C格式化库&#xff0c;提供了高效、安全的文本格式化功能&#xff0c;是C20中引入的std::format的基础实现。它比传统的printf和iostream更安全、更灵活、性能更好。 基本介绍 主要特点 类型安全&#xff1a…...

Linux部署私有文件管理系统MinIO

最近需要用到一个文件管理服务&#xff0c;但是又不想花钱&#xff0c;所以就想着自己搭建一个&#xff0c;刚好我们用的一个开源框架已经集成了MinIO&#xff0c;所以就选了这个 我这边对文件服务性能要求不是太高&#xff0c;单机版就可以 安装非常简单&#xff0c;几个命令就…...

Leetcode33( 搜索旋转排序数组)

题目表述 整数数组 nums 按升序排列&#xff0c;数组中的值 互不相同 。 在传递给函数之前&#xff0c;nums 在预先未知的某个下标 k&#xff08;0 < k < nums.length&#xff09;上进行了 旋转&#xff0c;使数组变为 [nums[k], nums[k1], …, nums[n-1], nums[0], nu…...