(转)Python 函数之函数定义、调用、传参

下面是学习过程中搜集资料发现的一篇文章,写的很好,学习一下。
原文连接:http://blog.csdn.net/SeeTheWorld518/article/details/48089235

函数是组织好的,可以被重复使用的,用来实现单一或相关功能的代码块。 
任何编程语言都提供了很多内嵌函数,同时还允许用户自己定义想要的函数。本节主要讲自定义函数。

1、定义函数的规则

我们自定义函数时,需要满足一些规则,否则编译器不会识别。 
(1)函数代码块以def关键词开头,后接函数标识符名和圆括号()。 
(2)任何传入实参和形参必须放在圆括号中。圆括号中写的是形参。 
(3)函数的第一行语句可以选择性地使用文档字符串—-用于存放函数说明。 
(4)函数内容以冒号起始,并且函数体要有缩进。 
(5)return[expression]结束函数,选择性地返回一个值给调用者。不带表达式的return相当于返回None。

2、定义函数语法

语法:

def functionname( parameters ) :
   "函数文档字符串(即函数说明)"
   function_suite(函数体)
   return [expression]
 
 
  • 1
  • 2
  • 3
  • 4

默认情况下,参数值和参数名称是按函数声明中定义的顺序匹配起来的。

3、函数调用

函数被定义后,本身是不会自动执行的,只有在被调用后,才会被执行。 
下面通过一个例子来讲解函数的调用。

def printStr(str1) :
  "打印任何传入的字符串"
  print str1
 
 
  • 1
  • 2
  • 3

调用上面的打印字符串函数 
原函数在定义时,定义了一个形参str1,所以在调用函数时,也必须传入一个参数,否则会报错。

printStr("调用字符串打印函数")
 
 
  • 1

执行结果:

调用字符串打印函数
 
 
  • 1

4、传参

先来看两个例子。 
实例1:传递变量

#!/usr/bin/python
#coding=utf-8

def changeNum(num) :
  num += 1
  print "自定义函数中的num = ",num

#定义变量num,赋初始值为10
num = 10
#
changeNum(num)
print "函数调用后num = ",num
 
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12

执行结果:

自定义函数中的num =  11
函数调用后num =  10
 
 
  • 1
  • 2

实例2 :传序列list

#!/usr/bin/python
# -*- coding: utf-8 -*-
def changeList(list1) :
  list1.append('newStr')
  print "函数中的list :",list1

#定义变量num,赋初始值为10
list1 = [1,2,3]
print "调用函数前的list: ",list1
changeList(list1)
print "调用函数后的list:",list1
 
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11

执行结果:

调用函数前的list:  [1, 2, 3]
函数中的list : [1, 2, 3, 'newStr']
调用函数后的list: [1, 2, 3, 'newStr']
 
 
  • 1
  • 2
  • 3

从上面的结果可以看出,在函数中改变num变量的值,但是在函数调用之外num变量的值却没有改变。而在第二个实例中,在函数中像list1中增加一个元素’newStr’,在函数外查看原list1,发现也增加了一个元素’newStr’。也就是说在函数中改变list1,原list1也跟着改变。但是从这样我们就能说在Python中,实例1是传值,实例2是传引用吗?

其实非也,和其他语言不一样,在Python中,一切皆对象, Python参数传递采用的都是“传对象引用”的方式。实际上,这种方式相当于传值和传引用的一种综合。如果函数收到的是一个可变对象(比如字典或者列表)的引用,就能修改对象的原始值,相当于通过“传引用”来传递对象。如果函数收到的是一个不可变对象(比如数字、字符或者元组)的引用,就不能直接修改原始对象,相当于通过“传值’来传递对象,此时如果想改变这些变量的值,可以将这些变量申明为全局变量。

--------------------------------------分割线------------------------------------------------------------------------------

又看到一篇很好的文章,看过之后也会帮助你理解
原文连接:https://www.cnblogs.com/loleina/p/5276918.html



 首先还是应该科普下函数参数传递机制传值和传引用是什么意思?

   函数参数传递机制问题在本质上是调用函数(过程)和被调用函数(过程)在调用发生时进行通信的方法问题。基本的参数传递机制有两种:值传递和引用传递。

  值传递(passl-by-value)过程中,被调函数的形式参数作为被调函数的局部变量处理,即在堆栈中开辟了内存空间以存放由主调函数放进来的实参的值,从而成为了实参的一个副本。值传递的特点是被调函数对形式参数的任何操作都是作为局部变量进行,不会影响主调函数的实参变量的值。

  引用传递(pass-by-reference)过程中,被调函数的形式参数虽然也作为局部变量在堆栈中开辟了内存空间,但是这时存放的是由主调函数放进来的实参变量的地址。被调函数对形参的任何操作都被处理成间接寻址,即通过堆栈中存放的地址访问主调函数中的实参变量。正因为如此,被调函数对形参做的任何操作都影响了主调函数中的实参变量。

      在python中实际又是怎么样的呢?

      先看一个简单的例子:

   

复制代码
from ctypes import *
import os.path  
import sys

def test(c):
    print "test before "
    print id(c)
    c+=2
    print "test after +"
    print id(c)
    return c

def printIt(t):
    for i in range(len(t)):
        print t[i]

if __name__=="__main__":
    a=2
    print "main before invoke test"
    print id(a)
    n=test(a)
    print "main afterf invoke test"
    print a
    print id(a)
    
复制代码

运行后结果如下:

 

复制代码
>>> 
main before invoke test
39601564
test before 
39601564
test after +
39601540
main afterf invoke test
2
39601564
复制代码

 

 id函数可以获得对象的内存地址.很明显从上面例子可以看出,将a变量作为参数传递给了test函数,传递了a的一个引用,把a的地址传递过去了,所以在函数内获取的变量C的地址跟变量a的地址是一样的,但是在函数内,对C进行赋值运算,C的值从2变成了4,实际上2和4所占的内存空间都还是存在的,赋值运算后,C指向4所在的内存。而a仍然指向2所在的内存,所以后面打印a,其值还是2.

   如果还不能理解,先看下面例子

复制代码

>>> a=1
>>> b=1
>>> id(a)
40650152
>>> id(b)
40650152
>>> a=2
>>> id(a)
40650140

复制代码

       a和b都是int类型的值,值都是1,而且内存地址都是一样的,这已经表明了在python中,可以有多个引用指向同一个内存(画了一个很挫的图,见谅),在给a赋值为2后,再次查看a的内存地址,都已经变化了

      

       而基于最前面的例子,大概可以这样描述:

        

      那python函数传参就是传引用?然后传参的值在被调函数内被修改也不影响主调函数的实参变量的值?再来看个例子。

复制代码
from ctypes import *
import os.path  
import sys

def test(list2):
    print "test before "
    print id(list2)
    list2[1]=30
    print "test after +"
    print id(list2)
    return list2

def printIt(t):
    for i in range(len(t)):
        print t[i]

if __name__=="__main__":
    list1=["loleina",25,'female']
    print "main before invoke test"
    print id(list1)
    list3=test(list1)
    print "main afterf invoke test"
    print list1
    print id(list1)
    
复制代码

      实际值为:

复制代码
>>> 
main before invoke test
64129944
test before 
64129944
test after +
64129944
main afterf invoke test
['loleina', 30, 'female']
64129944
复制代码

      发现一样的传值,而第二个变量居然变化,为啥呢?

      实际上是因为python中的序列:列表是一个可变的对象,就基于list1=[1,2] list1[0]=[0]这样前后的查看list1的内存地址,是一样的。

   

复制代码
>>> list1=[1,2]
>>> id(list1)
64185208
>>> list1[0]=[0]
>>> list1
[[0], 2]
>>> id(list1)
64185208
复制代码

 

      结论:python不允许程序员选择采用传值还是传引用。Python参数传递采用的肯定是“传对象引用”的方式。这种方式相当于传值和传引用的一种综合。如果函数收到的是一个可变对象(比如字典或者列表)的引用,就能修改对象的原始值--相当于通过“传引用”来传递对象。如果函数收到的是一个不可变对象(比如数字、字符或者元组)的引用,就不能直接修改原始对象--相当于通过“传值'来传递对象。


评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值