26

Under what circumstances is it possible, and when is it impossible, to assign to an instance's __class__ attribute in Python?

Error messages such as TypeError: __class__ assignment: only for heap types don't really do it for me.

2
  • 6
    @Rafe: actually, Django uses this trick in its authentication code Commented Jun 26, 2011 at 8:47
  • 2
    late to the party, but matplotlib does this in production code too (in some of the 3D plotting). Commented Mar 13, 2014 at 12:40

1 Answer 1

18

You can only assign to the __class__ attribute of an instance of a user-defined class (i.e. defined using the class keyword), and the new value must also be a user-defined class. Whether the classes are new-style or old-style does not matter. (You can't mix them, though. You can't turn an old-style class instance into a new-style class instance.) See also this issue in the Python bug tracker, which also complains that the error message is somewhat hard to understand.

Just to add what Rafe said in the above comment: Never do this in production.

Sign up to request clarification or add additional context in comments.

6 Comments

Also, from my experimenting it seems that the presence/absence of __slots__ also makes a difference: "X deallocator differs from Y" from instance_with_slots.__class__ = ClassWithoutSlots and instance_without_slots.__class__ = ClassWithSlots
Ah, I was confused because did a.__class__ = B.__class__ when I obviously should have been doing a.__class__ = B
Both matplotlib and django do this in production, according to the above comments
@dtheodor what a bold and frankly silly statement. Yes, Python chooses to expose some of its quite advanced and easy-to-misuse machinery, but it is a choice: __class__ could have been made a read-only attribute. Saying that these mature and exceptionally well supported projects are "doing it wrong" because they do something that we advise newcomers against on stackoverflow is like saying that engineers are doing it wrong because they're not using sliderules, or space shuttle welders doing it wrong because they're not using soldering irons.
@jedwards Not everything that language designers allow to do is actually a good idea, even if popular projects do it. I don't have the time to write a detailed analysis, but after looking at the code it is my conviction that both Django and Matplotlib would be better off not using this hack. It's too much magic for virtually no gain.
|

Your Answer

By clicking “Post Your Answer”, you agree to our terms of service and acknowledge you have read our privacy policy.

Start asking to get answers

Find the answer to your question by asking.

Ask question

Explore related questions

See similar questions with these tags.