一、文件处理
1.1 文件的高级应用
1.1.1 可读可写
r + t可读可写
with open('a.txt','r+t',encoding='utf-8') as fr:
print(fr.readable())
print(fr.writable())
True
True
w + t可写可读
with open('a.txt','w+t',encoding='utf-8') as fr:
print(fr.readable())
print(fr.writable())
True
True
a + t可追加可读
1.1.2 文件内指针移动
假如我们需要在文件指定的位置加入内容,使用基础r,a,w模式非常困难,因此需要操作指针移动。
1.1.2.1 seek
seek(offset,whence)中的offset代表文件指针的偏移量,单位是字节。
with open('a.txt','wt',encoding='utf-8') as fr:
print(f'fr.seek(4,0):{fr.seek(3,0)}')
fr.seek(4,0):3
注:seek后括号内的两个数代表不同的含义,第一个是所取得字节,第二个是0代表从文件头开始,是1代表从文件当前位置开始,是2代表从文件末尾开始。
1.1.2.2 tell()
每次统计都是从文件开头到当前指针所在位置
with open('a.txt','wt',encoding='utf-8') as fr:
fr.seek(4,0)
print(f'fr.tell():{fr.tell()}')
fr.tell():4
1.1.2.3 read(n)
只有在read(n)模式下,n代表的是字符的个数,除此之外,其他涉及文件指针的都是字节个数
with open('a.txt','rt',encoding='utf-8') as fr:
print(f"fr.read(3): {fr.read(3)}")
321
1.1.2.4 truncate(n)
truncate(n)是截断文件,所以文件的打开方式必须可写,但是不能用w或W+等方式打开,因为会直接清空文件。所以要在r+或a或a+等模式下使用,它的参照永远是文件头。并且truncate()不加参数,相当于清空文件。
1.2 文件修改的两种方式
文件数据存储在硬盘上,我们所说的修改,其实是覆盖,我们平时修改的文件,都是电脑模拟处理后的结果。
1.2.1 方法一
将硬盘存放的该文件内容全部加载到内存,在内存中进行修改,完成后,在覆盖到硬盘。
import os
with open('a.txt',encoding='UTF-8') as fr,
open('aa.txt','w',encoding='UTF-8') as fw:
data = fr.read()
data = data.replace('阿斯达','哈哈哈')
fw.write(data)
os.remove('a.txt')
os.rename('aa.txt','a.txt')
print('done')
1.2.2 方法二
将硬盘存放的文件内容一行一行的读入内存,修改完写入新文件,最后覆盖原文件。
import os
with open('a.txt',encoding='UTF-8') as fr,
open('aa.txt','w',encoding='UTF-8') as fw:
for line in fr:
line = line.replace('哈哈哈','略略略')
fw.write(line)
os.remove('a.txt')
os.rename('aa.txt','a.txt')
print('done')
总结:修改文件就是,以读的方式打开文件,以写的方式打开一个新文件,把原文件内容修改后写入新文件,之后使用os模块,把原文件删除,重命名新文件为原文件名,完成修改。
二、函数基础
2.1 函数的定义
2.1.1 什么是函数
在程序中,函数就是具备某一功能的工具,事先将函数定义好,遇到适用场景可以拿来就用。
2.1.2 为何使用函数
- 是程序不冗长
- 增加程序扩展性
- 增加程序可读性
2.1.3 如何使用函数
先定义函数,使用时调用。定义函数时,只检测语法,不执行函数体内的代码。
def register():
username = input('用户名:').strip()
pwd = input('密码:').strip()
with open('a.txt','a',encoding='utf8') as fa:
fa.write(f"{username}:{pwd}
")
fa.flush()
register()
def login():
inp_username = input('用户名:')
inp_pwd = input('密码:')
with open('a.txt','rt',encoding='utf8') as fr:
for user_info in fr:
user_info = user_info.strip('
')
user_info_list = user_info.split(':')
if inp_username == user_info_list[0] and inp_pwd == user_info_list[1]:
print('登陆成功')
break
else:
print("错误")
login()
2.2 定义函数的三种形式
定义函数时参数是函数体接收外部传值的一种媒介。
2.2.1 无参数
在函数阶段括号内没有参数,称为无参函数。需要注意的是:定义时无参,意味着调用时也无需传入参数。
如果函数体代码逻辑不需要依赖外部传入的值,必须得定义成无参函数。
def func():
print('hee')
func()
hee
2.2.2 有参数
在函数定义阶段括号内有参数,称为有参函数。需要注意的是:定义时有参,意味着调用时也必须传入参数。
如果函数体代码逻辑需要依赖外部传入的值,必须得定义成有参函数。
def sum_self(x, y):
res = x+y
print(res)
sum_self(1,2)
2.2.2 空参数
当你只知道你需要实现某个功能,但不知道该如何用代码实现时,你可以暂时写个空函数,然后先实现其他的功能。
def func():
pass
2.3 函数的返回值
2.3.1 什么是返回值
函数内部代码经过一系列逻辑处理获得的结果。
def func():
a = 2
return a
a = func()
print(a)
2.3.2 为什么有返回值
如果需要在程序中拿到函数的处理结果做进一步的处理,则需要函数必须要有返回值。
需要注意的是:
-
return是一个函数结束的标志,函数内可以有多个return,只要执行到return,函数就会执行。
-
return的返回值可以返回任意数据类型
-
return的返回值无个数限制,即可以使用逗号隔开返回多个值
- 0个:返回None
- 1个:返回值是该值本身
- 多个:返回值是元组
2.4 函数的调用
def add(x,y)即调用函数,会执行函数体代码,直到碰到return或者执行完函数体内所有代码结束。函数运行完毕所有代码,如果函数体不写return,则会返回None。
def add(x, y):
return x+y
res = add(10, 20) # 函数的调用
print(res*12)
2.5 函数的参数
2.5.1 形参和实参
2.5.1.1 形参
在函数定义阶段括号内定义的参数,称之为形式参数,简称形参,本质就是变量名。
def func(x, y):
print(x)
print(y)
2.5.1.2 实参
在函数调用阶段括号内传入的参数,称之为实际参数,简称实参,本质就是变量的值。
func(1, 2)
2.5.2 位置参数
2.5.2.1 位置实参
在函数定义阶段,按照从左到右的顺序依次定义的形参,称之为位置形参。
def func(x, y):
print(x)
print(y)
特点:按照位置定义的形参,都必须被传值,多一个不行,少一个也不行。
2.5.2.2 位置形参
在函数调用阶段,按照从左到右的顺序依次定义的实参,称之为位置实参。
func(1, 2)
特点:按照位置为对应的形参依次传值。
2.5.3 关键字实参
在调用函数时,按照key=value的形式为指定的参数传值,称为关键字实参。
func(y=2, x=1)
特点:可以打破位置的限制,但仍能为指定的形参赋值。
注意:
- 可以混用位置实参和关键字实参,但是位置实参必须在关键字实参的左边。
- 可以混用位置实参和关键字实参,但不能对一个形参重复赋值。
func(x, y=2)
func(y=2, x) # SyntaxError: positional argument follows keyword argument
func(x, x=1) # NameError: name 'x' is not defined
总结:函数的参数尽量不要超过3个; 需要的用的时候用,一般用位置形参+位置实参。