|
|||||||
|
|
|
![]() |
|
|
Strumenti |
|
|
#1 | ||
|
Senior Member
Iscritto dal: Feb 2003
Città: Stockholm (SE)
Messaggi: 1343
|
[C#/Vari] - Merge di N alberi con mapping table
Prima di tutto, ho taggato il topic come "vari" perché mi interessa anche un semplice consiglio algoritmico, al di lá della pura implementazione in C#.
Problema: nel database ho una view che é composta dai field ( IndexID, PlaceID, Name, ParentID, SiteID ) e che, per ciascun valore di IndexID, puó contenere diversi alberi univoci per SiteID. in breve, un Index puó essere genereato da diversi Site e ogni Site ha il suo proprio albero (di fatto esiste una "Stoccolma" per ciascun Site). Ora sto scrivendo una funzione che, per ciascun Index, dati gli alberi di ciascun Site, genera un unico albero utilizzando una tabella di mapping. La logica del merge dovrebbe essere la seguente: - se il nodo C1 é mappato al nodo D1, si usa il nodo D1 e si attaccano i figli del nodo C1 al nodo D1 fin qui tutto facile vero? bene, ora considerate questo, il nodo C13 (ovvero il terzo sottonodo del nodo C1), puó essere mappato al nodo D2. In questo caso deve essere rimosso dalla lista dei sottonodi D1. Piú interessante ora vero? Ed ora del sano codice. Iniziamo con le classi "contenitore"
Riporto una versione "concettuale" del Main Codice:
void Main()
{
IEnumerable<PlaceRow> placeRows = GetPlaceRows(93);
IEnumerable<IEnumerable<Place>> placeTrees = LoadTrees(placeRows);
IEnumerable<Mapping> mappings = GetMappings();
IEnumerable<Place> mergedTrees = MergeTrees(placeTrees, mappings);
}
Codice:
public IEnumerable<Place> MergeTrees (IEnumerable<IEnumerable<Place>> places, IEnumerable<Mapping> mappings)
{
var map = mappings.ToDictionary(k => k.SourceID, v => v.DestinationID);
throw new NotImplementedException("Epic fail");
}
La prima parte é banale. Per ciascuna lista, la si cicla e si controlla se qualche elemento é mappato. nel caso si aggiunge la lista di sottonodi al nodo di destinazione. Il problema é con i sottonodi mappati :S Grazie |
||
|
|
|
|
|
#2 |
|
Senior Member
Iscritto dal: Feb 2003
Città: Stockholm (SE)
Messaggi: 1343
|
Risolto! Cambiando totalmente approccio il problema era letteralmente banale.
Codice:
void Main()
{
IEnumerable<PlaceRow> placeRows = GetPlaceRows(93);
IEnumerable<Mapping> mappings = GetMappings();
IEnumerable<PlaceRow> mergedRows = MergeRows(placeRows, mappings);
IEnumerable<Place> placeTree = LoadTrees(mergedRows);
}
public IEnumerable<Place> LoadTrees(IEnumerable<PlaceRow> rows)
{
Func<PlaceRow, ILookup<int?, PlaceRow>, Place> transform = null;
transform = (row, grp) =>
{
Place place = new Place
{
ID = row.PlaceID,
Name = row.Name,
Places = from gr in grp[row.PlaceID]
select transform(gr, grp)
};
return place;
};
var lookup = rows.ToLookup( k=> k.ParentID);
return from row in lookup[null]
select transform(row, lookup);
}
public IEnumerable<PlaceRow> MergeRows(IEnumerable<PlaceRow> rows, IEnumerable<Mapping> mappings)
{
var map = mappings.ToDictionary(k => k.SourceID, v => v.DestinationID);
List<PlaceRow> newRowSet = new List<PlaceRow>();
foreach (var row in rows)
{
if (!map.ContainsKey(row.PlaceID))
{
newRowSet.Add(row);
if (row.ParentID.HasValue && map.ContainsKey(row.ParentID.Value))
row.ParentID = map[row.ParentID.Value];
}
}
return newRowSet;
}
In realtá mentre lo commentavo con un collega, per un nanosecondo avevo intravisto uno scenario non gestito. Ma me lo sono perso Ultima modifica di Kralizek : 15-06-2011 alle 14:29. |
|
|
|
|
| Strumenti | |
|
|
Tutti gli orari sono GMT +1. Ora sono le: 05:46.


















