文件操作
文本文件操作
文件操作主要分两大部分:打开文件、读写文件
打开文件
fp = open(path,mode='r')
该函数可以路径为Path的文件,默认以读权限打开,并返回一个打开文件对象
打开权限
'''
- r: 读
- w: 写
- a: 追加
'''
单纯使用这三种方式打开文件,只拥有一种打开权限,要么读,要么写
如果希望在打开文件的时候,读写权限兼备,那么可以在权限后带一个 + 号
'''
- r+: 读写 不创建新文件,文件读写指针在开头
- w+: 读写 创建新文件,读写指针在开头;如果文件存在则会清空这个文件之前的内容
- a+: 读写 创建新文件,读写指针在末尾;不会清空该文件之前的内容
'''
- 注意:含有写权限在打开一个不存在文件时,该文件将被创建
关闭文件
fp.close()
该函数可以关闭文件对象,并且刷新缓冲区
读文件
content = fp.read()
读取文件放到字符串变量中,参数num为指定读取的字符数,默认为全读
1.txt文件其中内容
aaa
bbb
读取文件内容
>>> fp = open('1.txt','r+')
>>> mystr = fp.read(2)
>>> print(mystr)
aa
>>> fp.close()
fp.readline()
读取一行到一个字符串变量中(读到换行符(\r\n),包括行末的标识符,或者是文件结束的标识(EOF) )
>>> fp = open('1.txt','r+')
>>> mystr = fp.readline()
>>> print(mystr)
aaa
>>> fp.close()
# 输出的多余空行为文件中一行结尾的换行符
fp.readlines()
读取整个文件到字符串列表。
>>> fp = open('1.txt','r+')
>>> str_list = fp.readlines()
>>> print(str_list)
['aaa\n','bbb']
>>> fp.close()
- 注意:每次调用 readlines(size) 函数,会返回大约200MB的数据,而且所返回的必然都是完整的行数据,大多数情况下,返回的数据的字节数会稍微比 size 指定的值大一点(除最后一次调用readlines(size) 函数的时候)。通常情况下,解释器会自动将用户指定的 size的值调整成内部缓存大小的整数倍
写文件
fp.write(str)
在文件中写入字符串
fp.writelines(list_of_string)
把字符串列表写入文件
>>> fp = open('1.txt','w')
>>> fp.write('abc\n')
>>> fp.writelines(['bbb\n','ccc\n']) #写入字符串列表
>>> fp.close()
注意:写入文件字符串的方法不会自动的加上换行符,所以需要大家在写入时,手动写入换行符标志**\n**
此外,写入文件的内容我们并不能直接在磁盘文件看到,这是因为写入的内容暂时被保存在了缓存中;我们可以通过使用 fp.close()或fp.flush()函数来进行缓冲区刷新操作,使写入文件的内容直接保存在磁盘中
刷新缓冲区
fp.close()
# 关闭文件可以刷新缓冲区
fp.flush()
# 手动刷新缓冲区
读写指针
每次读或写操作之后我们会发现,我们下次的都操作都是在这一次之后,这是因为我们在打开一个文件的同时,内存中会维护一个读写指针用来标识我们访问文件的位置,一个文件对象读写操作共享同一根指针
>>> fp = open('1.txt','r')
>>> fp.readline()
aaa\n
>>> fp.readline()
bbb\n
#这里第二次读操作继续在第一次操作之后
修改读写指针位置
fp.seek(offset[,whence])
'''
offset: 偏移量
whence: 从何处偏移;0从文件开头,1从当前位置,2从文件末尾处
'''
注意:以a或a+ 的模式打开,每次进行写操作时,文件操作标记会自动返回到文件末尾,如果此时你想读文件开头部分内容,需要修改读写指针位置为fp.seek(0, 0)
文件操作实现验证系统
在某些情况下,文件是一个非常棒的数据库系统; 我们在使用文件作为数据库使用时,可以使效率更高更快 并且使用文件系统作为数据库,可以在我们应对一些轻量级难题时,节省开发成本
在文件作为数据库时,我们需要把数据的保存格式进行设计 用户名密码以 account:passwd\n 形式保存在文件中(注意行末有换行)
用户注册
fp = open('user.txt','a+') # 使用a+权限打开文件,避免将之前的内容所覆盖。
reg_time = 1
while True:
is_reg = False # 判断是否已经注册的标志位
account = input('请输入你要注册的帐号:')
fp.seek(0,0) # 注册帐号之前,首先要确定该帐号没有被注册过。
for buf in fp:
account_buf = buf.split(':')[0]
# 我们依次会读取出文件中的每一行数据,按照我们的格式把帐号取出来
if account_buf == account:
is_reg = True
break # 跳出循环
# 读到最后一行也没有相等,那么说明我们文件中没有这个用户,继续向下,注册密码
if not is_reg:
passwd1 = input('请输入你要注册的密码:')
passwd2 = input('请再次输入你要注册的密码:')
if passwd2 == passwd1:#两次密码校验
str = '%s:%s\n'%(account,passwd1)#文件中写入帐号密码
fp.write(str)
print('注册成功')
break
else:
print('密码不相同,请再次输入')
continue
else:
if reg_time <= 3:
print('已经被注册,这是你第%d次尝试注册,你还剩余%d次机会'%(reg_time,4-reg_time))
reg_time += 1
else:
print('机会用尽,再见!')
break#超过重新注册次数,退出程序
fp.close()#关闭文件
用户登陆
myfile = open('user.txt','r')#登录时候,只需要读取权限
logintimes = 0
islogin = 1 #该变量等于1的情况下你是没有登录的。
while True:
if islogin == 1:#重试登陆次数最大为3
if logintimes == 3:
print('登录超过次数,再见')
break#跳出主循环
else:
print('这是你第%d次登录,你还剩余%d次机会。'%(logintimes,3-logintimes))
account = input('请输入你的帐号:')
passwd = input('请输入你的密码:')
myfile.seek(0,0)#为了避免因为重复登录而我们的文件描述符已经到了后面,导致之前的无法判断
for buf in myfile:
account_buf = buf.split(':')[0]
passwd_buf = buf.split(':')[1][:-1]#去掉行尾的换行符,同样可以使用str.strip
if account_buf == account and passwd == passwd_buf:
print('登录成功:',account_buf)
islogin = 0#登录成功则修改标志位
break
# 但是这里的break只能跳出最内圈循环,想要结束全部,我们可以设置一个标记位;用来确定我们是否已经登录
logintimes += 1
continue
else:
break