法一
思路:
打开文件后逐行读取同时记录指针位置,由于读取完一行后指针位于行尾,所以在读取完最后一行后需要将指针移到上一行末尾然后调用f.truncate()。为存储倒数第二行末尾的位置,我们需要一个1 × \times × 2的数组存储文件指针位置:
# 打开文件,由于只有二进制模式才支持移动指针操作,所以第二个参数要有'b'。
f = open('test.txt', 'rb+')
line = f.readline()
pos = [0, f.tell()]
# 读完最后一行后如果继续调用f.readline会返回 b'',也就是空行,
# 所以当line = b''时我们便可以确定文件已读完。
while line != b'':
line = f.readline()
temp = f.tell()
# 跳出循环时line的内容是 b'' 即此时已经读到了最后一行的下一行,
# 但由于它是空行所以指针的位置并不会变,但是如果继续更新数组pos
# 的内容会导致倒数第二行的位置信息被覆盖,所以加个判断。
if temp != pos[1]:
pos[0] = pos[1]
pos[1] = temp
# 得到了倒数第二行末尾的位置,我们就可以直接从文件起始位置移动到那里。
# f.seek第二个参数为0表示从文件起始位置移动。
# f.truncate没有参数时表示从文件开头截取到当前位置,剩余部分删除。
f.seek(pos[0], 0)
f.truncate()
f.close()
法二
思路:
直接从文件末尾向前移动足够长距离越过最后一行,然后读取指针后的全部内容,由于读取结果是列表,且每个元素就是一行内容,可以将除了最后一个元素以外的其他元素的长度求和,从而得到指针移动到倒数第二行末尾所需的移动距离:
# 打开文件。参数同法一。
f = open('test.txt', 'rb+')
# f.seek第一个参数是向前移动的字节数,须做好估计以保证能越过最后一行。注意是负值。
# f.seek第二个参数2表示从文件末尾移动。
f.seek(-50, 2)
content = f.readlines()
dist = 0
for i in content:
dist += len(i)
offset = dist - len(content[-1])
# 注意以上算出的不是从文件头到倒数第二行末尾的位置,所以要先将指针复位。
f.seek(-50, 2)
f.seek(offset, 1)
f.truncate()
f.close()
如果该文对你有帮助,请点个赞~