在python中,你可以让一个函数返回多个值。这是一个人为的例子:
def divide(x, y):
quotient = x/y
remainder = x % y
return quotient, remainder
(q, r) = divide(22, 7)
这看起来非常有用,但看起来它也可以被滥用(“嗯..函数 X 已经计算出我们需要的中间值。让我们也让 X 返回那个值”)。
什么时候应该画线并定义不同的方法?
在python中,你可以让一个函数返回多个值。这是一个人为的例子:
def divide(x, y):
quotient = x/y
remainder = x % y
return quotient, remainder
(q, r) = divide(22, 7)
这看起来非常有用,但看起来它也可以被滥用(“嗯..函数 X 已经计算出我们需要的中间值。让我们也让 X 返回那个值”)。
什么时候应该画线并定义不同的方法?
绝对(对于您提供的示例)。
有一个内置函数divmod()
可以做到这一点。
q, r = divmod(x, y) # ((x - x%y)/y, x%y) Invariant: div*y + mod == x
还有其他例子:zip
, enumerate
, dict.items
。
for i, e in enumerate([1, 3, 3]):
print "index=%d, element=%s" % (i, e)
# reverse keys and values in a dictionary
d = dict((v, k) for k, v in adict.items()) # or
d = dict(zip(adict.values(), adict.keys()))
顺便说一句,大多数时候括号不是必需的。来自Python 库参考的引用:
元组可以通过多种方式构建:
- 使用一对括号来表示空元组:()
- 对单例元组使用尾随逗号:a 或 (a,)
- 用逗号分隔项目:a, b, c 或 (a, b, c)
- 使用 tuple() 内置: tuple() 或 tuple(iterable)
因此他们应该返回一个对象。在您的情况下,此对象是一个元组。将元组视为一种特殊的复合数据结构。有些语言几乎每个函数都返回多个值(Lisp 中的列表)。
有时 return(x, y)
而不是Point(x, y)
.
随着 Python 2.6 中命名元组的引入,在许多情况下,最好返回命名元组而不是普通元组。
>>> import collections
>>> Point = collections.namedtuple('Point', 'x y')
>>> x, y = Point(0, 1)
>>> p = Point(x, y)
>>> x, y, p
(0, 1, Point(x=0, y=1))
>>> p.x, p.y, p[0], p[1]
(0, 1, 0, 1)
>>> for i in p:
... print(i)
...
0
1
首先,请注意 Python 允许以下内容(不需要括号):
q, r = divide(22, 7)
关于您的问题,无论哪种方式都没有硬性规定。对于简单(通常是人为的)示例,给定函数似乎总是有可能具有单一目的,从而产生单一值。但是,在将 Python 用于实际应用程序时,您很快就会遇到许多需要返回多个值的情况,这会导致代码更简洁。
所以,我想说做任何有意义的事,不要试图遵守人为的约定。Python 支持多个返回值,因此请在适当的时候使用它。
您给出的示例实际上是一个 python 内置函数,称为divmod
. 因此,在某个时间点,有人认为它足以包含在核心功能中。
对我来说,如果它使代码更干净,那就是 Pythonic。比较这两个代码块:
seconds = 1234
minutes, seconds = divmod(seconds, 60)
hours, minutes = divmod(minutes, 60)
seconds = 1234
minutes = seconds / 60
seconds = seconds % 60
hours = minutes / 60
minutes = minutes % 60
是的,返回多个值(即一个元组)绝对是 Pythonic。正如其他人所指出的,Python 标准库以及备受推崇的 Python 项目中有很多示例。两个额外的评论:
Python 对方法参数的处理需要能够直接返回多个值。例如,在 C++ 中,方法参数可以通过引用传递,因此除了正式的返回值之外,您还可以为它们分配输出值。在 Python 中,参数是“通过引用”传递的(但在 Java 的意义上,不是 C++)。您不能为方法参数分配新值并将其反映在方法范围之外。例如:
// C++
void test(int& arg)
{
arg = 1;
}
int foo = 0;
test(foo); // foo is now 1!
与之比较:
# Python
def test(arg):
arg = 1
foo = 0
test(foo) # foo is still 0
这绝对是pythonic。事实上,您可以从一个函数返回多个值,就像 C 语言中的样板文件一样,您需要为在某处返回的每种类型组合定义一个结构。
然而,如果你从一个函数返回一些疯狂的值,比如 10 个值,你应该认真考虑将它们捆绑在一个类中,因为那时它会变得笨拙。
返回一个元组很酷。另请注意在 python 2.6 中添加的新命名元组,这可能会让您更喜欢:http: //docs.python.org/dev/library/collections.html#collections.namedtuple
OT:RSRE 的 Algol68 有一个奇怪的“/:=”运算符。例如。
INT quotient:=355, remainder;
remainder := (quotient /:= 113);
商为 3,余数为 16。
注意:通常“(x/:=y)”的值被丢弃,因为商“x”是通过引用分配的,但在 RSRE 的情况下,返回的值是余数。
对于简单的函数(例如divmod
. 如果它使代码可读,那就是 Pythonic。
如果返回值开始变得混乱,请检查函数是否做的太多,如果是则拆分它。如果像对象一样使用大元组,则将其设为对象。另外,考虑使用命名元组,它将成为 Python 2.6 中标准库的一部分。
我对 Python 还很陌生,但是元组技术对我来说似乎很 Python。但是,我有另一个可以提高可读性的想法。使用字典允许通过名称而不是位置访问不同的值。例如:
def divide(x, y):
return {'quotient': x/y, 'remainder':x%y }
answer = divide(22, 7)
print answer['quotient']
print answer['remainder']