为什么我的元组值会发生变化(Python)?

时间:2016-05-16 17:17:00

标签: python artificial-intelligence machine-learning python-3.x

我有一个元组。第一个元素是一个浮点数,第二个元素是一个包含浮点数的嵌套列表的列表(不要担心这些值的含义)。

(0.2742965753780876, [[[33.119], [-8.326]], [[-34.084, -4.385], [-3.047, 18.546], [-10.757, 0.573], [21.395, 23.937], [5.011, -5.234]], [[-23.434, 9.989, 9.113, -23.253, 11.86], [-56.818, 4.771, -3.383, -27.143, 4.81], [-6.564, -40.132, -2.223, -2.663, -10.231], [-2.05, -15.989, 4.369, -20.051, 4.657]], [[-10.868, -39.934, 0.465, 38.103]], [-0.889, 1.129, 0.743]])
(0.2742965753780876, [[[21.633], [-8.972]], [[-35.754, -13.243], [-0.718, 17.724], [-16.452, 6.619], [24.151, 25.037], [1.76, -7.891]], [[-26.011, 9.072, 14.685, -20.044, 10.612], [-55.53, -0.131, 0.15, -27.031, 8.03], [-3.225, -36.499, -2.558, 0.253, -8.292], [-1.274, -22.561, 0.431, -23.405, 6.808]], [[-13.668, -47.758, -6.489, 43.27]], [-0.889, 1.129, 0.743]])
(0.2742965753780876, [[[22.435], [-6.71]], [[-47.591, -8.998], [-1.134, 16.529], [-16.399, 4.369], [23.344, 24.72], [2.175, -14.129]], [[-26.603, 11.472, 9.433, -21.13, 9.759], [-50.109, 1.084, 1.256, -18.826, 9.588], [-6.935, -27.957, 9.045, 1.291, 2.27], [-1.336, -29.908, -0.3, -27.242, 4.555]], [[-12.933, -42.377, 4.077, 38.864]], [-0.889, 1.129, 0.743]])
(0.2742965753780876, [[[26.688], [-4.315]], [[-49.478, -4.214], [0.116, 20.39], [-14.691, 3.496], [15.367, 23.116], [18.075, -2.748]], [[-25.588, 6.249, 4.364, -20.727, 19.639], [-55.524, -2.901, 4.639, -11.759, 11.794], [-8.633, -25.316, 11.841, 1.492, 1.36], [-0.797, -26.306, 1.379, -16.266, -0.291]], [[-24.726, -46.726, 12.765, 38.977]], [-0.889, 1.129, 0.743]])
(0.2742965753780876, [[[21.776], [-8.466]], [[-47.66, -5.868], [1.855, 23.062], [-19.521, 18.331], [29.251, 25.491], [21.32, -5.379]], [[-36.199, 7.786, -1.48, -27.042, 14.769], [-61.468, -12.218, -10.307, -6.156, 8.287], [-17.785, -33.124, 16.564, 2.249, -0.675], [-4.391, -18.11, 7.349, -9.234, -2.31]], [[-23.139, -55.043, 9.106, 35.827]], [-0.889, 1.129, 0.743]])

这些值是通过遗传算法训练人工神经网络获得的。正如你所看到的,我发布了五代顶级元组。如果您熟悉ML,您会注意到第一个元素是ANN在对训练数据进行分类时产生的均方误差,第二个元素是与该误差相关的ANN中的权重列表。

每当我创造新一代时,我都会采用顶级元组,然后将它们放回去#34;进入新候选人名单,这就是上述错误多次出现的原因。它是每一代人中的最佳候选人。

但是,你不需要知道这一点来理解我的问题。

重要的是,每当我使用遗传算法函数运行新一代时,顶级错误(第一个元素)可能不会改变。这是正常的,因为这意味着算法还没有找到更好的解决方案。当然,权重(第二个元素)也不应该改变,因为如果它们发生了,它们会产生完全不同的错误。

不幸的是,事实并非如此。我的元组中的数据以某种方式被更改。正如您在上面的打印输出中所看到的,每次运行新一代时,错误都会保持不变,但权重会发生变化。

为什么会这样?为什么元组中的值才会改变?

在这里查看我的一些代码。这是产生突变的代码。 mutate()方法逐步执行权重列表(浮点数),并对这些权重进行小的随机增量。这就是它对0.2742965753780876所做的事情,即使它不应该这样做,因为在改变权重和不更新错误方面没有任何意义。它甚至不可能,因为元组是不可变的。

for i in range(10):
    mList = weightList[random.randint(0,4)][1] # pick a set of weights from the top 5
    newList = mutate(mList, weightList[0][0]) # create a new list by mutating certain values of the unmutated list
    n.setWeights(listToGenome(newList,n)) # give the ANN the new weights
    error = trainSine(n) # train the network on the training set and return the mean-squared error
weightList.append((error, child))

您可能还想知道,每次创建新一代时,我都会根据错误对列表进行排序,然后选择前十位候选人继续下一代。

errorList.sort()
for i in range(10):
    survivorList.append(errorList[i])

关于它。当然,还有更多代码,但如果有人需要,我会分享它。有什么想法吗?

1 个答案:

答案 0 :(得分:6)

元组不可变,但里面的列表仍然是。您需要对列表进行深层复制,这些列表在复制时复制值,而不包含所有变量引用等。您可以使用copy模块执行此操作。语法如下:

import copy
myList = [1, 5, 3, 9, 4]
myOtherList = copy.deepcopy(myList)

使用上面的代码,如果您更改myList,或者myList导致的任何变量发生变化,myOtherList将保持不变,独立于其他变量。