一、函数是什么
定义:函数是指一组语句的集合通过一个名字(函数名)封装起来,想要执行这个函数,只需要调用函数名即可。
C中的函数叫function,java中的函数叫method,python中的函数用def做关键字。
二、使用函数的好处
1.提高代码的复用性
2.让代码更简洁、简化代码
3.代码可扩展
三、python中函数的定义
定义函数使用def关键字,后面是函数名,函数名不能重复
def sayHello(): #定义函数,函数名 print('hello') #函数体 #函数不调用是不会被执行的 sayHello() #调用函数
四、函数的参数
1、形参与实参
函数在调用的时候,可以传入参数,有形参和实参。形参就是函数接收的参数,而实参就是你实际传入的参数
形参:
只有在被调用时才分配内存单元,在调用结束时,即刻释放所分配的内存单元。因此,形参只在函数内部有效。
实参:
可以是常量、变量、表达式、函数等,无论实参是何种类型的量,在进行函数调用时,它们都必须有确定的值,以便把这些值传送给形参。函数调用结束返回主调用函数后则不能再使用该形参变量。
def calc(a,b): #形参:形式参数 print('%s * %s = %s'%(a,b,a*b)) calc(7,8) #实参:实际参数
2、函数的四种形参
(1) 位置参数
按照参数的位置来传参,calc这个函数中,a和b就是位置参数,是必传的,没有会报错
def calc(a,b): #位置参数,必填,不填会报错missing 2 required positional arguments print('%s * %s = %s'%(a,b,a*b)) calc(7,8)
有几个位置参数在调用的时候就要传几个,否则会报错,如果有多个位置参数的话,记不住哪个位置传哪个了,可以使用位置参数的名字来指定调用
def size(ength,width,height): print('Size is %s'%(ength * width * height)) size(height=7,ength=8,width=10)#关键字传参,参数过多可以使用未知参数的名字来指定调用
def op_mysql(host,port,username,passwd,charset,sql): print('连接数据库')
#一一对应传参 op_mysql('192.168.1.1',3306,'Amy','sfsd','utf-8','select * from user')
#不按顺序传参(关键字传参) op_mysql(sql = 'select * from user', host = '192.168.1.1', username = 'Amy', passwd = 'sfsd', charset = 'utf-8', port = '3306' )
#也可以将两种传参相结合 op_mysql('192.168.1.1',3306, sql = 'select * from user', username = 'Amy', passwd = 'sfsd', charset = 'utf-8', )
(2) 默认参数
在定义形参的时候,给函数默认赋一个值,比如说数据库的端口这样的,默认给它一个值,这样就算你在调用的时候没传入这个参数,它也是有值的。
默认参数不是必填的,如果给默认参数传值,它就会使用你传入的值。如果使用默认值参数的话,必须放在位置参数后面定义。
def op_file(file_name,content=None): #content:默认值参数,它是非必填的 f = open(file_name, 'a+', encoding='utf-8') f.seek(0) if content: #不为空代表写文件 f.write(content) f.flush() else: #为空代表读文件 all_user = f.read() #all_users是局部变量 return all_user #调用完函数之后,返回什么结果 f.close() print(op_file('username'))
(3) 非固定参数
位置参数和默认参数的参数个数都是固定的,如果一个函数参数是不固定的,以后也不知道会扩展成什么样,那么再用固定函数,后面程序就不好扩展了。这时候需要非固定参数。非固定参数有两种,可变参数和关键字参数。
(i) 可变参数
可变参数用*来接收,后面想传多少个参数就传多少个,如果位置参数、默认值参数、可变参数一起使用的的话,可变参数必须在位置参数和默认值参数后面。
a、非必填参数
b、它没有限制参数个数
b、它把传回来的参数放在元组里
def syz(name,*args): #参数组 # 位置参数、默认值参数、可变参数,可变参数会把后面多传的参数都放到args这个元组中 # args名字是随便写的,不过一般我们都写args print(name,args) syz('双鱼座') #双鱼座 (),非必填 syz('双鱼座','111') #双鱼座 ('111',) syz('asd','sdf',23224) #asd ('sdf', 23224),没有限制参数个数
(ii) 关键字参数
关键字参数用**来接收,后面的参数也是不固定的,位置参数、默认值参数、可变参数,关键字参数一起使用的话,关键字参数必须在最后面。
a、非必填参数
b、它没有限制参数个数
c、它把传回来的参数放在字典里
d、使用关键字参数的话,调用的时候必须使用关键字传参
def syz(name,**kwargs): #关键字参数,必须有k-v,两个参数 # 位置参数、关键字参数,调用的时候会把传入的关键字参数放到kwargs这个字典中 print(name,kwargs) syz('双鱼座') #非必填,双鱼座 {} syz('双鱼座','111') #报错,必须有k-v,两个参数 syz('双鱼座',room = '111') #双鱼座 {'room': '111'} syz('双鱼座',room = '111',addr = '北京') #没有限制参数个数,双鱼座 {'addr': '北京', 'room': '111'}
五、函数的返回值
每个函数都有返回值,如果没有在函数里指定返回值,则默认返回None,函数可以有多个返回值,如果有多个返回值,会把返回值放到一个元组中,返回的是一个元组。
def calc(x,y): res = x * y return x,y,res print(calc(2,3)) #多个返回值的函数,返回一个元组(2, 3, 6)
返回值作用:
1、把函数处理的结果返回回来,供后面的程序用。
2、结束函数,函数里面遇到return,函数会立即结束
def r(): for i in range(5): if i == 3: print(i) return #只写一个return的话,就返回None r() #3,遇到return函数立即结束 res = r() print(res) #默认返回None
import string def check(pwd): if len(pwd) >5 and len(pwd) <12: #长度6-11 if set(pwd) & set(string.ascii_letters) and set(pwd) & set(string.digits):#必须包含字符和数字 print('密码合法') else: print('密码不合法') else: print('密码不合法') res = check('ad1234') #调用函数,并把函数返回值赋给res # 函数里面如果没有return的话,默认返回none,return不是必须写的 print(res)
六、局部变量和全局变量
常量:
一个不变的值,通常用大写字母定义
PORT = 3306 #常量 FLIENAME = 'user.txt'
局部变量:
函数里面定义的变量,都是局部变量,只能在函数里面用,出了函数之后就不能用了
name = 'Lily' #全局变量 def sayName(): name = 'Kate'#局部变量 print(name) #kate sayName() print(name) #Lily
全局变量:
公共的变量,都可以用的变量。
在整个程序里面都生效的,在程序最前面定义的都是全局变量,全局变量如果要在函数中修改的话,需要加global关键字声明,如果是list、字典和集合的话,则不需要加global关键字,直接就可以修改
name = 'Lily' #字符串全局变量 names = [] #list全局变量 def sayName(): global name #修改全局变量name,加global关键字 name = 'Kate' names.append(name)#修改全局list变量,不用加global关键字 print(name) #kate sayName() print(name) #kate print(names) #['Kate']
最好少用全局变量的原因:
(1) 不安全,因为所有人都可以改
(2) 全局变量它会一直占着内存
七、递归
函数自己调用自己。
在函数内部,可以调用其他函数。如果一个函数在内部调用自身本身,这个函数就是递归函数。
递归调用的意思就是,在这个函数内部自己调用自己,就有点循环的意思。
def test1(): num = int(input('please enter a number:')) if num%2==0:#判断输入的数字是不是偶数 return True #如果是偶数的话,程序就退出了,返回True print('不是偶数请重新输入!') return test1()#如果不是偶数的话继续调用自己,输入值 print(test1())#调用test
递归调用的特性:
1. 必须有一个明确的结束条件
2. 每次进入更深一层递归时,问题规模相比上次递归都应有所减少
3. 递归效率不高,递归层次过多会导致栈溢出(在计算机中,函数调用是通过栈(stack)这种数据结构实现的,每当进入一个函数调用,栈就会加一层栈帧,每当函数返回,栈就会减一层栈帧。由于栈的大小不是无限的,所以,递归调用的次数过多,会导致栈溢出)
少用递归,递归最多递归999,递归的效率不高
i = 0 def test(): global i i += 1 print(i) test() test() #循环999次后会报错,3.6版本以后不会报错
八、高阶函数
如果一个函数的参数是一个函数的话,那么这个函数就是一个高阶函数
def s_int(n):#把传入的参数类型转换成int类型 return int(n) def add(x,y,z):#接收3个参数,x,y,z,z是一个函数 print(z(x)+z(y))#z是一个函数,把x和y的值传给z,然后用z函数的返回值把两个值相加 add('8','9',s_int)#调用,传入x和y的值,再把上面的那个定义好的函数传进去