Python的with语句:不止文件操作,数据库和多线程也能轻松搞定!
liuian 2025-08-31 04:04 26 浏览
你是不是也遇到过这样的问题:写文件操作时忘记调用close()导致资源泄漏?处理数据库连接时,因为异常没关闭连接让服务器崩溃?别慌!今天咱们聊聊Python里的"资源管家"——with语句,它可不只是用来打开文件那么简单,数据库连接、多线程锁都能轻松拿捏!
什么是with语句?一句话讲明白
with语句就像给资源配了个"智能管家",自动帮你完成"开门-使用-关门"全流程。不管你是正常用完资源,还是中途报错,它都能确保资源被正确释放,再也不用手动调用close()啦!
它的核心秘密在于上下文管理器协议——只要一个对象实现了__enter__()(开门)和__exit__()(关门)方法,就能用with语句管理。比如文件对象、数据库连接、线程锁,都是天生的"乖宝宝",会乖乖配合with语句~
场景一:文件操作——最经典的"自动关门"
说到with语句,文件操作绝对是它的"成名作"!以前写文件读写,得用try-finally手动关文件,又臭又长:
# 传统写法:手动关文件,生怕忘写finally
file = None
try:
file = open('data.txt', 'r')
content = file.read()
except FileNotFoundError:
print("文件不存在!")
finally:
if file:
file.close() # 漏写这行就等着资源泄漏吧!有了with语句,直接简化成一行,自动关文件,安全感拉满
# with语句:自动关文件,优雅!
with open('data.txt', 'r') as file:
content = file.read()
print(content)
# 出了with块,文件已经自动关闭,随便你怎么操作都不怕~进阶技巧:还能同时管理多个文件,比如读取一个文件并写入另一个,全程自动关:
# 同时操作两个文件,一个读一个写
with open('input.txt', 'r') as infile, open('output.txt', 'w') as outfile:
outfile.write(infile.read().upper()) # 把内容转大写写入新文件看图感受下with语句的简洁:
(代码编辑器截图:左侧文件结构,中间with语句操作文件,底部显示运行结果,全程无手动close())
场景二:数据库连接——再也不怕连接泄漏
数据库连接可是"娇贵资源",一个没关就可能让服务器崩溃!但有了with语句,连接自动关闭,安全感爆棚~
举个SQLite的例子:
import sqlite3
# with语句管理数据库连接,自动关闭
with sqlite3.connect('example.db') as conn:
cursor = conn.cursor()
cursor.execute('SELECT * FROM users') # 查询用户表
results = cursor.fetchall() # 获取结果
print(f"查到{len(results)}条数据!")
# 出了with块,连接已经自动关闭,服务器表示:谢谢你啊~高级玩家:异步数据库连接(asyncpg)
现在异步编程这么火,with语句也能跟上!比如用asyncpg操作PostgreSQL:
# 异步数据库连接,async with更配哦~
async with asyncpg.create_pool(**creds) as pool: # 创建连接池
async with pool.acquire() as conn: # 获取连接
await conn.execute("INSERT INTO logs VALUES($1)", datetime.now()) # 插入数据数据库连接的"自动管理"长这样:
(代码截图:with语句嵌套管理连接池和连接,底部显示运行结果,无需手动close())
场景三:多线程锁——线程安全"一键搞定"
多线程编程最怕啥?竞态条件!多个线程抢着改一个变量,结果乱套了…但用with语句管理线程锁,自动加锁解锁,线程安全so easy!
举个栗子:多线程计数器
import threading
lock = threading.Lock() # 创建锁
shared_counter = 0 # 共享变量
def worker():
global shared_counter
with lock: # 自动加锁,用完自动解锁
shared_counter += 1 # 线程安全的自增
# 创建5个线程并发执行
threads = [threading.Thread(target=worker) for _ in range(5)]
for t in threads:
t.start()
for t in threads:
t.join()
print(f"最终结果:{shared_counter}") # 输出5,完美!对比一下:加锁vs不加锁
不加锁时,5个线程可能同时改shared_counter,结果可能是3或4;加了with lock后,结果一定是5! 看下图效果:
(左侧代码加锁,右侧记事本显示结果:加锁后5行整齐输出,不加锁则混乱)
高级玩法:自定义上下文管理器,想管啥就管啥!
with语句不止能用内置对象,还能自己定义"管家"!比如写个计时器,自动统计代码块运行时间:
import time
class Timer:
def __enter__(self):
self.start = time.time() # 开门:记录开始时间
return self
def __exit__(self, *args):
print(f"耗时:{time.time()-self.start:.2f}秒") # 关门:计算耗时
# 用起来!
with Timer():
time.sleep(1.5) # 模拟耗时操作
# 输出:耗时:1.50秒甚至可以用contextlib装饰器简化:
from contextlib import contextmanager
@contextmanager
def temp_dir():
path = tempfile.mkdtemp() # 创建临时目录
try:
yield path # 返回目录路径
finally:
shutil.rmtree(path) # 自动删除目录
with temp_dir() as tmp:
print(f"使用临时目录:{tmp}") # 用完自动删,干净!避坑指南:这些错误别踩!
- AttributeError: enter
→ 原因:对象没实现上下文协议(没写__enter__和__exit__) → 解决:检查是否用错对象,比如with 普通变量:就会报错 - 嵌套with块
→ 不推荐:with A(): with B(): ...
→ 推荐:with A() as a, B() as b: 一行搞定多资源 - 异步同步混用
→ 错误:在同步函数里用async with
→ 解决:async with只能在async def函数里配合await使用
with语句,Pythoner必备"安全符"
不管是文件、数据库还是多线程,with语句都能帮你自动管理资源,既减少代码量,又避免资源泄漏,简直是"一举两得"!记住:写Python时,遇到需要"打开-关闭"的资源,先想想能不能用with语句,优雅又安全~
最后送大家一句口诀:"with一出,资源无忧;自动管理,bug溜走!"
参考来源:
CSDN博客:Python with语句深度实战指南Python官方文档:Context Managers
相关推荐
- win7 旗舰版和专业版哪个好(win7旗舰版专业版区别)
-
旗舰版比专业版功能要全,但是这多出的功能一般也用不上。至于是否影响电脑运行速度?同样的电脑配置,系统功能多的和功能少运行速度快,肯定是功能少点速度提升,但感觉上没什么区别。和政府机关内一般是和企业版本...
- 系统门窗一线品牌有哪些(中国系统门窗一线品牌有哪些)
-
轩尼斯门窗轩尼斯门窗创立于2004年,总部植根于佛山,生产基地在广东江苏共有4个,分别在凤岗厂区、盐城厂区、官窑厂区和范湖厂区;轩尼斯门窗在创建伊始就自主研发出了首款断桥铝门窗,产品技术专利超100项...
- 笔记本显卡天梯图排行榜(笔记本显卡天梯图最新版)
-
1、华硕ASUSROG-STRIX-RTX3080-O10G-GAMING1440-1935MHz华硕RTX3080被称为猛禽显卡,它将ROGSTRIX的轴流风扇进一步升级,扇叶数量也进行了...
- 惠普打印机售后维修官网(惠普打印机官方维修点查询)
-
是位于唐河县城区的惠普授权服务中心。这家维修点提供惠普打印机的售后服务和维修,包括故障排查、维修、更换零部件等。他们的技术人员都经过惠普的专业培训,能够提供高质量的服务。如果您的惠普打印机出现了问题,...
- 万能网卡驱动下载win11(万能网卡驱动windows7版2018最新版)
-
在windows11系统中点击桌面下方的开始图标,打开设置页面鼠标点击选择设备管理器选项。找到其中的网络适配器功能。右键选择显卡,点击卸载设备按钮等待卸载完成后重新安装驱动并重启计算机设备即可。想要修...
- cpu总是100使用率怎么回事呢
-
CPU占用率100%可能有多种原因。以下是一些可能的原因:驱动没有经过认证,这可能导致CPU资源占用100%。杀毒软件可能会占用大量的CPU资源,因为它们需要实时监控网页、邮件、个人隐私等功能。病毒或...
- 苹果6怎么升级系统版本(苹果6怎么升级系统版本最高能到多少)
-
要是喜欢自己动手可以自己去官网上下载,新系统在更新,他只是不支持自动更新的。如果不想自己动手,可以去拼多多或者是淘宝里面找一找,多的是那种帮你刷新系统的。也就是一点点钱的事情。现在选择很多活人不会被尿...
- windows2003镜像32位下载(win2003系统镜像)
-
虚拟光驱装系统,(win7,xp通用)具体步骤一、将从网上下载的win7旗舰版ISO系统文件存放到D盘。二、从网上下载虚拟光驱,打开安装后在任务栏右通知区显示“虚拟DAEMON管理器”图标,在我的电脑...
- win10电脑自动更新怎么关闭(win10电脑怎么关闭自动更新系统)
-
win10老推送win11打开的方法步骤如下,1,首先,打开设置,点击更新和安全2,打开后,点击windows预览体验计划3,打开后,点击开始4,然后按流程进行注册5,注册完成后,点击选择帐户6,然后...
- window7下载steam(window7下载一键重装如何恢复网络)
-
回答如下:要在Windows7上下载Steam,您可以按照以下步骤操作:1.打开您的浏览器,访问Steam官网(https://store.steampowered.com)。2.点击页面右上角...
- 系统还原没有还原点怎么办(系统还原点不动怎么办)
-
如果电脑没有创建还原点,就不能使用系统还原来回到之前的状态。但是,可以尝试使用其他备份工具或软件来恢复数据或重建系统。比如,可以使用第三方备份软件来备份重要文件和数据。如果是系统出现问题,可以尝试重新...
- 一周热门
-
-
飞牛OS入门安装遇到问题,如何解决?
-
如何在 iPhone 和 Android 上恢复已删除的抖音消息
-
Boost高性能并发无锁队列指南:boost::lockfree::queue
-
大模型手册: 保姆级用CherryStudio知识库
-
用什么工具在Win中查看8G大的log文件?
-
如何在 Windows 10 或 11 上通过命令行安装 Node.js 和 NPM
-
威联通NAS安装阿里云盘WebDAV服务并添加到Infuse
-
Trae IDE 如何与 GitHub 无缝对接?
-
idea插件之maven search(工欲善其事,必先利其器)
-
如何修改图片拍摄日期?快速修改图片拍摄日期的6种方法
-
- 最近发表
- 标签列表
-
- python判断字典是否为空 (50)
- crontab每周一执行 (48)
- aes和des区别 (43)
- bash脚本和shell脚本的区别 (35)
- canvas库 (33)
- dataframe筛选满足条件的行 (35)
- gitlab日志 (33)
- lua xpcall (36)
- blob转json (33)
- python判断是否在列表中 (34)
- python html转pdf (36)
- 安装指定版本npm (37)
- idea搜索jar包内容 (33)
- css鼠标悬停出现隐藏的文字 (34)
- linux nacos启动命令 (33)
- gitlab 日志 (36)
- adb pull (37)
- python判断元素在不在列表里 (34)
- python 字典删除元素 (34)
- vscode切换git分支 (35)
- python bytes转16进制 (35)
- grep前后几行 (34)
- hashmap转list (35)
- c++ 字符串查找 (35)
- mysql刷新权限 (34)
