3

我有没有按预期工作的 python 3 代码:

def addFunc(x,y):
    print (x+y)

def subABC(x,y,z):
    print (x-y-z)

def doublePower(base,exp):
    print(2*base**exp)

def  RootFunc(inputDict):
    for k,v in inputDict.items():
        if v[0]==1:
            d[k] = addFunc(*v[1:])
        elif v[0] ==2:
            d[k] = subABC(*v[1:])
        elif  v[0]==3:
            d[k] = doublePower(*v[1:])


d={"s1_7":[1,5,2],"d1_6":[2,12,3,3],"e1_3200":[3,40,2],"s2_13":[1,6,7],"d2_30":[2,42,2,10]}
RootFunc(d)

#test to make sure key var assignment works
print(d)

我得到:

{'d2_30': None, 's2_13': None, 's1_7': None, 'e1_3200': None, 'd1_6': None}

我期望:

{'d2_30': 30, 's2_13': 13, 's1_7': 7, 'e1_3200': 3200, 'd1_6': 6}

怎么了?

半相关:我知道字典是无序的,但是 python 选择这个顺序有什么原因吗?它是否通过随机化器运行密钥?

4

2 回答 2

12

print不返回值。它返回None,因此每次调用函数时,它们都会打印到标准输出并返回None。尝试将所有print语句更改为return

 def addFunc(x,y):
     return x+y

这会将值x+y返回给调用该函数的任何内容。

您的代码的另一个问题(除非您打算这样做)是您定义了一个字典d,然后当您定义您的函数时,您正在处理该字典d而不是“输入”字典:

def  RootFunc(inputDict):
    for k,v in inputDict.items():
        if v[0]==1:
            d[k] = addFunc(*v[1:])

您是否打算始终更改d不是您正在迭代的字典inputDict

可能还有其他问题(例如,在函数中接受可变数量的参数),但最好一次解决一个问题。


关于函数的附加说明:

下面是某种伪代码,试图传达函数的常用用法:

def sample_function(some_data):
     modified_data = []
     for element in some_data:
          do some processing
          add processed crap to modified_data
     return modified_data

函数被认为是“黑匣子”,这意味着您可以对它们进行结构化,以便您可以将一些数据转储到它们中,并且它们总是做同样的事情,您可以一遍又一遍地调用它们。它们将要么return值或yield值,要么更新某些值或属性或某些东西(后者称为“副作用”)。目前,只需注意return声明。

另一个有趣的事情是函数具有“范围”,这意味着当我使用参数的假名定义它时,我实际上不必有一个名为“some_data”的变量。我可以将任何我想要的东西传递给函数,但是在函数内部我可以引用假名并创建其他真正只在函数上下文中重要的变量。

现在,如果我们在上面运行我的函数,它将继续处理数据:

 sample_function(my_data_set)

但这通常是没有意义的,因为该函数应该返回一些东西,而我没有对它返回的东西做任何事情。我应该做的是将函数的值及其参数分配给某个容器,以便我可以保留处理后的信息。

my_modified_data = sample_function(my_data_set)

这是使用函数的一种非常常见的方式,您可能会再次看到它。


解决问题的一种简单方法:

考虑到所有这些,这是解决您的问题的一种方法,它来自一个真正常见的编程范式:

def  RootFunc(inputDict):
    temp_dict = {}
    for k,v in inputDict.items():
        if v[0]==1:
            temp_dict[k] = addFunc(*v[1:])
        elif v[0] ==2:
            temp_dict[k] = subABC(*v[1:])
        elif  v[0]==3:
            temp_dict[k] = doublePower(*v[1:])
    return temp_dict


  inputDict={"s1_7":[1,5,2],"d1_6":[2,12,3,3],"e1_3200":[3,40,2],"s2_13":[1,6,7],"d2_30"[2,42,2,10]}
  final_dict = RootFunc(inputDict)
于 2013-08-05T04:49:30.880 回答
2

正如 erewok 所说,您使用的是“打印”而不是“返回”,这可能是您错误的根源。就排序而言,您已经知道字典是无序的,至少根据 python doc,排序不是随机的,而是作为哈希表实现的。

摘自python 文档:[...]映射对象将可散列值映射到任意对象。映射是可变对象。目前只有一种标准映射类型,即字典。[...]

现在关键是元素的顺序并不是真正随机的。我经常注意到,无论我如何在某些值上构建字典,顺序都保持不变......使用 lambda 或直接创建它,顺序始终保持不变,所以它不能是随机的,但它绝对是随意的。

于 2013-08-05T04:57:23.113 回答