6.函数与lambda表达式

一、笔记

函数

(一)函数的定义

  • 函数以def关键词开头,后接函数名和被“()”括住的形参。
  • 函数执行的代码以“:”开始并且缩进。
  • 以“return【表达式】”结束函数。它的作用是选择性地返回一个值给调用方。不带表达式的“return”相当于返回“none”给调用方。

定义函数的格式:

def functionname(parameters):
	function_suite
	return [expression]

(二)函数的调用

举个例子:

def printcon(str1,str2):#函数名中不可含有“-”,否则报错。可以含有“_”。
    print(str1+str2)
printcon('hello','cook')#hellocook
value=printcon('fine','0')#fine0
print(value)#None

(注:定义完函数后,当键入函数名的前几个字母时,键入提示会出现自己已经定义的函数名。)
(三)函数参数
Python的参数形态由简到繁:
位置参数(positional argument)
默认参数(default argument)
可变参数(variable argument)
关键字参数(keyword argument)
命名关键字参数(name keyword argument)
参数组合
1.位置参数(positional argument)

def functionname(arg1):
	function_suite
	return [expression]
  • “arg1”——位置参数。调用函数(call function)时,这些参数的位置要固定。

2.默认参数(default argument)

def functionname(arg1,arg2=v):
	function_suite
	return [expression]
  • “arg2=v”——默认参数。调用函数时,如果默认参数的值没有传入,则被认为是默认值。
  • 默认参数要放在位置参数的后面,否则程序会报错。
def printB(fruit,price=6):
	print('fruit:{0},price:{1}'.format(fruit,price))
	
printB('banana')#fruit:banana,price:6
printB('banana',10)#fruit:banana,price:10
  • Python允许调用函数时参数顺序与声明的不一致,因为Python解释器能够用参数名匹配参数值。
def printB(fruit,price=6):
	print('fruit:{0},price:{1}'.format(fruit,price))
	

3.可变参数(variable argument)

def functionname(arg1,arg2=v,*args):
	function_suite
	return [expression]
  • “*args”——可变参数。个数可以是0到任意数,自动组装成元组。
  • 加了“*”的变量名会存放所有未命名的变量参数。
def printC(arg1,*args):
	print(arg1)
	print(args)
printC('A')
#A
#()
printC('A','B','C')
#A
#(B,C)
def printC(arg1,*args):
	print(arg1)
	for var in args:
		print(var)

printC('A')#A
printC('A','B','C')
#A
#B
#C

4.关键字参数(keyword argument)

#def functionname(arg1,arg2=v,*args,**kw):
	#function_suite
	#return [expression]
  • “**kw”——关键字参数。个数可以是0到任意数,自动组装成字典。
def printD(arg1,*args,**kwargs):
	print(arg1)
	print(args)
	print(kwargs)
	
printD(1,2,3,4)
#1
#(2,3,4)
#{}
printD(1,2,3,4,a=1,b=2)
#1
#(2,3,4)
#{'a':1,'b':2}

5.命名关键字参数(name keyword argument)

def functionname(arg1,arg2=v,*args,*,nkw,**kw):
	function_suite
	return [expression]
  • “*,nkw”——命名关键字参数。
  • 如果想要限制关键字参数的名字就可以用命名关键字参数。
  • 使用命名关键字参数时,要特别注意不能缺少参数名。
def printE(arg1,*,nkw,**kwargs):
	print(arg1)
	print(nkw)
	print(kwargs)
	
printE(10,nkw=20,a=1,b=2)
#10
#20
#{'a':1,'b':2}
printE(10,20,a=1,b=2)
#TypeError:printE() takes 1 positional argument but 2 were given
  • 没有写参数名nkw,所以10被当成位置参数。原函数只有一个位置参数,现在调用了2个,因此会报错。

(四)参数组合
在定义函数时,以上5种参数中最多可以用4个,并且还要遵循一定的顺序:

  • 位置参数、默认参数、可变参数、关键字参数。
  • 位置参数、默认参数、命名关键字参数、关键字参数。

在定义时注意各个参数的表示方法:

参数表示方法举例
位置参数arg1
默认参数arg2
可变参数*args(args接收的是一个tuple)
关键字参数**kwargs(kwargs接收的是一个dict)
命名关键字参数*,nkw

(五)函数的返回值

  • 定义函数时,“return[expression]”的作用是选择性地返回一个值给调用方。不带表达式的“return”相当于返回“none”给调用方。
def add(a,b):
	return a+b

print(add(1,2))#3
print(add([1,2],[2,3]))#[1,2,2,3]
def back():
	return [1,'my first name',3.14]
print(back())
#[1,'my first name',3.14]
def back():
	return 1,'my first name',3.14
print(back())
#(1,'my first name',3.14)
def printA(str):
    print(str)
printA('hello,nice to me you.')#hello,nice to me you.	
value=printA('fine')#fine
print(value)#none
print(type(value))#(class 'NoneType')

(六)变量作用域

  • Python中,程序的变量并不是在哪个位置都可以访问的,访问权限取决于这个变量是在哪个位置赋值的。
  • 定义在函数内部的变量拥有局部作用域,该变量称为局部变量。
  • 定义在函数外部的变量拥有全局作用域,该变量称为全局变量。
  • 局部变量只能在其被声明的函数内部访问,而全局变量可以在其声明的整个程序内访问。
  • 当内部作用域想修改外部作用域的变量时,就要用到global和nolocal关键字了。
num=1

def fun1():
	global num
	print(num)
	num=123
	print(num)
	
print(num)#1
fun1()
#1
#123
print(num)#123
print(num+1)#124

1.内嵌函数

def outer():
	print('This is outer.')
	def inner():
		print('This is inner')
	inner()#该函数只能在outer内部被调用(也就是现在这个位置)
		
outer()
#This is outer.
#This is inner.

2.闭包

  • 闭包是函数式编程的一个重要的语法结构,是一种特殊的内嵌函数。
  • 如果在一个内部函数里对外层非全局作用域的变量进行引用,那么内部函数就会被认为是闭包。
  • 通过闭包可以访问外层非全局作用域的变量,这个作用域被成为全局作用域。
def funX(x):
	def funY(y):
		return x*y
	return funY
	
i=funX(8)
print(type(i))#(class 'function')
print(i(5))#40
  • 闭包的返回值通常是函数。
def make_counter(init):
	counter=[init]
	def inc():counter[0]+=1
	def dec():counter[0]-=1
	def get():return counter[0]
	def reset():counter[0]=init
	return inc,dec,get,reset
	
inc,dec,get,reset=make_counter(0)
inc()
inc()
inc()
print(get())#3
dec()
print(get())#2
reset()
print(get())#0
  • 如果修改闭包作用域的值则需要nolocal关键字。
def outer():
	num=10
	def inner():
		nonlocal num
		num=100
		print(num)
	inner()
	print(num)

outer()
#100
#100

3.递归

  • 如果一个函数在内部调用自己本身,这个函数就是递归函数。
    例子:求阶乘。
#利用循环的方法实现
n=5
for k in range(1,5):
	n=n*k
print(n)#120

#利用递归函数实现
def factorial(n):
	if n==1:
		return 1
	return n*factorial(n-1)

print(factorial(5))#120

例子:斐波那契数列f(n)=f(n-1)+f(n-2),f(0)=0,f(1)=1

#利用循环实现
i=0
j=1
lst=list([i,j])
for k in range(2,11):
	k=i+j
	lst.append(k)
	i=j
	j=k
print(lst)
#[0,1,1,2,3,5,8,13,21,34,55]

#利用递归函数实现
def feibo(n):
	if n <= 1:
		return n
	return feibo(n-1)+feibo(n-2)

lst=list()
for k in range(11):
	lst.append(feibo(k))
print(lst)
#[0,1,1,2,3,5,8,13,21,34,55]
  • 可以自己设置递归的层数,Python默认递归的层数是100.
#利用递归函数实现阶乘
def factorial(n):
	if n==1:
		return 1
	return n*factorial(n-1)

print(factorial(1000))
#RecursionError:maximum recursion depth exceeded in comparion.
import sys
sys.setrecursionlimit(1000)

Lambda表达式

(一)匿名函数的定义

  • Python中的两类函数:

第一类:用关键词“def”定义的正规函数。
第二类:用关键词“lambda”定义的匿名函数。

  • 匿名函数的语法结构:
lambda argument_list:expression

lambda——定义匿名函数的关键词。
argument_list——函数参数(和正规函数中的参数类型一样)。
expression——表达式。

  • 注意:
    expression没有return语句,因为lambda不需要它来返回,表达式本身结果就是返回值。
    匿名函数拥有自己的命名空间,且不能返回自己参数列表以外或全局命名空间里的参数。
#用正规函数求平方
def sqr(x):
	return x**2

print(sqr)
#(function sqr at 0x0000000001E3B5E0)

y=[sqr(x) for x in range(10)]
print(y)
#[0,1,4,9,16,25,36,49,64,81]
#用lambda求平方
sqr1=lambda x:x**2
print(sqr1)
#(function (lambda) at 0X000000000BA60)

y=[sqr1(x) for x in range(10)]
print(y)
#[0,1,4,9,16,25,36,49,64,81]
summary=lambda a,b: a+b
print(summary(1,2))#3

summary1=lambda *args:sum(args) 
print(summary1(1,2,3,4))#10

(二)匿名函数的应用

二、练习

怎么给函数编写⽂档?
怎么给函数参数和返回值注解?
闭包中,怎么对数字、字符串、元组等不可变元素更新。
分别根据每一行的首元素和尾元素大小对二维列表 a = [[6, 5], [3, 7], [2, 8]] 排序。(利用lambda表达式)
利用python解决汉诺塔问题?
有a、b、c三根柱子,在a柱子上从下往上按照大小顺序摞着64片圆盘,把圆盘从下面开始按大小顺序重新摆放在c柱子上,尝试用函数来模拟解决的过程。(提示:将问题简化为已经成功地将a柱上面的63个盘子移到了b柱)

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值