python中列表的值会更改,而我不会尝试更改它们

2024-09-30 22:19:28 发布

您现在位置:Python中文网/ 问答频道 /正文

我正在尝试创建一个程序来设置不同颜色的阈值,在这个程序中我有一个返回到上一个颜色范围的函数,但是每次我尝试向保存这些旧值的列表(minse,maxse)添加新值时,列表中特定颜色的所有列表都会转到新的附加列表。你知道吗

import cv2
import numpy as np
import cPickle as pickle
from matplotlib import pyplot as plt

def nothing(x):
    pass

cap = cv2.VideoCapture(0)
cv2.namedWindow('image')
#last values
minse = [[],[],[]]
maxse = [[],[],[]]
p = 0
try:
    mins, maxs= pickle.load(open("varvid.p", "rb"))
except:
    mins = np.array([[255,255,255],[255,255,255],[255,255,255]])
    maxs = np.array([[0,0,0],[0,0,0],[0,0,0]])


cv2.createTrackbar('brush_size','image',1,10, nothing)
cv2.createTrackbar('vahe','image',0,50, nothing)


# mouse callback function
def choose_color(event,x,y,flags,param):
    global mins, maxs, minse, maxse, p, brush_size, vahe

    if event == cv2.EVENT_LBUTTONDOWN:
        nx = brush_size
        ny = brush_size
        pixs = hsv[y-ny:y+ny,x-nx:x+nx,:] 
        print minse[p]
        print mins[p]
        minse[p].append(mins[p])
        print minse[p]
        maxse[p].append(maxs[p])
        mins[p]=np.minimum(mins[p], pixs.min(0).min(0) - vahe).clip(0, 255).astype('uint8')
        print mins[p]
        maxs[p]=np.maximum(maxs[p], pixs.max(0).max(0) + vahe).clip(0, 255).astype('uint8')

#deletes lasts maxs and mins
def eelmised_varvid():
    global maxs, maxse, mins, minse, p
    try:    

        maxs[p] = maxse[p].pop()
        mins[p] = minse[p].pop()

    except:
        mins[p] = np.array([255,255,255])
        maxs[p] = np.array([0,0,0])

cv2.namedWindow('tava')
cv2.setMouseCallback('tava', choose_color)

print("For exiting press 'q', to save the values 's', ball colors - 'p', yellow - 'y' and blue -  'b', default is  the ball")

while(True):
    # Capture frame-by-frame
    ret, frame = cap.read()

    hsv = cv2.cvtColor(frame,cv2.COLOR_BGR2HSV) 
    brush_size =cv2.getTrackbarPos('brush_size','image')
    vahe = cv2.getTrackbarPos('brush_size','image')

    # Display the resulting frame
    median = cv2.medianBlur(hsv,5)
    mask = cv2.inRange(hsv, mins[p], maxs[p])


    cv2.imshow('tava', median)
    cv2.imshow('mask', mask)
    k = cv2.waitKey(1) & 0xFF

    if k == ord('q'):
        break
    elif k == ord('p'):
        p = 0
    elif k == ord('y'):
        p = 1
    elif k == ord('b'):
        p = 2
    elif k == ord('s'):
        pickle.dump([mins,maxs], open("varvid.p", "wb"))
    elif k == ord('e'):
        eelmised_varvid()
# When everything done, release the capture
cap.release()
cv2.destroyAllWindows()

问题应该在代码的这一部分:

def choose_color(event,x,y,flags,param):
    global mins, maxs, minse, maxse, p, brush_size, vahe

    if event == cv2.EVENT_LBUTTONDOWN:
        nx = brush_size
        ny = brush_size
        pixs = hsv[y-ny:y+ny,x-nx:x+nx,:]
        print"The value that should be appended to the old value list:",mins[p]
        print"The old value list before appending:", minse[p]
        minse[p].append(mins[p])
        print"The old value list after appending:",minse[p]
        maxse[p].append(maxs[p])
        mins[p]=np.minimum(mins[p], pixs.min(0).min(0) - vahe).clip(0, 255).astype('uint8')
        maxs[p]=np.maximum(maxs[p], pixs.max(0).max(0) + vahe).clip(0, 255).astype('uint8')

例如:第一次点击: 应附加到旧值列表的值:[255] 追加之前的旧值列表:[] 追加后的旧值列表:[array([255,255,255])] 第二次点击: 应附加到旧值列表的值:[84 33 237] 追加之前的旧值列表:[array([84,33,237])] 追加后的旧值列表:[array([84,33,237]),array([84,33,237])]

编辑: 我试着对它进行更多的调试,发现这不是.append()函数的问题,但出于某种原因,这行代码:

mins[p]=np.minimum(mins[p], pixs.min(0).min(0) - vahe).clip(0, 255).astype('uint8')

更改minse[p]的值

如有任何帮助,恕不另行通知并提前感谢:)


Tags: 列表sizenparraycv2printnxbrush
1条回答
网友
1楼 · 发布于 2024-09-30 22:19:28

关键是,Python列表并不真正存储“数据”,它只存储对其他对象的引用。您的问题可以减少:

list_ = []
thing = [0, 0]
list_.append(thing)
print list_
# [[0, 0]]

thing[0] = 1
list_.append(thing)
print list_
# [[1, 0], [1, 0]]

因此,当您执行上述操作时,会得到一个类似[thing, thing]的列表。thing的瞬时内容决定了list_的内容。你知道吗

另外,请注意,现在执行以下操作时:

thing = [2, 2]
list_.append(thing)
print list_
# [[1, 0], [1, 0], [2, 2]]

列表的旧内容没有更改,因为您没有更改标签“thing”引用的对象。相反,您为标签“thing”指定了一个新对象,这不会与以前附加到该标签的对象发生冲突。你知道吗

只有当列表(或其他集合类型)中的内容是“可变”对象时,才需要考虑此行为,例如listdict或Numpy数组。有关更多信息,请参见this question。你知道吗

解决方法:复制你放在列表中的对象。就你而言:

minse[p].append(mins[p].copy())

相关问题 更多 >