python函数,lambda表达式,三目运算,列表解析,递归

时间:2024-01-07 22:20:20

一、自定义函数

定义函数时,函数体不执行;只有在调用函数时,函数体才执行。函数的结构:

1. def

2. 函数名

3. 函数体

def func_name():
函数体

4. 返回值

如果没有声明返回值,返回值默认为None

def func_name():
pass
return True def func_name():
myString = 'this is to test how to return'
return mySting

5. 参数

  • 形式参数
    • 定义函数时出现的参数,称之为形式参数,有几个形参就写几个,eg. def func(args1,args2)
    • 如果没有形式参数,就留空
  • 实际参数
    • 调用函数时传入的参数,称之为实际参数,又叫实参, eg. func('kaye',21)
  • 默认参数
    • 在定义函数时,通过name=value这种形式,指定形参的默认值,这种形参,我们称之为默认参数,eg. def func(args1, args2 = 'default')
    • 默认参数必须放在形参列表的最后,如果有多个默认参数也都放在形参列表最后
  • 关键字参数,又称指定参数
    • 在调用函数时,通过name=value这种形式,将实参传递给指定的形参。这样的实参,我们称之为关键字参数、或指定参数,
    • 通常,实参通过从左到右的位置顺序匹配形参;但是,如果传入了指定参数,优先按参数名称将实参匹配给形参
    • eg. def func(args1,args2), func(args2 = 21,args1 = 'kaye') 和 func(args1 = 'kaye', args2 = 21) 的传参效果是一样的
  • 动态参数,又称可变参数
    • 在定义函数时,使用*args的形式,意味着可以接收任意多个额外的位置相关的参数到元组中;使用**kwargs的形式,意味着可以接收任意多个额外的关键字参数到字典中
      • eg. func('kaye','leo','jack')
      • eg. func('kaye'=21,'leo'=25,'jack'=30)
    • 在调用函数时,使用*args的形式,意味着我们可以在序列args(可以是列表、字符串、元组)中封装任意多个位置相关的对象,并且在将这个序列args传递给函数时,将序列args解包为分开的、单个的参数
      • eg.  func(*['kaye','leo','jack']),相当于args被赋值为('kaye,'leo','jack'), args[0] = 'kaye', args[1] = 'leo',...
      • 序列args中的元素可以是列表、字典、元组等等,由此可以获得多层嵌套的序列化数据类型
    • 在调用函数时,使用**kwargs的形式,意味着我们可以在字典中封装任意多个关键字参数,,并且在将这个字典传递给函数时,将字典解包为分开的、单个的参数(键值对)
      • eg. func(**{'kaye':21,'leo':25,'jack':30})
      • kwargs从最外层看是字典,而字典的键值对可以是列表、字典、元组等等,由此可以获得多层嵌套的字典
  • keyword-only参数
    • 在定义函数时,跟在*args或一个单独的*之后的所有参数(可以有1个也可以有多个),都称之为keyword-only参数
    • 对于这种参数,在调用函数时,必须使用关键字参数传参
    • eg1.  def f1(a,*b,c)   =>   f1(1,'2,3,c=4)
    • eg2.  def f2(a,*b,c,d)  =>  f2(1,2,3,4,5,c=6,d=7)
  • 万能参数,即混合模式
    • 定义函数时,def func(*args, **kwargs),args接收任意多个位置相关的参数,kwargs接收任意多个关键字参数(键值对)
    • 可以同时接收序列化数据类型(字典、元组、字符串)和字典,作为其传入参数
  • python内部使用以下步骤在赋值前进行参数匹配:
    • 通过位置分配非关键字参数
    • 通过匹配变量名分配关键字参数(即调用时的name=value)
    • 其他额外的非关键字参数分配到*name元组中(即*args)
    • 其他额外的关键字参数分配到**name字典中(即**kwargs)
    • 用默认值分配给未得到分配的参数(即函数定义中的name=value)

6. 补充:

  • 函数重新定义(以后再说函数重载)

def f1

def f1

python解释器从上到下执行命令,当函数被重新定义,调用该函数时会使用重新定义的这个函数

  • 调用函数传入形参时,是传入了对实参的引用,也就意味着,调用函数时对形参的操作,是直接作用在实参上的,会引起实参的变化
  • 全局变量
    • 全局变量放在程序最前面,从实际作用上看相当于常量,应该全部大写
    • 全局变量可以被所有函数直接读取
    • 如果要在函数中对全局变量重新赋值,必须在变量名前面加上global
    • 如果全局变量是字典/列表,要对字典列表中的元素进行增删改,则不需要加global,直接按平常代码写法书写即可。这是因为字典和列表是可变类型,每一次赋值和更改,都是在原地修改的!
  • 注释
    • 一定要写注释,让别人一看就知道这个函数是在完成什么功能
    • 一般在函数定义第一行之后输入注释内容,用三个双引号括起来,"""comment"""
      • :param param1:
      • :return:...

二、三元运算(三目运算)

对简单的条件语句,可以用三元运算简写。三元运算只能写在一行代码里面

# 书写格式

result = 值1 if 条件 else 值2

# 如果条件成立,那么将 “值1” 赋值给result变量,否则,将“值2”赋值给result变量

examples:

result = 'the result if the if succeeds' if option == True else 'the result if the if fails and falls to the else part'

三、lambda表达式

对于简单的函数,也存在一种简便的表示方式,即:lambda表达式

# ###################### 普通函数 ######################
# 定义函数(普通方式)
def func(arg):
return arg + 1 # 执行函数
result = func(123) # ###################### lambda ###################### # 定义函数(lambda表达式)
my_lambda = lambda arg : arg + 1 # 执行函数
result = my_lambda(123)

四、列表解析、字典解析

myList = [1,2,3,4]
newList =[i*2 for i in myList] #经过列表解析后,newList = [2,4,6,8]
myList = [1,2,3,4]
myDic = {i:i*2 for i in myList} #得到字典{1:2,2:4,3:6,4:8}

五、递归

利用函数编写如下数列:

斐波那契数列指的是这样一个数列 0, 1, 1, 2, 3, 5, 8, 13, 21, 34, 55, 89, 144, 233,377,610,987,1597,2584,4181,6765,10946,17711,28657,46368...

def func(arg1,arg2):
if arg1 == 0:
print arg1, arg2
arg3 = arg1 + arg2
print arg3
func(arg2, arg3) # 在当前函数定义中调用当前函数,体现了递归 func(0,1)