With語句是什么?
有一些任務,可能事先需要設置,事后做清理工作。對於這種場景,Python的with語句提供了一種非常方便的處理方式。一個很好的例子是文件處理,你需要獲取一個文件句柄,從文件中讀取數據,然后關閉文件句柄。 如果不用with語句,代碼如下:
1
2
3
|
file
=
open
(
"/tmp/foo.txt"
)
data
=
file
.read()
file
.close()
|
1
2
3
4
5
|
file
=
open
(
"/tmp/foo.txt"
)
try
:
data
=
file
.read()
finally
:
file
.close()
|
1
2
|
with
open
(
"/tmp/foo.txt"
) as
file
:
data
=
file
.read()
|
with如何工作?
這看起來充滿魔法,但不僅僅是魔法,Python對with的處理還很聰明。基本思想是with所求值的對象必須有一個__enter__()方法,一個__exit__()方法。 緊跟with后面的語句被求值后,返回對象的__enter__()方法被調用,這個方法的返回值將被賦值給as后面的變量。當with后面的代碼塊全部被執行完之后,將調用前面返回對象的__exit__()方法。 下面例子可以具體說明with如何工作:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
|
#!/usr/bin/env python
# with_example01.py
class
Sample:
def
__enter__(
self
):
print
"In
__enter__()"
return
"Foo"
def
__exit__(
self
,
type
, value, trace):
print
"In
__exit__()"
def
get_sample():
return
Sample()
with get_sample() as sample:
print
"sample:"
, sample
|
1
2
3
4
|
bash
-
3.2
$ .
/
with_example01.py
In __enter__()
sample: Foo
In __exit__()
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
|
#!/usr/bin/env python
# with_example02.py
class
Sample:
def
__enter__(
self
):
return
self
def
__exit__(
self
,
type
, value, trace):
print
"type:"
,
type
print
"value:"
, value
print
"trace:"
, trace
def
do_something(
self
):
bar
=
1
/
0
return
bar
+
10
with Sample() as sample:
sample.do_something()
|
1
2
3
4
5
6
7
8
9
10
|
bash
-
3.2
$ .
/
with_example02.py
type
: <
type
'exceptions.ZeroDivisionError'
>
value: integer division
or
modulo by zero
trace: <traceback
object
at
0x1004a8128
>
Traceback (most recent call last):
File
"./with_example02.py"
, line
19
,
in
<module>
sample.do_something()
File
"./with_example02.py"
, line
15
,
in
do_something
bar
=
1
/
0
ZeroDivisionError: integer division
or
modulo by zero
|
(轉)