问 2 个关于 Python 的简单问题。困扰我很久

2017-10-17 15:32:11 +08:00
 northisland

操作 json 文件相关的。

我要读文件进来,

第一种写法

json_obj=None
with open(path) as f:
  json_obj=json.loads(f.read())

第二种写法

json_obj=json.loads(open(path).read())

问题 1:

第一种写法里,第一行能否删掉?

换言之,就是 with 里的变量 json_obj 能否自动 with 在这个过程后存在?

问题 2:

第二种写法,是否正确?临时变量 open(path)在结束时,是否释放了文件句柄?

提前谢谢各位帮助答疑解惑

4560 次点击
所在节点    Python
25 条回复
WhoAmI233
2017-10-17 15:40:17 +08:00
第一种写法第一行不能删,第二种写法正确的,open(path)在结束时肯定释放了。
whx20202
2017-10-17 15:46:56 +08:00
第一种为什么不能删啊
WhoAmI233
2017-10-17 15:48:10 +08:00
@whx20202 删了,那你定义这个变量有何作用啊?思考一下变量的作用域
mgna17
2017-10-17 15:49:09 +08:00
第一种:第一行可以删,with 中用到的变量在退出 with 之后可以继续使用

第二种:python3 的文件对象有个 __del__ 函数,应该会自动释放的。python2 没有这个 __del__ 函数,不清楚
XIVN1987
2017-10-17 15:51:37 +08:00
json_obj=None 可以删掉

第二个问题文件应该不会被自动 close(),我试了在 ipython 命令行中执行 open('test.txt'),然后去删除 test.txt ,,系统报错说 test.txt 文件已经被打开了,,不让删除,,所以应该在你的软件执行完退出后文件句柄才能被释放
ChristopherWu
2017-10-17 15:54:07 +08:00
其实,第一行可以删。
不过加了 json_obj=None, 是保证代码即便出现错误时,json_obj 也会有初始值。
Yinz
2017-10-17 15:54:28 +08:00
第一种,可以删掉第一行。
第二种,以前写爬虫的时候试过这么写然后跑光了句柄,程序卡死,但是不记得当时是用的 PY2 还是 PY3 了
XIVN1987
2017-10-17 15:57:05 +08:00
更正下,,刚才说错了

如果是在函数内打开的文件,比如
def test(): open('test.txt')

在命令行中执行 test()

之后可以立即在文件浏览器中删掉 test.txt 这个文件,,

说明函数退出后文件句柄就被释放了
justou
2017-10-17 15:57:24 +08:00
with 是上下文管理器语法, 它保证它经手的资源在退出 with 块后被正确释放, 即使发生了异常.

如果没有发生异常, 第一种情形可以不用 json_obj=None, 退出 with 块后 json_obj 继续存在;

如果发生了异常, 第一种情形 json_obj 未定义, 但是文件会被正确关闭. 但是第二种情形文件不会被正确关闭(json_obj 同样未定义), 直到程序结束.
BBCCBB
2017-10-17 16:00:14 +08:00
第一行那个可以删, Python 作用域和其他的不太同, 和 js 的很像,

我也不知道第二种对不对, 也不想尝试... 推荐使用 with, 显示释放
tamlok
2017-10-17 16:04:42 +08:00
标题写法应该是“两”,不是“ 2 ”😁
ChristopherWu
2017-10-17 16:06:39 +08:00
第二种严格来说可以不关掉。
可以自己测试一下:
```
import json
# json_obj=None
json_obj=json.loads(open("/Users/yonghaohu/a").read())
open("/Users/yonghaohu/te/a")
while(1):
pass
```

运行程序后,lsof -p 该程序, 查看打开的文件, 是没有打开任何 a 文件的。

如果你在 while 里, 不断调用 open,lsof -p 则会看到该文件。

Python 85861 yonghaohu 3r REG 1,4 7 108089581 /Users/yonghaohu/te/a
ChristopherWu
2017-10-17 16:18:49 +08:00
接上,gc 删掉打开的文件对象时, 会自动删掉掉句柄。但是这个过程是不可控的,最好还是自己手动 close
SuperMild
2017-10-17 16:21:02 +08:00
不懂 python,但是,不是自己实际测试一下就能确定吗?
xlui
2017-10-17 16:27:40 +08:00
第一行可以删,with 块跟函数和模块不同,没有定义新作用域。
第二个临时变量在语句结束时丢失引用被垃圾回收,CPython 和 IPython 会刷新并且关闭文件。而 PyPy 和 JPython 采用不同的垃圾回收机制所以会有不同的情况出现。
虽然 CPython/IPython 会在文件对象丢失引用时刷新并关闭,还是推荐使用 with 语句。
这两个问题我记得 流畅的 Python ( fluent python ) 上都有简单的提到,这本书真的太好了。
enenaaa
2017-10-17 16:38:23 +08:00
@justou 第 2 种写法里,如果发生异常。 并不是直到程序结束才释放, 而是直到该异常被处理后才释放。
刚刚试过。
dongxiaozhuo
2017-10-17 16:41:01 +08:00
py2 的话,第一种情况,第一行可以删除。不过为啥不用 json.load 方法,直接读取文件对象,不是更好?
strahe
2017-10-17 17:07:17 +08:00
一楼瞎说, 1, 第一行可以删, 而且建议删,
第二种不建议这么写, open 以后要主动 close
stebest
2017-10-17 17:10:24 +08:00
同认为第一行可以删,Python 不需要预先定义。
oroge
2017-10-17 17:23:00 +08:00
https://stackoverflow.com/questions/38660609/when-how-does-an-anonymous-file-object-close

上面说“测试一下不就行了”的人不怕遇到 undefined behavior ?

这是一个专为移动设备优化的页面(即为了让你能够在 Google 搜索结果里秒开这个页面),如果你希望参与 V2EX 社区的讨论,你可以继续到 V2EX 上打开本讨论主题的完整版本。

https://www.v2ex.com/t/398359

V2EX 是创意工作者们的社区,是一个分享自己正在做的有趣事物、交流想法,可以遇见新朋友甚至新机会的地方。

V2EX is a community of developers, designers and creative people.

© 2021 V2EX