473,796 Members | 2,737 Online
Bytes | Software Development & Data Engineering Community
+ Post

Home Posts Topics Members FAQ

override a property

Is there a way to override a data property in the instance? Do I need to create
another class with the property changed?
--
Robin Becker

Oct 17 '05
18 4764
Kay Schluehr wrote:
Robin Becker wrote:
Is there a way to override a data property in the instance? Do I need to create
another class with the property changed?
--
Robin Becker

It is possible to decorate a method in a way that it seems like
property() respects overridden methods. The decorator cares
polymorphism and accesses the right method.

def overridable(f):
def __wrap_func(sel f,*args,**kwd):
func = getattr(self.__ class__,f.func_ name)
if func.func_name == "__wrap_fun c":
return f(self,*args,** kwd)
else:
return func(self,*args ,**kwd)
return __wrap_func
class A(object):
def __init__(self, x):
self._x = x

@overridable
def get_x(self):
return self._x

x = property(get_x)

class B(A):

def get_x(self):
return self._x**2

class C(B):pass

a = A(7)
a.x
7
b = B(7)
b.x
49
c = C(7)
c.x


49


I thought that methods were always overridable. In this case the lookup on the
class changes the behaviour of the one and only property.

--
Robin Becker

Oct 18 '05 #11
Robin Becker wrote:
## my silly example
class ObserverPropert y(property):
def __init__(self,n ame,observers=N one,validator=N one):
self._name = name
self._observers = observers or []
self._validator = validator or (lambda x: x)
self._pName = '_' + name
property.__init __(self,
fset=lambda inst, value: self.__notify_f set(inst,value) ,
)

def __notify_fset(s elf,inst,value) :
value = self._validator (value)
for obs in self._observers :
obs(inst,self._ pName,value)
inst.__dict__[self._pName] = value

def add(self,obs):
self._observers .append(obs)

def obs0(inst,pName ,value):
print 'obs0', inst, pName, value

def obs1(inst,pName ,value):
print 'obs1', inst, pName, value

class A(object):
x = ObserverPropert y('x')

a=A()
A.x.add(obs0)

a.x = 3

b = A()
b.x = 4

#I wish I could get b to use obs1 instead of obs0
#without doing the following
class B(A):
x = ObserverPropert y('x',observers =[obs1])

b.__class__ = B

b.x = 7


Can you add the object to be observed as another parameter to the add
method?

py> class ObservablePrope rty(property):
.... def __init__(self, *args, **kwargs):
.... super(Observabl eProperty, self).__init__( *args, **kwargs)
.... self._observers = {}
.... def __set__(self, obj, value):
.... super(Observabl eProperty, self).__set__(o bj, value)
.... for observer in self._observers .get(obj, []):
.... observer(obj)
.... def add(self, obj, observer):
.... self._observers .setdefault(obj , []).append(observ er)
....
py> class A(object):
.... def _getx(self):
.... return self._x
.... def _setx(self, value):
.... self._x = value
.... x = ObservablePrope rty(_getx, _setx)
....
py> def obs1(obj):
.... print 'obs1:', obj.x
....
py> def obs2(obj):
.... print 'obs2:', obj.x
....
py> a = A()
py> a.x = 3
py> A.x.add(a, obs1)
py> a.x = 4
obs1: 4
py> A.x.add(a, obs2)
py> a.x = 5
obs1: 5
obs2: 5
py> b = A()
py> b.x = 6
py> A.x.add(b, obs2)
py> b.x = 7
obs2: 7

Probably "self._observer s" should use some sort of weakref dict instead
of a regular dict, but hopefully the idea is clear.

STeVe
Oct 18 '05 #12
Steven Bethard wrote:
Robin Becker wrote:
........
Can you add the object to be observed as another parameter to the add
method?

py> class ObservablePrope rty(property):
... def __init__(self, *args, **kwargs): ....... py> A.x.add(b, obs2)
py> b.x = 7
obs2: 7

Probably "self._observer s" should use some sort of weakref dict instead
of a regular dict, but hopefully the idea is clear.

STeVe


yes I think this is what Alex is proposing. It probably means abandoning the
class based observers entirely otherwise there would have to be a decision on
whether the instance observers take priority and some argument convention on
whether the class or the instance was being added to.
--
Robin Becker

Oct 19 '05 #13
Robin Becker wrote:
I thought that methods were always overridable.
In this case the lookup on the
class changes the behaviour of the one and only property.


How can something be made overridable that is actually overridable? I
didn't know how to better express the broken polymorphism of Pythons
properties than by stating it as a pleonasm about the used get and set
methods. This way a property don't ever have to be redefined in
subclasses if get_x, set_x etc. are changed.

Kay

Oct 20 '05 #14
Kay Schluehr wrote:
Robin Becker wrote:

I thought that methods were always overridable.
In this case the lookup on the
class changes the behaviour of the one and only property.

How can something be made overridable that is actually overridable? I
didn't know how to better express the broken polymorphism of Pythons
properties than by stating it as a pleonasm about the used get and set
methods. This way a property don't ever have to be redefined in
subclasses if get_x, set_x etc. are changed.

Kay


well I guess that's the ambiguity of human language. Clearly when I
assign to a normal attribute I am changing its value; assigning to a
property or descriptor does something that is not so obvious. Changing
the behaviour of such an attribute could be done by inheritance as
suggested. The new class has overridden the property. When I want to do
that on an instance I have first to create a mutable version of the
descriptor where the mutability is on the instance not the class. I call
the action of changing the base descriptor behaviour 'overriding', but
perhaps that's not the correct word. What do you suggest?
--
Robin Becker
Oct 21 '05 #15
On Tue, 18 Oct 2005 08:00:51 +0000, Robin Becker <ro***@SPAMREMO VEjessikat.fsne t.co.uk> wrote:
Bruno Desthuilliers wrote:
Robin Becker a écrit :
Is there a way to override a data property in the instance? Do I need
to create another class with the property changed?

Do you mean attributes or properties ?


I mean property here. My aim was to create an ObserverPropert y class
that would allow adding and subtracting of set/get observers. My current
implementati on works fine for properties on the class, but when I need
to specialize an instance I find it's quite hard.

Sorry, but my news feed went belly-up for a few days and I had to go to
google groups to see what had transpired.

ISTM you are already a fair way along the lines of Alex's suggestion of a custom
descriptor class, having chosen to create it by subclassing property.

If it is ok to add attributes to instances, you could put overriding
observer functions there and just check for them in your __notify_fset
loop, e.g., by checking for a matching name with an '__override_' prefixed
to the name of the obs function you want to override (see example below).

If it's not ok to add instance attributes, you could store the information
elsewhere, using weak-ref stuff as Alex suggests.

E.g. (I'm pasting in as a quote what I copied from google):

bruno modulix wrote:
.....

Could you elaborate ? Or at least give an exemple ?.....
in answer to Bengt & Bruno here is what I'm sort of playing with. Alex suggests
class change as an answer, but that looks really clunky to me. I'm not sure what
Alex means by

A better design might be to use, instead of the builtin
type 'property', a different custom descriptor type that is specifically
designed for your purpose -- e.g., one with a method that instances can
call to add or remove themselves from the set of "instances overriding
this ``property''" and a weak-key dictionary (from the weakref module)
mapping such instances to get/set (or get/set/del, if you need to
specialize "attribute deletion" too) tuples of callables. I see it's clear how to modify the behaviour of the descriptor instance, but is
he saying I need to mess with the descriptor magic methods so they know what
applies to each instance?
I think maybe only insofar as __notify_fset is magic ;-)
## my silly example
class ObserverPropert y(property):
def __init__(self,n ame,observers=N one,validator=N one):
self._name = name
self._observers = observers or []
self._validator = validator or (lambda x: x)
self._pName = '_' + name
property.__init __(self,
fset=lambda inst, value: self.__notify_f set(inst,value) ,
) def __notify_fset(s elf,inst,value) :
value = self._validator (value)
for obs in self._observers : obs = inst.__dict__.g et('__override_ '+obs.func_name , obs)
obs(inst,self._ pName,value)
inst.__dict__[self._pName] = value def add(self,obs):
self._observers .append(obs) def obs0(inst,pName ,value):
print 'obs0', inst, pName, value def obs1(inst,pName ,value):
print 'obs1', inst, pName, value class A(object):
x = ObserverPropert y('x') a=A()
A.x.add(obs0 ) a.x = 3 b = A()
b.x = 4 #I wish I could get b to use obs1 instead of obs0
#without doing the following I think your class assignment would eliminate all x-observing functions,
but did you mean strictly just to "use obs1 instead of obs0" -- which
would mean to leave others operable?

## >class B(A):
## > x = ObserverPropert y('x',observers =[obs1])
##
## >b.__class__ = B

b.__override_ob s0 = obs1

# Of course you could wrap that last line functionality in some helper thing ;-)
b.x = 7


With the above mods put in becker.py, I get:
import becker obs0 <becker.A object at 0x02EF3C2C> _x 3
obs0 <becker.A object at 0x02EF3C4C> _x 4
obs1 <becker.A object at 0x02EF3C4C> _x 7

But adding another observer doesn't eliminate the other(s):
def obs3(inst,pName ,value): ... print 'obs3', inst, pName, value
... becker.A.x.add( obs3)
becker.b.x = 777 obs1 <becker.A object at 0x02EF3C4C> _x 777
obs3 <becker.A object at 0x02EF3C4C> _x 777 becker.a.x = 777

obs0 <becker.A object at 0x02EF3C2C> _x 777
obs3 <becker.A object at 0x02EF3C2C> _x 777

HTH

Regards,
Bengt Richter
Oct 21 '05 #16
Robin Becker <ro***@reportla b.com> wrote:
...
in answer to Bengt & Bruno here is what I'm sort of playing with. Alex
suggests class change as an answer, but that looks really clunky to me.
I'm not sure what
Changing class is indeed 'clunky', though it might have been necessary
depending on how one interpreted your original specs.
Alex means by
A better design might be to use, instead of the builtin
type 'property', a different custom descriptor type that is specifically
designed for your purpose -- e.g., one with a method that instances can
call to add or remove themselves from the set of "instances overriding
this ``property''" and a weak-key dictionary (from the weakref module)
mapping such instances to get/set (or get/set/del, if you need to
specialize "attribute deletion" too) tuples of callables.
I see it's clear how to modify the behaviour of the descriptor instance,
but is he saying I need to mess with the descriptor magic methods so they
know what applies to each instance?


If (e.g.) __set__ needs to behave differently when applied to certain
instances rather than others, then it had better be "messed with"
(overridden) compared to property.__set_ _ since the latter has no such
proviso. Of course, your architecture as sketched below (taking
advantage of the fact that property.__set_ _ always calls a certain
callable, and you get to control that callable) is OK too.

## my silly example
class ObserverPropert y(property):
def __init__(self,n ame,observers=N one,validator=N one):
self._name = name
self._observers = observers or []
self._validator = validator or (lambda x: x)
self._pName = '_' + name
property.__init __(self,
fset=lambda inst, value: self.__notify_f set(inst,value) ,
)
Why not just fset=self.__not ify_fset ? I fail to see the added value of
this lambda. Anyway...:
def __notify_fset(s elf,inst,value) :
value = self._validator (value)
for obs in self._observers :
obs(inst,self._ pName,value)
inst.__dict__[self._pName] = value

def add(self,obs):
self._observers .append(obs)
....this class only offers sets of observers *per-descriptor instance*,
not ones connected to a specific 'inst' being observed. My point is,
you could add the latter pretty easily.

def obs0(inst,pName ,value):
print 'obs0', inst, pName, value

def obs1(inst,pName ,value):
print 'obs1', inst, pName, value

class A(object):
x = ObserverPropert y('x')

a=A()
A.x.add(obs0)

a.x = 3

b = A()
b.x = 4

#I wish I could get b to use obs1 instead of obs0
#without doing the following
class B(A):
x = ObserverPropert y('x',observers =[obs1])

b.__class__ = B

b.x = 7


You can, if you have a way to call, say, b.x.add_per_ins t(b, obs1).
Such as, adding within ObserverPropert y:

self._observers _per_inst = {}

in the init, and changing the notification method to do:

def __notify_fset(s elf,inst,value) :
value = self._validator (value)
observers = self._observers _per_inst.get(i nst)
if not observers: observers = self._observers
for obs in observers:
obs(inst,self._ pName,value)
inst.__dict__[self._pName] = value

and a new method add_per_inst:

def add_per_inst(se lf, inst, obs):
self._observers _per_inst.setde fault(inst,[]).append(obs)

Of course, you most likely want to use weak rather than normal
references here (probably to both instances and observers), but that's a
separate issue.
Alex
Oct 22 '05 #17
Alex Martelli wrote:
.......
If (e.g.) __set__ needs to behave differently when applied to certain
instances rather than others, then it had better be "messed with"
(overridden) compared to property.__set_ _ since the latter has no such
proviso. Of course, your architecture as sketched below (taking
advantage of the fact that property.__set_ _ always calls a certain
callable, and you get to control that callable) is OK too.
...... I think I at last got this
.......
Why not just fset=self.__not ify_fset ? I fail to see the added value of
this lambda. Anyway...:
duh just being Homerish
def __notify_fset(s elf,inst,value) :
value = self._validator (value) ........ def add(self,obs):
self._observers .append(obs)

...this class only offers sets of observers *per-descriptor instance*,
not ones connected to a specific 'inst' being observed. My point is,
you could add the latter pretty easily.

.......
......... You can, if you have a way to call, say, b.x.add_per_ins t(b, obs1).
Such as, adding within ObserverPropert y:

self._observers _per_inst = {}

in the init, and changing the notification method to do:

def __notify_fset(s elf,inst,value) :
value = self._validator (value)
observers = self._observers _per_inst.get(i nst)
if not observers: observers = self._observers
for obs in observers:
obs(inst,self._ pName,value)
inst.__dict__[self._pName] = value

and a new method add_per_inst:

def add_per_inst(se lf, inst, obs):
self._observers _per_inst.setde fault(inst,[]).append(obs)

Of course, you most likely want to use weak rather than normal
references here (probably to both instances and observers), but that's a
separate issue.


.......
yes I begin at last to see the full complexity of this. There are really
three possible attachments, the descriptor, instance class and the
instance. Since the descriptor is attached to the class or a base class
one could argue about whether observers should be inherited etc etc, but
perhaps that's a step too far.

Thanks to Alex and Bengt and others for clarifying a bunch of issues.
--
Robin Becker
Oct 22 '05 #18
Robin Becker wrote:
Kay Schluehr wrote:
Robin Becker wrote:

I thought that methods were always overridable.
In this case the lookup on the
class changes the behaviour of the one and only property.

How can something be made overridable that is actually overridable? I
didn't know how to better express the broken polymorphism of Pythons
properties than by stating it as a pleonasm about the used get and set
methods. This way a property don't ever have to be redefined in
subclasses if get_x, set_x etc. are changed.

Kay


well I guess that's the ambiguity of human language. Clearly when I
assign to a normal attribute I am changing its value; assigning to a
property or descriptor does something that is not so obvious. Changing
the behaviour of such an attribute could be done by inheritance as
suggested. The new class has overridden the property. When I want to do
that on an instance I have first to create a mutable version of the
descriptor where the mutability is on the instance not the class. I call
the action of changing the base descriptor behaviour 'overriding', but
perhaps that's not the correct word. What do you suggest?
--
Robin Becker


I would suggest to take a step back and start with Raymond Hettingers
descriptor definition:

"In general, a descriptor is an object attribute with "binding
behavior", one whose attribute access has been overridden by methods in
the descriptor protocol. Those methods are __get__, __set__, and
__delete__. If any of those methods are defined for an object, it is
said to be a descriptor."

http://users.rcn.com/python/download/Descriptor.htm

The definition is a little confusing since we have to identify the
descriptor object that implements one of the descriptor methods
__get__, __set__ and __del__ with an object attribute that is assigned
by the descriptor ( as object not attribute ). Otherwise we can assert
that a descriptor becomes effective only if it is used as an object
attribute. The meaning / essence of a descriptor is to assign
attributes by a descriptor to alter binding behaviour but it's
essentially an object that can be handled quite differently. To make
the destinction clear I will talk about "descripted " attributes.

Now we can talk unambigously about "overriding " the descriptor by means
of overriding the descriptor methods in subclasses. In case of
"properties " we pass certain functions into property() that will be
wrapped into descriptor methods. Thereby property() is itself a
descriptor. If we use methods of the class where the descripted
attribute is defined, overriding has no effect on the descriptor. The
inheritance hierarchies of descriptors and classes that define
descripted attributes do not correspond.

One way of establishing a pseudo-correspondence I've already presented.
But maybe one can do it better without decorators? Remember that
property() is a descriptor factory and there is no intrinsic need to
pass functions into a factory function. Why not passing strings that
are names of methods?

For brevity only __get__ should be defined here:

class described_by(ob ject):
def __init__(self, fget=""):
assert isinstance(fget , str)
self.fget = fget

def __get__(self, obj, objtype=None):
if obj is None:
return self
if self.fget is None:
raise AttributeError, "unreadable attribute"
return getattr(obj.__c lass__, self.fget)(obj)
class A(object):
def __init__(self, x):
self._x = x

def get_x(self):
return self._x

x = described_by("g et_x")

class B(A):
def get_x(self):
return self._x**2
a = A(7)
a.x 7 b = B(7)
b.x

49

Regards,
Kay

Oct 22 '05 #19

This thread has been closed and replies have been disabled. Please start a new discussion.

Similar topics

7
2075
by: Thomas Kehl | last post by:
Hello. I provided a class, which derives from the class DataView. The only thing, which I would like to do in this class am to overwrite the Property COUNT. I wars it however simply not. I get always following message from the compiler, although COUNT of the DataView (according to MSDN) is virtual. the element ' System.Data.DataView.Count.get ' cannot be overwritten, because as virtually, abstract or do not overwrite it is marked Can...
5
5919
by: Eric Johannsen | last post by:
I have a simple object that inherits from CollectionBase and overrides the Count property: namespace MyTest { public class CollTest : System.Collections.CollectionBase { public override int Count { get { return 0; }
5
6852
by: Darius | last post by:
I'm writing here in hopes that someone can explain the difference between the new and virtual/override keywords in C#. Specifically, what is the difference between this: public class Window { public void Draw() { Console.WriteLine("The WINDOW Draw method is running!");
7
6451
by: Dave Y | last post by:
I am a newbie to C# and am having trouble trying to override a ListView property method. I have created a new class derived from the Forms.Listview and I cannot figure out the syntax to override ListView.Items.Add(), . I see that it is a virtual method so it should be easy to do. If anyone can help I would appreciate it greatly. I can do what I need to do in a different way this would just make everything significantly cleaner and eaasier...
1
5341
by: Å krat Bolfenk | last post by:
How can I override property in C++/CLI. Well I know how to override when the input and output types are the same (and number) ..., but I would like the derived (overridden) property to return different type than base. Thanks
8
5499
by: bdeviled | last post by:
I am deploying to a web environment that uses load balancing and to insure that sessions persist across servers, the environment uses SQL to manage sessions. The machine.config file determines how all applications will use sessions and to insure that all application use this method, the session properties cannot be overriden. Within the sessionstate tags, the webadmin (upon my request)r emoved the property for timeout, hoping that...
4
1996
by: R. MacDonald | last post by:
Hello, Group, I have been curious about a statement I see occasionally in the VB (v2003) help. Sometimes the information about a property will indicate that it is overridable, and I have been making use of this. Sometimes there is also a statement such as: "You are not required to override both the get and set methods of the .... property; you can override only one if needed."
14
33294
by: Dave Booker | last post by:
It looks like the language is trying to prevent me from doing this sort of thing. Nevertheless, the following compiles, and I'd like to know why it doesn't work the way it should: public class ComponentA { static string s_name = "I am the root class."; public string Name { get {return s_name;} } }
1
3128
by: Joshua | last post by:
I'm creating a web control that has an image and a text box. I would like to then override the Visible and Text property of the usercontrol, so when you reference the visible property of the user control it only set the visible property of the textbox. When i call these newly created property I get the following error: Object reference not set to an instance of an object. Does anybody have any idea? Thank you.
0
9535
by: Hystou | last post by:
Most computers default to English, but sometimes we require a different language, especially when relocating. Forgot to request a specific language before your computer shipped? No problem! You can effortlessly switch the default language on Windows 10 without reinstalling. I'll walk you through it. First, let's disable language synchronization. With a Microsoft account, language settings sync across devices. To prevent any complications,...
0
10465
Oralloy
by: Oralloy | last post by:
Hello folks, I am unable to find appropriate documentation on the type promotion of bit-fields when using the generalised comparison operator "<=>". The problem is that using the GNU compilers, it seems that the internal comparison operator "<=>" tries to promote arguments from unsigned to signed. This is as boiled down as I can make it. Here is my compilation command: g++-12 -std=c++20 -Wnarrowing bit_field.cpp Here is the code in...
0
10242
jinu1996
by: jinu1996 | last post by:
In today's digital age, having a compelling online presence is paramount for businesses aiming to thrive in a competitive landscape. At the heart of this digital strategy lies an intricately woven tapestry of website design and digital marketing. It's not merely about having a website; it's about crafting an immersive digital experience that captivates audiences and drives business growth. The Art of Business Website Design Your website is...
0
10021
tracyyun
by: tracyyun | last post by:
Dear forum friends, With the development of smart home technology, a variety of wireless communication protocols have appeared on the market, such as Zigbee, Z-Wave, Wi-Fi, Bluetooth, etc. Each protocol has its own unique characteristics and advantages, but as a user who is planning to build a smart home system, I am a bit confused by the choice of these technologies. I'm particularly interested in Zigbee because I've heard it does some...
0
9061
agi2029
by: agi2029 | last post by:
Let's talk about the concept of autonomous AI software engineers and no-code agents. These AIs are designed to manage the entire lifecycle of a software development project—planning, coding, testing, and deployment—without human intervention. Imagine an AI that can take a project description, break it down, write the code, debug it, and then launch it, all on its own.... Now, this would greatly impact the work of software developers. The idea...
0
6800
by: conductexam | last post by:
I have .net C# application in which I am extracting data from word file and save it in database particularly. To store word all data as it is I am converting the whole word file firstly in HTML and then checking html paragraph one by one. At the time of converting from word file to html my equations which are in the word document file was convert into image. Globals.ThisAddIn.Application.ActiveDocument.Select();...
0
5453
by: TSSRALBI | last post by:
Hello I'm a network technician in training and I need your help. I am currently learning how to create and manage the different types of VPNs and I have a question about LAN-to-LAN VPNs. The last exercise I practiced was to create a LAN-to-LAN VPN between two Pfsense firewalls, by using IPSEC protocols. I succeeded, with both firewalls in the same network. But I'm wondering if it's possible to do the same thing, with 2 Pfsense firewalls...
1
4127
by: 6302768590 | last post by:
Hai team i want code for transfer the data from one system to another through IP address by using C# our system has to for every 5mins then we have to update the data what the data is updated we have to send another system
3
2931
bsmnconsultancy
by: bsmnconsultancy | last post by:
In today's digital era, a well-designed website is crucial for businesses looking to succeed. Whether you're a small business owner or a large corporation in Toronto, having a strong online presence can significantly impact your brand's success. BSMN Consultancy, a leader in Website Development in Toronto offers valuable insights into creating effective websites that not only look great but also perform exceptionally well. In this comprehensive...

By using Bytes.com and it's services, you agree to our Privacy Policy and Terms of Use.

To disable or enable advertisements and analytics tracking please visit the manage ads & tracking page.