2
2
3
3
#函数(3)
4
4
5
- 在设计函数的时候 ,有时候我们能够确认参数的个数,比如一个用来计算圆面积的函数,它所需要的参数就是半径(πr^2),这个函数的参数是确定的。
5
+ 对于函数的参数 ,有时候我们能够确认参数的个数,比如一个用来计算圆面积的函数,它所需要的参数就是半径(πr^2),这个函数的参数是确定的。
6
6
7
7
> 你能不能写一个能够计算圆面积的函数呢?
8
8
9
- 然而,这个世界不总是这么简单的,也不总是这么确定的,反而不确定性是这个世界常常存在的。如果看官了解量子力学 ——好多人听都没有听过的东西——就更理解真正的不确定性了。当然,不用研究量子力学也一样能够体会到,世界充满里了不确定性。不是吗?塞翁失马焉知非福,这不就是不确定性吗?
9
+ 然而,这个世界不总是这么简单的,也不总是这么确定的,反而不确定性是这个世界常常存在的。如果读者了解量子力学 ——好多人听都没有听过的东西——就更理解真正的不确定性了。当然,不用研究量子力学也一样能够体会到,世界充满里了不确定性。不是吗?塞翁失马焉知非福,这不就是不确定性吗?
10
10
11
11
##参数收集
12
12
13
- 既然有很多不确定性,那么函数的参数的个数,也当然有不确定性,函数怎么解决这个问题呢?python用这样的方式解决参数个数的不确定性:
13
+ 世界是不确定的,那么函数参数的个数,也当然有不确定的时候,怎么解决这个问题呢?Python用这样的方式解决参数个数的不确定性。
14
14
15
- def func(x,*arg):
16
- print x #输出参数x的值
15
+ def func(x, *arg):
16
+ print x #Python 3请自动修改为print()的格式,下同,从略。
17
17
result = x
18
18
print arg #输出通过*arg方式得到的值
19
19
for i in arg:
20
20
result +=i
21
21
return result
22
22
23
- print func(1,2,3,4,5,6,7,8, 9) #赋给函数的参数个数不仅仅是2个
23
+ print func(1, 2, 3, 4, 5, 6, 7, 8, 9) #赋给函数的参数个数不仅仅是2个
24
24
25
25
运行此代码后,得到如下结果:
26
26
27
27
1 #这是函数体内的第一个print,参数x得到的值是1
28
28
(2, 3, 4, 5, 6, 7, 8, 9) #这是函数内的第二个print,参数arg得到的是一个元组
29
29
45 #最后的计算结果
30
30
31
- 从上面例子可以看出,如果输入的参数个数不确定,其它参数全部通过* arg,以元组的形式由arg收集起来。对照上面的例子不难发现:
31
+ 从上面例子可以看出,如果输入的参数个数不确定,其它参数全部通过` *arg ` ,以元组的形式由arg收集起来。对照上面的例子不难发现:
32
32
33
- - 值1传给了参数x
34
- - 值2,3,4,5,6.7.8.9被塞入一个tuple里面,传给了arg
33
+ - 值1传给了参数 ` x `
34
+ - 值2,3,4,5,6,7,8,9被塞入一个元组里面,传给了 ` arg `
35
35
36
- 为了能够更明显地看出* args(名称可以不一样,但是* 符号必须要有),可以用下面的一个简单函数来演示:
36
+ 为了能够更明显地看出` *args ` (名称可以不一样,但是* 符号必须要有),可以用下面的一个简单函数来演示:
37
37
38
38
>>> def foo(*args):
39
- ... print args #打印通过这个参数得到的对象
39
+ ... print args #Python 3: print(args)
40
40
...
41
41
42
42
下面演示分别传入不同的值,通过参数* args得到的结果:
43
43
44
- >>> foo(1,2, 3)
44
+ >>> foo(1, 2, 3)
45
45
(1, 2, 3)
46
46
47
- >>> foo("qiwsir","qiwsir.github.io","python")
47
+ >>> foo("qiwsir", "qiwsir.github.io", "python")
48
48
('qiwsir', 'qiwsir.github.io', 'python')
49
49
50
- >>> foo("qiwsir",307,["qiwsir",2],{"name":"qiwsir","lang":"python"})
50
+ >>> foo("qiwsir", 307, ["qiwsir", 2], {"name":"qiwsir", "lang":"python"})
51
51
('qiwsir', 307, ['qiwsir', 2], {'lang': 'python', 'name': 'qiwsir'})
52
52
53
- 不管是什么,都一股脑地塞进了tuple中 。
53
+ 不管是什么,都一股脑地塞进了元组中 。
54
54
55
55
>>> foo("python")
56
56
('python',)
57
57
58
- 即使只有一个值,也是用tuple收集它 。特别注意,在tuple中 ,如果只有一个元素,后面要有一个逗号。
58
+ 即使只有一个值,也是用元组收集它 。特别注意,在元组中 ,如果只有一个元素,后面要有一个逗号。
59
59
60
60
还有一种可能,就是不给那个` *args ` 传值,也是许可的。例如:
61
61
62
62
>>> def foo(x, *args):
63
- ... print "x:",x
63
+ ... print "x:",x #Python 3: print("x:"+str(x))
64
64
... print "tuple:",args
65
65
...
66
66
>>> foo(7)
67
67
x: 7
68
68
tuple: ()
69
69
70
- 这时候` *args ` 收集到的是一个空的tuple 。
70
+ 这时候` *args ` 收集到的是一个空的元组 。
71
71
72
72
> 在各类编程语言中,常常会遇到以foo,bar,foobar等之类的命名,不管是对变量、函数还是后面要讲到的类。这是什么意思呢?下面是来自维基百科的解释。
73
73
77
77
78
78
> 单词“foobar”或分离的“foo”与“bar”常出现于程序设计的案例中,如同Hello World程序一样,它们常被用于向学习者介绍某种程序语言。“foo”常被作为函数/方法的名称,而“bar”则常被用作变量名。
79
79
80
- 除了用* args这种形式的参数接收多个值之外 ,还可以用** kargs的形式接收数值,不过这次有点不一样:
80
+ 除了用` *args ` 这种形式的参数接收多个值之外 ,还可以用** kargs的形式接收数值,不过这次有点不一样:
81
81
82
82
>>> def foo(**kargs):
83
- ... print kargs
83
+ ... print kargs #Python 3: print(kargs)
84
84
...
85
85
>>> foo(a=1,b=2,c=3) #注意观察这次赋值的方式和打印的结果
86
86
{'a': 1, 'c': 3, 'b': 2}
94
94
95
95
如果用` **kargs ` 的形式收集值,会得到dict类型的数据,但是,需要在传值的时候说明“键”和“值”,因为在字典中是以键值对形式出现的。
96
96
97
- 看官到这里可能想了 ,不是不确定性吗?我也不知道参数到底会可能用什么样的方式传值呀,这好办,把上面的都综合起来。
97
+ 读者到这里可能想了 ,不是不确定性吗?我也不知道参数到底会可能用什么样的方式传值呀,这好办,把上面的都综合起来。
98
98
99
99
>>> def foo(x,y,z,*args,**kargs):
100
- ... print x
100
+ ... print x #Python 3用户请修改为print()格式,下同
101
101
... print y
102
102
... print z
103
103
... print args
124
124
125
125
很good了,这样就能够足以应付各种各样的参数要求了。
126
126
127
- ##另外一种传值方式
127
+ ##一种优雅的姿势
128
128
129
- >>> def add(x,y):
129
+ >>> def add(x, y):
130
130
... return x + y
131
131
...
132
- >>> add(2,3)
132
+ >>> add(2, 3)
133
133
5
134
134
135
- 这是通常的函数调用方法,在前面已经屡次用到 。这种方法简单明快,很容易理解。但是,世界总是多样性的,有时候你秀出下面的方式,甚至在某种情况用下面的方法可能更优雅。
135
+ 这是通常的函数调用时的传值方法 。这种方法简单明快,很容易理解。但是,世界总是多样性的,有时候你秀出下面的方式,甚至在某种情况用下面的方法可能更优雅。
136
136
137
- >>> bars = (2,3)
137
+ >>> bars = (2, 3)
138
138
>>> add(*bars)
139
139
5
140
140
141
141
先把要传的值放到元组中,赋值给一个变量` bars ` ,然后用` add(*bars) ` 的方式,把值传到函数内。这有点像前面收集参数的逆过程。注意的是,元组中元素的个数,要跟函数所要求的变量个数一致。如果这样:
142
142
143
- >>> bars = (2,3, 4)
143
+ >>> bars = (2, 3, 4)
144
144
>>> add(*bars)
145
145
Traceback (most recent call last):
146
146
File "<stdin>", line 1, in <module>
150
150
151
151
这是使用一个星号` * ` ,是以元组形式传值,如果用` ** ` 的方式,是不是应该以字典的形式呢?理当如此。
152
152
153
- >>> def book(author,name):
154
- ... print "%s is writing %s" % ( author,name)
153
+ >>> def book(author, name):
154
+ ... print "{0}} is writing {1}".format (author,name) #Python 3: print("{0}}is writing {1}".format ( author,name) )
155
155
...
156
- >>> bars = {"name":"Starter learning Python","author":"Kivi"}
156
+ >>> bars = {"name":"Starter learning Python", "author":"Kivi"}
157
157
>>> book(**bars)
158
158
Kivi is writing Starter learning Python
159
159
160
160
这种调用函数传值的方式,至少在我的编程实践中,用的不多。不过,不代表读者不用。这或许是习惯问题。
161
161
162
- ##复习
162
+ ##融会贯通
163
163
164
- python中函数的参数通过赋值的方式来传递引用对象 。下面总结通过总结常见的函数参数定义方式,来理解参数传递的流程。
164
+ Python中函数的参数通过赋值的方式来传对象引用 。下面总结通过总结常见的函数参数定义方式,来理解参数传递的流程。
165
165
166
- ###def foo(p1,p2,p3,...)
166
+ ###def foo(p1, p2, p3, ...)
167
167
168
168
这种方式最常见了,列出有限个数的参数,并且彼此之间用逗号隔开。在调用函数的时候,按照顺序以此对参数进行赋值,特备注意的是,参数的名字不重要,重要的是位置。而且,必须数量一致,一一对应。第一个对象(可能是数值、字符串等等)对应第一个参数,第二个对应第二个参数,如此对应,不得偏左也不得偏右。
169
169
170
- >>> def foo(p1,p2,p3):
171
- ... print "p1==>",p1
170
+ >>> def foo(p1, p2, p3):
171
+ ... print "p1==>",p1 #Python 3用户修改为print()格式,下同
172
172
... print "p2==>",p2
173
173
... print "p3==>",p3
174
174
...
175
- >>> foo("python",1, ["qiwsir","github","io"]) #一一对应地赋值
175
+ >>> foo("python", 1, ["qiwsir","github","io"])
176
176
p1==> python
177
177
p2==> 1
178
178
p3==> ['qiwsir', 'github', 'io']
@@ -182,25 +182,25 @@ python中函数的参数通过赋值的方式来传递引用对象。下面总
182
182
File "<stdin>", line 1, in <module>
183
183
TypeError: foo() takes exactly 3 arguments (1 given) #注意看报错信息
184
184
185
- >>> foo("python",1,2, 3)
185
+ >>> foo("python",1, 2, 3)
186
186
Traceback (most recent call last):
187
187
File "<stdin>", line 1, in <module>
188
188
TypeError: foo() takes exactly 3 arguments (4 given) #要求3个参数,实际上放置了4个,报错
189
189
190
- ###def foo(p1=value1,p2=value2,...)
190
+ ###def foo(p1=value1, p2=value2, ...)
191
191
192
- 这种方式比前面一种更明确某个参数的赋值 ,貌似这样就不乱子了,很明确呀。颇有一个萝卜对着一个坑的意味。
192
+ 这种方式比前面一种更明确某个参数的值 ,貌似这样就不乱子了,很明确呀。颇有一个萝卜对着一个坑的意味。
193
193
194
194
还是上面那个函数,用下面的方式赋值,就不用担心顺序问题了。
195
195
196
- >>> foo(p3=3,p1=10,p2=222)
196
+ >>> foo(p3=3, p1=10, p2=222)
197
197
p1==> 10
198
198
p2==> 222
199
199
p3==> 3
200
200
201
201
也可以采用下面的方式定义参数,给某些参数有默认的值
202
202
203
- >>> def foo(p1,p2=22,p3=33): #设置了两个参数p2,p3的默认值
203
+ >>> def foo(p1, p2=22, p3=33): #设置了两个参数p2, p3的默认值
204
204
... print "p1==>",p1
205
205
... print "p2==>",p2
206
206
... print "p3==>",p3
@@ -209,16 +209,16 @@ python中函数的参数通过赋值的方式来传递引用对象。下面总
209
209
p1==> 11
210
210
p2==> 22
211
211
p3==> 33
212
- >>> foo(11,222) #按照顺序,p2=222,p3依旧维持原默认值
212
+ >>> foo(11, 222) #按照顺序,p2=222, p3依旧维持原默认值
213
213
p1==> 11
214
214
p2==> 222
215
215
p3==> 33
216
- >>> foo(11,222,333) #按顺序赋值
216
+ >>> foo(11, 222, 333) #按顺序赋值
217
217
p1==> 11
218
218
p2==> 222
219
219
p3==> 333
220
220
221
- >>> foo(11,p2=122)
221
+ >>> foo(11, p2=122)
222
222
p1==> 11
223
223
p2==> 122
224
224
p3==> 33
@@ -230,30 +230,30 @@ python中函数的参数通过赋值的方式来传递引用对象。下面总
230
230
231
231
###def foo(* args)
232
232
233
- 这种方式适合于不确定参数个数的时候,在参数args前面加一个* ,注意,仅一个哟。
233
+ 这种方式适合于不确定参数个数的时候,在参数args前面加一个` * ` ,注意,仅一个哟。
234
234
235
- >>> def foo(*args): #接收不确定个数的数据对象
235
+ >>> def foo(*args):
236
236
... print args
237
237
...
238
- >>> foo("qiwsir.github.io") #以tuple形式接收到,哪怕是一个
238
+ >>> foo("qiwsir.github.io")
239
239
('qiwsir.github.io',)
240
240
>>> foo("qiwsir.github.io","python")
241
241
('qiwsir.github.io', 'python')
242
242
243
- #### def foo(** args)
243
+ ###def foo(** args)
244
244
245
- 这种方式跟上面的区别在于,必须接收类似arg=val形式的 。
245
+ 这种方式跟上面的区别在于,必须接收类似 ` arg=val ` 形式的 。
246
246
247
- >>> def foo(**args): #这种方式接收,以dictionary的形式接收数据对象
247
+ >>> def foo(**args):
248
248
... print args
249
249
...
250
250
251
- >>> foo(1,2,3) #这样就报错了
251
+ >>> foo(1,2,3)
252
252
Traceback (most recent call last):
253
253
File "<stdin>", line 1, in <module>
254
254
TypeError: foo() takes exactly 0 arguments (3 given)
255
255
256
- >>> foo(a=1,b=2,c=3) #这样就可以了,因为有了键值对
256
+ >>> foo(a=1,b=2,c=3)
257
257
{'a': 1, 'c': 3, 'b': 2}
258
258
259
259
下面来一个综合的,看看以上四种参数传递方法的执行顺序
@@ -289,6 +289,8 @@ python中函数的参数通过赋值的方式来传递引用对象。下面总
289
289
targs_tuple==> ('3t1', '3t2')
290
290
dargs_dict==> {'d2': '4d2', 'd1': '4d1'}
291
291
292
+ 对函数的基本内容已经介绍完毕,但是,并不意味着结束,因为还有更深刻的东西没说呢,且看下节。
293
+
292
294
------
293
295
294
296
[ 总目录] ( ./index.md )   ;  ;  ; |  ;  ;  ; [ 上节:函数(2)] ( ./202.md )   ;  ;  ; |  ;  ;  ; [ 下节:函数(4)] ( ./204.md )
0 commit comments