Path: csiph.com!v102.xanadu-bbs.net!xanadu-bbs.net!eternal-september.org!feeder.eternal-september.org!border1.nntp.ams1.giganews.com!nntp.giganews.com!bcyclone05.am1.xlned.com!bcyclone05.am1.xlned.com!newsfeed.xs4all.nl!newsfeed2a.news.xs4all.nl!xs4all!newsgate.cistron.nl!newsgate.news.xs4all.nl!post.news.xs4all.nl!not-for-mail Return-Path: X-Original-To: python-list@python.org Delivered-To: python-list@mail.python.org X-Spam-Status: OK 0.000 X-Spam-Evidence: '*H*': 1.00; '*S*': 0.00; '*not*': 0.07; 'method,': 0.07; 'override': 0.07; 'dict': 0.09; 'meaningful': 0.09; 'methods,': 0.09; 'overridden': 0.09; 'received:80.91': 0.09; 'received:80.91.229': 0.09; 'received:gmane.org': 0.09; 'received:list': 0.09; 'subclass': 0.09; 'subject:fields': 0.09; 'anyway': 0.11; 'exception': 0.13; 'def': 0.14; 'suggest': 0.15; 'arg):': 0.16; 'foo"': 0.16; 'foo.': 0.16; 'overriding': 0.16; 'param': 0.16; 'paulo': 0.16; 'received:80.91.229.3': 0.16; 'received:dip0.t-ipconnect.de': 0.16; 'received:plane.gmane.org': 0.16; 'received:t-ipconnect.de': 0.16; 'subject:object': 0.16; 'surprising': 0.16; 'wrote:': 0.16; 'later': 0.16; 'attribute': 0.18; 'runs': 0.18; '>>>': 0.20; 'suggested': 0.20; '"",': 0.22; 'code.': 0.23; '2015': 0.23; 'this:': 0.23; '(most': 0.24; 'raise': 0.24; 'sort': 0.25; 'header:User-Agent:1': 0.26; 'header:X-Complaints-To:1': 0.26; 'skip:_ 20': 0.26; 'bugs': 0.27; 'have,': 0.27; "doesn't": 0.28; 'prints': 0.29; 'code:': 0.29; 'fri,': 0.31; 'keyword': 0.31; 'skip:_ 10': 0.32; 'class': 0.33; "d'aprano": 0.33; 'instead,': 0.33; 'steven': 0.33; 'traceback': 0.33; 'this?': 0.34; 'file': 0.34; 'add': 0.34; 'to:addr:python- list': 0.35; 'instance': 0.35; 'something': 0.35; 'sometimes': 0.35; 'but': 0.36; 'there': 0.36; '(and': 0.36; 'two': 0.37; 'should': 0.37; 'subject:: ': 0.37; 'skip:i 20': 0.37; 'received:org': 0.38; 'method': 0.39; 'to:addr:python.org': 0.39; 'received:de': 0.40; 'your': 0.60; 'repeat': 0.67; 'risk': 0.67; 'special': 0.72; 'careful': 0.91 X-Injected-Via-Gmane: http://gmane.org/ To: python-list@python.org From: Peter Otten <__peter__@web.de> Subject: Re: Passing new fields to an object Date: Sat, 13 Jun 2015 09:49:34 +0200 Organization: None References: <557b868c$0$11125$c3e8da3@news.astraweb.com> Mime-Version: 1.0 Content-Type: text/plain; charset="ISO-8859-1" Content-Transfer-Encoding: 7Bit X-Gmane-NNTP-Posting-Host: p57bd9d98.dip0.t-ipconnect.de User-Agent: KNode/4.13.3 X-BeenThere: python-list@python.org X-Mailman-Version: 2.1.20+ Precedence: list List-Id: General discussion list for the Python programming language List-Unsubscribe: , List-Archive: List-Post: List-Help: List-Subscribe: , Newsgroups: comp.lang.python Message-ID: Lines: 84 NNTP-Posting-Host: 2001:888:2000:d::a6 X-Trace: 1434181789 news.xs4all.nl 2948 [2001:888:2000:d::a6]:34316 X-Complaints-To: abuse@xs4all.nl X-Received-Bytes: 5589 X-Received-Body-CRC: 658444861 Xref: csiph.com comp.lang.python:92585 Steven D'Aprano wrote: > On Fri, 12 Jun 2015 16:53:08 +0100, Paulo da Silva wrote: > >> I would like to do something like this: >> >> class C: >> def __init__(self,**parms): >> ... >> >> c=C(f1=1,f2=None) >> >> I want to have, for the object >> self.f1=1 >> self.f2=None >> >> for an arbitrary number of parameters. >> >> What is the best way to achieve this? > > > Others have suggested that you update the instance dict with the keyword > parameters: > > self.__dict__.update(parms) > > > But I suggest that you should be very careful with this technique, > because it can lead to surprising problems and bugs in your code. If your > class has methods (and what sort of class doesn't have methods?), this > will override them and lead to mysterious failures in your code: > > instance = C(a=1, b=2, method=3) > # much later > result = instance.method(args) # will raise exception > > > You should use SimpleNamespace, as Peter suggests, but *not* subclass it. > If you subclass it and add methods: > > class C(SimpleNamespace): > def foo(self, arg): > print("called foo") > > > then you risk overriding foo method, as above. If you don't add methods, > there is no need to subclass. I sometimes do it anyway if only to get a meaningful class name. > Instead, use composition: your class should *contain* a SimpleNamespace, > not *be* one: > > class C: > def __init__(self, **param): > self.ns = SimpleNamespace(param) **param > def __getattr__(self, attrname): > return getattr(self.ns, attrname) > def foo(self, arg): > print("called foo") > > > instance = C(a=1, b=2, foo=3) > # later > instance.foo("x") # prints "called foo" >>> c = C(ns=42) >>> assert c.ns == 42 Traceback (most recent call last): File "", line 1, in AssertionError Yes, it's less likely, but now that the OP has two convenient ways to produce a name clash I think it's time to repeat that there is one bullet- proof approach: use a dict ;) > The special method __getattr__ only runs if the attribute name is not > found in the usual way, so the method foo will continue to be found and > not be overridden by the param foo.