标签:
什么是函数,函数说白了就是将一系列代码封装起来,实现代码的重用。
什么是代码重用?
假设我有这样的需求:
但是我还是觉得太麻烦了,每次想吃饭的时候都要重复这样的步骤。此时,我希望有这样的机器:
将重复的工作封装到一起,我们只要向机器里放入东西,就能得到我们想要的。
这也就是所谓的代码重用。
自定义函数
知道了函数是干什么用的之后,我们就开始学习自定义函数,也就是动手来造这个神奇的机器。
看代码示例:
defdfb(a):‘‘‘一系列操作‘‘‘
return ‘一碗%s饭‘ %a
a= dfb(‘米‘)
b = dfb(‘米‘)print a
print b
这样我们就得到了两碗饭,真是方便快捷。
现在来解释里面都有什么:
1. def 是python的关键字,是专门用来自定义函数的。
2. dfb是函数名,用来以后调用的。
3.(a)中的a为函数的参数,为函数里面的操作提供数据的。
4.return用来返回一个对象,这个对象可以是函数的处理结果,也可以是函数的处理状态等等。
1.def
没什么好解释的,语法规定。
2.函数名
函数名就类似于变量名。
例如我写了一个函数,但我不调用它,那会怎么样。
defdfb(a):‘‘‘一系列操作‘‘‘
return ‘一碗%s饭‘ %a
什么也没有输出,那是不是意味着函数不存在呢?
我们看看内存里有没有:
print id(dfb)
很明显,函数在内存中,能够找得到。
所以,当我们定义一个函数的时候,python就会将函数加载到内存中,只不过不调用的时候,函数内部的代码就不执行。
很明显,和变量赋值原理差不多,所以要注意一个问题:如果函数名和变量名冲突了,相当于重新赋值。而python解释是从上到下的,也就是说此时谁在下面谁占用这个变量名。剩下的那个就只能在内存中等待垃圾回收了。
defdfb(a):‘‘‘一系列操作‘‘‘
return ‘一碗%s饭‘ %a
dfb= 1
print dfb
print dfb()
那么函数名的命名有什么要求吗?
函数名的要求比变量名严格一点,除了遵守变量名的要求之外,还不能够使用数字。
所以函数名只能使用字母和下划线(_),同时还要避开python的关键字。
另外,在pep8标准中,函数名提倡都要小写。
2.函数的作用于域
参数函数的一大重难点,很多人不明白所谓的形参和实参到底是怎么回事。
要明白其中的区别,首先还解释函数的作用域是什么回事。
所谓的函数作用域,就是这样。
python在执行函数里面的代码的时候,会将其放在一个新的环境中,而这种新的环境就像电脑安装了虚拟机一样。新环境中并没有之前所创建的对象,相当于我在虚拟机中找不到我本机的文件一样。
而函数运行之后,python会销毁函数的运行环境,也就是完成任务后就将虚拟机删除掉,里面的数据也就删除了。
这是,我问这样的一个问题:
a = 123
defdfb(a):‘‘‘一系列操作‘‘‘
return ‘一碗%s饭‘ %a
b= ‘米‘
print dfb(b)
函数会输出什么,a现在指向谁?
因为函数是有作用域的,虽然全局中有a这变量,但是函数执行的时候是在新环境执行的,也就自然没有a这个变量。而当我函数执行完毕后,执行环境就销毁了,而且本来就是相互不影响的。自然a的指向就没有被改变。
正是因为函数里面的a执行完毕后就消失了,所以它是形式上的,对全局来说没有什么意义。所以称其为形参,仅仅只是函数内部处理时用的。而当我们调用函数时,实际给的参数,如这里的 dfb(b) 中的b这个实际的参数,就称为实参。
而在函数内的变量就是局部变量,外部的就是全局变量。所以在函数中定义的变量,也就是局部变量,只在函数内部有效。
2.global
如果我要强制在函数里面进行全局变量的声明怎么办?
a = 123
defdfb(a):‘‘‘一系列操作‘‘‘b= 1
return ‘一碗%s饭‘ %a
dfb(‘米‘)print b
此时可以使用global关键字:
a = 123
defdfb(a):‘‘‘一系列操作‘‘‘
global b #我先声明我要创建一个全局变量b
b = 1 #然后我再为b赋值
return ‘一碗%s饭‘ %a
dfb(‘米‘)print b
这样就可以在函数内创建全局变量了。
此时有机智的同学就要问了,那我在函数内部声明的全局变量和之前的冲突,是不是会重新赋值呢?
答案:是的。
但要注意一个问题:
a = 123
defdfb(a):‘‘‘一系列操作‘‘‘
global a #我声明全局变量a,想要覆盖之前的赋值,而传参的时候,相对于进行了a=b的操作
return ‘一碗%s饭‘ %a
b= ‘米‘
print dfb(b)
然而报错了:
说明我们不能将形式参数声明为全局的。
那我们改一下形式参数的名称:
a = 123
defdfb(c):‘‘‘一系列操作‘‘‘
global a #我声明全局变量a,想要覆盖之前的赋值
a =creturn ‘一碗%s饭‘ %a
b= ‘米‘dfb(b)print a
可以了,说明我们的思路是正确的,在函数内声明全局变量确实会覆盖已经有的变量。
3.传参
传参是函数的又一大重点和难点。
传参是为了能使函数适用更多的情况,我们在上面的示例中都写列带参数的函数,是不是说函数一定要参数才行呢?
deftest():print ‘然而我并没有参数‘test()
可以看出没有参数也是可以的,只是没有参数的时候,无论怎么调用得到的结果都是一样的,灵活性太低,所以为了提高灵活性,就有了参数存在的必要。
而参数又分为普通参数,默认参数,动态参数,下面逐一说明。
1.普通参数
deftest(a,b,c):printaprintbprintc
test(1,2,3)
可以看出参数是按照顺序传递进去的,这种写法就叫普通函数。
但是这里有一个问题:
deftest(a,b,c):printaprintbprintc
test(1,2)
本来要传3个参数的,但是我仅传了2个,这样就报错了。但我不想这样,我希望当我不传参数的时候,参数有一个默认的值,此时就需要默认参数了。
2.默认参数
def test(a,b,c=3):printaprintbprintc
test(1,2)
传参就相当于为形参赋上实参的值,你给了参数,我就按顺序执行 a = 1 ,b = 2,但此时c已经赋值为3了,所以就算不传,参数的数量也够了。
当然也可以将参数传够, text(1,2,3) 就相当于为c重新赋值了。这样就能达到你传了参数就按照传的参数来处理,而没传就按默认的值处理。
所以我们可以为全部的参数都设置默认值。
但是可能会出现这样的情况:
def test(a=1,b,c=3):printaprintbprintc
test(1,2)
并不允许这样的写法,因为这样没有默认值的参数很难处理,所以当默认参数和普通参数同时存在时,要将普通参数放在前面:
def test(b,a=1,c=3):printaprintbprintc
test(1,2)
之所以要规定这样写,是因为要配合传参的方法:我们可以显式地规定哪个值是传给哪个参数的:
def test(b,a=1,c=3):printaprintbprintc
test(a=1,b=2,c=3)
如果是这样的显式传参的话,传参的顺序是任意的。也就是 text(a=1,c=3,b=2) 也可以,反正最后赋值的结果是一样的。
但是,如果是普通传参和显式传参配合使用时,就必须将默认传参放在开头: text(2,a=1,c=3,) 普通的按顺序传,显式可以不按顺序传。这也就是我们在写函数的时候要求普通的参数都放在前面,默认的参数都放在后面,就是为了和这里对应。
3.动态参数
因为我们的函数最后可能并不是只有自己用,而是给用户或其他人调用,但是其他人不一定了解我这里接受多少个参数。
传少了我们可以使用默认参数来解决,但是传多了怎么办,一旦传多了就报错用户体验就不好了,为了提高函数的适应能力,就出现了动态参数。
def test(a,*args,**kwargs):printa
test(1,2,c=3)
这样函数的适应性就更高了,那么这里的 *args 和 **kwargs 分别是什么意思。
我们先来看去是什么类型:
def test(a,*args,**kwargs):printtype(args),type(kwargs)
test(1,2,c=3)
它们是元祖和字典,这里注意前面的*号只是表示这个是什么类型的,*表示元祖,**代表字典,而真正的变量名是*后面的。这里并没有规定一定要用args命名元祖,kwargs命名字典。但是这是个规范的写法,为的是让别的程序员能一眼看出这是个什么东西。
接下来我们看看里面存的是什么:
def test(a,*args,**kwargs):printargsprintkwargs
test(1,2,c=3)
它将默认传参方式多传的值放在一个元祖里,而用显式传参多传的用其参数名为键,参数值为值,组成了字典。
你可以无视它们,也可以将处理它们,赋值操作也好,循环也好,成员判断也好,各种需要看个人。
4.return
当函数遇到return语句时,表示函数执行完毕,此时返回一个对象,然后销毁函数的执行环境。
但是你在上面的示例中看到也有这样的写法:
deftest():print ‘我并没写return‘test()
发现没有return语句还是能执行的,在调用的时候输出了东西,执行一遍后也停止了。
注意,在函数内没有写return语句的时候,默认return的是一个空对象。也就是就算没写,python内部也做了处理。
此时,有部分人分不清函数的输出和返回值的区别。
这样说吧,在函数里print之类的操作能够输出内容,是因为虽然函数的执行环境是独立的,但代码还是有效的。外部能进行的操作,函数内部也可以。但是并不是所有的函数在执行完毕后都有如此明显的输出效果,此时我们需要查看函数是否成功,或者说我放了米进去,你操作一番之后总要把饭给我拿出来吧。
这就是函数中return的意义。返回一个对象。这个对象可以是对执行状态的说明,也可以是处理后的结果等等。
deftest():‘‘‘一系列操作‘‘‘
return ‘搞定了‘test()
但运行还是没看到东西。
那是因为函数虽然返回了对象,但是这个对象还在内存中,你并没有把它拿出了,当然什么也看不到。
print test()
这样就看到了,当然你可以进行变量的赋值,如 a = test() ,之后调用变量a就行了。当然返回结果多用True和False代表成功和失败,然后可以和条件控制语句配合使用。
deftest():‘‘‘一系列操作‘‘‘
print test()
当然不写就默认返回空对象None了。
deftest(a,b):
c= a +breturncprint test(1,2)
返回处理后的结果也是可以的,反正看个人需求。
最后,虽说遇到return代表函数结束,但并意味着一个函数里面只有一个return。
deftest(a,b):if a !=0:return a +breturn ‘a不能为0‘
print test(1,2)print test(0,2)
可以配合条件控制语句实现不同情况返回不同的对象,这样就函数就能处理更多的情况了。
关于自定义函数就先说到这里,如有什么错误和需要补充的后面会相应修改。
标签: