|
|||||||
|
|
|
![]() |
|
|
Strumenti |
|
|
#1 |
|
Member
Iscritto dal: Apr 2008
Messaggi: 120
|
[PYTHON] metaclasse e deep copy con classe list
Qualcuno ha idea di come si possa implementare una metaclasse affinchè ridefinisca la copia di una lista in python?
Normalmente la copia in python fra list è una shallow copy come dice qua: http://docs.python.org/library/copy.html. Il main è questo: Codice:
l=list() l.append(1) l.append(2) l.append(3) # l viene copiata in l1 l1=l # il contenuto di l1 viene modificato l1[2] = ’B’ l1 [1, 2, ’B’] # ma il contenuto di l rimane invariato l [1, 2, 3] Io avevo pensato a questo come soluzione: Codice:
def funz(obj):
import copy
print('entro')
return str(copy.deepcopy(obj))
class metacl(type):
def __new__(meta,classname,supers,classdict):
classdict['__boh__'] = funz
return type.__new__(meta,classname,supers,classdict)
class list(list,metaclass=metacl): pass
ma = non è un operatore che si può ridefnirire come + (attraverso __add__) avevo pensato a __getattr__ a __copy__ ma non mi ridefinisce niente! Ultima modifica di postgres : 22-02-2012 alle 18:38. Motivo: titolo più dettagliato |
|
|
|
|
|
#2 |
|
Senior Member
Iscritto dal: Jan 2002
Città: Germania
Messaggi: 26110
|
Hai 2 problemi. Il primo è che una lista la puoi creare anche così:
Codice:
MiaLista = [1, 2, 3] Il secondo è che la copia puoi farla anche così: Codice:
NuovaLista = MiaLista[ : ]
__________________
Per iniziare a programmare c'è solo Python con questo o quest'altro (più avanzato) libro @LinkedIn Non parlo in alcun modo a nome dell'azienda per la quale lavoro Ho poco tempo per frequentare il forum; eventualmente, contattatemi in PVT o nel mio sito. Fanboys |
|
|
|
|
|
#3 |
|
Member
Iscritto dal: Apr 2008
Messaggi: 120
|
L'obiettivo è definire una metaclasse che implementi deep copy sull'operatore di assegnamento affinchè la classe list copi con deep copy..
|
|
|
|
|
|
#4 |
|
Senior Member
Iscritto dal: Jan 2002
Città: Germania
Messaggi: 26110
|
Ti anticipo subito che è impossibile, perché in Python non esiste l'operatore di assegnamento, ma l'istruzione di assegnamento, che non è soggetta a overloading.
In buona sostanza, se scrivi: Codice:
Lista = [1, 2, 3] AltraLista = Lista Ti riporto cosa succede a basso livello con Python con quel codice: Codice:
Python 2.6.6 (r266:84297, Aug 24 2010, 18:46:32) [MSC v.1500 32 bit (Intel)] on win32
Type "help", "copyright", "credits" or "license" for more information.
>>> import dis
>>> def f():
... Lista = [1, 2, 3]
... AltraLista = Lista
...
>>> dis.dis(f)
2 0 LOAD_CONST 1 (1)
3 LOAD_CONST 2 (2)
6 LOAD_CONST 3 (3)
9 BUILD_LIST 3
12 STORE_FAST 0 (Lista)
3 15 LOAD_FAST 0 (Lista)
18 STORE_FAST 1 (AltraLista)
21 LOAD_CONST 0 (None)
24 RETURN_VALUE
Dall'offset 0 a 9 viene costruita la lista. 12 la memorizza nella variabile Lista (da notare l'istruzione usata: STORE_FAST). Questo rappresenta la prima istruzione. 15 e 18 rappresentano, invece, la seconda istruzione, che si occupano rispettivamente di prelevare il valore da Lista e memorizzarlo in AltraLista. E' un meccanismo di basso livello (una banale copia, alla fine) che Python utilizza per QUALUNQUE oggetto (valore) manipoli, siano esse lista, tuple, numeri, o quant'altro. Ed è un meccanismo che, come puoi capire, non fa scattare alcun metodo speciale (tipo __getattr__, __setattr__, ecc. per intenderci) che sia eventualmente possibile intercettare. Spero sia chiaro. Se hai ancora dubbi, chiedi pure. P.S. 21 e 24 sono istruzioni di chiusura della funzione, perché Python restituisce sempre un valore, anche se non lo specifichi appositamente. Ignorale.
__________________
Per iniziare a programmare c'è solo Python con questo o quest'altro (più avanzato) libro @LinkedIn Non parlo in alcun modo a nome dell'azienda per la quale lavoro Ho poco tempo per frequentare il forum; eventualmente, contattatemi in PVT o nel mio sito. Fanboys |
|
|
|
|
|
#5 |
|
Junior Member
Iscritto dal: Sep 2011
Messaggi: 12
|
se provassi a ridefinire copy, si potrebbe riuscire a farlo?
|
|
|
|
|
|
#6 |
|
Senior Member
Iscritto dal: Jan 2002
Città: Germania
Messaggi: 26110
|
Non risolverebbe il problema, che nel suo caso riguarda l'assegnazione.
L'assegnazione, per quanto detto prima, non fa uso della copy.
__________________
Per iniziare a programmare c'è solo Python con questo o quest'altro (più avanzato) libro @LinkedIn Non parlo in alcun modo a nome dell'azienda per la quale lavoro Ho poco tempo per frequentare il forum; eventualmente, contattatemi in PVT o nel mio sito. Fanboys |
|
|
|
|
|
#7 |
|
Junior Member
Iscritto dal: Sep 2011
Messaggi: 12
|
non si riesce ad intercettare STORE_FAST?
mi sembra strano che diano un esercizio d'esame impossibile da risolvere |
|
|
|
|
|
#8 | |
|
Senior Member
Iscritto dal: Jan 2002
Città: Germania
Messaggi: 26110
|
Decisamente no.
Quote:
__________________
Per iniziare a programmare c'è solo Python con questo o quest'altro (più avanzato) libro @LinkedIn Non parlo in alcun modo a nome dell'azienda per la quale lavoro Ho poco tempo per frequentare il forum; eventualmente, contattatemi in PVT o nel mio sito. Fanboys |
|
|
|
|
|
|
#9 |
|
Member
Iscritto dal: Apr 2008
Messaggi: 120
|
Ecco il testo:
Traditionally object-oriented programming provides two different modes for cloning an instance or a structured data type: shallow and deep copy. The former has the effect to clone exclusively the external shell and not its content originating a quite fastidious aliasing effect. The latter, instead, copies the shell and its content recursively creating two completely separate copies of the instance. As you know, Python's programs suffer of the aliasing effect when you copy an instance of a class or a structured type (e.g., a list) with the = operator. As showed in the following example: Codice:
>>> l=[1,2,3] >>> l1=l >>> l1[2] = ’B’ >>> l1 [1, 2, ’B’] >>> l [1, 2, ’B’] The exercise consists of defining a meta-class which implements the deep copy on the assignment operator and binding this to the standard class list. Such that the following behavior can be yielded Codice:
>>> l=list() >>> l.append(1) >>> l.append(2) >>> l.append(3) >>> l1=l >>> l1[2] = ’B’ >>> l1 [1, 2, ’B’] >>> l [1, 2, 3] |
|
|
|
|
|
#10 |
|
Senior Member
Iscritto dal: Jan 2002
Città: Germania
Messaggi: 26110
|
Letto tutto, ma rimane il problema: non esiste alcun operatore di assegnazione in Python, per cui l ed l1 contengono esattamente lo stesso oggetto.
Si può intercettare quest'operazione: Codice:
l1[2] = 'B'
__________________
Per iniziare a programmare c'è solo Python con questo o quest'altro (più avanzato) libro @LinkedIn Non parlo in alcun modo a nome dell'azienda per la quale lavoro Ho poco tempo per frequentare il forum; eventualmente, contattatemi in PVT o nel mio sito. Fanboys |
|
|
|
|
|
#11 | |
|
Junior Member
Iscritto dal: Sep 2011
Messaggi: 12
|
Quote:
|
|
|
|
|
|
|
#12 |
|
Senior Member
Iscritto dal: Jan 2002
Città: Germania
Messaggi: 26110
|
Col metodo speciale __setitem__.
__________________
Per iniziare a programmare c'è solo Python con questo o quest'altro (più avanzato) libro @LinkedIn Non parlo in alcun modo a nome dell'azienda per la quale lavoro Ho poco tempo per frequentare il forum; eventualmente, contattatemi in PVT o nel mio sito. Fanboys |
|
|
|
|
|
#13 |
|
Junior Member
Iscritto dal: Sep 2011
Messaggi: 12
|
ho provato cosi:
Codice:
import copy
def setItemNostro(obj,q,b):
print('chiamo il 1')
a = copy.deepcopy(obj)
a.pop(q)
a.insert(q,b)
print('eccola',a)
return a
class metacl(type):
def __new__(meta,classname,supers,classdict):
classdict['__setitem__'] = setItemNostro
return type.__new__(meta,classname,supers,classdict)
class list(list,metaclass=metacl): pass
Codice:
if __name__ == "__main__":
l=list()
l.append(1)
l.append(2)
l.append(3)
l1=l
l1[2] = 'B'
print(l1)
print(l)
Codice:
chiamo il 1 eccola [1, 2, 'B'] [1, 2, 3] [1, 2, 3] |
|
|
|
|
|
#14 |
|
Senior Member
Iscritto dal: Jan 2002
Città: Germania
Messaggi: 26110
|
Come dicevo prima, non puoi farlo, perché dal testo di cui sopra dovresti riuscire a intercettare l'operazione di assegnazione fra variabili, mentre quella che hai intercettato è soltanto l'operazione di assegnazione di un elemento a un oggetto in grado di rispondere a quest'interfaccia.
__setitem_ sei riuscito a intercettarlo, ma non te ne fai niente. In pratica hai creato una copia di obj, l'hai modificata, ma quando esci dalla funzione "manipolatrice" tutto il lavoro che hai fatto viene buttato via. Per renderlo "persistente", dovresti modificare direttamente obj, ma se lo fai anche la variabile l verrà modificata di conseguenza. Non puoi nemmeno scrivere obj = a all'interno di quella funzione, perché ovviamente non funzionerebbe (semplicemente la variabile obj conterrebbe il nuovo valore, ma la lista originale rimarrebbe immutata).
__________________
Per iniziare a programmare c'è solo Python con questo o quest'altro (più avanzato) libro @LinkedIn Non parlo in alcun modo a nome dell'azienda per la quale lavoro Ho poco tempo per frequentare il forum; eventualmente, contattatemi in PVT o nel mio sito. Fanboys |
|
|
|
|
|
#15 |
|
Member
Iscritto dal: Apr 2008
Messaggi: 120
|
su un sito ho letto questo:
If you really want to override assignment, one way would be to define a property. Instead of 'var = value' you would then write 'var.propertyname = value'. Using a property, assignments are translated into calls to getter or setter methods, which can be overridden fairly easily, and which can modify the object 'in place'. Quindi si potrebbe integrare una metaclass con una proprierty con getter e setter per modificare l'oggetto? con la metaclasse invece di intercettare = che è impossibile intercettiamo la classe lista aggiungendogli questa proprietà (nstead of 'var = value' you would then write 'var.propertyname = value'). può essere fattibile? |
|
|
|
|
|
#16 |
|
Senior Member
Iscritto dal: Jan 2002
Città: Germania
Messaggi: 26110
|
Certamente, e lo fai coi metodi speciali __getattr__ e __setattr__ che servono proprio allo scopo.
Però è una cosa diversa da quella di cui abbiamo discusso finora.
__________________
Per iniziare a programmare c'è solo Python con questo o quest'altro (più avanzato) libro @LinkedIn Non parlo in alcun modo a nome dell'azienda per la quale lavoro Ho poco tempo per frequentare il forum; eventualmente, contattatemi in PVT o nel mio sito. Fanboys |
|
|
|
|
|
#17 |
|
Member
Iscritto dal: Apr 2008
Messaggi: 120
|
ma cosa ci metto dentro al setitem? come la uso deepcopy la dentro? setitem mi intercetta [1,2,3].
Io ho provato così, ma mi da l'effetto opposto, ovvero stampa sempre [1,2,3]!!! E se gli dico di mettere b senza usare la deepcopy mi ritroverei sempre con due [1,2,B]. Comef accio a fargli stampare due liste diverse ridefinendo setitem? Modificando setitem però alla fine invece di ottenere sempre [1, 2, 3] Codice:
import copy
stack = ()
def psetItem(obj,q,b): #[], posti 2, valore B
global stack
stack = copy.deepcopy(obj)
return obj
class metacl(type):
def __new__(meta,classname,supers,classdict):
classdict['__setitem__'] = psetItem
return type.__new__(meta,classname,supers,classdict)
class list(list, metaclass = metacl): pass
if __name__ == "__main__":
l=list()
l.append(1)
l.append(2)
l.append(3)
#print("llllll",l)
l1=l
l1[2] = 'B'
print("lll",l)
print("l1l1l1",l1)
print("lll",l)
|
|
|
|
|
| Strumenti | |
|
|
Tutti gli orari sono GMT +1. Ora sono le: 10:00.




















