Я пытаюсь глубоко клонировать следующее использование класса AutoMapper:
public class MainData
{
public MainData()
{
Details = new List<Detail>();
}
public int Id { get; private set; }
public DateTime LastUpdate { get; private set; }
public IList<Detail> Details { get; private set; }
public int Prop1 { get; set; }
public int Prop2 { get; set; }
public void AddDetail(Detail detail)
{
Details.Add(detail);
}
public void RemoveDetail(Detail detail)
{
Details.Remove(detail);
}
public MainData Clone()
{
Mapper.Reset();
Mapper.CreateMap<MainData, MainData>().ForMember(d => d.Id, o => o.Ignore());
// Mapper.CreateMap<Detail, Detail>().ForMember(d => d.Id, o => o.Ignore()); // REMOVED
var newMainData = new MainData();
Mapper.Map(this, newMainData);
newMainData.Details = this.Details.Select(item => item.Clone()).ToList(); // ADDED
return newMainData;
}
}
public class Detail
{
public int Id { get; private set; }
public string Name { get; set; }
public double Area { get; set; }
public double Height { get; set; }
public Detail Clone() // ADDED
{
Mapper.CreateMap<Detail, Detail>().ForMember(d => d.Id, o => o.Ignore());
var newDetail = new Detail();
Mapper.Map(this, newDetail);
return newDetail;
}
}
Clone
метод хорошо работает для свойств MainData, но, кажется, только делает мелкую копию списка Деталей. Я попытался добавить .ForMember(d => d.Details, o => o.UseDestinationValue())
но это не копирует список Деталей вообще. Как я могу получить список Деталей, глубоко клонированный также т.е., таким образом, я заканчиваю с двумя полностью независимыми объектами включая все элементы списка?
ОБНОВЛЕНИЕ: Я должен исключить свойство Id, поскольку я использую эти объекты с NHibernate, таким образом, не уверенным, если сериализуемое решение сделает это.
UPDATE2: Измененный вышеупомянутый код для клонирования IList также. Это, кажется, хорошо работает, поскольку я могу исключить свойства, которые заставляют NHibernate думать, что он был уже сохранен.
вот одно решение с ValueInjecter
var clone = new MainData();
clone.InjectFrom(mainData);//mainData is your source
mainData.Details.AsParallel.ForAll(detail =>
{
var dc = new Detail();
dc.InjectFrom(detail);
clone.AddDetail(dc);
});
, свойства, у которых есть частные сеттеры, не будут устанавливаться (выглядит разумным)
удачи;)
РЕДАКТИРОВАТЬ: Я сделал лучшее решение, посмотрите здесь
AutoMapper на самом деле не является API клонирования. Вместо этого я бы использовал этот трюк клонирования:
public static object CloneObject(object obj)
{
using (MemoryStream memStream = new MemoryStream())
{
BinaryFormatter binaryFormatter = new BinaryFormatter(null,
new StreamingContext(StreamingContextStates.Clone));
binaryFormatter.Serialize(memStream, obj);
memStream.Seek(0, SeekOrigin.Begin);
return binaryFormatter.Deserialize(memStream);
}
}
Он работает не во всех ситуациях, но довольно удобен.