打开文件的模式(二)

对于非文本文件,我们只能使用b模式,"b"表示以字节的方式操作
(而所有文件也都是以字节的形式存储的,使用这种模式无需考虑文本文件的字符编码、图片文件的jgp格式、视频文件的avi格式)

rb:    以字节方式读文件
wb:  以字节方式写文件
ab:   以字节方式追加文件

注:以b方式打开时,读取到的内容是字节类型,写入时也需要提供字节类型,所以不能指定编码

1. rb

#错误举例
f = open ('test1.py', 'rb', encoding = 'utf -8')
data = f.read()
print(data)
f.close() #执行结果:
报错
f = open ('test1.py', 'rb', encoding = 'utf-8')
ValueError: binary mode doesn't take an encoding argument (以b方式打开时,因为读取到的内容是字节类型,所以不能指定编码方式,否则会报错)
#正确
f = open ('test1.py', 'rb')
data = f.read()
print(data)
f.close() #执行结果:
b"'hello'\r\n'\xe5\xb0\x8f\xe7\x81\xab\xe9\x94\x85'\r\n'666'"

test1.py中的内容如下:

'hello'
'小火锅'
'666'

分析该程序执行结果:

1. python在windows操作系统下,换行符为在windows操作系统下,换行符为 \r\n
2. ‘字符串’-------encode-------》bytes
     bytes---------decode-------》‘字符串’

所以,我们想让执行结果为字符串,可在print时做decode处理

f = open ('test1.py', 'rb')
data = f.read()
print(data.decode('utf-8'))
f.close() #执行结果:
'hello'
'小火锅'
''

2. wb

#错误举例

f = open ('test1.py', 'wb')
f.write('hello')
f.close() #执行结果: TypeError: a bytes-like object is required, not 'str' (以b方式写入时需要提供字节类型,所以不能写入字符串类型)
#正确

f = open ('test1.py', 'wb')
f.write(bytes('hello\n小火锅', encoding = 'utf-8' ))
f.close()

3. ab

#举例
f = open ('test1.py', 'ab')
f.write(bytes('hello\n小火锅\n', encoding = 'utf-8' ))
f.close()

关于文件的其他操作介绍

1. .encoding 读取文件打开时后的编码方式(即open时指定的编码方式)

#举例

f = open ('test1.py', 'w', encoding = 'GB2312')
f.close()
print(f.encoding) #执行结果: GB2312

2. .closed 确定文件是否关闭

#举例

f = open ('test1.py', 'w', encoding = 'utf-8')
f.close()
print(f.closed) #执行结果: True

3. .flush 刷新操作(将文件内容从内存刷到硬盘)

4. .tell 读取光标所在位置

补充:文件内光标移动

一: read(3):

   1. 文件打开方式为文本模式时,代表读取3个字符

   2. 文件打开方式为b模式时,代表读取3个字节

二: 其余的文件内光标移动都是以字节为单位如seek,tell,truncate

test1中的内容
aaa
小火锅
666

#举例
f = open ('test1.py', 'r', encoding = 'utf-8')
print(f.tell())
f.readline()
print(f.tell())
f.readline()
print(f.tell())
f.close() #执行结果:
0
5
16
#结果分析:
第一个tell判断光标位置时候,光标在文件开头,即在位置0
读取了一行之后,光标跑到文件第二行开头,但是在Windows操作系统,换行符为 \r\n,占两个字节,而且tell光标移动以字节为单位,所以光标位置为 5
读取了第二行之后,光标跑到了文件第三行开头,光标经历了三个汉字和换行符,在编码方式为 utf-8 时,走过了3*3+2=11个字节,所以光标位置为 16

5. .seek 控制光标的移动

#举例
f = open ('test1.py', 'r', encoding = 'utf-8')
f.seek(2) #光标从默认位置0开始,往后移动两个字节
data = f.read()
print(data) #打印光标后的内容
f.close() #执行结果:
a
小火锅
666

如果,我将test1.py的内容改为

小火锅
aaa
666

再次执行上述程序

#报错:UnicodeDecodeError: 'utf-8' codec can't decode byte 0x8f in position 0: invalid start byte

原因:一个汉字在utf-8编码方式为3个字节,seek(2)移动两个字节,难道有神奇的功能将汉字劈开嘛,所以肯定会报错啊!

关于seek的一些补充:

seek有三种模式,分别为

  • 0模式 默认从0开始(不用指定)
  • 1模式 从上次相对位置开始
  • 2模式 从文件末尾开始seek(第一个参数需要为负数)
举例:
f = open ('test1.py', 'r', encoding = 'utf-8')
f.seek(3)
print(f.tell())
f.seek(9)
print(f.tell())
f.close() #执行结果
3
9
f = open ('test1.py', 'rb')
f.seek(3,1)
print(f.tell())
f.seek(10,1)
print(f.tell())
f.close() #执行结果
3
13
f = open ('test1.py', 'rb')
f.seek(-3,2) #seek 在模式 2 时,第一个参数为负数
print(f.tell())
print(f.read())
f.close() #执行结果
18
b'6\r\n'

任务:应用:打开一个日志文件,并且读取最新日志(核心:倒着读文件内容)

日志文件内容如下:
2018/11/20 aaa 上网听歌
2018/11/20 bbb 上网网购
2018/11/20 ccc 上网学习

#方法一
f = open('日志文件', 'rb')
data = f.readlines() #将日志文件的内容以列表形式读到内存中
print(data[-1].decode('utf-8')) #以切片方式读取列表中最后一个元素,即文件最后一行内容 #执行结果
2018/11/20 ccc 上网学习
#方法二
f = open('日志文件', 'rb')
for i in f: #文件循环方式
offs = -10 #设置初始偏移量
while True: #设置一个死循环来读取文件的最后一行内容,读取到break
f.seek(offs, 2) #seek模式2,光标倒着移动
data = f.readlines() #以列表形式读取光标后的内容
if len(data) > 1: #如果该列表长度大于1,说明光标移动到最后一行之前,最后一行内容已被读出,break,反则最后一行内容还未全部读出,将偏移量扩大,直到独处最后一行全部内容
  print('文件最后一行:%s'%(data[-1].decode('utf-8')))
  break
offs *= 2 #执行结果
文件最后一行:2018/11/20 ccc 上网学习

方法一看着简单,但是方法一需要将日志文件的内容一列表形式全部读到内存中,占用较多内存
方法二的思想就是我用最后一行内容,我倒着读,只关注我想得到的信息

6. .truncate 截取文件内容(实质为文件内容的改写,所以在 open 文件时,需要设置正确的打开文件的模式)

举例:

f = open('日志文件', 'r+',encoding = 'utf-8')
data = f.truncate(8)
print(data)

注意:

  • 打开方式不可以是r(报错,实质为文件的改写)
  • 打开方式不可以是w w+(这两个模式将内容全部删除,所以截取不到任何内容)

Python小白学习之路(二十)—【打开文件的模式二】【文件的其他操作】的更多相关文章

  1. Python小白学习之路(十九)—【文件操作步骤】【文件操作模式】

    一.文件操作步骤 step1:打开文件,得到文件句柄并赋值给一个变量step2:通过句柄对文件进行操作step3:关闭文件 举例: a = open('hello world', 'r', encod ...

  2. Python小白学习之路(十二)—【前向引用】【风湿理论】

    前向引用 风湿理论(函数即变量) 理论总是很抽象,我个人理解: 代码从上到下执行,一旦遇到定义的函数体,内存便为其开辟空间,并用该函数的名字作为一个标识但是该函数体内具体是什么内容,这个时候并不着急去 ...

  3. Python小白学习之路(十)—【函数】【函数返回值】【函数参数】

    写在前面: 昨天早睡之后,感觉今天已经恢复了百分之八十的样子 又是活力满满的小伙郭 今日份鸡汤: 我始终相信,在这个世界上,一定有另一个自己,在做着我不敢做的事,在过着我想过的生活.-------宫崎 ...

  4. Python小白学习之路(十五)—【map()函数】【filter()函数】【reduce()函数】

    一.map()函数 map()是 Python 内置的高阶函数 有两个参数,第一个是接收一个函数 f(匿名函数或者自定义函数都OK啦):第二个参数是一个 可迭代对象 功能是通过把函数 f 依次作用在 ...

  5. Python小白学习之路(十八)—【内置函数三】

    一.对象操作 help() 功能:返回目标对象的帮助信息 举例: print(help(input)) #执行结果 Help on built-in function input in module ...

  6. Python小白学习之路(十四)—【作用域】【匿名函数】【编程方法论】【高阶函数】

    吧啦吧啦内心戏 在没有具体学作用域之前,我在之前的学习笔记中就有提到 我开始以为是自己自创的词儿 没想到这个词早已经存在(手动捂脸) 真是个无知的小火锅(不知者无罪) 我发现自己最擅长做的事情,就是给 ...

  7. Python小白学习之路(十六)—【内置函数一】

    将68个内置函数按照其功能分为了10类,分别是: 数学运算(7个) abs()   divmod()  max()  min()  pow()  round()  sum() 类型转换(24个) bo ...

  8. FastAPI 学习之路(十八)表单与文件

    系列文章: FastAPI 学习之路(一)fastapi--高性能web开发框架 FastAPI 学习之路(二) FastAPI 学习之路(三) FastAPI 学习之路(四) FastAPI 学习之 ...

  9. Python小白学习之路(二十二)—【生成器】

    一.什么是生成器? 生成器可以理解成是一种数据类型,特殊地是生成器可以自动实现迭代器协议其他的数据类型需要调用自己内置的__iter__方法所以换种说法,生成器就是可迭代对象 !回忆:很重要的迭代器协 ...

随机推荐

  1. tensorflow的transpose

    从图中看出来perm=[1,0,2] 表示第一个维度和第二个维度进行交换. 默认的是[0,1,2]   所以perm=[1,0,2] 表示第一个维度和第二个维度进行交换.0,1,2表示index.

  2. cocos2d-js IOS接facebook插件

    当前测试版本:cocos2d-x 3.8.1       3.7也试用,之下的版本没测过,一般是路径改变,文件名称一般不会变 注:当前工程是通过控制台new的工程,不是cocosStudio创建的工程 ...

  3. 2018.10.16 NOIP模拟 长者(主席树+hash)

    传送门 考试的时候开始sb的以为需要可持久化trietrietrie树,发现建树时空都是O(n2)O(n^2)O(n2)的. 然后发现由于每次只从原来的字符串改一个字符. 因此直接主席树维护区间has ...

  4. hdu-2191(完全背包+二进制优化模板)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=2191 思路:完全背包模板 #include<iostream> #include<c ...

  5. ardunio 实现RS485通讯-下位机

    #include <SoftwareSerial.h> SoftwareSerial mySerial(,); byte ZERO=0x00; byte Addr=0x03; byte S ...

  6. 【Unity】1.2 HelloWorld--测试桌面和Android游戏能否正常运行

    分类:Unity.C#.VS2015 创建日期:2016-03-23 一.简介 这一节先搞一个最简单的Unity游戏,目的是为了验证Unity的桌面游戏开发环境和Android游戏开发环境是否有问题. ...

  7. python编码(三)

    python 有str object 和 unicode object 两种字符串,都可以存放字符的字节编码,但是他们是不同的type,这一点很重要,也是为什么会有encode(编码)和decode( ...

  8. (用了map) Registration system

    http://acm.hust.edu.cn/vjudge/contest/view.action?cid=93241#problem/C (654123) http://codeforces.com ...

  9. (并查集) Wireless Network --POJ --2236

    链接: http://poj.org/problem?id=2236 http://acm.hust.edu.cn/vjudge/contest/view.action?cid=82830#probl ...

  10. android public.xml 用法

    一.android的pulibc.xml文件 如果你用 apktoool 反编译过 apk 就知道,反编译后res/values 下有一个 public.xml 文件,内容如图   这个东西有什么用呢 ...