Decorators are much, much simpler and more limited -- and therefore should be preferred whenever the desired effect can be achieved with either a metaclass or a class decorator.
Anything you can do with a class decorator, you can of course do with a custom metaclass (just apply the functionality of the "decorator function", i.e., the one that takes a class object and modifies it, in the course of the metaclass's __new__
or __init__
that make the class object!-).
There are many things you can do in a custom metaclass but not in a decorator (unless the decorator internally generates and applies a custom metaclass, of course -- but that's cheating;-)... and even then, in Python 3, there are things you can only do with a custom metaclass, not after the fact... but that's a pretty advanced sub-niche of your question, so let me give simpler examples).
For example, suppose you want to make a class object X
such that print X
(or in Python 3 print(X)
of course;-) displays peekaboo!
. You cannot possibly do that without a custom metaclass, because the metaclass's override of __str__
is the crucial actor here, i.e., you need a def __str__(cls): return "peekaboo!"
in the custom metaclass of class X
.
The same applies to all magic methods, i.e., to all kinds of operations as applied to the class object itself (as opposed to, ones applied to its instances, which use magic methods as defined in the class -- operations on the class object itself use magic methods as defined in the metaclass).
与恶龙缠斗过久,自身亦成为恶龙;凝视深渊过久,深渊将回以凝视…