函数
通过函数,可以定义一段代码块,之后通过函数名可以反复调用
定义一个函数:
def alert(): "打印Hello World" print("Hello World")
使用def来定义函数,第二行建议使用使用文档字符串进行函数说明。
之后使用函数名就能够反复调用函数。
alert()
函数的返回值
函数的最后可以使用return定义返回值,没用retrun或者return后面不带表达式,返回值都是None。一上面的函数为例
def alert(): "打印Hello World" print("Hello World") return # 这行有和没有都一样,之后可以去掉这行再试一下 a = alert() # 调用函数,并把返回值赋值给a print(a) # 来看一下a是不是None
有返回值的情况,返回值应该可以是任意类型,并且可以返回多个值
def test1(): print("This is test1") return 0 def test2(): print("This is test2") return 1,"Hello",["Sunday","Monday"],{"name":"Bob"} a = test1() b = test2() print(a) print(b)
返回多个值的情况,应该也是按一个值来处理的。这里看到最终结果是将所有的值放到一个元组里来作为返回值。
返回值的总结:
- return数量=0:返回None
- return数量=1:返回object
- return数量>1:返回tuple
函数的参数
定义一个带参数的函数
def alert(msg): "打印msg,如果是字符串返回1,否则返回0" print(msg) return 1 if type(msg) is str else 0 a = alert(123) # 调用函数时,也必须带上参数 print(a) # 返回是0,因为是数字不是字符串 b = alert("123") print(b) # 返回是1,参数是字符串
形参:变量只有在被调用时才分配内存单元,在调用结束时,即刻释放所分配的内存单元。因此,形参只在函数内部有效。函数调用结束返回主调用函数后则不能再使用该形参变量
实参:可以是常量、变量、表达式、函数等,无论实参是何种类型的量,在进行函数调用时,它们都必须有确定的值,以便把这些值传送给形参。因此应预先用赋值,输入等办法使参数获得确定值
def power(x,y): "计算x的y次方并返回" return(x**y) a,b = 3,2 c = power(a,b) print(c)
上面函数中的x和y就是形参。而后面调用函数时的a和b就是实参。
关键参数
上面调用函数时应用的参数必须一一对应,数量和位置都不能错。也可以用关键参数来引用。
def power(x,y): "计算x的y次方并返回" return(x**y) a,b = 3,2 c = power(y=a,x=b) print(c)
也可以这样混用,但是关键参数必须在位置参数后面
def power(x,y): "计算x的y次方并返回" return(x**y) a,b = 3,2 c = power(a,y=b) print(c)
参数a是位置参数,值传给了x;剩下的y=b是关键参数,b的值也能传给y。
默认参数
看一个默认参数的例子:
def introduce(name,age,country="China"): print("My name is %s"%name) print("I am %d years old"%age) print("I am from %s"%country) introduce("Jack",23) introduce("Jack",23,"Amarica")
如果定义函数的时候给参数预先赋了值,那么这个参数就有了默认参数。在调用的时候可以省略这个默认参数,如果省略就取默认值,如果有值就使用调用的值
其他运用场景:默认安装路径,连接服务的默认端口号
非固定参数
定义函数时,可以使用*args和**kwargs。这样在调用函数的时候,多余的实参会传入这2个形参里。
def test1(x,*args): # *args会把多传入的参数变成一个元组 print(x,args) test1(1,2,3,4,5) test1(1,*[2,3,4,5]) # 也可以用这种形式调用 def test2(x,*args,**kwargs): # **kwargs会把多传入的关键参数变成一个字典 print(x,args,kwargs) test2(1,2,3,a=4,b=5) test2(1,*[2,3],**{‘a‘:4,‘b‘:5}) # 也可以用这种形式调用
这里*和**后面理论上可以使用任意的符合变量命名规则的名字,但是建议使用固定的名字*args和**kwargs。
局部变量和全局变量
在子程序中定义的变量称为局部变量,在程序中定义的变量称为全局变量。
全局变量作用域是整个程序,局部变量作用域是定义该变量的子程序。
当全局变量与局部变量同名时:在定义局部变量的子程序内,局部变量起作用;在其它地方全局变量起作用。
name = "Alice" # 先定义2个全局变量 age = 23 def test1(): print(name,age) # 这里的2个变量还是全局变量 def test2(): name = "Jerry" # 这里定义了局部变量name,对全局变量并没有影响 print(name,age) # 这里的name就是局部变量,age还是全局变量 test1() test2() print(name,age) # 这里的2个全局变量的值并没有改变
所以在子程序中既可以使用全局变量也可以使用自己的局部变量。如果全局变量和局部变量同名的情况下,改变局部变量也不会影响到全局变量。
但是如果想在子程序用改变全局变量,可以通过global声明。
name = "Alice" # 先定义2个全局变量 age = 23 def test2(): global name # 声明name是全局变量 name = "Jerry" # 这里改变了name的值,看看全局变量是否也变了 print(name,age) print(name,age) # 先打印2个全局变量 test2() # 调用一次函数,其中修改了name print(name,age) # 这里的全局变量也被修改了
方法能实现,但是尽量不用这么用,不要在子程序中改变全局变量,更不要去定义全局变量。
最后再看一下列表在子程序中的情况,列表、字典、集合这类数据类型没有局部变量。
name = [‘Alice‘,‘Bob‘,‘Carl‘] # 定义全局变量 def test(): print(name[1]) name[1] = "Jack" # 修改数组的一个元素 print(name) # 先打印一次数组 test() # 调用函数,函数中改变了数组中的一个元素 print(name) # 虽然函数中未使用global,但是数组中的元素还是被改变了
递归
在函数内部,可以调用其他函数。如果一个函数调用的是其自身,那么这个函数就是递归函数。
递归的特性:
- 必须有一个明确的结束条件
- 每次进入更深一层递归时,问题规模相比上次递归都应有所减少
- 递归效率不高,递归层次过多会导致栈溢出。(Python最多递归999层,到了会报错)
递归的例子:
def halve(n): "每次减半并转为整形,直到0为止" print(n) n = int(n/2) if n > 0 : halve(n) halve(100)
高阶函数
变量可以指向函数,函数的参数能接收变量。另外,一个函数也可以接收另一个函数作为参数,这种函数就称之为高阶函数。举例说明:
def fxy(x,y,f): "返回x和y经过f函数处理之后的值" return f(x,y) a = fxy(3,6,min) # min是内置函数,求最小值 b = fxy(3,6,max) # max是内置函数,求最大值 c = fxy(3,6,pow) # pow是内置函数,求幂。3的6次方是729 print(a,b,c)
作业
工资管理系统
Alex 100000
Rain 80000
Egon 50000
Yuan 30000
-----以上是info.txt文件-----
实现效果:
从info.txt文件中读取员工及其工资信息,最后将修改或增加的员工工资信息也写入原info.txt文件。
效果演示:
1. 查询员工工资
2. 修改员工工资
3. 增加新员工记录
4. 退出
>>:1
请输入要查询的员工姓名(例如:Alex):Alex
Alex的工资是:100000。
1. 查询员工工资
2. 修改员工工资
3. 增加新员工记录
4. 退出
>>:2
请输入要修改的员工姓名和工资,用空格分隔(例如:Alex 10):Alex 10
修改成功!
1. 查询员工工资
2. 修改员工工资
3. 增加新员工记录
4. 退出
>>:3
请输入要增加的员工姓名和工资,共空格分割(例如:Eric 100000):Eric 100000
增加成功!
1. 查询员工工资
2. 修改员工工资
3. 增加新员工记录
4. 退出
>>:4
再见!