【Linux】线程控制
目录
一、原生线程库:
二、线程控制:
1、线程创建:
2、线程等待:
自定义类型的接收对象:
编辑
3、线程终止:
pthread_exit:
pthread_cancel:
4、线程ID:
线程库的底层原理
三、线程栈:
创建多线程:
栈区变量:
局部性存储:
拿到别的线程执行流数据:
四、线程分离:
一、原生线程库:
在编译代码的时候,需要新加选项-lpthread(在部分环境下是不用加的,如果出现链接错误就加上,没有就不用加),因为要使用pthread的原生线程库
什么是原生线程库:
我们知道,在Linux中是没有真正意义上线程的概念的,只有轻量级进程的概念,
所以Linux工程师在系统层和用户层之间封装了原生线程库方便用户进行使用
在Linux中,原生线程库也是pthread库,在使用的时候包含即可
二、线程控制:
1、线程创建:
pthread_t:这其实就是无符号长整型:unsigned long int
接下来看看参数:
参数1:pthread_t* thread
这是一个输出型参数,表示线程的tid
参数2:设置线程属性,一般直接nullptr即可
参数3:这是一个函数指针,这是一个很重要的参数,发现这个函数返回值是void*,参数也是void*,这是一个回调函数,当线程启动的时候会自动回调此函数
参数4:这是void*类型的,就是线程启动时,传递给上述函数的形参
返回值:成功返回0,失败返回一个错误码
void* newthread(void* argc)
{while(1){cout << "我是新线程,我正在运行,pid : " << getpid() <<endl;sleep(1);}return nullptr;
}int main()
{pthread_t tid;pthread_create(&tid,nullptr,newthread,nullptr); sleep(1);while(1){cout << "我是主线程,我正在运行,pid : " << getpid() << endl;sleep(1);}return 0;
}
如上,写一段线程小代码,运行后:
如上,可以发现:这两个线程的pid也就是进程id是一样的,也就证明了线程是进程内部的分支
可以通过指令ps -aL来观察线程的属性
如上,可以发现每个线程有其独特的LWP,这就类似于进程的pid,CPU在调度线程的时候是通过LWP来进行调度的
如上,仔细观察还会发现有一个线程的LWP和PID是相等的,此时这个线程就叫做主线程
多线程错误:
同时对于这些线程如果一个线程发生了错误,那么所有共享一个进程的线程都会退出
如下,我们修改代码为存在除零错误
那么当主线程出现错误的时候,尽管别的进程没有错误,但是仍然会退出
同样的,向一个线程发送终止信号,其他线程也会被终止:
全局变量:
同样的,线程也共享者全局变量区:
如上,当在主线程中进行num的++,在新线程中打印出num的值也会++,所以线程间通信是比较方便的
2、线程等待:
线程和进程是差不多的,父进程需要等待子进程,防止子进程僵尸
主线程也需要等待新线程,在原生线程库中,封装了一个接口:pthread_join
参数1:thread:这是待等待线程的ID
参数2:retval:输出型参数,用于获取新线程的返回值,
返回值:成功返回0,失败返回错误码
void* newthread(void* arr)
{const char* name = (const char*)arr;int cnt = 5;while(1){cout << name << "new thread : pid : " << getpid() << endl;sleep(1);cnt--;if(cnt == 0) break;}return (void*)11;
}int main()
{pthread_t tid;pthread_create(&tid,nullptr,newthread,(void*)"Thread 1 ");sleep(7);void* retval;pthread_join(tid,&retval);cout << "ret : " << (long)retval << endl;cout << "main quit success ..." << endl;return 0;
}
自定义类型的接收对象:
#include <iostream>
#include <unistd.h>
#include <pthread.h>
#include <string>using namespace std;class Request
{
public:Request(int start,int end,const string& threadname):_start(start),_end(end),_threadname(threadname){}public:int _start;int _end;string _threadname;
};class Response
{
public:Response(int retval,int retcode):_retval(retval),_retcode(retcode){}
public:int _retval;int _retcode;
};void* sumcount(void* argc)
{Request* rq = static_cast<Request*>(argc);Response* rsp = new Response(0,0);for(int i = rq->_start;i <= rq->_end;i++){rsp->_retval += i;}delete rq;return rsp;
}int main()
{pthread_t tid;Request* rq = new Request(1,100,"mythread");pthread_create(&tid,nullptr,sumcount,rq);void* ret;pthread_join(tid,&ret);Response *rsp = static_cast<Response*>(ret);cout << "rsp->result: " << rsp->_retval << ", exitcode: " << rsp->_retcode << endl;delete rsp;return 0;
}
如上,这就是计算两个数之间的和
上述有两个地方要注意:
1、在主函数中pthread_create中的第三个参数不用强转,而之前的字符串需要强转,这是因为之前的字符串默认是const char* 的,强制转换成void*这样可以避免权限放大
而这里自定义类型不用强转,尽管类型不同,这是因为void*类型的指针可以接受任意类型的指针
2、里面运用到了static_cast,这是C++引入的,比()强转更安全,因为:会在编译时进行类型检查,如果转换不合法(例如无关类型之间的转换),编译器会报错,而()强制类型转换,基本上都是可以转换的
3、线程终止:
pthread_exit:
将一个线程终止有许多方法,比如发信号,exit等等
如果在代码中调用exit,这是将进程退出,那么就会导致所有线程都会退出,这显然是不对的,
那么在原生线程库中就封装了接口:pthread_exit
参数retval:这就是类似于exit中的参数,退出码,返回值
这里的retval和pthread_join里的retval形参名字怎么是一样的呢?
首先我们要知道,主线程和创建出来的新线程是在不同的栈帧中的,那么想要远程修改就需要赋值传地址
如下,将新线程中的return可以改为pthread_exit(),这样优雅多了
pthread_cancel:
参数:指的是取消哪一个线程,并且退出的线程,退出码为-1
返回值:成功返回0,失败返回错误码
void* newthread(void* arr)
{const char* name = (const char*)arr;int cnt = 5;while(1){cout << name << "new thread : pid : " << getpid() << endl;sleep(1);cnt--;if(cnt == 0) break;}pthread_exit((void*)11);
}int main()
{pthread_t tid;pthread_create(&tid,nullptr,newthread,(void*)"Thread 1 ");sleep(2);pthread_cancel(tid);void* retval;pthread_join(tid,&retval);cout << "retval : " << (long)retval << endl;cout << "main quit success ..." << endl;return 0;
}
如上,我们不应该取消成功后返回0吗,这里为什么是-1呢?
注意:这里的-1是指的是取消线程的退出码,而不是pthread_cancel的退出码
4、线程ID:
如上,这个接口是获取当前线程的tid
void* newthread(void* arr)
{const char* name = (const char*)arr;int cnt = 5;while(1){cout << name << "new thread : pid : " << getpid() << " tid : " << pthread_self() << endl;sleep(1);cnt--;if(cnt == 0) break;}pthread_exit((void*)11);
}int main()
{pthread_t tid;pthread_create(&tid,nullptr,newthread,(void*)"Thread 1 ");cout <<"主线程的tid : " << pthread_self() << endl; cout <<"创建的新线程的tid : " << tid << endl;void* retval;pthread_join(tid,&retval);cout << "retval : " << (long)retval << endl;cout << "main quit success ..." << endl;return 0;
}
如上,发现这个tid和新线程中的pthread_self是一样的
但是这个tid和我们ps -aL中的LWP是不一样的,本来二者也都是不一样的,因为LWP是操作系统层面的概念,操作系统自己知道即可,我们用户只关心tid
这个线程ID到底是什么东西呢?和tid是不是同一个东西呢?
线程库的底层原理
我们先理解理解如下的图
1、在Linux中是没有线程的概念的,只有轻量级进程的概念
2、所以我们在Linux中的线程的概念是库给我们维护的------所以线程库要维护线程的概念(这是用户关心的,比如说线程的tid,线程的时间片有没有到等)又因为这个线程在底层是维护的轻量级进程的执行流,是不用维护线程的执行流的
3、所以这个原生线程库要不要加载到内存中?------当然要,不然怎么用------那么加载到哪里?----- 加载到内存中,通过页表把pthread库映射到我们的共享区
4、我们每创建一个线程,都要在线程库中创建一个线程控制块供用户维护,
这个线程控制块对上提供信息如独立栈在哪?线程ID是什么,回调函数在哪等等,
更重要的是对下提供该线程的LWP指向底层的哪个轻量级进程的执行流
如上,这样就能够很好的管理我们的线程了,当想要拿到对应的线程ID时,直接访问这个描述好的结构体里直接拿就行了
而tid是一个地址:是每一个线程库级别的tcb起始地址,
为什么要以每个tcb的起始地址作为tid呢?
1、它里面存放的是地址
2、它是在用户空间的
3、是虚拟地址,可以直接访问的
三、线程栈:
每一个线程在其运行时有独立的栈结构
因为每一个线程有其独特的调用链
也就是必须有其调用链对应的栈帧结构
这个栈结构会保存任何一个执行流在其运行中的所有临时变量比如压栈的形参,返回值等等
其中主线程直接用地址空间结构中的栈结构即可
其他被主线成pthread_create创建的都是轻量级进程,这些线程用的不是地址空间的栈,而是用的是线程库中独立的栈结构
其做法是首先在pthread.so里面创建创建对应的描述这个线程的线程控制块tcb,这个tcb的起始地址就作为这个线程的tid,里面有一块默认大小的空间,这就是线程栈
当在内核中创建执行流,就是在库里面调用clone()接口,然后把线程执行的方法,已经刚刚创建的线程栈作为第一第二参数传递给clone(),这个函数不是我们使用的,Linux程序员已经将这个函数封装起来了,我们只需要使用这个函数封装后的接口如pthread_create,pthread_join等等
接下来用代码来理解理解:
创建多线程:
首先,我们用一个vector数组来存储多线程的tid,然后定义一个类对象和10进制转16进制的函数,接着就可以实现多线程的实现了
#include <iostream>
#include <string>
#include <vector>
#include <pthread.h>
#include <unistd.h>using namespace std;#define N 3class threadDate
{
public:threadDate(int number){_threadname = "thread-" + to_string(number); // thread-0}public:string _threadname;
};string toHex(pthread_t tid)
{char buffer[128];snprintf(buffer,sizeof(buffer),"0x%x",tid);return buffer;
}void* threadrun(void* argc)
{threadDate* td = static_cast<threadDate*>(argc);int i = 0;while(i < 10){printf("pid : %d,tid : %s,threadname : %s\n",getpid(),toHex(pthread_self()).c_str(),td->_threadname.c_str());sleep(1);i++;}delete td;
}int main()
{vector<pthread_t> tids;//创建多线程for(int i = 0; i < N; i++){pthread_t tid;threadDate* td = new threadDate(i);pthread_create(&tid,nullptr,threadrun,td);tids.push_back(tid);sleep(1);}for(int i = 0;i < tids.size();i++){pthread_join(tids[i],nullptr);}return 0;
}
栈区变量:
我们在线程执行代码块中加上一个变量并对其进行打印:
如上,我们可以发现:他们各自都是独立的,和全局存储互相影响是不一样的,他们是互不影响的,也就是说当我们执行对应的代码的时候,各个线程就会在自己的线程栈中开辟对应的栈帧
局部性存储:
我们上述知道线程会共享全局区的数据
如上,但是如果想要他们访问独立起来,怎么办呢?----- 当然是给val这个变量加上__thread进行修饰
如上,他们的值就是各自++了,并且其地址也是不一样的了
拿到别的线程执行流数据:
这个是通过全局变量拿到的
所以,在线程中并不存在真正的独立,我们前面所讲的独立栈本质还是在进程地址空间的,所以只要是想拿还是能够被访问的 ---- 所以,线程与线程之间是没有秘密的,线程栈上的数据也是能够被其他线程访问的
四、线程分离:
在默认情况下,新线程最后是被主线程所等待pthread_join的,这是为了防止新线程没有被释放而造成内存泄漏,但是还有一种方法,能够做到当线程退出的时候,主线程不需要等待,而是在线程退出的时候自动释放资源
接口:
参数:所分离线程的tid
返回值:成功返回0,失败返回错误码
主线程分离:
自己分离:
如上,我们发现无论是自我分离,还是主线程分离,都只执行了几行代码,这是因为main函数已经执行完了,然后return退出了,这个时候进程就是退出了,进程的资源也就没有了,然而线程又是属于进程的,所以自然而然线程就会退出
线程分离是一种属性状态,当创建线程的时候是不分离状态,当在描述线程的结构体tcb中这个状态为0或者1表示这个线程是不分离状态还是分离状态,但是本质上还是在共享同一份资源,只是分离后的线程退出和主线程没有关系了
相关文章:
【Linux】线程控制
目录 一、原生线程库: 二、线程控制: 1、线程创建: 2、线程等待: 自定义类型的接收对象: 编辑 3、线程终止: pthread_exit: pthread_cancel: 4、线程ID: 线程库的底层原…...
pyqt联合designer的运用和设置
PyQt Designer 简介 PyQt Designer 是一个用于创建和设计 PyQt 应用程序用户界面的可视化工具。它允许用户通过拖放方式添加和排列各种控件,如按钮、文本框、滑块等,并设置它们的属性和样式,从而快速构建出美观且功能完整的 UI 界面。 Windows版本:【免费】安装包别管啊啊…...
spring boot3.4.3+MybatisPlus3.5.5+swagger-ui2.7.0
使用 MyBatis-Plus 操作 books 表。我们将实现以下功能: 创建实体类 Book。 创建 Mapper 接口 BookMapper。 创建 Service 层 BookService 和 BookServiceImpl。 创建 Controller 层 BookController。 配置 MyBatis-Plus 和数据库连接。 1. 项目结构 src ├─…...
利用微软的 HTML 应用程序宿主程序的攻击
mshta.exe 是微软的 HTML 应用程序宿主程序(Microsoft HTML Application Host),属于 Windows 系统组件。它的核心功能是运行 .hta(HTML Application)文件,允许通过 HTML、JavaScript、VBScript 等技术创建交…...
【深度学习】读写文件
读写文件 到目前为止,我们讨论了如何处理数据,以及如何构建、训练和测试深度学习模型。 然而,有时我们希望保存训练的模型,以备将来在各种环境中使用(比如在部署中进行预测)。 此外,当运行一个…...
Bert的使用
一、Data.py # data负责产生两个dataloader from torch.utils.data import DataLoader, Dataset from sklearn.model_selection import train_test_split #给X,Y 和分割比例, 分割出来一个训练集和验证机的X, Y import torchdef read_file(path):data []label …...
Unity使用UGUI制作无限滑动列表
原理参照上一篇使用NGUI的制作无限滑动列表的文章 Unity 使用NGUI制作无限滑动列表_unity 滑动列表很多物体-CSDN博客 准备工作: 新建一个空物体命名为LoopList,并调整其大小, 并增加Scroll Rect组件(用于滑动)、Re…...
ThinkPHP6用户登录系统的全过程
ThinkPHP6用户登录系统的全过程涉及请求处理、数据传输、路由分发、控制器逻辑、模型验证及中间件协作等多个模块的交互。详细的过程解析如下: 1. 前端请求与路由分发 前端发起请求:用户在前端页面(如Vue组件或HTML表单)输入用户…...
C++全栈聊天项目(2) 单例模式封装Http管理者
完善注册类界面 先在注册类构造函数里添加lineEdit的模式为密码模式 ui->lineEdit_Passwd->setEchoMode(QLineEdit::Password); ui->lineEdit_Confirm->setEchoMode(QLineEdit::Password);我们在注册界面的ui里添加一个widget,widget内部包含一个tip居…...
【鸿蒙开发】OpenHarmony调测工具hdc使用教程(设备开发者)
00. 目录 文章目录 00. 目录01. OpenHarmony概述02. hdc简介03. hdc获取04. option相关的命令05. 查询设备列表的命令06. 服务进程相关命令07. 网络相关的命令08. 文件相关的命令09. 应用相关的命令10. 调试相关的命令11. 常见问题12. 附录 01. OpenHarmony概述 OpenHarmony是…...
ORACLE EBS数据库RELINK方式搭建克隆环境
ORACLE EBS系统的数据库,一般都安装了很多特定功能的小补丁来解决特定的BUG;因此对于已经安装好的系统,想要克隆一套测试环境、搭建一个新的备机做测试等,如果按照生产环境标准,则需要安装大量补丁,带来很大…...
MySQL regexp 命令
REGEXP命令是一种用于进行正则表达式匹配的运算符,允许在查询中使用正则表达式来匹配字符串模式1。 基本语法 基本的语法结构如下: SELECT * FROM table_name WHERE column_name REGEXP pattern; 这里,pattern是你要匹配的正则表达式模…...
前端实习到工作的经历
看了很多人的程序员生涯之路,我突然意识到我也该记录一些东西,因此有感而发。 我是一个24届毕业生,大三下就开始找前端实习,当时学校不让走,我们都是先面着然后准备放假就去。当时周围小伙伴都找好了,考完…...
Vue3——Fragment
文章目录 一、Fragment的核心意义1. 解决Vue2的单根限制问题2. 减少不必要的 DOM 嵌套3. 语义化和结构化 二、Fragment 的实现原理三、Fragment 使用方式1. 基本用法2. 结合条件渲染3. 动态组件 四、实际应用场景1. 列表/表格组件2. 布局组件3. 语义化标签 五、注意事项1. 属性…...
Linux_16进程地址空间
CPU内的寄存器只有一套,但是CPU内寄存器的数据可能会有多份! 一、程序地址空间 下面这个图对应的是内存吗?(实际上是虚拟的进程地址空间) 32位机器内存最大为多少? 32位操作系统的地址总线为32位&#x…...
职坐标机器学习编程实战:调试优化与自动化测试精要
内容概要 在机器学习编程实践中,代码调试优化与自动化测试工具的应用是构建高可靠性系统的核心环节。本书聚焦从数据预处理到模型部署的全流程,通过特征工程优化、训练过程监控及持续集成方案的设计,系统化解决算法工程化中的典型问题。在特…...
git文件过大导致gitea仓库镜像推送失败问题解决(push failed: context deadline exceeded)
问题描述: 今天发现gitea仓库推送到某个镜像仓库的操作几个月前已经报错终止推送了,报错如下: 首先翻译报错提示可知是因为git仓库大小超过1G限制。检查本地.git文件,发现.git文件大小已达到1.13G。确定是.git文件过大导致&…...
llvm数据流分析
llvm数据流分析 1.数据流分析2.LLVM实现2.1.常量传播2.2.活跃性分析 相关参考文档:DataFlowAnalysisIntro、ustc编译原理课程、南大程序分析课程1、南大程序分析课程2。 1.数据流分析 数据流分析在编译优化等程序分析任务上都有重要应用。通常数据流分析可被抽象为…...
Vite为什么选用Rollup打包?
Vite 在生产阶段使用 Rollup 打包,但这不是唯一选择。它的设计背后有明确的权衡和考量,同时开发者也可以选择其他替代方案。 一、为什么 Vite 默认使用 Rollup? 1. Rollup 的核心优势 • Tree-shaking:Rollup 的静态分析能力极强&…...
Docker 入门与实战指南
Docker 入门与实战指南 一、Docker 简介 Docker 是一个开源的容器化平台,允许开发者将应用程序及其依赖打包成一个可移植的容器。容器可以在任何安装了 Docker 的环境中运行,确保应用的一致性和可移植性。 1.1 为什么使用 Docker? 环境一…...
C# 常用数据类型
C# 数据类型分为 值类型、引用类型 和 特殊类型,以下是详细分类及对应范围/说明: 一、值类型(Value Types) 值类型直接存储数据,分配在栈内存中,默认不可为 null。 简单类型 整数类型…...
深入解读 JavaScript 中 `this` 的指向机制:覆盖所有场景与底层原理
this 是 JavaScript 中最容易引发困惑的核心概念之一,它的指向在不同场景下呈现截然不同的行为。本文将系统性地解析 this 的所有使用场景,结合代码示例和底层原理,帮助你彻底掌握其运行机制。 一、全局环境下的 this 1. 浏览器环境 在浏览器…...
无人机全景应用解析与技术演进趋势
无人机全景应用解析与技术演进趋势 ——从立体安防到万物互联的空中革命 一、现有应用场景全景解析 (一)公共安全领域 1. 立体安防体系 空中哨兵:搭载 77 GHz 77\text{GHz} 77GHz毫米波雷达(探测距离 5 km 5\text{km} 5km&…...
手写简易Tomcat核心实现:深入理解Servlet容器原理
目录 一、Tomcat概况 1. tomcat全局图 2.项目结构概览 二、实现步骤详解 2.1 基础工具包(com.qcby.util) 2.1.1 ResponseUtil:HTTP响应生成工具 2.1.2 SearchClassUtil:类扫描工具 2.1.3 WebServlet:自定义注解…...
【音视频】ffmpeg命令提取像素格式
1、提取YUV数据 提取yuv数据,并保持分辨率与原视频一致 使用-pix_fmt或-pixel_format指定yuv格式提取数据,并保持原来的分辨率 ffmpeg -i music.mp4 -t "01:00" -pixel_format yuv420p music.yuv提取成功后,可以使用ffplay指定y…...
深度剖析Redis:双写一致性问题及解决方案全景解析
在高并发场景下,缓存与数据库的双写一致性是每个开发者必须直面的核心挑战。本文通过5大解决方案,带你彻底攻克这一技术难关! 一、问题全景图:当缓存遇到数据库 1.1 典型问题场景 // 典型问题代码示例 public void updateProduc…...
Redis----大key、热key解决方案、脑裂问题
文章中相关知识点在往期已经更新过了,如果有友友不理解可翻看往期内容 出现脑裂问题怎么保证集群还是高可用的 什么是脑裂问题 脑裂说的就是当我们的主节点没有挂,但是因为网络延迟较大,然后和主节点相连的哨兵通信较差,之后主…...
Android 调用c++报错 exception of type std::bad_alloc: std::bad_alloc
一、报错信息 terminating with uncaught exception of type std::bad_alloc: std::bad_alloc 查了那部分报错c++代码 szGridSize因为文件太大,初始化溢出了 pEGM->pData = new float[szGridSize]; 解决办法 直接抛出异常,文件太大就失败吧 最后还增加一个日志输出,给…...
【从零开始学习计算机科学】操作系统(五)处理器调度
【从零开始学习计算机科学】操作系统(五)处理器调度 处理器调度一些简单的短程调度算法的思路先来先服务(First-Come-First-Served,FCFS)优先级调度及其变种最短作业优先调度算法(SJF)--非抢占式最短作业优先调度算法(SJF)--抢占式最高响应比优先调度算法轮转调度算法…...
LeetCode1871 跳跃游戏VII
LeetCode 跳跃游戏 IV:二进制字符串的跳跃问题 题目描述 给定一个下标从 0 开始的二进制字符串 s 和两个整数 minJump 和 maxJump。初始时,你位于下标 0 处(保证该位置为 0)。你需要判断是否能到达字符串的最后一个位置…...
ResNet50深度解析:原理、结构与PyTorch实现
ResNet50深度解析:原理、结构与PyTorch实现 1. 引言 ResNet(残差网络)是深度学习领域的一项重大突破,它巧妙解决了深层神经网络训练中的梯度消失/爆炸问题,使得构建和训练更深的网络成为可能。作为计算机视觉领域的里…...
MATLAB 控制系统设计与仿真 - 24
PID 控制器分析- 控制器的形式 连续控制器的结构: 为滤波时间常数,这类PID控制器在MATLAB系统控制工具箱称为并联PID控制器,可由MATLAB提供的pid函数直接输入,格式为: 其他类型的控制器也可以由该函数直接输入&#x…...
数字IC后端设计实现教程 |Innovus ICC2 Routing Pin Access Setting设置方法
默认情况下routing 引擎可以在标准单元可以打孔的任何地方(via region)打孔,甚至工具还会先拉出一块metal,然后再打孔过渡到高层。 随之工艺节点越做越小,标准单元内部的结构也越来越复杂。此时如果还沿用传统工艺的走…...
mysql经典试题共34题
1、准备数据 -- drop drop table if exists dept; drop table if exists emp; drop table if exists salgrade;-- CREATE CREATE TABLE dept (deptno int NOT NULL COMMENT 部门编号,dname varchar(14) CHARACTER SET utf8mb4 COLLATE utf8mb4_0900_ai_ci DEFAULT NULL COMM…...
网络编程-----服务器(多路复用IO 和 TCP并发模型)
一、单循环服务器模型 1. 核心特征 while(1){newfd accept();recv();close(newfd);}2. 典型应用场景 HTTP短连接服务(早期Apache)CGI快速处理简单测试服务器 3. 综合代码 #include <stdio.h> #include <sys/types.h> /* See NO…...
GitHub 项目版本管理与 Release 发布流程记录
GitHub 项目版本管理与 Release 发布流程记录 1. 项目环境设置 1.1 打开 VS Code 并进入项目目录 E:\adb\Do>code .1.2 配置 Git 用户信息 E:\adb\Do>git config --global user.name "n" E:\adb\Do>git config --global user.email "**gmail.com&q…...
GStreamer —— 2.15、Windows下Qt加载GStreamer库后运行 - “播放教程 1:Playbin 使用“(附:完整源码)
运行效果 介绍 我们已经使用了这个元素,它能够构建一个完整的播放管道,而无需做太多工作。 本教程介绍如何进一步自定义,以防其默认值不适合我们的特定需求。将学习: • 如何确定文件包含多少个流,以及如何切换 其中。…...
Python+DeepSeek:开启AI编程新次元——从自动化到智能创造的实战指南
文章核心价值 技术热点:结合全球最流行的编程语言与国产顶尖AI模型实用场景:覆盖代码开发/数据分析/办公自动化等高频需求流量密码:揭秘大模型在编程中的创造性应用目录结构 环境搭建:5分钟快速接入DeepSeek场景一:AI辅助代码开发(智能补全+调试)场景二:数据分析超级助…...
使用OpenCV和MediaPipe库——驼背检测(姿态监控)
目录 驼背检测的运用 1. 驾驶姿态与疲劳关联分析 2. 行业应用案例 1. 教育场景痛点分析 2. 智能教室系统架构 代码实现思路 1. 初始化与配置 2. MediaPipe和摄像头设置 3. 主循环 4. 资源释放 RGB与BGR的区别 一、本质区别 二、OpenCV的特殊性 内存结构示意图&…...
maven的项目构建
常用构建命令 命令说明mvn clean清理编译结果(删掉target目录)mvn compile编译核心代码,生成target目录mvn test-compile编译测试代码,生成target目录mvn test执行测试方法mvn package打包,生成jar或war文件mvn insta…...
光电感知赋能智能未来 灵途科技护航新质生产力发展
2024年《政府工作报告》将大力推进现代化产业体系建设,加快发展新质生产力作为首要工作任务。这是“新质生产力”首次出现在《政府工作报告》中。 发展新质生产力具体包括 新兴产业 :推动商业航天、低空经济等新兴产业实现安全健康发展。 未来产业 &a…...
文件上传靶场(10--20)
目录 实验环境: 具体内容实现: 第十关(双写绕过): 第十一关:(%00截断,此漏洞在5.2版本中) 正确用法 错误用法 思路: 操作过程: 第十二关…...
deepseek在pycharm中的配置和简单应用
对于最常用的调试python脚本开发环境pycharm,如何接入deepseek是我们窥探ai代码编写的第一步,熟悉起来总没坏处。 1、官网安装pycharm社区版(免费),如果需要安装专业版,需要另外找破解码。 2、安装Ollama…...
Linux 生成静态库
文章目录 前提小知识生成和使用.a库操作步骤 在应用程序中,有一些公共的代码需要反复使用的,可以把这些代码制作成“库文件”;在链接的步骤中,可以让链接器在“库文件”提取到我们需要使用到的代码,复制到生成的可执行…...
yolo-TensorRT相关代码逐步详解-pt转engine
基于TensorRT 的推论运行速度会比仅使用CPU 快40倍,提供精度INT8 和FP16 优化,支援TensorFlow、Caffe、Mxnet、Pytorch 等深度学习框架,其中Mxnet、Pytorch 需先转换为ONNX 格式。 TensorRT的构建流程大致分为几个步骤:创建构建器和网络、解析模型、配置构建参数、构建引擎…...
简记_ MCU管脚的防静电处理
一、分析(一) 接口处的信号要先过 ESD/TVS 管,然后拉到被保护器件; 建个 ESD 电路发生器的模型,代入到我们的电路中去分析: 继电器实现这两个“开关”,并且还会感应出一些额外的RLC寄生。 ES…...
C语言实现算法(二)
以下是 “10个不重复的C语言经典算法案例“,包含可运行代码、开发环境配置及系统要求。所有代码基于标准C语法,已在GCC 9.3.0环境下测试通过。 开发环境配置 编译器:GCC(推荐) Windows:安装 MinGW 或 Visual Studio Linux:sudo apt-get install gcc macOS:通过Xcode Co…...
transformer模型介绍——大语言模型 LLMBook 学习(二)
1. transformer模型 1.1 注意力机制 **注意力机制(Attention Mechanism)**在人工智能中的应用,实际上是对人类认知系统中的注意力机制的一种模拟。它主要模仿了人类在处理信息时的选择性注意(Selective Attention)&a…...
K8s 1.27.1 实战系列(十一)ConfigMap
ConfigMap 是 Kubernetes 中管理非敏感配置的核心资源,通过解耦应用与配置实现灵活性和可维护性。 一、ConfigMap 的核心功能及优势 1、配置解耦 将配置文件(如数据库地址、日志级别)与容器镜像分离,支持动态更新而无需重建镜像。 2、多形式注入 环境变量:将键值…...
下降路径最⼩和(medium)
题目描述: 给你一个 n x n 的 方形 整数数组 matrix ,请你找出并返回通过 matrix 的下降路径 的 最小和 。 下降路径 可以从第一行中的任何元素开始,并从每一行中选择一个元素。在下一行选择的元素和当前行所选元素最多相隔一列(…...