Penso che il modo più comune di aggiungere qualcosa a una raccolta sia utilizzare un tipo di Add
metodo fornito da una raccolta:
class Item {}
var items = new List<Item>();
items.Add(new Item());
e in realtà non c'è nulla di insolito in questo.
Mi chiedo comunque perché non lo facciamo in questo modo:
var item = new Item();
item.AddTo(items);
sembra in qualche modo più naturale del primo metodo. Ciò avrebbe il vantaggio che quando la Item
classe ha una proprietà come Parent
:
class Item
{
public object Parent { get; private set; }
}
puoi rendere privato il setter. In questo caso, ovviamente, non è possibile utilizzare un metodo di estensione.
Ma forse mi sbaglio e non ho mai visto questo schema prima perché è così insolito? Sai se esiste un tale schema?
In C#
un metodo di estensione sarebbe utile per questo
public static T AddTo(this T item, IList<T> list)
{
list.Add(item);
return item;
}
Che ne dici di altre lingue? Immagino che nella maggior parte di essi la Item
classe abbia dovuto fornire un'interfaccia chiamiamola così ICollectionItem
.
Update-1
Ci ho pensato un po 'di più e questo modello sarebbe davvero utile, ad esempio, se non si desidera aggiungere un elemento a più raccolte.
ICollectable
interfaccia di test :
interface ICollectable<T>
{
// Gets a value indicating whether the item can be in multiple collections.
bool CanBeInMultipleCollections { get; }
// Gets a list of item's owners.
List<ICollection<T>> Owners { get; }
// Adds the item to a collection.
ICollectable<T> AddTo(ICollection<T> collection);
// Removes the item from a collection.
ICollectable<T> RemoveFrom(ICollection<T> collection);
// Checks if the item is in a collection.
bool IsIn(ICollection<T> collection);
}
e un'implementazione di esempio:
class NodeList : List<NodeList>, ICollectable<NodeList>
{
#region ICollectable implementation.
List<ICollection<NodeList>> owners = new List<ICollection<NodeList>>();
public bool CanBeInMultipleCollections
{
get { return false; }
}
public ICollectable<NodeList> AddTo(ICollection<NodeList> collection)
{
if (IsIn(collection))
{
throw new InvalidOperationException("Item already added.");
}
if (!CanBeInMultipleCollections)
{
bool isInAnotherCollection = owners.Count > 0;
if (isInAnotherCollection)
{
throw new InvalidOperationException("Item is already in another collection.");
}
}
collection.Add(this);
owners.Add(collection);
return this;
}
public ICollectable<NodeList> RemoveFrom(ICollection<NodeList> collection)
{
owners.Remove(collection);
collection.Remove(this);
return this;
}
public List<ICollection<NodeList>> Owners
{
get { return owners; }
}
public bool IsIn(ICollection<NodeList> collection)
{
return collection.Contains(this);
}
#endregion
}
utilizzo:
var rootNodeList1 = new NodeList();
var rootNodeList2 = new NodeList();
var subNodeList4 = new NodeList().AddTo(rootNodeList1);
// Let's move it to the other root node:
subNodeList4.RemoveFrom(rootNodeList1).AddTo(rootNodeList2);
// Let's try to add it to the first root node again...
// and it will throw an exception because it can be in only one collection at the same time.
subNodeList4.AddTo(rootNodeList1);
add(item, collection)
, ma non è un buon stile OO.
item.AddTo(items)
supponiamo di avere una lingua senza metodi di estensione: naturale o no, per supportare addTo ogni tipo avrebbe bisogno di questo metodo e fornirlo per ogni tipo di raccolta che supporta l'aggiunta. Questo è come il miglior esempio di introduzione di dipendenze tra tutto ciò che io abbia mai sentito: P - Penso che la falsa premessa qui stia cercando di modellare un po 'di astrazione programmatica nella vita "reale". Questo spesso va storto.