Python多线程编程理解面试题解析
一、多线程介绍
Python 的多线程是一种实现并发编程的方式,允许程序同时执行多个任务。然而,由于 Python 的全局解释器锁(GIL)的存在,多线程在某些场景下可能无法充分利用多核 CPU 的性能。以下是对 Python 多线程的理解和用法的详细说明。
二、多线程理解和使用
1. 理解多线程
1.1 什么是线程?
线程是操作系统能够调度的最小单位,一个进程可以包含多个线程。
同一进程中的线程共享内存空间,因此它们之间的通信比进程间通信更高效。
1.2 多线程的优势
I/O 密集型任务:多线程适合处理 I/O 操作(如文件读写、网络请求),因为线程可以在等待 I/O 完成时切换到其他任务。
资源共享:线程之间可以轻松共享数据,无需复杂的通信机制。
1.3 Python 的 GIL 限制
GIL(Global Interpreter Lock):Python 的 CPython 解释器中存在 GIL,它确保同一时刻只有一个线程执行 Python 字节码。
影响:
在计算密集型任务中,多线程无法利用多核 CPU 的优势。
对于 I/O 密集型任务,多线程仍然有效,因为线程在等待 I/O 时会释放 GIL。
1.4 多线程的特点
- 共享内存:同一进程中的所有线程共享内存地址空间,因此线程可以直接访问全局变量和资源。
- 轻量级:线程是轻量级的,创建和销毁的成本低于进程。
- 上下文切换:线程之间的上下文切换比进程之间的切换开销小,但仍然存在一定的开销。
1.5 多线程的应用场景
- I/O密集型应用:如网络通信、文件读取和写入等。
- GUI应用:避免界面卡顿,提高用户体验。
- 任务并发执行:如批量处理任务、定时任务等。
2. 多线程模块:threading
Python 提供了 threading 模块来实现多线程编程。以下是常用类和方法:
2.1创建线程
使用 threading.Thread 类创建线程对象:
import threading
def task(name):print(f"线程 {name} 正在运行")
# 创建线程
t1 = threading.Thread(target=task, args=("A",))
t2 = threading.Thread(target=task, args=("B",))
# 启动线程
t1.start()
t2.start()
# 等待线程完成
t1.join()
t2.join()
print("所有线程已完成")
2.2 自定义线程类
通过继承 threading.Thread 类自定义线程:
class MyThread(threading.Thread):def __init__(self, name):super().__init__()self.name = namedef run(self):print(f"线程 {self.name} 正在运行")
# 创建并启动线程
t1 = MyThread("A")
t2 = MyThread("B")
t1.start()
t2.start()
t1.join()
t2.join()
2.3 线程同步
当多个线程访问共享资源时,可能会出现竞争条件(Race Condition)。为了解决这个问题,可以使用线程同步机制。
(1) 使用锁(Lock)
threading.Lock 可以确保同一时间只有一个线程访问共享资源。
import threading
# 共享资源
counter = 0
lock = threading.Lock()
def increment():global counterfor _ in range(100000):with lock: # 加锁counter += 1
# 创建线程
t1 = threading.Thread(target=increment)
t2 = threading.Thread(target=increment)
t1.start()
t2.start()
t1.join()
t2.join()
print(f"最终计数器值:{counter}")
(2) 使用信号量(Semaphore)
threading.Semaphore 用于控制同时访问资源的线程数量。
import threading
semaphore = threading.Semaphore(2) # 最多允许 2 个线程同时访问
def worker(name):with semaphore:print(f"{name} 开始工作")threading.Event().wait(1) # 模拟工作print(f"{name} 完成工作")
threads = [threading.Thread(target=worker, args=(f"线程-{i}",)) for i in range(5)]
for t in threads:t.start()
for t in threads:t.join()
2.4 多线程与多进程对比
对于计算密集型任务,建议使用 multiprocessing 模块。
3. 高级用法:线程池
concurrent.futures.ThreadPoolExecutor 提供了更高级的线程管理方式。
from concurrent.futures import ThreadPoolExecutor
import time
def task(n):print(f"任务 {n} 开始")time.sleep(1)print(f"任务 {n} 完成")return n * n
# 创建线程池
with ThreadPoolExecutor(max_workers=3) as executor:futures = [executor.submit(task, i) for i in range(5)]results = [future.result() for future in futures]
print(f"所有任务结果:{results}")
三、多线程面试经典问题
3.1 什么是线程安全?如何保证线程安全?
定义:线程安全是指在多线程环境下,程序能够正确处理共享资源而不出现数据不一致或错误。
保证方法:
使用锁(如互斥锁、读写锁)。
使用原子操作(如 AtomicInteger)。
避免共享可变状态(使用不可变对象或线程本地存储)。
使用线程安全的数据结构(如 ConcurrentHashMap)。
3.2 实现多线程的方式
(1)Python 中的多线程实现
Python 提供了多种实现多线程的方式:
- a. 使用 threading 模块
- b. 继承 Thread 类
方法同以上,不再赘述。
(2)Java 中的多线程实现(此处,拓展java知识)
Java 提供了多种实现多线程的方式:
a. 继承 Thread 类
class MyThread extends Thread {public void run() {System.out.println("Thread " + Thread.currentThread().getName() + " is running");}
}
public class Main {public static void main(String[] args) {MyThread t1 = new MyThread();MyThread t2 = new MyThread();t1.start();t2.start();}
}
b. 实现 Runnable 接口
class MyRunnable implements Runnable {public void run() {System.out.println("Thread " + Thread.currentThread().getName() + " is running");}
}
public class Main {public static void main(String[] args) {Thread t1 = new Thread(new MyRunnable());Thread t2 = new Thread(new MyRunnable());t1.start();t2.start();}
}
3.3 你知道哪些多线程的优化技巧?
(1)使用线程池
线程池可以复用线程,减少线程创建和销毁的开销。
Python 示例:
from concurrent.futures import ThreadPoolExecutor
def task(n):print(f"Processing {n}")return n * n
with ThreadPoolExecutor(max_workers=3) as executor:results = executor.map(task, range(10))for result in results:print(result)
Java 示例(此处,拓展java知识):
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
class Task implements Runnable {private int n;public Task(int n) { this.n = n; }public void run() {System.out.println("Processing " + n);}
}
public class Main {public static void main(String[] args) {ExecutorService pool = Executors.newFixedThreadPool(3);for (int i = 0; i < 10; i++) {pool.submit(new Task(i));}pool.shutdown();}
}
(2)避免过度同步
问题:过度同步会导致性能瓶颈。
优化方法:
- 尽量缩小同步代码块的范围。
- 使用无锁算法(如 CAS)。
(3)使用异步编程
异步编程(如 Python 的 asyncio 或 Java 的 CompletableFuture)可以在单线程中实现高效的并发。
3.4 什么是 GIL?它对 Python 多线程有什么影响?
GIL(Global Interpreter Lock):Python 解释器的一个互斥锁,确保同一时刻只有一个线程执行 Python 字节码。
影响:
- 在 CPU 密集型任务中,多线程无法充分利用多核 CPU。
- 在 I/O 密集型任务中,多线程仍然有效,因为线程在等待 I/O 时会释放 GIL。
3.5 什么是 CAS(Compare-And-Swap)?
定义:CAS 是一种无锁算法,通过比较并交换内存值来实现原子操作。
优点:避免使用锁,提高性能。
缺点:可能导致“ABA 问题”。
3.6 什么是死锁,如何避免死锁?
死锁发生在两个或多个线程互相等待对方释放资源的情况。避免死锁的策略包括:
避免嵌套锁:尽量不在持有一个锁时去请求其他锁。
固定加锁顺序:确保所有线程以相同的顺序请求锁。
使用超时策略:在请求锁时设置超时时间。
3.7 在Python中,何时使用多线程,何时使用多进程?
使用多线程适合I/O密集型任务,如网络请求、数据库操作等。
使用多进程适合CPU密集型任务,因为它们可以绕过GIL,每个进程有自己的Python解释器和内存空间。
3.8 如何处理线程中的异常?
可以在目标函数内部捕获异常,或者使用Thread类的join()方法,结合is_alive()来检查线程状态。
import threading
def worker():try:# 可能会引发异常的代码raise ValueError("An error occurred")except Exception as e:print(f"Error in thread: {e}")thread = threading.Thread(target=worker)
thread.start()
thread.join()
3.9 什么是条件变量(Condition)?
条件变量是一种线程间的同步机制,可以让线程在满足某个条件之前阻塞,并在条件满足时通知其他线程。适用于生产者-消费者问题等场景。
condition = threading.Condition()
def producer():with condition:# 生产物品condition.notify() # 通知消费者
def consumer():with condition:condition.wait() # 等待生产者的通知# 消费物品
3.10 你如何监控多线程的执行状态?
可以使用threading模块中的active_count()和current_thread()等方法,也可以使用日志记录线程的执行状态。
3.11 什么是异步编程?异步编程和多线程的区别?
定义:异步编程是一种单线程并发模型,通过事件循环和回调机制实现高效的 I/O 操作。
适用场景:I/O 密集型任务(如网络请求、文件读写)。
示例:
import asyncio
async def task(n):print(f"Start task {n}")await asyncio.sleep(1)print(f"End task {n}")
async def main():await asyncio.gather(task(1), task(2), task(3))
asyncio.run(main())
异步编程与多线程的区别:
3.12 工作中有用过多线程吗?举例说一下?
简单举例,可以说用过多线程爬虫,同时抓取多个页面,示例代码如下:
import threading
from queue import Queue
# 共享队列,存储待抓取的 URL
url_queue = Queue()
# 存储结果的列表
results = []
lock = threading.Lock() # 线程锁,确保线程安全
def worker():while not url_queue.empty():url = url_queue.get() # 从队列中获取 URLtry:data = fetch_house_data(url)with lock: # 确保线程安全results.extend(data)finally:url_queue.task_done() # 标记任务完成
def multi_thread_crawler(base_url, num_pages, num_threads=5):# 将所有页面 URL 放入队列for page in range(1, num_pages + 1):url_queue.put(f"{base_url}/page/{page}")# 创建并启动线程threads = []for _ in range(num_threads):t = threading.Thread(target=worker)t.start()threads.append(t)# 等待所有任务完成url_queue.join()# 等待所有线程结束for t in threads:t.join()
# 调用
base_url = "https://example.com/house"
multi_thread_crawler(base_url, num_pages=10, num_threads=5)
# 打印结果
for item in results:print(item)
四、多线程总结
1. 多线程注意点
(1)线程安全
如果多个线程访问共享资源,必须使用锁或其他同步机制。
避免死锁(Deadlock),即多个线程互相等待对方释放资源。
(2)调试多线程程序
多线程程序的调试较为复杂,可以使用日志记录或工具(如 threading.enumerate())查看线程状态。
(3)性能瓶颈
对于计算密集型任务,考虑使用多进程或异步编程(asyncio)。多线程适合处理 I/O 密集型任务,但受制于 GIL,不适合计算密集型任务。
2. 概括
(1)Python中的多线程使得程序能够在同一进程中并行处理多个任务,尤其在I/O密集型操作中表现优异。理解线程的基本概念、创建管理及其同步机制对于实现高效稳定的多线程应用至关重要。
(2)我们可以使用 threading 模块可以轻松实现多线程编程,配合锁、信号量等同步机制避免竞争条件。对于更复杂的任务,则更推荐使用线程池(ThreadPoolExecutor)简化管理。如果需要更高的性能,可以结合多进程或异步编程(asyncio)。
相关文章:
Python多线程编程理解面试题解析
一、多线程介绍 Python 的多线程是一种实现并发编程的方式,允许程序同时执行多个任务。然而,由于 Python 的全局解释器锁(GIL)的存在,多线程在某些场景下可能无法充分利用多核 CPU 的性能。以下是对 Python 多线程的理…...
网络协议相关知识有哪些?
前言 网络协议的基础是OSI和TCP/IP模型,这两个模型是理解协议分层的关键。 正文(仅是个人理解,如有遗漏望海涵) 网络协议是网络中设备间通信的规则和标准,涉及数据传输、路由、错误控制等多个方面。以下是网络协议相关知识的系统梳理: 一、网络协议分层模型 1、OSI七…...
【并发压测】高并发下Linux流量监控
在高并发环境下,Linux流量监控至关重要,可以帮助您确保网络性能和稳定性。以下是一些常用的Linux流量监控工具和方法: 1. **iftop**:iftop 是一款实时的网络流量监控工具,可以显示当前服务器上每个网络接口的流量使用情…...
Spring Boot项目中解决跨域问题(四种方式)
目录 一,跨域产生的原因二,什么情况下算跨域三,实际演示四,解决跨域的方法 1,CrossOrigin注解2,添加全局过滤器3,实现WebMvcConfigurer4,Nginx解决跨域5,注意 开发项目…...
革新之力:数字科技——重塑未来的超越想象之旅
在21世纪的科技浪潮中,数字科技如同一股不可阻挡的洪流,正以前所未有的速度和广度改变着我们的生活、工作乃至整个社会的结构。它不仅是技术的简单迭代,更是对人类社会认知边界的拓宽,对经济模式、社会治理、文化形态等多方面的深…...
matlab和java混合编程经验分享
最常用的就是可以查到再控制栏deploytool选择library complier打包,但是有问题就是比如果用了外部的求解器比如yalmip或者cplex的话用这个方法会找不到外部的求解器,网上找了很多,基本都大同小异。 后面分享一个亲测有效的打包方法࿰…...
rk3588/3576板端编译程序无法运行视频推理
图片推理可以,但是视频不行,运行视频推理报错:segment fault. 我遇到的问题原因是ffmpeg安装有问题,可以先在板端运行:ffmpeg -version ffmpeg version 4.2.4-1ubuntu1.0firefly6 Copyright (c) 2000-2020 the FFmpe…...
MATLAB在数据分析和绘图中的应用:从基础到实践
引言 股票数据分析是金融领域中的重要研究方向,通过对历史价格、成交量等数据的分析,可以帮助投资者更好地理解市场趋势和做出决策。MATLAB作为一种强大的科学计算工具,提供了丰富的数据处理和可视化功能,非常适合用于股票数据的…...
【CS285】高斯策略对数概率公式的学习笔记
公式介绍 在【CS285】中提到了高斯策略对数概率公式的公式如下: log π θ ( a t ∣ s t ) − 1 2 ∥ f ( s t ) − a t ∥ Σ 2 const \log \pi_{\theta}(\mathbf{a}_t | \mathbf{s}_t) -\frac{1}{2} \left\| f(\mathbf{s}_t) - \mathbf{a}_t \right\|_{\S…...
windows环境下用docker搭建php开发环境dnmp
安装WSL WSL即Linux子系统,比虚拟机占用资源少,安装的前提是系统必须是win10以上。 WSL的安装比较简单,网上有很多教程,例如:WSL简介与安装流程(Windows 下的 Linux 子系统)_wsl安装-CSDN博客&…...
区块链中的递归长度前缀(RLP)序列化详解
文章目录 1. 什么是RLP序列化?2. RLP的设计目标与优势3. RLP处理的数据类型4. RLP编码规则详解字符串的编码规则列表的编码规则 5. RLP解码原理6. RLP在以太坊中的应用场景7. 编码示例分析8. 总结 1. 什么是RLP序列化? 递归长度前缀(RLP&…...
PHP建立MySQL持久化连接(长连接)及mysql与mysqli扩展的区别
如果在 PHP 5.3 的版本以前想要创建MySQL的持久化连接(长连接),需要显式调用 pconnect 创建: $con mysql_pconnect($server[host], $server[username], $server[password]); if (!($con false)) { if (mysql_select_db($server[database], $con) fals…...
基于Python+Django+Vue的旅游景区推荐系统系统设计与实现源代码+数据库+使用说明
运行截图 功能介绍 前台功能包括:首页、详情页、订单、用户中心。后台功能包括:首页、轮播图管理、管理员、卖家管理、买家管理、景区管理、订单管理非开源功能(分类管理,地区管理,收藏管理,评论管理&a…...
架构学习第七周--Prometheus
目录 一、监控系统基础 二、Prometheus介绍 三、Prometheus单机部署 四、服务发现与告警功能 4.1,服务发现 4.2,告警功能实现 五、Prometheus与Kubernetes 5.1,Kubernetes指标 5.2,Prometheus集群部署 一、监控系统基础…...
基于Nanopi duo2的WiFi智能摄像头
1.固件包烧录 https://wiki.friendlyelec.com/wiki/index.php/NanoPi_Duo2/zh#.E8.BF.9E.E6.8E.A5WiFi 固件包链接以及烧录工具都在上面链接中 烧录过程 使用读卡器将SD卡插入到电脑,然后打开烧录工具 2.通过串口工具连接板子使其连接WiFi 对应的串口工具,就是这个HyperT…...
Hive Orc表数据导出和导入
导出到hdfs:hive执行 INSERT OVERWRITE DIRECTORY /test/hdfs_dir ROW FORMAT DELIMITED FIELDS TERMINATED BY \t STORED AS ORC SELECT * FROM hive_table; HDFS导出到本地:shell执行 hdfs dfs -get /test/hdfs_dis/file_name /linux_dir/xxx 本…...
Python爬虫实战:从零到一构建数据采集系统
文章目录 前言一、准备工作1.1 环境配置1.2 选择目标网站 二、爬虫实现步骤2.1 获取网页内容2.2 解析HTML2.3 数据保存 三、完整代码示例四、优化与扩展4.1 反爬应对策略4.2 动态页面处理4.3 数据可视化扩展 五、注意事项六、总结互动环节 前言 在大数据时代,数据采…...
Ubuntu 的RabbitMQ安装
目录 1.安装Erlang 查看erlang版本 退出命令 2. 安装 RabbitMQ 3.确认安装结果 4.安装RabbitMQ管理界面 5.启动服务并访问 1.启动服务 2.查看服务状态 3.通过IP:port 访问界面 4.添加管理员用户 a)添加用户名:admin,密码࿱…...
七星棋牌源码高阶技术指南:6端互通、200+子游戏玩法深度剖析与企业级搭建实战(完全开源)
在棋牌游戏行业高速发展的今天,如何构建一个具备高并发、强稳定性与多功能支持的棋牌游戏系统成为众多开发者和运营团队关注的焦点。七星棋牌全开源修复版源码 凭借其 六端互通、200子游戏玩法、多省区本地化支持,以及 乐豆系统、防沉迷、比赛场、AI智能…...
cuda安装
cuda WSL2急速搭建CUDA体验环境_wsl2 cuda-CSDN博客 cudnn cuDNN Archive | NVIDIA Developer pip install torch torchvision torchaudio --index-url https://download.pytorch.org/whl/cu118...
Ubuntu 下 nginx-1.24.0 源码分析 - ngx_os_init 函数
ngx_os_init 声明在 src/os/unix/ngx_os.h ngx_int_t ngx_os_init(ngx_log_t *log); 定义在 src\os\unix\ngx_posix_init.c ngx_int_t ngx_os_init(ngx_log_t *log) {ngx_time_t *tp;ngx_uint_t n; #if (NGX_HAVE_LEVEL1_DCACHE_LINESIZE)long size; #endif#if (NGX…...
记录一次部署PC端网址全过程
当我查看我之前写的文章时、顿时惊奇发出感慨:啥时候写的?是我写的么?疑惑重重… 所以说,好记性不如烂笔头。 记录一次部署PC端网址全过程 部署PC端网址分是三步:第一步:申请域名并映射到外网IP ࿰…...
QML 实现一个动态的启动界面
QML 实现一个动态的启动界面 一、效果查看二、源码分享三、所用到的资源下载 一、效果查看 二、源码分享 工程结构 main.qml import QtQuick import QtQuick.Controls import QtQuick.Dialogs import Qt.labs.platformWindow {id:windowwidth: 640height: 400visible: truetit…...
视频HDR技术详解,你的电脑怎么播放HDR视频?
闲聊:前两天在b站上面看到影视飓风的视频,让我有点疑惑,我不知道为什么播放视频有设备撑不住一说,所以感兴趣去ytb下载了4k原片30hz刷新的,然后测试一下我的电脑能不能播放,发现还是可以的,视觉…...
Spring统一功能处理:拦截器、响应与异常的统一管理
目录 一.拦截器 二.统一数据返回格式 三.统一异常处理 一.拦截器 拦截器是Spring框架提供的核功能之,主要来拦截的请求,在指定法前后,根据业务需要执预先设定的代码。 也就是说,允许开发员提前预定义些逻辑,在的请…...
2025年度福建省职业院校技能大赛高职组“信息安全管理与评估”赛项规程
2025 年度福建省职业院校技能大赛 高 职组“ 信息安全管理与评估 ”赛项规程 一、赛项名称 省赛编号:GZ032 赛项名称:信息安全管理与评估 赛项组别:高职组 竞赛形式: 团体赛 二、竞赛目的 为全面贯彻落实国家网络强国战略&#x…...
Vue 中 nextTick 的原理详解
1. 为什么需要 nextTick Vue 采用 异步渲染机制,当响应式数据发生变化时,Vue 并不会立即更新 DOM,而是将这些变化放入一个 队列 中,并在 同一事件循环(Event Loop)中合并相同的修改,最后执行批…...
vue 手写分页
【先看效果】 (1)内容小于2页 不展示页码 (2)1 < 内容页数< 限定展示页码 展示:页码、上下页;隐藏:首页、末页图标,上、下一区间码。即:(页数&#…...
跟着柳叶刀数字健康,学习如何通过病理切片预测分子分类对预后的影响|项目复现
小罗碎碎念 项目复现 今天和大家分享一个非常具有参考价值的项目,手把手带着大家复现一篇发表在柳叶刀数字健康的文章。 花了六个小时才完成的这篇推送,信息量非常大,遇到了很多报错问题,但是解决以后的感觉是非常爽的,先给大家展示一下最终的成果——在同一张切片上,通…...
DuodooBMS源码解读之 sale_change模块
销售变更模块用户使用手册 一、模块概述 本扩展模块主要包含两个主要的 Python 文件:sale_change/report/sale_change_report.py 和 sale_change/wizard/sale_change_download.py,提供了销售变更报表查看和销售变更单下载的功能。以下是详细的使用说明…...
基于 Highcharts 实现 Vue 中的答题统计柱状图组件
在现代 Web 开发中,数据可视化是一个重要的组成部分,而 Highcharts 是一个广泛使用的 JavaScript 图表库,可以帮助开发者在 Web 页面上轻松地绘制丰富的图表。在本文中,我们将基于 Highcharts 创建一个用于答题统计的柱状图&#…...
MySQL 插入更新语句(insert…on duplicate key update语句 )
我们日常在使用 insert into 语句向表中插入数据时,一定遇到过主键或唯一索引冲突的情况,当遇到这种情况时,MySQL默认的反应是报错并停止执行后续的语句,为了避免这种情况,你有3种选择: 使用insert ignore…...
Jenkins整合Jmeter实现接口自动化测试
🍅 点击文末小卡片,免费获取软件测试全套资料,资料在手,涨薪更快 一、安装jmeter 下载:http://jmeter.apache.org/download_jmeter.cgi 这里我用了一台Windows安装jmeter用来写接口测试的脚本,启动前修改j…...
基于LM Arena 的 LLM 基准测试排行榜:DeepSeek-R1 排名第 5
打开 Arena 网站:https://lmarena.ai/,点开 Leaderboard 可以看到上图的排行榜,可以看到 DeepSeek-R1 排名第 5。...
游戏引擎学习第116天
回顾昨天的工作 本次工作内容主要集中在游戏开发的低级编程优化,尤其是手动优化软件渲染。工作目的之一是鼓励开发者避免依赖外部库,而是深入理解代码并进行优化。当前阶段正进行SIMD(单指令多数据)优化,使用Intel推荐…...
【AI绘画】大卫• 霍克尼风格——自然的魔法(一丹一世界)
大卫• 霍克尼,很喜欢这个老头,“艺术是一场战斗”。老先生零九年有了iphone,开始用iphone画画,一零年开始用ipad画画,用指头划拉,据说五分钟就能画一幅,每天早上随手画几幅送给身边的朋友。很c…...
当湖南家具遇上DeepSeek:极满家开启智能家居新时代
在湖南的街头巷尾,总流传着这样一句话:家具有温度,生活才有味道。走进铂乐极满家的展厅,只有被阳光浸润的原木香气,和智能家居跳动的温暖建议。DeepSeek就推荐了能自动调节高度的智能款,连护眼灯角度都算好…...
#渗透测试#批量漏洞挖掘#畅捷通T+远程命令执行漏洞
免责声明 本教程仅为合法的教学目的而准备,严禁用于任何形式的违法犯罪活动及其他商业行为,在使用本教程前,您应确保该行为符合当地的法律法规,继续阅读即表示您需自行承担所有操作的后果,如有异议,请立即停止本文章读。 目录 一、漏洞概况 二、攻击特征 三、应急处置…...
Docker挂载数据显式挂载和隐式挂载的区别
项目使用的Docker file 创建数据卷挂载点,结果发现宿主机目录中的数据卷路径下是空的,才知道docker file中创建的数据卷是隐式挂载,并不会在宿主机上留下持久化数据,随着容器被删除隐式挂载的数据卷也会跟着被删除 后面改为在jen…...
深度解析应用层协议-----HTTP与MQTT(涵盖Paho库)
HTTP协议概述 1.1 HTTP的基本概念 HTTP是一种应用层协议,使用TCP作为传输层协议,默认端口是80,基于请求和响应的方式,即客户端发起请求,服务器响应请求并返回数据(HTML,JSON)。在H…...
MySQL八股学习笔记
文章目录 一、MySQL结构1.宏观结构1.1.Server层1.2.存储引擎层 2.建立链接-连接器3.查询缓存4.解析SQL-解析器(1)词法分析(2)语法分析 5.执行SQL5.1.预处理器 prepare5.2.优化器 optimize5.3.执行器 execute(1…...
C++ 设计模式-备忘录模式
游戏存档实现,包括撤销/重做、持久化存储、版本控制和内存管理 #include <iostream> #include <memory> #include <deque> #include <stack> #include <chrono> #include <fstream> #include <sstream> #include <ct…...
Jenkins 环境搭建---基于 Docker
前期准备 提前安装jdk、maven、nodeJs(如果需要的话) 创建 jenkins 环境目录,用来当做挂载卷 /data/jenkins/ 一:拉取 Jenkins 镜像 docker pull jenkins/jenkins:lts 二:设置 Jenkins挂载目录 mkdir -p ~/jen…...
【动态规划篇】:解析背包问题--动态规划塑造的算法利器
✨感谢您阅读本篇文章,文章内容是个人学习笔记的整理,如果哪里有误的话还请您指正噢✨ ✨ 个人主页:余辉zmh–CSDN博客 ✨ 文章所属专栏:动态规划篇–CSDN博客 文章目录 一.01背包问题1.模板题2.例题1.分割等和子集2.目标和3.最后…...
ok113i平台——多媒体播放器适配
1. 视频播放支持 1.1 在Linux平台交叉编译ffmpeg动态库,详情查看《ok113i平台——交叉编译音视频动态库》 提取如下动态库: libavcodec.so.58.134.100 libavdevice.so.58.13.100 libavfilter.so.7.110.100 libavformat.so.58.76.100 libavutil.so.56.…...
【保姆级教程】DeepSeek R1+RAG,基于开源三件套10分钟构建本地AI知识库
一、总体方案 目前在使用 DeepSeek 在线环境时,页面经常显示“服务器繁忙,请稍后再试”,以 DeepSeek R1 现在的火爆程度,这个状况可能还会持续一段时间,所以这里给大家提供了 DeepSeek R1 RAG 的本地部署方案。最后实现…...
线程与进程的深入解析及 Linux 线程编程
在操作系统中,进程和线程是进行并发执行的两种基本单位。理解它们的区别和各自的特点,能够帮助开发者更好地进行多任务编程,提高程序的并发性能。本文将探讨进程和线程的基础概念,及其在 Linux 系统中的实现方式,并介绍…...
微信问题总结(onpageshow ,popstate事件)
此坑描述 订单详情某按钮点击,通过window.location.href跳转到(外部)第三方链接后,回退后,在ios中生命周期和路由导航钩子都失效了,无法触发。 在安卓中无视此坑, 回退没有问题 解决 原因&am…...
自己安装一台DeepSeek的服务器
找一台还可以的Linux服务器,登录后执行: curl -fsSL https://ollama.com/install.sh | sh 等待安装完成: 执行命令,根据服务器能力安装不同版本的AI模型: ollama run llama3.2 下一步就开始对话吧: llam…...
面阵工业相机提高餐饮业生产效率
餐饮行业是一个快节奏、高要求的领域,该领域对生产过程中每一个阶段的效率和准确性都有很高的要求。在食品加工、包装、质量控制和库存管理等不同生产阶段实现生产效率的优化是取得成功的关键步骤。面阵工业相机能够一次性捕捉对象的二维区域图像,并支持…...