如何将具有多个键的字典创建为一个值?

时间:2012-04-12 12:45:37

标签: python dictionary

我对我要制作的字典有疑问。我的目标是为单个值设置多个键,如下所示:

dictionary = {('a', 'b'): 1, ('c', 'd'): 2}
assert dictionary['a'] == 1
assert dictionary['b'] == 1

有什么想法吗?

5 个答案:

答案 0 :(得分:29)

我猜你的意思是:

class Value:
    def __init__(self, v=None):
        self.v = v

v1 = Value(1)
v2 = Value(2)

d = {'a': v1, 'b': v1, 'c': v2, 'd': v2}
d['a'].v += 1

d['b'].v == 2 # True
  • Python的字符串和数字是不可变的对象,
  • 因此,如果您希望d['a']d['b']在更改时指向“更新”的相同值,请将值引用到 mutable 对象(如上所述的用户定义类,或dictlistset)。
  • 然后,当您在d['a']修改对象时,d['b']会同时更改,因为它们都指向同一个对象。

答案 1 :(得分:5)

如果使用fromkeys,您的示例会创建多个键:值对。如果您不想这样做,可以使用一个密钥并为密钥创建别名。例如,如果您使用寄存器映射,则您的密钥可以是寄存器地址,而别名可以是寄存器名称。这样,您就可以在正确的寄存器上执行读/写操作。

>>> mydict = {}
>>> mydict[(1,2)] = [30, 20]
>>> alias1 = (1,2)
>>> print mydict[alias1]
[30, 20]
>>> mydict[(1,3)] = [30, 30]
>>> print mydict
{(1, 2): [30, 20], (1, 3): [30, 30]}
>>> alias1 in mydict
True

答案 2 :(得分:4)

如果你要经常添加到这本词典,你会想要采用基于类的方法,类似于@Tutty在这个问题2d-dictionary-with-many-keys-that-will-return-the-same-value中的答案。

但是,如果您有一个静态字典,并且只需要通过多个键访问值,那么您可以使用两个字典的非常简单的路径。一个用于存储别名键关联,另一个用于存储实际数据:

alias = {
    'a': 'id1',
    'b': 'id1',
    'c': 'id2',
    'd': 'id2'
}

dict = {
    'id1': 1,
    'id2': 2
}

dict[alias['a']]

如果你需要添加到字典中,你可以编写这样的函数来使用这两个字典:

def add(key, id, value=None)
    if id in dict:
        if key in alias:
            # Do nothing
            pass
        else:
            alias[key] = id
    else:
        dict[id] = value
        alias[key] = id

add('e', 'id2')
add('f', 'id3', 3)

虽然这很有效,但我认为最终如果你想做这样的事情,那么编写自己的数据结构可能就好了,尽管它可以使用类似的结构。

答案 3 :(得分:4)

很简单。首先要了解Python解释器的设计。如果任何两个或更多变量具有与它映射到该值的相同值,它基本上不为所有变量分配内存。

让我们转到代码示例

In [6]: a = 10

In [7]: id(a)
Out[7]: 10914656

In [8]: b = 10

In [9]: id(b)
Out[9]: 10914656

In [10]: c = 11

In [11]: id(c)
Out[11]: 10914688

In [12]: d = 21

In [13]: id(d)
Out[13]: 10915008

In [14]: e = 11

In [15]: id(e)
Out[15]: 10914688

In [16]: e = 21

In [17]: id(e)
Out[17]: 10915008

In [18]: e is d
Out[18]: True
In [19]: e = 30

In [20]: id(e)
Out[20]: 10915296

从上面的输出中,变量a和b共享相同的内存,当我创建一个新的变量e时,c和d有不同的内存并存储一个已经存在于变量c中的值(11),因此它映射到当我将变量e中存在的值更改为已经存在于变量d中的21时,内存位置并不会创建新内存,所以现在变量d和e共享相同的内存位置。最后,我将变量e中的值更改为30,而不是存储在任何其他变量中,因此它为e创建了一个新的内存。

所以任何具有相同值的变量都会共享内存。

  

不适用于列表和字典对象

让我们回答你的问题。

当多个键具有相同的值时,则所有键共享相同的内存,因此您期望的内容已经存在于python中。

你可以像这样使用它

In [49]: dictionary = {
    ...:     'k1':1,
    ...:     'k2':1,
    ...:     'k3':2,
    ...:     'k4':2}
    ...:     
    ...:     

In [50]: id(dictionary['k1'])
Out[50]: 10914368

In [51]: id(dictionary['k2'])
Out[51]: 10914368

In [52]: id(dictionary['k3'])
Out[52]: 10914400

In [53]: id(dictionary['k4'])
Out[53]: 10914400

从上面的输出中,键k1和k2映射到相同的地址,这意味着值1只存储在内存中,这是多键单值字典,这就是你想要的东西。 :P

答案 4 :(得分:0)

短版和中版

有多个键

#!/usr/bin/env python3

def get_keys(s):
    # Lower the user's entry to easily manipulate data
    s = s.lower()
    
    # Create a list to simulate multiple keys
    numbers = ['uno', 'one', 'um', 'eins', 'ein']
    
    # Lambda for input validation
    validator = lambda x: x if x in numbers else 'no-key-found'  # return me x if x is found in the list numbers, contratiwise return me 'no-key-found'
    
    dic = {
        validator(s):'1',
        'no-key-found':'Key does not exist'
    }
    
    
    return dic[validator(s)]


print(get_keys(input('Type in word: ')))

更简单的版本

#!/usr/bin/env python3

import sys


def week_days():
    # Assets
    number_day = ['1', '2', '3', '4', '5', '6', '7']
    
    # Introduction
    print('Welcome to the Week Day Finder')
    
    # User input
    day = input('Please, enter the day you want to check: ').lower()
    WEEK_COLOR = {'RED': '\u001b[31m', 'GREEN': '\u001b[32m'}
    
    # Day validator
    days = {
        '1' if day in number_day else 'sunday': 'Weekend Day',
        '2' if day in number_day else 'monday': 'Week Day',
        '3' if day in number_day else 'tuesday': 'Week Day',
        '4' if day in number_day else 'wednesday': 'Week Day',
        '5' if day in number_day else 'thursday': 'Week Day',
        '6' if day in number_day else 'friday': 'Week Day',
        '7' if day in number_day else 'saturday': 'Weekend Day'
    }
    
    # Logical trial
    try:
        if days[day] == 'Week Day':
            print(WEEK_COLOR['GREEN'], days[day])
            
        elif days[day] == 'Weekend Day':
            print(WEEK_COLOR['RED'], days[day])
            
    except KeyError:
        print('** Invalid Day **', file=sys.stderr)
        
        
def main():
    week_days()
    
    
if __name__ == '__main__':
    main()