Python文件操作

94 阅读5分钟

文件操作

文本文件操作

文件操作主要分两大部分:打开文件、读写文件

打开文件

 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