python实验_python实验之函数

本文详细介绍了Python中函数的使用,包括内置函数、自定义函数、关键字参数、可变参数、命名关键字参数、参数组合及递归函数的用法,并提供了多个示例进行解析,帮助读者深入理解Python函数的精髓。
摘要由CSDN通过智能技术生成

步骤1常用内置函数

比如int函数可以把其他类型的数据转化为整数:>>> int('123')>>> int(12.34)>>> float('12.34')>>> str(1.23)>>> str(100)>>> bool(1)>>> bool('')

步骤2函数名函数名其实就是指向一个函数对象的引用,完全可以把函数名赋给一个变量,相当于给这个函数起了一个“别名”:

>>> a = abs

#变量a指向abs函数

>>> a(-1) #所以也可以通过a调用abs函数

步骤3定义函数在Python中,定义一个函数要使用def语句,依次写出函数名、括号、括号中的参数和冒号:,然后,在缩进块中编写函数体,函数的返回值用return语句返回。我们以自定义一个求绝对值的my_abs函数为例:def my_abs(x):

if x>=0:

return x

else:

return -x如果想定义一个什么事也不做的空函数,可以用pass语句,可以用来作为占位符。修改一下my_abs的定义,对参数类型做检查,只允许整数和浮点数类型的参数。数据类型检查可以用内置函数isinstance()实现:

def my_abs(x):

if not isinstance(x, (int, float)):

raise TypeError('bad operand type')

if x >= 0:

return x

else:

return -x步骤4关键字参数可变参数允许你传入0个或任意个参数,这些可变参数在函数调用时自动组装为一个tuple。而关键字参数允许你传入0个或任意个含参数名的参数,这些关键字参数在函数内部自动组装为一个dict。def person(name, age,**kw):

print('name:', name, 'age:', age,'other:', kw)

def writePerson(file,name,age, **kw):

f = open(file,"w")

result1 ="name:%s,age:%d,other:%s"%(name,age,str(kw))

f.write(result1)

f.close()函数person除了必选参数name和age外,还接受关键字参数kw。在调用该函数时,可以只传入必选参数:>>> person('Michael', 30)name: Michael age: 30 other: {}也可以传入任意个数的关键字参数:>>> person('Bob', 35, city='Beijing')name: Bob age: 35 other: {'city': 'Beijing'}>>> person('Adam', 45, gender='M',job='Engineer')name: Adam age: 45 other: {'gender': 'M', 'job':'Engineer'}

和可变参数类似,也可以先组装出一个dict,然后,把该dict转换为关键字参数传进去:>>> extra = {'city': 'Beijing', 'job': 'Engineer'}

>>> person('Jack', 24, city=extra['city'], job=extra['job'])

name: Jack age: 24 other: {'city': 'Beijing', 'job': 'Engineer'}当然,上面复杂的调用可以用简化的写法:>>> extra = {'city': 'Beijing', 'job': 'Engineer'}

>>>writePerson("/home/hcna-ai/exam4/result1.txt",'Jack', 24,**extra)**extra表示把extra这个dict的所有key-value用关键字参数传入到函数的**kw参数,kw将获得一个dict,注意kw获得的dict是extra的一份拷贝,对kw的改动不会影响到函数外的extra。步骤5命名关键字参数如果要限制关键字参数的名字,就可以用命名关键字参数,例如,只接收city和job作为关键字参数。这种方式定义的函数如下:def person(name, age, *, city, job):

print(name, age,city, job)和关键字参数**kw不同,命名关键字参数需要一个特殊分隔符*,*后面的参数被视为命名关键字参数。调用方式如下:>>> person('Jack', 24, city='Beijing', job='Engineer')

Jack 24 Beijing Engineer如果函数定义中已经有了一个可变参数,后面跟着的命名关键字参数就不再需要一个特殊分隔符*了:def person(name, age, *args, city, job):

print(name, age,args, city, job)命名关键字参数必须传入参数名,这和位置参数不同。如果没有传入参数名,调用将报错命名关键字参数可以有缺省值,从而简化调用:def person(name, age, *, city='Beijing', job):

print(name, age,city, job)

def writePerson(file,name, age, *, city='Beijing', job):

f =open(file,"w")

result2 ="name:%s,age:%d,city:%s,job:%s"%(name,age,city,job)

f.write(result2)

f.close()由于命名关键字参数city具有默认值,调用时,可不传入city参数:>>> writePerson("/home/hcna-ai/exam4/result2.txt",'Jack',24, job='Engineer')使用命名关键字参数时,要特别注意,如果没有可变参数,就必须加一个*作为特殊分隔符。如果缺少*,Python解释器将无法识别位置参数和命名关键字参数。步骤6参数组合在Python中定义函数,可以用必选参数、默认参数、可变参数、关键字参数和命名关键字参数,这5种参数都可以组合使用。但是请注意,参数定义的顺序必须是:必选参数、默认参数、可变参数、命名关键字参数和关键字参数。比如定义一个函数,包含上述若干种参数:

def f1(a, b, c=0, *args, **kw):

print('a=', a, 'b =', b, 'c =', c, 'args =', args, 'kw =', kw)

def f2(a, b, c=0, *, d, **kw):

print('a=', a, 'b =', b, 'c =', c, 'd =', d, 'kw =', kw)

def f2_write(file,a, b, c=0, *, d,**kw):

f= open(file,"w")

result3= "a=%s,b=%s,c=%s,d=%s,kw=%s"%(a,b,c,d,str(kw))

f.write(result3)

f.close()

在函数调用的时候,Python解释器自动按照参数位置和参数名把对应的参数传进去。>>> f1(1, 2)

a = 1 b = 2 c = 0 args = () kw = {}

>>> f1(1, 2, c=3)

a = 1 b = 2 c = 3 args = () kw = {}

>>> f1(1, 2, 3, 'a', 'b')

a = 1 b = 2 c = 3 args = ('a', 'b') kw = {}

>>> f1(1, 2, 3, 'a', 'b', x=99)

a = 1 b = 2 c = 3 args = ('a', 'b') kw = {'x': 99}

>>> f2(1, 2, d=99, ext=None)

a = 1 b = 2 c = 0 d = 99 kw = {'ext': None}最神奇的是通过一个tuple和dict,你也可以调用上述函数:>>> args = (1, 2, 3, 4)

>>> kw = {'d': 99, 'x': '#'}

>>> f1(*args, **kw)

a = 1 b = 2 c = 3 args = (4,) kw = {'d': 99, 'x': '#'}

>>> args = (1, 2, 3)

>>> kw = {'d': 88, 'x': '#'}

>>> f2(*args, **kw)

a = 1 b = 2 c = 3 d = 88 kw = {'x': '#'}

>>>f2_write("/home/hcna-ai/exam4/result3.txt",*args,**kw)所以,对于任意函数,都可以通过类似func(*args, **kw)的形式调用它,无论它的参数是如何定义的。步骤7递归函数使用递归函数需要注意防止栈溢出。在计算机中,函数调用是通过栈(stack)这种数据结构实现的,每当进入一个函数调用,栈就会加一层栈帧,每当函数返回,栈就会减一层栈帧。由于栈的大小不是无限的,所以,递归调用的次数过多,会导致栈溢出。栈溢出解决方法:尾递归优化:解决递归调用栈溢出的方法是通过尾递归优化,事实上尾递归和循环的效果是一样的,所以,把循环看成是一种特殊的尾递归函数也是可以的。尾递归是指,在函数返回的时候,调用自身本身,并且,return语句不能包含表达式。这样,编译器或者解释器就可以把尾递归做优化,使递归本身无论调用多少次,都只占用一个栈帧,不会出现栈溢出的情况。上面的fact(n)函数由于return n * fact(n - 1)引入了乘法表达式,所以就不是尾递归了。要改成尾递归方式,需要多一点代码,主要是要把每一步的乘积传入到递归函数中:def fact(n):

return fact_iter(n,1)

def fact_iter(num, product):

if num == 1:

return

product

return fact_iter(num- 1, num * product)可以看到,return fact_iter(num - 1, num * product)仅返回递归函数本身,num - 1和num * product在函数调用前就会被计算,不影响函数调用。

>>> f4 =open("/home/hcna-ai/exam4/result4.txt","w")

>>>f4.write(str(fact(10)))

>>>f4.close()

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值