【Linux笔记】——线程同步条件变量与生产者消费者模型的实现
🔥个人主页🔥:孤寂大仙V
🌈收录专栏🌈:Linux
🌹往期回顾🌹:【Linux笔记】——线程互斥与互斥锁的封装
🔖流水不争,争的是滔滔不息
- 一、线程同步的简介
- 二、条件变量
- 三、条件变量函数
- 四、条件变量的封装
- 五、生产者消费者模型
- 基于BlockingQueue的生产者消费者模型
一、线程同步的简介
定义
线程同步是指在多线程环境中,通过某种机制确保多个线程在访问共享资源时能够有序、协调地进行,以避免数据不一致或竞争条件的问题。线程同步的主要目的是控制线程的执行顺序,确保共享资源在某一时刻只能被一个线程访问或修改。
必要性
在多线程编程中,多个线程可能会同时访问和修改共享资源(如变量、文件、数据库等)。如果没有适当的同步机制,可能会导致以下问题:
- 数据竞争:多个线程同时修改同一数据,导致数据状态不一致。
- 死锁:多个线程相互等待对方释放资源,导致程序无法继续执行。
- 资源争用:多个线程竞争同一资源,导致性能下降或程序崩溃。
为实现线程同步这里引入条件变量
二、条件变量
条件变量是一种用于线程同步的机制,通常与互斥锁(mutex)结合使用,用于在多线程环境中实现线程间的协调。条件变量的主要作用是允许线程在某些条件不满足时进入等待状态,并在条件满足时被唤醒继续执行。
在多线程并发中,光有互斥锁只能保证临界区的互斥访问,但有些情况线程即使拿到了锁,也无法立刻执行任务,比如:线程A 拿到锁准备从任务队列里取任务,结果发现队列是空的 ,此时线程A该怎么办?它不能一直占着锁,否则别的线程(比如生产任务的线程B)就无法加锁,也就无法往队列中添加任务了。
条件变量的作用:在资源条件未满足时,挂起当前线程的执行(释放锁)并进入等待状态,直到条件满足被唤醒,再继续执行。
假如苹果是共享资源,不能被多个线程同时访问(临界资源),盘子是共享区(临界区),锁是控制访问盘子的唯一通道谁持有锁谁能动苹果,线程们是一群想吃苹果的人,条件变量是通知机制,用来告诉排队的人可以去拿锁。
有一堆线程想去盘子里拿到苹果。只有拿到锁得的人才能查看到盘子里是否有苹果。如果拿到锁发现没有苹果,就把锁放回去,然后进入等待队列。条件变量上场,等有线程放了苹果病通知,就会“敲钟”,让等待队列里的线程醒来。被唤醒的线程们就会重新竞争锁,拿到锁的线程再次检查苹果是否存在,存在就吃掉,不存在就继续等。
条件变量是一个“协调者”,帮你把拿不到锁的线程有序地放到队列里睡觉,然后在条件满足时“敲钟”叫醒他们,大家再去抢锁判断是否可以操作资源。
三、条件变量函数
初始化
int pthread_cond_init(pthread_cond_t *restrict cond,const pthread_condattr_t
*restrict attr);
参数:cond:要初始化的条件变量,attr:NULL
销毁
int pthread_cond_destroy(pthread_cond_t *cond)
等待条件满足
int pthread_cond_wait(pthread_cond_t *restrict cond,pthread_mutex_t *restrict
mutex);
参数:cond:要在这个条件变量上等待,mutex:互斥量
唤醒等待
int pthread_cond_broadcast(pthread_cond_t *cond);
int pthread_cond_signal(pthread_cond_t *cond);
实例
#include <iostream>
#include <vector>
#include <pthread.h>
#include <unistd.h>
#include <stdio.h>
#include <string>const int NUM=5;
int cnt=1000;using namespace std;pthread_mutex_t glock = PTHREAD_MUTEX_INITIALIZER; //定义全局锁
pthread_cond_t gcond = PTHREAD_COND_INITIALIZER; //定义全局信号量void* pthreadrun(void* args)
{string name=static_cast<const char*>(args);while(true){ pthread_mutex_lock(&glock); //上锁pthread_cond_wait(&gcond,&glock); //等待队列std::cout << name << " 计算: " << cnt << std::endl;cnt--;pthread_mutex_unlock(&glock); //解锁}
}int main()
{vector<pthread_t> pthreads;for(int i=0;i<NUM;i++){pthread_t tid;char* name=new char [64];snprintf(name,64,"thread-%d", i);int n=pthread_create(&tid,nullptr,pthreadrun,name);if(n==0){cout<<"创建新线程成功"<<endl;}pthreads.push_back(tid);sleep(1);}sleep(3);while(true){std::cout << "唤醒所有线程... " << std::endl;pthread_cond_broadcast(&gcond);sleep(1);}for(auto &id: pthreads){int n=pthread_join(id,nullptr);if(n==0){cout<<"进程回收成功"<<endl;}}
}
我们注意到等待队列是在临界区内,加锁和解锁之间写的
while(true){ pthread_mutex_lock(&glock); //上锁pthread_cond_wait(&gcond,&glock); //等待队列std::cout << name << " 计算: " << cnt << std::endl;cnt--;pthread_mutex_unlock(&glock); //解锁}
上面苹果的那个比喻,我们不是先申请锁然后去临界区内申请资源吗,没资源释放锁进入等待队列。流程:线程先申请锁成功,判断资源是否可用如果不可用就要自动释放锁进入等待队列(注意这里的的释放锁并进入等待队列是pthread_cond_wait提供的自动的),等唤醒时重新自动上锁,再重新判断资源状态。如果拿到资源执行完正常的释放锁。
注意:
pthread_cond_wait 中的“释放” → 是为了让别的线程能进入临界区补充资源。
最后的 pthread_mutex_unlock → 是整个线程工作完之后的“正常释放锁”。
四、条件变量的封装
#include <iostream>
#include <pthread.h>
#include "Mutex.hpp"using namespace MutexModule;
using namespace std;namespace CondModule
{class Cond{public:Cond(){pthread_cond_init(&_cond,nullptr);}void Wait(Mutex& mutex) //阻塞队列进行等待{int n=pthread_cond_wait(&_cond,mutex.Get());}void Signal() //唤醒条件变量下等待的进程{int n=pthread_cond_signal(&_cond);}void Broadcast() //唤醒条件变量下所有等待的进程{ int n=pthread_cond_broadcast(&_cond);}~Cond(){pthread_cond_destroy(&_cond);}private:pthread_cond_t _cond;};
}
直接pthread库封装成类就完事了。
五、生产者消费者模型
生产者消费者模型是一种经典的并发编程模型,用于解决多线程环境下的资源分配和同步问题。该模型涉及两类线程:生产者线程和消费者线程。生产者负责生成数据并将其放入共享缓冲区,而消费者则从缓冲区中取出数据进行处理。
生产者消费者模式就是通过⼀个容器来解决生产者和消费者的强耦合问题。生产者和消费者彼此之间不直接通讯,而通过阻塞队列来进行通讯,所以生产者生产完数据之后不用等待消费者处理,直接扔给阻塞队列,消费者不找生产者要数据,而是直接从阻塞队列里取,阻塞队列就相当于⼀个缓冲区,平衡了生产者和消费者的处理能力。这个阻塞队列就是用来给生产者和消费者解耦的。
生产者消费者模型:三种要素,生产者、消费者、一个交易场所。中间的"交易场所就是一快"内存"空间"。
三种关系:
生产者之间:竞争关系,互斥关系。
消费者和消费者之间:互斥关系。
生产者和消费者之间:互斥和同步
两种角色:
生产者角色和消费者角色(线程承担)。
一个交易场所:
以特定结构构成的一种"内存空间"。
生产者消费者模型的优势:
生产过程和消费过程解耦,支持忙闲不均,提高效率(不体现在“交易场所”上,而是未来获取任务和处理任务)。
基于BlockingQueue的生产者消费者模型
在多线程编程中阻塞队列(Blocking Queue)是⼀种常用于实现生产者和消费者模型的数据结构。其与普通的队列区别在于,当队列为空时,从队列获取元素的操作将会被阻塞,直到队列中被放入了元素;当队列满时,往队列里存放元素的操作也会被阻塞,直到有元素被从队列中取出(以上的操作都是基于不同的线程来说的,线程在对阻塞队列进程操作时会被阻塞)
源码代码地址:生产者消费者模型源码
Blockqueue.hpp
#pragma once
#include <iostream>
#include <pthread.h>
#include <queue>
#include "Cond.hpp"using namespace std;
using namespace CondModule;
using namespace MutexModule;const int num = 5;template <typename T>
class Blockqueue
{
public:Blockqueue(int cap = num): _cap(cap), _csleep_num(0), _psleep_num(0)//{ }void Equeue(const T &in) // 生产者写入数据{LockGuard lockguard(_mutex); // RAII自动上锁解锁while (Qfull()) // 如果阻塞队列是满的{_psleep_num++;_full_cond.Wait(_mutex); // 生产者线程进入等待队列_psleep_num--;}_q.push(in);if (_csleep_num > 0){_empty_cond.Signal(); // 唤醒消费者cout<<"唤醒消费者"<<endl;}}T pop() // 消费者读数据{LockGuard lockguard(_mutex);while (Qempty()) // 如果阻塞队列是空的{_csleep_num++;_empty_cond.Wait(_mutex);//消费者线程进入等待队列_csleep_num--;}T data=_q.front();_q.pop();if(_psleep_num >0){_full_cond.Signal(); //唤醒生产者cout<<"唤醒生产者"<<endl;}return data;}~Blockqueue(){}bool Qfull(){return _q.size() >= num;}bool Qempty(){return _q.empty();}private:queue<T> _q;int _cap;Mutex _mutex;Cond _full_cond;Cond _empty_cond;int _psleep_num;int _csleep_num;
};
void Equeue(const T &in) // 生产者写入数据{LockGuard lockguard(_mutex); // RAII自动上锁解锁while (Qfull()) // 如果阻塞队列是满的{_psleep_num++;_full_cond.Wait(_mutex); // 生产者线程进入等待队列_psleep_num--;}_q.push(in);if (_csleep_num > 0){_empty_cond.Signal(); // 唤醒消费者cout << "唤醒消费者" << endl;}}T pop() // 消费者读数据{LockGuard lockguard(_mutex);while (Qempty()) // 如果阻塞队列是空的{_csleep_num++;_empty_cond.Wait(_mutex); // 消费者线程进入等待队列_csleep_num--;}T data = _q.front();_q.pop();if (_psleep_num > 0){_full_cond.Signal(); // 唤醒生产者cout << "唤醒生产者" << endl;}return data;}
生产者写入数据与消费者读数据放一起聊
生产者写入数据,无可厚非先上锁,判断如果阻塞队列是满的,那么我们就要用条件变量让生产者线程进行等待队列。但是这里一定要注意生产者线程写入和消费者线程读是并发执行的,如果不计数就可能在某些场景中浪费唤醒操作,甚至唤醒失败。当消费者读数据的时候,读完数据要唤醒生产者继续写数据,这时候如果写成
if(true)
{_full_cond.Signal();
}
想想这样粒度不够细,会不会白敲钟,造成效率浪费。计数后
if (_psleep_num > 0)
{_full_cond.Signal(); // 唤醒生产者cout << "唤醒生产者" << endl;}
这样有设计确实有生产者在等,才敲钟,是不是更精细,效率更高。
接着上面生产者线程进入等待队列后,顺一下流程。生产者往阻塞队列中写入数据,这时候判断消费者线程是否阻塞,唤醒消费者(这里看到可能有点懵,这么设计就是源于生产者消费者模型的并发执行的)。消费者读数据跟上述生产者写数据思路一样。
这段代码最绕的就是这个计数了,_psleep_num记录当前有多少个生产者线程"等待着空位",_csleep_num记录有多少个消费者线程在“等着有数据”。这样设计就是为了避免无意义的唤醒,提高效率,避免无意义的浪费。
main.cc
#include "Blockqueue.hpp"
#include <iostream>
#include <pthread.h>
#include <unistd.h>
using namespace std;void *consumer(void *args) // 消费者
{Blockqueue<int> *bq = static_cast< Blockqueue<int> *>(args);while(true){sleep(20);int data=bq->pop();cout<<"消费者消费数据->"<<data<<endl;}
}void *productor(void *args) // 生产者
{Blockqueue<int> *bq = static_cast< Blockqueue<int> *>(args);int data=1;while (true){sleep(1);cout<<"生产者写入数据-> "<<data<<endl;bq->Equeue(data);data++;}
}int main()
{Blockqueue<int> *bq = new Blockqueue<int>();pthread_t c, p;pthread_create(&c, nullptr, consumer, bq);pthread_create(&p, nullptr, productor, bq);pthread_join(c,nullptr);pthread_join(p,nullptr);return 0;
}
创建阻塞队列对象,创建线程,对了这里写的是单线程,消费者调用pop()让消费者读数据,生产者调用Equeue()让消费者写数据。
运行结果
代码地址:生产者消费者模型源码
相关文章:
【Linux笔记】——线程同步条件变量与生产者消费者模型的实现
🔥个人主页🔥:孤寂大仙V 🌈收录专栏🌈:Linux 🌹往期回顾🌹:【Linux笔记】——线程互斥与互斥锁的封装 🔖流水不争,争的是滔滔不息 一、线程同步的…...
Popeye
概览与定位 Popeye 是由 derailed 团队开源的 Kubernetes 集群资源 “Sanitizer”,它以只读方式扫描集群内的各种资源(如 Pod、Service、Ingress、PVC、RBAC 等),并基于社区最佳实践给出问题等级及修复建议,覆盖配置误…...
ES(ES2023/ES14)最新更新内容,及如何减少内耗
截至2023年10月,JavaScript(ECMAScript)的最新版本是 ES2023(ES14)。 ES2023 引入了许多新特性,如findLast、toSorted等,同时优化了性能。通过减少全局变量、避免内存泄漏、优化循环、减少DOM操作、使用Web Workers、懒加载、缓存、高效数据结构和代码压缩,可以显著降低…...
电子数据取证(数字取证)技术全面指南:从基础到实践
为了后续查阅方便,推荐工具先放到前面 推荐工具 数字取证基础工具 综合取证平台 工具名称类型主要功能适用场景EnCase Forensic商业全面的证据获取和分析、强大的搜索能力法律诉讼、企业调查FTK (Forensic Toolkit)商业高性能处理和索引、集成内存分析大规模数据处…...
【通用智能体】Serper API 详解:搜索引擎数据获取的核心工具
Serper API 详解:搜索引擎数据获取的核心工具 一、Serper API 的定义与核心功能二、技术架构与核心优势2.1 技术实现原理2.2 对比传统方案的突破性优势 三、典型应用场景与代码示例3.1 SEO 监控系统3.2 竞品广告分析 四、使用成本与配额策略五、开发者注意事项六、替…...
基于 STM32 的手持式安检金属探测器设计与实现
一、硬件设计:芯片与功能模块选型及接线 1. 主控芯片选型 芯片型号:STM32F103C8T6 核心优势: 32 位 Cortex-M3 内核,主频 72MHz,满足实时数据处理需求64KB Flash+20KB SRAM,支持程序存储与数据缓存丰富外设:2 路 USART、2 路 SPI、1 路 I2C、12 位 ADC,适配多模块通信…...
虚幻引擎5-Unreal Engine笔记之Default Pawn与GamMode、Camera的关系
虚幻引擎5-Unreal Engine笔记之Default Pawn与GamMode、Camera的关系 code review! 文章目录 虚幻引擎5-Unreal Engine笔记之Default Pawn与GamMode、Camera的关系1.Default Pawn与Camera的关系1.1. Default Pawn 是什么?1.2. Default Pawn 的主要组件1.3. Default…...
Spring源码主线全链路拆解:从启动到关闭的完整生命周期
Spring源码主线全链路拆解:从启动到关闭的完整生命周期 一文看懂 Spring 框架从启动到销毁的主线流程,结合原理、源码路径与伪代码三位一体,系统学习 Spring 底层机制。 1. 启动入口与环境准备 原理说明 Spring Boot 应用入口是标准 Java 应…...
飞帆控件:可配置post/get接口
先上链接: post_get_ithttps://fvi.cn/796看一下这个控件的配置: 当 url 有某个 get 参数时,例如某些接口回传的参数。使用这个接口会发生这些: 如果检测到 url 中有该 url 参数则继续执行选择是否从 url 中删除该参数将这个参数…...
Android 自定义悬浮拖动吸附按钮
一个悬浮的拨打电话按钮,使用CardViewImageView可能会出现适配问题,也就是图片显示不全,出现这种问题,就直接替换控件了,因为上述的组合控件没有FloatingActionButton使用方便,还可以有拖动和吸附效果不是更…...
Spring AI Alibaba集成阿里云百炼大模型应用
文章目录 1.准备工作2.引入maven依赖3.application.yml4.调用4.1.非流式调用4.2.流式调用 阿里云百炼推出的智能体应用、工作流应用和智能体编排应用,有效解决了大模型在处理私有领域问题、获取最新信息、遵循固定流程以及自动规划复杂项目等方面的局限,…...
UI-TARS本地部署
UI-TARS本地部署 UI-TARS本地部署 UI-TARS 论文(arXiv) UI-TARS 官方仓库:包含部署指南、模型下载链接及示例代码。 UI-TARS-Desktop 客户端:支持本地桌面应用的交互控制。 模型部署框架:vLLM本地部署 1.下载项目…...
如何利用内网穿透实现Cursor对私有化部署大模型的跨网络访问实践
文章目录 前言1.安装Ollama2.QwQ-32B模型安装与运行3.Cursor安装与配置4. 简单使用测试5. 调用本地大模型6. 安装内网穿透7. 配置固定公网地址总结 前言 本文主要介绍如何在Windows环境下,使用Cursor接入Ollama启动本地部署的千问qwq-32b大模型实现辅助编程&#x…...
Linux的进程概念
目录 1、冯诺依曼体系结构 2、操作系统(Operating System) 2.1 基本概念 编辑 2.2 目的 3、Linux的进程 3.1 基本概念 3.1.1 PCB 3.1.2 struct task_struct 3.1.3 进程的定义 3.2 基本操作 3.2.1 查看进程 3.2.2 初识fork 3.3 进程状态 3.3.1 操作系统的进程状…...
(10)python开发经验
文章目录 1 cp35 cp36什么意思2 找不到pip3 subprocess编码错误4 导出依赖文件包含路径5 使用自己编译的python并且pyinstall打包程序 更多精彩内容👉内容导航 👈👉Qt开发 👈👉python开发 👈 1 cp35 cp36什…...
什么是时间戳?怎么获取?有什么用
时间戳的定义 时间戳(Timestamp)是指记录某个事件发生的具体时间点,通常以特定的格式表示。它可以精确到秒、毫秒甚至更小的单位,用于标识某个时刻在时间轴上的位置。 获取时间戳的方法 在不同的编程语言中,获取时间…...
Zookeeper 入门(二)
4. Zookeeper 的 ACL 权限控制( Access Control List ) Zookeeper 的ACL 权限控制,可以控制节点的读写操作,保证数据的安全性,Zookeeper ACL 权 限设置分为 3 部分组成,分别是:权限模式(Scheme)、授权对象(…...
[创业之路-361]:企业战略管理案例分析-2-战略制定-使命、愿景、价值观的失败案例
一、失败案例 1、使命方面的失败案例 真功夫创业者内乱:真功夫在创业过程中,由于股权结构不合理,共同创始人及公司大股东潘宇海与实际控制人、董事长蔡达标产生管理权矛盾。双方在公司发展方向、管理改革等方面无法达成一致,导致…...
dijkstra算法加训上 之 分层图最短路
来几个分层图的题练习下哈 P4568 [JLOI2011] 飞行路线 P4568 [JLOI2011] 飞行路线 - 洛谷https://www.luogu.com.cn/problem/P4568 题目描述 Alice 和 Bob 现在要乘飞机旅行,他们选择了一家相对便宜的航空公司。该航空公司一共在 n 个城市设有业务,设这…...
赋予AI更强的“思考”能力
刚刚!北大校友、OpenAI前安全副总裁Lilian Weng最新博客来了:Why We Think 原文链接:Why We Think by Lilian Weng 这篇文章关注:如何让AI不仅仅是“知道”答案,更能“理解”问题并推导出答案。通过赋予AI更强的“思…...
微服务项目->在线oj系统(Java版 - 1)
相信自己,终会成功 目录 C/S架构与B/S架构 C/S架构(Client/Server,客户端/服务器架构) 特点: 优点: 缺点: 典型应用: B/S架构(Browser/Server,浏览器/服务器架构&a…...
【深度学习】使用块的网络(VGG)
虽然 AlexNet 证明深层神经网络卓有成效,但它没有提供一个通用的模板来指导后续的研究人员设计新的网络。 也就是说尽管我知道了更深更大的网络更有效,但是不清楚怎么让它更深更大,从而起到一个更好的效果。 于是,研究人员开始从单…...
Python数据可视化 - Pyecharts绘图示例
文章目录 一、Pyecharts简介及安装1. Pyecharts简介2. 安装Pyecharts 二、准备数据三、饼图示例1. 初始化选项配置2. 饼图相关设置3. 全局配置项3.1 标题配置项3.2 图例配置项3.3 提示框配置项3.4 工具箱配置项3.5 视觉映射配置项 4. 系列配置项4.1 标签选项配置4.2 图元样式配…...
Day29
复习日 知识点回顾 类的装饰器装饰器思想的进一步理解:外部修改、动态类方法的定义:内部定义和外部定义 作业:复习类和函数的知识点,写下自己过去29天的学习心得,如对函数和类的理解,对python这门工具的理…...
Python列表全面解析:从入门到精通
文章目录 Python列表全面解析:从入门到精通一、列表基础1. 什么是列表?2. 列表特性总结表 二、列表的基本操作(基础)1. 访问元素2. 修改列表 三、列表的常用方法(基础)1. 添加元素的方法2. 删除元素的方法3. 查找和统计方法4. 排序和反转 四、列表的高级…...
Nacos数据写入流程
在 3 节点的 Nacos 集群中,数据写入流程和主节点(Leader)的角色基于 Nacos 的分布式一致性协议(通常使用 Raft 协议)来实现。以下以 Markdown 格式详细说明 3 节点 Nacos 集群的数据写入流程以及主节点的角色和确定方式…...
《P4551 最长异或路径》
题目描述 给定一棵 n 个点的带权树,结点下标从 1 开始到 n。寻找树中找两个结点,求最长的异或路径。 异或路径指的是指两个结点之间唯一路径上的所有边权的异或。 输入格式 第一行一个整数 n,表示点数。 接下来 n−1 行,给出…...
Ansible模块——文件属性查看,文件或目录创建和属性修改
ansible.builtin.stat 可以查看文件信息。 选项 类型 默认值 描述 pathstrnull 要检查的文件或目录的完整路径(必需)。 followboolfalse 如果是符号链接,是否跟随到目标路径上获取其状态。 get_attributesbooltrue 是否返回扩展属性&#…...
【图像生成大模型】Wan2.1:下一代开源大规模视频生成模型
Wan2.1:下一代开源大规模视频生成模型 引言Wan2.1 项目概述核心技术1. 3D 变分自编码器(Wan-VAE)2. 视频扩散 Transformer(Video Diffusion DiT)3. 数据处理与清洗 项目运行方式与执行步骤1. 环境准备2. 安装依赖3. 模…...
AGI大模型(25):LangChain提示词模版
我们也可以创建prompt template, 并引入一些变量到prompt template中,这样在应用的时候更加灵活。 1 代码实现 # 我们也可以创建prompt template, 并引入一些变量到prompt template中,这样在应用的时候更加灵活 from langchain_core.prompts import ChatPromptTemplate from…...
mybatis中的resultMap的association及collectio的使用
目录 1.reusltmap的说明 2.association的使用 3.collection的使用 4.总结 1.reusltmap的说明 resultmap定义了数据库的结果映射到java对象的规则,resultmap包含4个属性: id: ResultMap 的唯一标识 type: 映射到的 Java 类型(全限定类名或…...
静态网站部署:如何通过GitHub免费部署一个静态网站
GitHub提供的免费静态网站托管服务可以无需担心昂贵的服务器费用和复杂的设置步骤,本篇文章中将一步步解如何通过GitHub免费部署一个静态网站,帮助大家将创意和作品快速展现给世界。 目录 了解基础情况 创建基础站点 在线调试站点 前端项目部署 部署…...
Android 手写签名功能详解:从原理到实践
Android 手写签名功能详解 1. 引言2. 手写签名核心实现:SignatureView 类3. 交互层实现:MainActivity 类4. 布局与配置5. 性能优化与扩展方向 1. 引言 在电子政务、金融服务等移动应用场景中,手写签名功能已成为提升用户体验与业务合规性的关…...
【iOS(swift)笔记-9】WKWebView无法访问网络
对于iOS 在info中添加App Transport Security Settings,然后在App Transport Security Settings里添加Allow Arbitrary Loadstrue 对于macOS 除了上面的操作,还需在项目信息的App Sandbox里有个Network打钩选项...
Adapter适配器模式
Adapter适配器模式是一种结构设计模式,用于解决接口不兼容的问题,通过适配器类,可以将一个类的接口转换为客户渴望的另一个接口,从而使原来无法协作的对象能够一起工作。 角色和职责: 目标接口(Target&…...
七、xlib窗口渲染
文章目录 1.渲染图片2.双缓冲3.混合图片4.渐变窗口 1.渲染图片 在上篇文章中的最后,我们使用libpng加载了一个png图片,并显示到窗口上,但是我们可以看到显示到窗口的图片周边有黑色的背景。原因是在我测试的操作系统下使用xlib创建的窗口默认…...
python中http.cookiejar和http.cookie的区别
在Python中,http.cookiejar和http.cookie(通常指http.cookies模块)是两个不同的模块,它们的主要区别如下: 1. 功能定位 http.cookiejar 用于管理HTTP客户端的Cookie,提供自动化的Cookie存储、发送和接收功…...
架构设计模式:构建健壮、可扩展的 Serverless 应用
架构设计模式:构建健壮、可扩展的 Serverless 应用 到目前为止,我们已经掌握了 Serverless 的基本概念,了解了 FaaS 和 BaaS 如何协同工作,学会了使用框架进行开发部署,并知道了如何监控和排查问题。现在,是时候从“能用”向“好用”迈进了。 仅仅将代码部署到 Lambda 函…...
2- PyTorch
文章目录 1. Overview2. 线性模型 1. Overview 在人的智能中,最经常做的事情是推理和预测,在机器学习中也是如此。我们在以往的算法课中,所接触的穷举、贪心、分治和动规等算法都是由人设计的,而在机器学习中,算法是由…...
MinIO:从入门到精通,解锁云原生存储的奥秘
一、引言:为什么 MinIO 正在重塑存储世界? 在云计算和大数据时代,传统存储系统面临扩展性差、成本高、兼容性不足等挑战。MinIO 凭借其 S3 兼容性、分布式架构、高性能存储 等特性,成为企业构建现代化存储基础设施的首选。 本文…...
【LeetCode 热题100】739:每日温度(详细解析)(Go语言版)
🌡️ LeetCode 739:每日温度(详解 单调栈 多种思路对比) 📌 题目描述 给定一个整数数组 temperatures,表示每天的温度,返回一个数组 answer,其中 answer[i] 是指在第 i 天之后&am…...
Linux学习笔记|GCC编译指令基础|静动态库|makefile
一、GCC 编译指令基础 基本编译命令 gcc -o code code.c和gcc code.c -o code:这两条命令功能相同,都是使用 GCC 编译器将code.c源文件编译成名为code的可执行文件。-o选项用于指定输出文件名,选项位置在源文件前后不影响最终结果。 编译过程…...
【LeetCode 热题100】17:电话号码的字母组合(详细解析)(Go语言版)
☎️ LeetCode 17. 电话号码的字母组合(回溯 DFS 详解) 📌 题目描述 给定一个仅包含数字 2-9 的字符串,返回所有它能表示的字母组合。答案可以按任意顺序返回。 数字到字母的映射如下(与电话按键相同)…...
C++学习:六个月从基础到就业——C++17:std::optional/variant/any
C学习:六个月从基础到就业——C17:std::optional/variant/any 本文是我C学习之旅系列的第四十七篇技术文章,也是第三阶段"现代C特性"的第九篇,主要介绍C17引入的三个重要工具类型:std::optional、std::varia…...
Go语言中函数 vs 方法
函数(Function):不属于任何类型,是全局可调用的。 方法(Method):绑定在某个类型上的函数,调用时依赖于这个类型的值或指针。 一、函数(Function) func 函数…...
代码随想录算法训练营第六十五天| 图论10—卡码网94. 城市间货物运输 I,95. 城市间货物运输 II
被学校课程轰炸了一周,回过头发现训练营已经要结束了,抓紧时间补完。不过算法这边也很难,感觉每天都是勉强理解在干什么的状态。 94. 城市间货物运输 I 94. 城市间货物运输 I SPFA算法,也是Bellman_ford 队列优化算法 优化原理…...
TDengine 在新能源领域的价值
能源数据的定义 能源数据是指记录和描述能源产业各个方面的信息,包括能源生产、供应、消费、储备、价格、排放以及相关政策和技术的数据。这些数据可以通过各种途径收集和整理,如能源企业的统计报表、政府部门的调查和监测、国际组织的发布数据等。 能…...
浅谈Frida 检测与绕过
目录 ptrace 占位与进程名检测端口检测与 D-Bus 协议通信扫描 /proc 目录(maps、task、fd)定位 so 中的 SVC syscall内存动态释放代码 1. ptrace 占位与进程名检测 检测方式 遍历运行进程列表,检查是否存在 frida-server 或相关进程名&…...
WaterStamp —— 一个实用的网页水印生成器开发记
我正在参加CodeBuddy「首席试玩官」内容创作大赛,本文所使用的 CodeBuddy 免费下载链接:腾讯云代码助手 CodeBuddy - AI 时代的智能编程伙伴 最近,我和 CodeBuddy 一起完成了一个名为 WaterStamp 的网页水印生成器项目。这个小工具主要用于给…...
【MySQL】存储过程,存储函数,触发器
目录 准备工作 一. 存储过程 1.1.什么是存储过程 1.2.创建存储过程 1.3.创建只显示大于等于指定值的记录的存储过程 1.4.显示,删除存储过程 二. 存储函数 2.1.什么是存储函数 2.2.使用存储函数 2.2.1.使用存储函数之前 2.2.2.使用存储函数计算标准体重 …...