一般我们读取的文件和编写的Python
文件位于同一目录下,例如在当前目录下我们已经创建了要处理的文件test.txt
,里面包含的内容为:
Hello,world!
Hello,Python!
Hello,my brothers.
我们运行在同一目录下的Python
文件test.py
,代码如下:
with open('test.txt') as file_object:
contents = file_object.read()
print(contents)
程序运行结果:
Hello,world!
Hello,Python!
Hello,my brothers.
test.py
文件中的第一行代码中有函数open()
,用于打开文件,这是我们处理文件的第一步。函数open()
中的参数'test.txt'
就是要打开的文件。函数open()
返回的是打开文件的对象,第一行代码就是把文本文件test.txt
打开,并将其对象保存在file_object
变量中。
关键字with
的功能是在不再需要访问文件后自动将文件关闭。所以我们在这里只是open()
打开了文件,但是没有加入close()
代码关闭文件,因为Python
会在处理文件之后自动将文件关闭。
test.py
文件中的第二行代码是使用read()
方法读取文本文件test.txt
的全部内容,并将内容保存在字符串变量contents
中,然后通过print()
将结果都输出。
如果我们要处理的文本文件和Python
程序文件不在同一目录下,那么我们就要在open()
函数中传入文本文件的绝对路径。
我们也可以将文本文件中的数据进行逐行读取,我们要处理的文本文件还是上面的test.txt
,这时我们可以逐行将test.txt
的内容输出,代码如下:
with open('test.txt') as file_object:
for line in file_object:
print(line)
输出结果:
Hello,world!
Hello,Python!
Hello,my brothers.
我们会发现输出结果中每一行内容后面都多了一个空行,这时因为在文件中每一行的末尾都会有一个换行符,而每条print()
语句也会加上一个换行符,所以每行末尾都有两个换行符,所以输出之后就会多一个空行。
我们可以采取rstrip()
方法消除空行,代码如下:
with open('test.txt') as file_object:
for line in file_object:
print(line.rstrip())
这时输出的结果中就没有多余的空行了:
Hello,world!
Hello,Python!
Hello,my brothers.
在上文中,函数open()
返回的对象只在with
代码块内可用,但是我们想在with
代码块之外的位置使用,这就需要在with
代码块内创建一个包含文本文件test.txt
各行内容的列表。例如:
with open('test.txt') as file_object:
lines = file_object.readlines()
上述代码中readlines()
方法就是从文本文件test.txt
中依次读取每一行,并保存在lines
列表中。
要将信息写入文本文件中,我们依然用open()
方法,只不过除了将文本文件名当作参数传入函数open()
中去之外,还需要再传入写参数w
,例如下面这个Python
的程序test2.py
:
with open('test2.txt','w') as example:
example.write('Hello world!')
程序运行结果就是在test2.py
文件所在目录生成了一个名为test2.txt
的文本文件,文本文件中的内容为:
Hello world!
在这个例子中,调用函数open()
,传入两个参数,一个是我们要创建的文件名test2.txt
,还有一个就是参数w
,代表的是写入命令。
我们也可以往test2.txt
中传入多行信息,例如:
with open('test2.txt','w') as example:
example.write('Hello world!\n')
example.write('Hello python!\n')
该段程序中第二行write()
函数中还加入了换行符\n
,这样就可以将加入的消息分行了。
程序运行之后生成的test2.txt
的内容为:
Hello world!
Hello python!
我们要注意的是,使用w
命令打开一个待写入的文本文件时,如果该文本文件原来已经存在了,Python
将会在写入内容前将文本文件中原来的内容全部清空。所以我们如果要在已经存在的文本文件中添加信息内容,而不是将原来的信息内容都清除,就要采取附加模式打开文件。
我们用附加模式打开文件时,Python
会将我们写入的数据直接加入到文本文件原有信息的末尾,如果指定的文本文件不存在,才会新建一个文本文件。
例如现在已经存在了文本文件test2.txt
,里面的内容为:
Hello world!
Hello python!
我们现在想在文本文件test2.txt
中再加入两句话:
Hello my brothers!
Hello my sisters!
实现代码如下:
with open('test2.txt','a') as example:
example.write('Hello my brothers!\n')
example.write('Hello my sisters!\n')
我们在函数open()
中传入了参数a
,告诉Python
程序我们是将信息加入到文本文件的末尾,而不是覆盖文件。
程序运行之后,文本文件test2.txt
中的内容变为:
Hello world!
Hello python!
Hello my brothers!
Hello my sisters!
try-except
代码块处理异常我们一般使用try-except
代码块处理异常,try-except
代码块让Python
程序在遇到错误时执行指定的操作,而不是产生traceback
。
例如,我们编写如下程序:
try:
print(2/0)
except:
print("We can't divide by zero!")
程序运行结果:
We can't divide by zero!
我们将print(2/0)
语句放在了try
代码块中,很显然这是一个错误语句,因为0
不能当除数。try-except
代码块的作用就是当try
代码块中的程序代码运行没有错误,则跳过except
代码块;反之如果try
代码块中的程序代码运行出现错误,则运行except
代码块中的内容,一般except
代码块中的内容都是输出错误信息。
如果except
代码块后面还有其他代码,程序将继续运行,因为Python
已经在程序报错后执行了except
代码块中的内容,告诉了错误信息,所以可以继续运行程序。
raise
显式抛出异常信息除了系统自动抛出异常外,程序员也可以通过Python
的raise
显式抛出自己的包含特定信息的异常。一旦执行了raise
语句,raise
之后的语句将不能执行。
def read_C():
try:
C = float(sys.argv[1])
except ValueError:
raise ValueError('Degrees must be number, not "%s"' % sys.argv[1])
if C < -273.15:
raise ValueError('C=%g is a non-physical value!' % C)
return C
在raise关键字之后抛出一个ValueError
类型的异常信息,并在后面的参数中定义具体的异常信息。
else
代码块可以在try-except
代码块后面添加try
代码块,当try
代码块中的程序运行没有异常时,程序将运行else
代码块中的内容,例如:
a = int(input())
b = int(input())
try:
answer = a/b
except:
print("We can't divide by zero!")
else:
print(answer)
该段程序表示:先输入两个整数a
和b
,执行try
代码块中的内容,计算整数a
和b
相除的结果,并将计算结果保存在变量answer
中,如果b
为0
,则try
代码块中运行错误,程序执行except
代码块中的内容,输出提醒信息We can't divide by zero!
,而不是traceback
中的内容。
当try
代码块中的程序运行正确,即b
不为0
,则执行else
代码块的内容,将answer
输出。
try-except-else
代码块的原理大致如下:先运行try
代码块中的程序内容,如果运行出错则执行except
代码块中的程序内容,如果运行正确则执行else
代码块中的程序内容。