目录
文件和异常
1. 从文件中读取数据
(1)读取整个文件
with open('11.txt') as file_object: #函数open()打开文件
contents = file_object.read() #读取文件的内容
print(contents)
关键字 with: 在不再需要访问文件后将其关闭。通过with可让Python去确定:你只管打开文件,并在需要时使用它,Python自会在合适的时候自动将其关闭。
函数 open(): 要以任何方式使用文件,都得先打开文件,这样才能访问它。
read(): 读取这个文件的全部内容
(2)文件路径
相对文件路径: 该位置是相对于当前运行的程序所在目录的。
绝对文件路径: 该文件与当前程序不在同一文件夹下,则需要提供完整的文件路径。
file_path='D:\python\other_files\text_files\11.txt'
with open(file_path) as file_object:
注意:在 Windows 系统中,在文件路径中使用反斜杠(\),而在 Linux 和 OS X 中用斜杠(/)。
(3)逐行读取
要以每次一行的方式检查文件,可对文件对象使用 for 循环。
filename = '11.txt'
with open(filename) as file_object:
for line in file_object:
print(line.rstrip()) #rstrip()消除多余的空白行
(4)创建一个包含文件各行内容的列表
使用关键字 with 时,open() 返回的文件对象只在 with 代码块内可用。 如果要在 with 代码块外访问文件的内容,可在 with 代码块内将文件的各行存储在一个列表中,并在 with 代码块外使用该列表。
filename = '11.txt'
with open(filename) as file_object:
lines = file_object.readlines() #readlines()从文件中读取每一行,并将其存储在一个列表中
#在with代码块外打印内容
for line in lines:
print(line.rstrip()) #rstrip()消除多余的空白行
(5)使用文件的内容
filename = '11.txt'
with open(filename) as file_object:
lines = file_object.readlines()
pi = '' #变量pi用于存储圆周率的值
for line in lines:
pi += line.strip() #strip()消除左右的空格
print(pi) #输出3.141592653589793238462643383279
print(len(pi)) #输出32
注意:读取文本文件时,Python 将其中的所有文本都解读为字符串。如果你读取的是数字,并要将其作为数值使用,就必须使用函数 int() 将其转换为整数,或使用函数 float() 将其转换为浮点数。
2. 写入文件
保存数据的最简单的方式之一是将其写入到文件中。通过将输出写入文件,即便关闭包含程序输出的终端窗口,这些输出也依然存在。
(1)写入空文件
filename = 'program.txt'
#open()提供两个参数:要打开的文件的名称、'w'写入模式
with open(filename, 'w') as file_object:
file_object.write("Hello, Aiolei!") #write()将一个字符串写入文件
打开文件时,可指定读取模式:读取模式 (‘r’)、写入模式 (‘w’)、附加模式(‘a’)或让你能够读取和写入文件的模式(‘r+’)。如果你省略了模式实参,Python 将以默认的只读模式打开文件。
如果你要写入的文件不存在,函数 open() 将自动创建它。然而,以写入模式打开文件时千万要小心,因为如果指定的文件已经存在,Python 将在返回文件对象前清空该文件。
注意:Python 只能将字符串写入文本文件。要将数值数据存储到文本文件中,必须先使用函数 str() 将其转换为字符串格式。
(2)写入多行
函数 write() 不会在你写入的文本末尾添加换行符,如果写入多行需要自己加 ’\n’
filename = 'program.txt'
with open(filename, 'w') as file_object:
file_object.write("Hello, Aiolei!\n")
file_object.write("2020")
(3)附加到文件
如果你要给文件添加内容,而不是覆盖原有的内容,可以附加模式打开文件。以附加模式打开文件时,Python 不会在返回文件对象前清空文件,而你写入到文件的行都将添加到文件末尾。
filename = 'program.txt'
with open(filename, 'a') as file_object: #'a'附加模式
file_object.write("Today is a good day!\n")
file_object.write("Happy\n")
我们会发现:文件原来的内容还在,后两排是刚添加的内容,没有覆盖文件原来的内容。
3. 异常
每当发生让 Python 不知所措的错误时,它都会创建一个异常对象。如果你编写了处理该异常的代码,程序将继续运行;如果你未对异常进行处理,程序将停止,并显示一个 traceback,其中包含有关异常的报告。
异常是使用 try-except 代码块处理的。try-except 代码块让 Python 执行指定的操作,同时告诉 Python 发生异常时怎么办。使用了 try-except 代码块时,即便出现异常,程序也将继续运行:显示你编写的友好的错误消息,而不是令用户迷惑的 traceback。
(1)处理 ZeroDivisionError 异常
如果 try 代码块中的代码运行起来没有问题,Python 将跳过 except 代码块;如果 try 代码块中的代码导致了错误,Python 将查找这样的 except 代码块,并运行其中的代码。
try:
print(6/0)
except ZeroDivisionError:
print("You can't divide by 0!")
如果 try-except 代码块后面还有其他代码,程序将接着运行。
(2)try-except-else 代码块
try-except-else 代码块的工作原理大致如下:Python 尝试执行 try 代码块中的代码;只有可能引发异常的代码才需要放在 try 语句中。有时候,有一些仅在 try 代码块成功执行时才需要运行的代码;这些代码应放在 else 代码块中。
print("Give me two numbers, and I'll divide them.")
print("Enter 'q' to quit.")
while True:
first_number = input("\nFirst number: ")
if first_number == 'q':
break
second_number = input("Second number: ")
if second_number == 'q':
break
try:
answer = int(first_number) / int(second_number)
except ZeroDivisionError:
print("You can't divide by 0!")
else: #依赖于try 代码块成功执行的代码都应放到else代码块中
print(answer)
(3)分析文本
filename = 'program.txt'
try:
with open(filename) as file_object:
contents = file_object.read()
except FileNotFoundError:
msg = "Sorry, the file " + filename + " does not exist."
print(msg)
else:
#split() 以空格为分隔符将字符串分拆成多个部分,并将这些部分都存储到一个列表中
words = contents.split()
num_words = len(words) #计算文件大致包含多少个单词
print("The file " + filename + " has about " + str(num_words) + " words.")
(4)使用多个文件
def count_words(filename): #计算单词个数函数
try:
with open(filename) as file_object:
contents = file_object.read()
except FileNotFoundError:
msg = "Sorry, the file " + filename + " does not exist."
print(msg)
else:
#split() 以空格为分隔符将字符串分拆成多个部分,并将这些部分都存储到一个列表中
words = contents.split()
num_words = len(words) #计算文件大致包含多少个单词
print("The file " + filename + " has about " + str(num_words) + " words.")
filenames = ['program.txt', '11.txt', 'result.txt', 'data.txt']
for filename in filenames:
count_words(filename)
(5)发生异常时一声不吭
有时候你希望程序在发生异常时一声不吭,就像什么都没有发生一样继续运行。 要让程序在失败时一声不吭,在 except 代码块中明确地告诉 Python 什么都不要做,用 pass 语句。
try:
......
except FileNotFoundError:
pass #什么也不做
else:
......
4. 存储数据
模块 json 能够将简单的 Python 数据结构转储到文件中,并在程序再次运行时加载该文件中的数据。还可以使用 json 在 Python 程序之间分享数据。更重要的是,JSON 数据格式并非 Python 专用的,这能够将以 JSON 格式存储的数据与使用其他编程语言的人分享。
(1)使用 json.dump() 和 json.load()
json.dump():存储数字
函数 json.dump() 接受两个实参:要存储的数据以及可用于存储数据的文件对象。
import json #导入模块json
numbers = [2,3,4,5,7,8]
filename = 'numbers.json' #使用文件扩展名.json来指出文件存储的数据为JSON格式
with open(filename, 'w') as f_obj:
json.dump(numbers, f_obj)
json.load():将列表读取到内存中
import json #导入模块json
filename = 'numbers.json'
with open(filename) as f_obj:
numbers = json.load(f_obj)
print(numbers)
(2)保存和读取用户生成的数据
import json
#若以前存储了用户名,就加载它;否则就提示用户输入用户名并存储它
filename = 'username.json'
try:
with open(filename) as f_obj:
username = json.load(f_obj) #加载
except FileNotFoundError:
username = input("What is your name?") #输入用户名
with open(filename, 'w') as f_obj:
json.dump(username, f_obj) #存储用户名
else:
print("Welcome back, " + username + "!")
(3)重构
将代码划分为一系列完成具体工作的函数,这样的过程被称为重构。重构让代码更清晰、更易于理解、更容易扩展。
import json
def get_username(): #如果存储了用户名就获取它
filename = 'username.json'
try:
with open(filename) as f_obj:
username = json.load(f_obj) #加载
except FileNotFoundError:
return None
else:
return username
def greet(): #问候用户
username = get_username()
if username:
print("Welcome back, " + username + "!")
else:
username = input("What is your name?") #输入用户名
filename = 'username.json'
with open(filename, 'w') as f_obj:
json.dump(username, f_obj) #存储用户名
greet()