day10函数进阶
1.讲在课前
# 1.组内同学有问题及时反映给组长 # 2.关于课上走神的问题: # 早睡 增加课间时间 课上的配合 # 3.关于老师的进度 # 4.关于课上的段子 # 5.博客 —— 博客地址 : # 代码为主,简单的注释 # —— 项目地址 # 6.复习 # —— 给你们出题 跟着以往的进度 # —— 画思维导图 # —— 基本数据类型 # ——文件操作 # —— 函数 # —— 常用模块 # —— 面向对象 # —— 网络编程 # —— 并发编程
2.复习
# 函数 # 可读性强 复用性强 # def 函数名(): # 函数体 #return 返回值 # 所有的函数 只定义不调用就一定不执行 #先定义后调用 #函数名() #不接收返回值 #返回值 = 函数名() #接收返回值 #返回值 # 没有返回值 :默认返回None # 不写return:函数内的代码执行完毕自动结束 # 只写return:结束一个函数 # return None # 返回一个值 :结束了函数且返回一个值,可以是任意的值 # 返回多个值 : 多个值之间用逗号隔开,接收的时候可以用一个变量接收(元祖),也可以用等量的多个变量接收 # def f(a): # return '栗子' # # ret = f('苹果') # print(f('苹果')) #参数 #形参 定义函数的时候 # 位置参数 :必须传 # *args :动态参数 可以接收任意多个按位置传入的参数 # 默认参数 : 可以不传 —— 陷阱 # **kwargs : 动态参数 可以接受任意多个按关键字传入的参数 #实参 调用函数的时候 # 按照位置传参数 # 按照关键字传参数 # 可以混用 位置参数必须在关键字传参之前 # 不能对一个参数重复赋值 # def 娃哈哈(*args): # print(args) # # # 娃哈哈(1,2,3,4) # l = [1,2,3,4] # 娃哈哈(*l)
3.默认参数的陷阱
def qqxing(k,l = {}): # l.append(1) l[k] = 'v' print(l) qqxing(1) #[1] qqxing(2) #[1,1] qqxing(3) #[1,1,1] # 如果默认参数的值是一个可变数据类型, # 那么每一次调用函数的时候, # 如果不传值就公用这个数据类型的资源
4.作业讲解
#2、写函数,检查获取传入列表或元组对象的所有奇数位索引对应的元素,并将其作为新列表返回给调用者。 # def func(l): # return l[1::2] #切片 # print(func([1,2,3,4,5])) # 3、写函数,判断用户传入的值(字符串、列表、元组)长度是否大于5。 # def func(x): # return len(x)>5 # if func('abcd'): # print('长度确实大于5') # 4、写函数,检查传入列表的长度,如果大于2, # 那么仅保留前两个长度的内容,并将新内容返回给调用者。 # def func(l): # return l[:2] # # print(func([1,2,3,4])) # 5、写函数,计算传入字符串中【数字】、【字母】、【空格】 以及 【其他】的个数,并返回结果。 # def func(s): #'skghfoiw8qpeuqkd' # dic = {'num':0,'alpha':0,'space':0,'other':0} # for i in s: # if i.isdigit(): # dic['num']+=1 # elif i.isalpha(): # dic['alpha'] += 1 # elif i.isspace(): # dic['space'] += 1 # else: # dic['other'] += 1 # return dic # print(func('+0-0skahe817jashf wet1')) # 6、写函数,检查用户传入的对象(字符串、列表、元组) # 的每一个元素是否含有空内容,并返回结果。 # def func(x): # if type(x) is str and x: #参数是字符串 # for i in x: # if i == ' ': # return True # elif x and type(x) is list or type(x) is tuple: #参数是列表或者元组 # for i in x: # if not i: # return True # elif not x: # return True # # print(func([])) #7、写函数,检查传入字典的每一个value的长度,如果大于2, # 那么仅保留前两个长度的内容,并将新内容返回给调用者。 # dic = {"k1": "v1v1", "k2": [11,22,33,44]} # PS:字典中的value只能是字符串或列表 # def func(dic): # for k in dic: # if len(dic[k]) > 2: # dic[k] = dic[k][:2] # return dic # dic = {"k1": "v1v1", "k2": [11,22,33,44]} # print(func(dic)) # 8、写函数,接收两个数字参数,返回比较大的那个数字。 # def func(a,b): # if a>b: # return a # else: # return b # print(func(1,5)) # def func(a,b): # return a if a > b else b # print(func(5,1)) # 三元运算 # a = 1 # b = 5 # c = a if a>b else b #三元运算 # print(c) # 变量 = 条件返回True的结果 if 条件 else 条件返回False的结果 #必须要有结果 #必须要有if和else #只能是简单的情况 # 9、写函数,用户传入修改的文件名,与要修改的内容, # 执行函数,完成整个文件的批量修改操作(进阶)。 # def func(filename,old,new): # with open(filename, encoding='utf-8') as f, open('%s.bak'%filename, 'w', encoding='utf-8') as f2: # for line in f: # if old in line: # 班主任:星儿 # line = line.replace(old,new) # # 写文件 # f2.write(line) # 小护士:金老板 # # import os # os.remove(filename) # 删除文件 # os.rename('%s.bak'%filename, filename) # 重命名文件
5.函数的命名空间
# 函数进阶 # a = 1 # def func(): # print(a) # func() # 命名空间和作用域 # print() # input() # list # tuple #命名空间 有三种 #内置命名空间 —— python解释器 # 就是python解释器一启动就可以使用的名字存储在内置命名空间中 # 内置的名字在启动解释器的时候被加载进内存里 #全局命名空间 —— 我们写的代码但不是函数中的代码 # 是在程序从上到下被执行的过程中依次加载进内存的 # 放置了我们设置的所有变量名和函数名 #局部命名空间 —— 函数 # 就是函数内部定义的名字 # 当调用函数的时候 才会产生这个名称空间 随着函数执行的结束 这个命名空间就又消失了 #在局部:可以使用全局、内置命名空间中的名字 #在全局:可以使用内置命名空间中的名字,但是不能用局部中使用 #在内置:不能使用局部和全局的名字的 # def func(): # a = 1 # # func() # print(a) # def max(l): # print('in max func') # # print(max([1,2,3])) #在正常情况下,直接使用内置的名字 #当我们在全局定义了和内置名字空间中同名的名字时,会使用全局的名字 #当我自己有的时候 我就不找我的上级要了 #如果自己没有 就找上一级要 上一级没有再找上一级 如果内置的名字空间都没有 就报错 # 多个函数应该拥有多个独立的局部名字空间,不互相共享 # def input(): # print('in input now') # def func(): # # input = 1 # print(input) # func() # def fun1(): # a = 1 # # def fun2():pass #func --> 函数的内存地址 #函数名() 函数的调用 #函数的内存地址() 函数的调用 # 作用域两种 # 全局作用域 —— 作用在全局 —— 内置和全局名字空间中的名字都属于全局作用域 ——globals() # 局部作用域 —— 作用在局部 —— 函数(局部名字空间中的名字属于局部作用域) ——locals() # a = 1 # def func(): # global a # a = 2 # # func() # print(a) # 对于不可变数据类型 在局部可是查看全局作用域中的变量 # 但是不能直接修改 # 如果想要修改,需要在程序的一开始添加global声明 # 如果在一个局部(函数)内声明了一个global变量,那么这个变量在局部的所有操作将对全局的变量有效 # a = 1 # b = 2 # def func(): # x = 'aaa' # y = 'bbb' # print(locals()) # print(globals()) # # func() # print(globals()) # print(locals()) #本地的 #globals 永远打印全局的名字 #locals 输出什么 根据locals所在的位置 # a = 1 # def func(a): # a = 2 # return a # # a = func(a) # print(a)
6.函数的嵌套和作用域链
# def max(a,b): # return a if a>b else b # # def the_max(x,y,z): #函数的嵌套调用 # c = max(x,y) # return max(c,z) # # print(the_max(1,2,3)) #函数的嵌套定义 #内部函数可以使用外部函数的变量 # a = 1 # def outer(): # a = 1 # def inner(): # a = 2 # def inner2(): # nonlocal a #声明了一个上面第一层局部变量 # a += 1 #不可变数据类型的修改 # inner2() # print('##a## : ', a) # inner() # print('**a** : ',a) # outer() # print('全局 :',a) #nonlocal 只能用于局部变量 找上层中离当前函数最近一层的局部变量 #声明了nonlocal的内部函数的变量修改会影响到 离当前函数最近一层的局部变量 # 对全局无效 # 对局部 也只是对 最近的 一层 有影响 # a = 0 # def outer(): # # a = 1 # def inner(): # # a = 2 # def inner2(): # nonlocal a # print(a) # inner2() # inner() # # # outer() # def func(): # print(123) # # # func() #函数名就是内存地址 # func2 = func #函数名可以赋值 # func2() # # l = [func,func2] #函数名可以作为容器类型的元素 # print(l) # for i in l: # i() def func(): print(123) def wahaha(f): f() return f #函数名可以作为函数的返回值 qqxing = wahaha(func) # 函数名可以作为函数的参数 qqxing()
7.闭包
#闭包:嵌套函数,内部函数调用外部函数的变量 # def outer(): # a = 1 # def inner(): # print(a) # inner() # outer() def outer(): a = 1 def inner(): print(a) return inner inn = outer() inn() # import urllib #模块 from urllib.request import urlopen # ret = urlopen('http://www.xiaohua100.cn/index.html').read() # print(ret) # def get_url(): # url = 'http://www.xiaohua100.cn/index.html' # ret = urlopen(url).read() # print(ret) # # get_url() def get_url(): url = 'http://www.xiaohua100.cn/index.html' def get(): ret = urlopen(url).read() print(ret) return get get_func = get_url() get_func()