With关键字的使用 | 手把手教你入门Python之七十八

简介: with语句实质上是⼀个上下⽂管理器,很多需要手动关闭的连接,比如说,文件连接,socket连接,数据库的连接都能使用with关键字来自动关闭连接。

上一篇:面向对象案例练习 | 手把手教你入门Python之七十七
下一篇:自定义异常 | 手把手教你入门Python之七十九

本文来自于千锋教育在阿里云开发者社区学习中心上线课程《Python入门2020最新大课》,主讲人姜伟。

With关键字的使用

对于系统资源如⽂件、数据库连接、socket ⽽⾔,应⽤程序打开这些资源并执⾏完业务逻辑之后,必须做的⼀件事就是要关闭(断开)该资源。

⽐如 Python 程序打开⼀个⽂件,往⽂件中写内容,写完之后,就要关闭该⽂件,否则会出现什么情况呢?极端情况下会出现 "Too many open files" 的错误,因为系统允许你打开的最⼤⽂件数量是有限的。

同样,对于数据库,如果连接数过多⽽没有及时关闭的话,就可能会出现 "Can not connect to MySQL server Too many connections",因为数据库连接是⼀种⾮常昂贵的资源,不可能⽆限制的被创建。

来看看如何正确关闭⼀个⽂件。
普通版:

def m1():
    f = open("output.txt", "w")
    f.write("python之禅")
    f.close()

这样写有⼀个潜在的问题,如果在调⽤ write 的过程中,出现了异常进⽽导致后续代码⽆法继续执⾏,close⽅法⽆法被正常调⽤,因此资源就会⼀直被该程序占⽤者释放。那么该如何改进代码呢?

进阶版:

def m2():
    f = open("output.txt", "w")
    try:
        f.write("python之禅")
    except IOError:
        print("oops error")
    finally:
        f.close()

改良版本的程序是对可能发⽣异常的代码处进⾏ try 捕获,使⽤ try/finally 语句,该语句表示如果在 try 代码块中程序出现了异常,后续代码就不再执⾏,⽽直接跳转到 except 代码块。⽽⽆论如何,finally 块的代码最终都会被执⾏。因此,只要把 close 放在 finally 代码中,⽂件就⼀定会关闭。

⾼级版:

def m3():
    with open("output.txt", "r") as f:
        f.write("Python之禅")  # 不需要再手动去关闭文件
try:
    with open('01-练习.py', 'r')as file:
        file.read() # 不需要再手动的关闭文件
except FileNotFoundError:
    print('文件未找到')

⼀种更加简洁、优雅的⽅式就是⽤ with 关键字。open ⽅法的返回值赋值给变量 f,当离开 with 代码块的时候,系统会⾃动调⽤ f.close() ⽅法, with 的作⽤和使⽤ try/finally 语句是⼀样的。

上下文管理器

with语句实质上是⼀个上下⽂管理器,很多需要手动关闭的连接,比如说,文件连接,socket连接,数据库的连接都能使用with关键字来自动关闭连接。
with语句后的对象都会有 __enter__()__exit__() ⽅法。
在进⼊到上下⽂时,会⾃动调⽤ __enter__() ⽅法,程序正常执⾏完成,或者出现异常中断的时候,都会调⽤ __exit__() ⽅法。

class MyContext(object):
    def __init__(self, name, age):
        self.name = name
        self.age = age
        
    def __enter__(self):
        print('调⽤了enter⽅法')
        return self
        
    def test(self):
        1 / 0
        print(self.name + '调⽤了test⽅法')
        
    def __exit__(self, exc_type, exc_val, exc_tb):
        print('调⽤了exit⽅法')
        print(exc_type, exc_val, exc_tb)
        
with MyContext('zhangsan', 18) as context:
    context.test()
class Demo(object):
    def __init__(self):
        print('__enter__方法被执行了')
        return self

    def __exit__(self, exc_type, exc_val, exc_tb):
        print('__exit__方法被调用了')


def create_obj():
    x = Demo()
    return x

# y = create_obj()
# d = y.__enter__()


with create_obj() as d:  # as 变量名
    # 变量  d 不是 create_obj的返回结果
    # 它是创建的对象 y 调用 __enter__ 之后的返回结果
    print(d)

配套视频

相关文章
|
26天前
|
存储 数据采集 人工智能
Python编程入门:从零基础到实战应用
本文是一篇面向初学者的Python编程教程,旨在帮助读者从零开始学习Python编程语言。文章首先介绍了Python的基本概念和特点,然后通过一个简单的例子展示了如何编写Python代码。接下来,文章详细介绍了Python的数据类型、变量、运算符、控制结构、函数等基本语法知识。最后,文章通过一个实战项目——制作一个简单的计算器程序,帮助读者巩固所学知识并提高编程技能。
|
12天前
|
C语言 Python
[oeasy]python054_python有哪些关键字_keyword_list_列表_reserved_words
本文介绍了Python的关键字列表及其使用规则。通过回顾`hello world`示例,解释了Python中的标识符命名规则,并探讨了关键字如`if`、`for`、`in`等不能作为变量名的原因。最后,通过`import keyword`和`print(keyword.kwlist)`展示了Python的所有关键字,并总结了关键字不能用作标识符的规则。
26 9
|
27天前
|
IDE 程序员 开发工具
Python编程入门:打造你的第一个程序
迈出编程的第一步,就像在未知的海洋中航行。本文是你启航的指南针,带你了解Python这门语言的魅力所在,并手把手教你构建第一个属于自己的程序。从安装环境到编写代码,我们将一步步走过这段旅程。准备好了吗?让我们开始吧!
|
26天前
|
测试技术 开发者 Python
探索Python中的装饰器:从入门到实践
装饰器,在Python中是一块强大的语法糖,它允许我们在不修改原函数代码的情况下增加额外的功能。本文将通过简单易懂的语言和实例,带你一步步了解装饰器的基本概念、使用方法以及如何自定义装饰器。我们还将探讨装饰器在实战中的应用,让你能够在实际编程中灵活运用这一技术。
38 7
|
28天前
|
开发者 Python
Python中的装饰器:从入门到实践
本文将深入探讨Python的装饰器,这一强大工具允许开发者在不修改现有函数代码的情况下增加额外的功能。我们将通过实例学习如何创建和应用装饰器,并探索它们背后的原理和高级用法。
42 5
|
27天前
|
机器学习/深度学习 人工智能 算法
深度学习入门:用Python构建你的第一个神经网络
在人工智能的海洋中,深度学习是那艘能够带你远航的船。本文将作为你的航标,引导你搭建第一个神经网络模型,让你领略深度学习的魅力。通过简单直观的语言和实例,我们将一起探索隐藏在数据背后的模式,体验从零开始创造智能系统的快感。准备好了吗?让我们启航吧!
69 3
|
1月前
|
Python
Python编程入门:从零开始的代码旅程
本文是一篇针对Python编程初学者的入门指南,将介绍Python的基本语法、数据类型、控制结构以及函数等概念。文章旨在帮助读者快速掌握Python编程的基础知识,并能够编写简单的Python程序。通过本文的学习,读者将能够理解Python代码的基本结构和逻辑,为进一步深入学习打下坚实的基础。
|
存储 数据库 Python
Python - with 语句
Python - with 语句
127 0
|
数据库连接 数据库 Python
Python中的With语句
在Python中,您需要通过打开文件来访问文件。您可以使用 open()函数来实现。Open 返回一个文件对象,该文件对象具有用于获取有关已打开文件的信息和对其进行操作的方法和属性。
164 0
|
C++ Python 编译器
为什么Python没有属性赋值的“with”语句?
Python有一个 'with' 语句,它封装了块的执行,在块的入口和出口调用代码。有些语言的结构是这样的: a = 1 # equivalent to obj.a = 1 total = total + 1 # obj.total = obj.total + 1 在Python中,这样的结构是不明确的。