Python20-Day03
##########文件操作相关##########
一、文件操作
文件就是操作系统提供给应用程序来操作硬盘虚拟概念,用户或应用程序通过操作文件,可以将自己的数据永久保存下来。
文件操作的流程:
* 打开文件,得到文件句柄并赋值给一个变量
* 通过句柄对文件进行操作
* 关闭文件
python中的文件操作:
1. 打开文件,得到文件句柄并赋值给一个变量
f = open('a.txt','r',encoding='utf-8') #默认打开模式就为r
2. 通过文件句柄对文件进行操作
data = f.read()
3. 关闭文件
f.close()
python的资源回收
打开一个文件包含两部分资源:操作系统级打开的文件+应用程序的变量。在操作完毕一个文件时,必须把与该文件的这两部分资源一个不落地回收,回收方法为:
1、f.close() #回收操作系统级打开的文件
2、del f #回收应用程序级的变量
操作完毕文件后,一定要f.close()
with关键字来帮助管理上下文:
with open('a.txt','r',encoding='utf-8') as f:
pass
二、打开文件的模式
文件句柄 = open('文件路径','模式')
模式可以是以下方式以及他们之间的组合:
Character | Meaning |
‘r' | open for reading (default) |
‘w' | open for writing, truncating the file first |
‘a' | open for writing, appending to the end of the file if it exists |
‘b' | binary mode |
‘t' | text mode (default) |
‘+' | open a disk file for updating (reading and writing) |
‘U' | universal newline mode (for backwards compatibility; should not be used in new code) |
1. 打开文件的模式(默认为文本模式)
r,只读模式 默认模式 文件必须存在,不存在抛出异常
w,只写模式,不可读,不存在创建,存在则清空内容
a,追加模式 不可读,不存在则创建,存在只增加内容
2. 对于非文本文件,只能使用b模式,"b"表示以字节的方式操作,所有文件都是以字节的形式存储的,使用这种模式无序考虑文本文件的字符编码,
rb wb ab
三、操作文件的方法:
#掌握 f.read() #读取所有内容,光标移动到文件末尾 f.readline() #读取一行内容,光标移动到第二行首部 f.readlines() #读取每一行内容,存放于列表中 f.write('1111\n222\n') #针对文本模式的写,需要自己写换行符 f.write('1111\n222\n'.encode('utf-8')) #针对b模式的写,需要自己写换行符 f.writelines(['333\n','444\n']) #文件模式 f.writelines([bytes('333\n',encoding='utf-8'),'444\n'.encode('utf-8')]) #b模式 #了解 f.readable() #文件是否可读 f.writable() #文件是否可读 f.closed #文件是否关闭 f.encoding #如果文件打开模式为b,则没有该属性 f.flush() #立刻将文件内容从内存刷到硬盘 f.name
练习,利用b模式,编写一个cp工具,要求如下:
1. 既可以拷贝文本又可以拷贝视频,图片等文件
2. 用户一旦参数错误,打印命令的正确使用方法,如usage: cp source_file target_file
提示:可以用import sys,然后用sys.argv获取脚本后面跟的参数
import sys if len(sys.argv) != 3: print('usage: cp source_file target_file') # source_file,target_file = sys.argv[1],sys.argv[2] _,source_file,target_file = sys.argv with open(source_file,'rb') as read_f,open(target_file,'wb') as write_f: for line in read_f: write_f.write(line)
四、文件修改
方式一:将硬盘存放的该文件的内容全部加载到内存,在内存中是可以修改的,修改完毕后,再由内存覆盖到硬盘(word,vim,nodpad++等编辑器)
import os with open('a.txt','r') as read_f,open('.a.txt.swap','w') as write_f: data = read_f.read() data = data.replace('1111','zhaoliying') write_f.write(data) os.remove('a.txt') os.rename('.a.txt.swap','a.txt')
方式二:将硬盘存放的该文件的内容一行一行地读入内存,修改完毕就写入新文件,最后用新文件覆盖源文件
import os with open('a.txt','r') as read_f,open('.a.txt.swap','w') as write_f: data = read_f.read() data = data.replace('1111','zhaoliying') write_f.write(data) os.remove('a.txt') os.rename('.a.txt.swap','a.txt')
练习题:
1. 文件a.txt内容:每一行内容分别为商品名字,价钱,个数,求出本次购物花费的总钱数
apple 10 3
tesla 100000 1
mac 3000 2
lenovo 30000 3
chicken 10 3
goods_price_all = 0 with open('a.txt','r') as f: for line in f: line_list = line.split() price = int(line_list[1]) goods_num = int(line_list[2]) goods_price = price * goods_num goods_price_all+=goods_price print('',goods_price_all)
##########函数相关##########
一、函数基础
1、函数分类
(1)内置函数:如len(),max(),sum()
(2)自定义函数
2、定义函数
自定义函数的语法
#语法 def 函数名(参数1,参数2,参数3,...): '''注释''' 函数体 return 返回的值 #函数名要能反映其意义
函数使用的原则:先定义,再调用
#测试一 def foo(): print('from foo') bar() foo() #报错 #测试二 def bar(): print('from bar') def foo(): print('from foo') bar() foo() #正常 #测试三 def foo(): print('from foo') bar() def bar(): print('from bar') foo() #会报错吗? #结论:函数的使用,必须遵循原则:先定义,后调用 #我们在使用函数时,一定要明确地区分定义阶段和调用阶段 #定义阶段 def foo(): print('from foo') bar() def bar(): print('from bar') #调用阶段 foo()
函数在定义阶段都做了什么?
只检查语法,不执行代码,也就是说,语法错误在函数定义阶段就会检测出来,而代码的逻辑只有在执行时才会知道
2、 定义函数的3中形式
1. 无参 :应用场景仅仅只是执行一些操作,比如用户交互,打印
2. 有参: 需要根据外部传来的参数,才能执行相应的逻辑,比如统计长度,求最大值最小值
3. 空函数: 涉及代码结构
#定义阶段 def tell_tag(tag,n): #有参数 print(tag*n) def tell_msg(): #无参数 print('hello world') #调用阶段 tell_tag('*',12) tell_msg() tell_tag('*',12) ''' ************ hello world ************ ''' #结论: #1、定义时无参,意味着调用时也无需传入参数 #2、定义时有参,意味着调用时则必须传入参数
3、调用函数
1. 调用函数
函数名加括号
1 先找到名字
2 根据名字调用代码
2. 函数返回值
无 return --> None
return 1个值 -> 返回1个值
return 逗号分隔多个值,元组
什么时候该有返回值?
调用函数,经过一系列的操作,最后要拿到一个明确的结果,则必须要有返回值,通常有参函数需要有返回值,输入参数,经过计算,得到一个最终结果。
什么时候不需要有返回值?
调用函数,仅仅只是执行一系列的操作,最后不需要得到什么结果,则无需又返回值。通常无参函数不需要有返回值
3. 函数调用的三种形式
1 语句形式:foo()
2 表达式形式: 3*len('hello')
3 当中另外一个函数的参数:range(len('hello'))
4. 函数的参数
1 形参与实参
形参即变量名,实参即变量值,函数调用时,将值绑定到变量名上,函数调用结束,解除绑定。
2 具体应用
#1、位置参数:按照从左到右的顺序定义的参数 位置形参:必选参数 位置实参:按照位置给形参传值 #2、关键字参数:按照key=value的形式定义的实参 无需按照位置为形参传值 注意的问题: 1. 关键字实参必须在位置实参右面 2. 对同一个形参不能重复传值 #3、默认参数:形参在定义时就已经为其赋值 可以传值也可以不传值,经常需要变得参数定义成位置形参,变化较小的参数定义成默认参数(形参) 注意的问题: 1. 只在定义时赋值一次 2. 默认参数的定义应该在位置形参右面 3. 默认参数通常应该定义成不可变类型 #4、可变长参数: 可变长指的是实参值的个数不固定 而实参有按位置和按关键字两种形式定义,针对这两种形式的可变长,形参对应有两种解决方案来完整地存放它们,分别是*args,**kwargs ===========*args=========== def foo(x,y,*args): print(x,y) print(args) foo(1,2,3,4,5) def foo(x,y,*args): print(x,y) print(args) foo(1,2,*[3,4,5]) def foo(x,y,z): print(x,y,z) foo(*[1,2,3]) ===========**kwargs=========== def foo(x,y,**kwargs): print(x,y) print(kwargs) foo(1,y=2,a=1,b=2,c=3) def foo(x,y,**kwargs): print(x,y) print(kwargs) foo(1,y=2,**{'a':1,'b':2,'c':3}) def foo(x,y,z): print(x,y,z) foo(**{'z':1,'x':2,'y':3}) ===========*args+**kwargs=========== def foo(x,y): print(x,y) def wrapper(*args,**kwargs): print('====>') foo(*args,**kwargs) #5、命名关键字参数:*后定义的参数,必须被传值(有默认值的除外),且必须按照关键字实参的形式传递 可以保证,传入的参数中一定包含某些关键字 def foo(x,y,*args,a=1,b,**kwargs): print(x,y) print(args) print(a) print(b) print(kwargs) foo(1,2,3,4,5,b=3,c=4,d=5) 结果: 1 2 (3, 4, 5) 1 3 {'c': 4, 'd': 5}
#1、写函数,,用户传入修改的文件名,与要修改的内容,执行函数,完成批了修改操作
import sys import os if len(sys.argv) != 4: print('Uasge: 函数 文件名 修改前 修改后') # change_file = sys.argv[1] # before_content = sys.argv[2] # after_content = sys.argv[3] change_file,before_content,after_content=sys.argv[1],sys.argv[2],sys.argv[3] def change(change_file,before_content,after_content): with open(change_file,'r') as read_f,open('%s.swap' % change_file,'w') as write_f: for line in read_f: if before_content in line: line = line.replace(before_content,after_content) write_f.write(line) change(change_file,before_content,after_content) os.remove(change_file) os.rename('%s.swap' %change_file,change_file)
#2、写函数,计算传入字符串中【数字】、【字母】、【空格】 以及 【其他】的个数
def calc_num(calc_str): digit_num = 0 space_num = 0 zimu_num = 0 other_num = 0 for item in calc_str: if item.isdigit(): digit_num=digit_num+1 elif item.isspace(): space_num=space_num+1 elif item.isalpha(): zimu_num=zimu_num+1 else: other_num=other_num+1 print(digit_num,zimu_num,space_num,other_num) calc_num('helloworld123123132 321312 %%%%%% 321213 fadsfasf 4123412 ')
#3、写函数,判断用户传入的对象(字符串、列表、元组)长度是否大于5
def object_len(object_user): if len(object_user) > 5: print('%s is 大于5' % object_user) else: print('其他') # object_len('zhaoliying') # object_len((11,111,22))
#4、写函数,检查传入列表的长度,如果大于2,那么仅保留前两个长度的内容,并将新内容返回给调用者。
def list_len(in_list): if not isinstance(in_list,list): print('%s is not list' % (in_list)) else: # print('%s is list' % in_list) if len(in_list) > 2: in_list = in_list[0:2] return in_list # print(list_len([1,2,3,4,5])) # a = list_len([1,2,3,4,5]) # list_len([1,2,3,4,5])
#5、写函数,检查获取传入列表或元组对象的所有奇数位索引对应的元素,并将其作为新列表返回给调用者
new_list = ['a','b','c','d','e','f','g','h','i',] print(new_list[1::2]) def user_list(in_list): if not isinstance(in_list, list): print('%s is not list.' %in_list) else: print('%s is list.' %in_list) new_list=in_list[1::2] return new_list a = user_list(['a','b','c','d','e','f','g','h','i',]) print(a)
#6、写函数,检查字典的每一个value的长度,如果大于2,那么仅保留前两个长度的内容,并将新内容返回给调用者。
#dic = {"k1": "v1v1", "k2": [11,22,33,44]} #PS:字典中的value只能是字符串或列表 def user_dic(in_dic): d = {} if not isinstance(in_dic,dict): print('%s is not dic.' % in_dic) else: for key in in_dic: if len(in_dic[key]) > 2: d[key] = in_dic[key][0:2] # val_dic = in_dic[key] # res_dic = val_dic[0:2] return d # a = user_dic({'hello':['zhaoliying','liying','wangxiaochen']}) a = user_dic({'hello':'zhaoliying'}) print(a)
##########函数对象、函数嵌套、名称空间与作用域、装饰器##########
一、函数对象
函数是第一类对象,即函数可以当做数据传递
#可以被引用
#可以当做参数传递
#返回值可以是函数
#可以当做容器类型的元素
利用这个特性,可以优雅的取代多分支的if
def foo(): print('From foo') def bar(): print('From bar') dic = { 'foo' : foo, 'bar' : bar, } while True: choice = input('Input your choice:>').strip() if choice in dic: dic[choice]()
二、函数嵌套
函数嵌套调用
函数的嵌套定义
三、名称空间与作用域
什么是名称空间?
名称空间:存放名字x与1绑定关系的地方
名称空间的加载顺序
python test.py
#python解释器先启动,首先加载:内置名称空间
#执行test.py文件,加载全局名称空间
#执行文件的过程中,如果调用函数,则临时产生局部名称空间
名字的查找顺序
局部名称空间--->全局名称空间--->内置名称空间
#在全局无法查看局部的,在局部可以查看全局的
四、作用域
作用域即范围,
---全局范围:全局存活,全局有效
---局部范围:临时存活,局部有效
作用域关系在函数定义阶段就已经固定,与函数调用位置无关
五、闭包函数
#闭包函数包含对外部作用域而非全局作用域的引用
闭包的意义与应用
闭包的意义:返回的函数对象,不仅仅是一个函数对象,在该函数外还包裹了一层作用域,这使得,该函数无论在何处调用,优先使用自己外层包裹的作用域
应用领域: 延迟计算
六、装饰器
为何要用装饰器?
开放封闭原则:对修改封闭,对扩展开放
装饰器的原则:
不修改被装饰对象的源代码
不修改被装饰对象的调用方式
装饰器的目的:
在遵循1和2的前提下,为被装饰对象添加上新功能
#无参装饰器
import time def timmer(func): def wrapper(*args,**kwargs): start_time=time.time() res=func(*args,**kwargs) stop_time=time.time() print('run time is %s' %(stop_time-start_time)) return res return wrapper @timmer def foo(): time.sleep(3) print('from foo') foo()
有参装饰器:
def auth(driver='file'): def auth2(func): def wrapper(*args,**kwargs): name=input("user: ") pwd=input("pwd: ") if driver == 'file': if name == 'egon' and pwd == '123': print('login successful') res=func(*args,**kwargs) return res elif driver == 'ldap': print('ldap') return wrapper return auth2 @auth(driver='file') def foo(name): print(name) foo('egon')