Non è possibile utilizzare la stessa implementazione come oggetto risultato di os.stat() e altri. Comunque Python 2.6 ha una nuova funzione di fabbrica che crea un tipo di dati simile chiamato tupla. Una tupla con nome è una tupla i cui slot possono anche essere indirizzati per nome. La tupla chiamata non dovrebbe richiedere più memoria, secondo la documentazione, rispetto a una tupla normale, poiché non hanno un dizionario per istanza. La firma funzione di fabbrica è:
collections.namedtuple(typename, field_names[, verbose])
Il primo argomento specifica il nome del nuovo tipo, il secondo argomento è una stringa (spazio o separati da virgola) che contiene i nomi dei campi e, infine, se verbose è vero, il la funzione factory stamperà anche la classe generata.
Esempio
Supponiamo di avere una tupla contenente un nome utente e una password.Per accedere al nome utente che si ottiene l'elemento in posizione zero e la password si accede alla posizione uno:
credential = ('joeuser', 'secret123')
print 'Username:', credential[0]
print 'Password:', credential[1]
Non c'è niente di sbagliato in questo codice, ma la tupla non è auto-documentazione. Devi trovare e leggere la documentazione sul posizionamento dei campi nella tupla. Qui è dove tuple può venire in soccorso. Siamo in grado di ricodificare l'esempio precedente come segue:
import collections
# Create a new sub-tuple named Credential
Credential = collections.namedtuple('Credential', 'username, password')
credential = Credential(username='joeuser', password='secret123')
print 'Username:', credential.username
print 'Password:', credential.password
Se sei interessato di ciò che il codice è simile per la credenziale di tipo appena creata è possibile aggiungere verbose = True alla lista degli argomenti durante la creazione del tipo, in questo caso particolare si ottiene il seguente risultato:
import collections
Credential = collections.namedtuple('Credential', 'username, password', verbose=True)
class Credential(tuple):
'Credential(username, password)'
__slots__ =()
_fields = ('username', 'password')
def __new__(_cls, username, password):
return _tuple.__new__(_cls, (username, password))
@classmethod
def _make(cls, iterable, new=tuple.__new__, len=len):
'Make a new Credential object from a sequence or iterable'
result = new(cls, iterable)
if len(result) != 2:
raise TypeError('Expected 2 arguments, got %d' % len(result))
return result
def __repr__(self):
return 'Credential(username=%r, password=%r)' % self
def _asdict(t):
'Return a new dict which maps field names to their values'
return {'username': t[0], 'password': t[1]}
def _replace(_self, **kwds):
'Return a new Credential object replacing specified fields with new values'
result = _self._make(map(kwds.pop, ('username', 'password'), _self))
if kwds:
raise ValueError('Got unexpected field names: %r' % kwds.keys())
return result
def __getnewargs__(self):
return tuple(self)
username = _property(_itemgetter(0))
password = _property(_itemgetter(1))
la tupla denominato non si limita a fornire accesso ai campi di nome, ma contiene anche funzioni di supporto come la funzione _make() che aiuta a creare un'istanza credenziali da una sequenza o iterabile. Per esempio:
cred_tuple = ('joeuser', 'secret123')
credential = Credential._make(cred_tuple)
La documentazione della libreria Python per namedtuple ha più informazioni e esempi di codice, quindi suggerisco che si take a peek.