开发者

Python permanent assignment variables

开发者 https://www.devze.com 2023-01-13 02:16 出处:网络
I have just started python and came across something kind of strange. The following code assigns a co-ordinate of x=1 and y=2 to the variable test. The test2 variable assigns itself the same value as

I have just started python and came across something kind of strange.

The following code assigns a co-ordinate of x=1 and y=2 to the variable test. The test2 variable assigns itself the same value as test and then the [x] value for test2 is changed to the old [x] value minus 1. This works fine, however, when the last part is executed, not only does it minus 1 from the [x] value in test2, it does the same to the [x] value in the test variable too.

test = [1,2];
test2 = test;
test2[1] = test2[1] - 1;

I found doing the following worked fine but I still don't understand why the first method开发者_如何学Go changes the test value as well as the test2 value.

test = [1,2];
test2 = test;
test2 = [test2[0] -1 ,test2[1]];

Could someone please explain why this happens.

Thank You TheLorax


In Python, like in Java, assignment per se never makes a copy -- rahter, assignment adds another reference to the same object as the right-hand side of the =. (Argument passing works the same way). Strange that you've never heard of the concept, when Python is quite popular and Java even much more so (and many other languages tend to work similarly, possibly with more complications or distinguos).

When you want a copy, you ask for a copy (as part of the expression that is on the right-hand side, or gets used to get the argument you're passing)! In Python, depending on your exact purposes, there are several ways -- the copy module of the standard library being a popular ones (with functions copy, for "shallow" copies, and deepcopy -- for "deep" copies, of course, i.e., ones where, not just the container, but every contained item, also gets deep copied recursively).

Often, though, what you want is "a new list" -- whether the original sequence is a list, a copy, or something else that's iterable, you may not care: you want a new list instance, whose items are the same as those of "something else". The perfect way to express this is to use list(somethingelse) -- call the list type, which always makes a new instance of list, and give it as an argument the iterable whose items you want in the new list. Similarly, dict(somemapping) makes a new dict, and set(someiterable) makes a new set -- calling a type to make a new instance of that type is a very general and useful concept!


It's because when you do test2 = test you are not copying the contents of the list, but simply assigning test2 a reference to the original list. Thus, any changes to the test2 will also affect test.

The right way to do this is using deepcopy() from the copy module (or copy() if you can get away with a shallow copy.

import copy
test2 = copy.deepcopy(test) # deep copy

test2 = copy.copy(test)) # shallow copy

test2 = test[:] # shallow copy using slices

See this page for a more in-depth explanation as well as other methods to copy a list.


[1,2] is an array and when you assign it to test you assign a reference to that array to test. So when you do test2=test, you are assigning the reference to yet another variable. Make changes to any variable and the array will get changed.

Your second example works, because you create a new array altogether. You might as well have done it like this:

test = [1,2]
test2 = [test[0]-1, test[1]]


Python does not do assignment.

It does name binding.

There is a difference, and it is explained wonderfully and in detail here.

In your first example: Your first line binds the name "test" to a list object created by the "[1,2]" expression. Your second line binds the name "test2" to the same object that is bounded to "test". Your third line mutates the object bound to "test2", which is the same as the object bound to by "test".

In your second example: Your first line binds the name "test" to a list object created by the "[1,2]" expression. Your second line binds the name "test2" to the same object that is bounded to "test". Your third line binds the name "test2" to a new object, created by the expression "[test2[0]-1, test21]".

If you really want two independent lists with the same values bound to different variables, you need to:

>>> test = [1, 2]
>>> test2 = list(test)
>>> test2[0] = 3
>>> print(test)
[1, 2]
>>> print(test2)
[3, 2]

Note that this is a shallow-copy list, so changing the state of the object of an element in test2 will, of course, effect test if the the same object is found in that list.

p.s. Python. Semi-colons. Ugly. Unnecessary. Do not use.

0

精彩评论

暂无评论...
验证码 换一张
取 消