19
votes

When we iterate over the dictionary below, each iteration returns(correctly) a key,value pair

for key, value in dict.items():
    print "%s key has the value %s" % (key, value)

'some key' key has the value 'some value' (repeated however many times there are a k,v pair)

The above makes sense to me, however if we do this:

for key in dict.items():
    print "%s key has the value %s" % (key, value)

("some key", "some value") has the value "some value" (the left tuple will iterate through each key value pair and the right value will just stay at the first value in the dict and repeat)

We end up getting each k,v pair returned in the first %s (key) and the 2nd %s (value) does not iterate, it just returns the first value in the dict for each iteration of the for loop.

I understand that if you iterate with only for key in dict then you are iterating over the keys only. Here since we are iterating a set of tuples (by using dict.items()) with only the key in the for loop, the loop should run for the same number of times as the first example, since there are as many keys as key,value pairs.

What I'm having trouble grasping is why python gives you the entire tuple in the second example for key.


Thanks for the help all -- I'd like to add one more question to the mix.

for a,a in dict.items():
    print a

Why does the above print the value, and if i print a,a - obviously both values are printed twice. If I had typed for a,b I would be iterating (key,value) pairs so I would logically think I am now iterating over (key,key) pairs and would therefore print key rather than value. Sorry for the basic questions just playing around in interpreter and trying to figure stuff out.

2
No, if you only want the key do for key in d.keys(): (don't call your own variable dict) or just for key in d. - jonrsharpe
@Solaxun: I think the answer to your additional question (the "for a,a") is that (a,a) iterates over (key, value) pairs, so that a is assigned key and then immediately assigned value every iteration, so you always see a=value. I don't see a use-case for this though... - Moot

2 Answers

27
votes

The first example is utilizing something known as "tuple unpacking" to break what is REALLY the same tuple as in your separate example down into two different variables.

In other words this:

for key, value in dict.items():

Is just this:

for keyvalue in dict.items():
    key, value = keyvalue[0], keyvalue[1]

Remember that a for loop always iterates over the individual elements of the iterator you give it. dict.items() returns a list-like object of tuples, so every run through the for loop is a new tuple, automatically unpacked into key, value if you define it as such in the for loop.

It may help to think of it this way:

d = {'a':1, 'b':2, 'c':3}
list(d)          # ['a', 'b', 'c']             the keys
list(d.keys())   # ['a', 'b', 'c']             the keys
list(d.values()) # [1, 2, 3]                   the values
list(d.items())  # [('a',1), ('b',2), ('c',3)] a tuple of (key, value)

N.B. that the only reason your code

for key in dict.items():
    print "%s key has value: %s" % (key, value)

Does not throw a NameError is because value is already defined from elsewhere in your code. Since you do not define value anywhere in that for loop, it would otherwise throw an exception.

0
votes

In the second example you gave you are not assigning "value" to anything:

Notice the small edit here:

for key in dict:  ##Removed call to items() because we just want the key, 
                  ##Not the key, value pair
    value = dict[key]  # Added this line
    print "%s key has the value %s (key, value)

Note:

In the second example, you could now call dict.keys() or just dict (referencing a dictionary in a for loop will return it's keys). Calling dict.items() will confusingly assign key=(, ) which is probably not what you want.