使用递归集合的 C# 克隆 Class
C# Clone Class with recursive Collections
我有一个 Class 像这样绑定到 wpf 树视图
public class Vorlageneintrag : INotifyPropertyChanged
{
public string erw_Ausgabe { get; set; }
public string erw_Pfad { get; set; }
public string erw_Mailadresse { get; set; }
...
public ObservableCollection<Vorlageneintrag> Items {get; set;}
...
public Vorlageneintrag Copy()
{
return (Vorlageneintrag)this.MemberwiseClone();
}
}
工作正常,现在我添加了一个过滤器功能,它删除了所有不匹配“Vorlageneintrag”的过滤器。按我的意愿工作。但现在我的问题:
如何让我恢复原始列表以重置我的过滤器?
通过使用副本中的 .toList() 或 .MemberwiseClone() 重置 Itemssource 仅适用于第一个“层”,但不适用于 Vorlageneintrag.Items 或 Vorlageneintrag.Items[0]。Items 和更深的(可以有“无限的“子子子......树)。还是 create/fill 一个全新的对象会更好?
编辑:
筛选:
List<Vorlageneintrag> temp =((ObservableCollection<Vorlageneintrag>)tree_vorlagen.ItemsSource).ToList();
ObservableCollection<Vorlageneintrag> orglist=((ObservableCollection<Vorlageneintrag>)tree_vorlagen.ItemsSource);
foreach (var item in temp)
{
if(Loopandfilter(item).Items.Count==0)
{
orglist.Remove(item);
}
else
{
item.IsNodeExpanded = true;
}
}
private Vorlageneintrag Loopandfilter(Vorlageneintrag item)
{
if(item.Items.Count==0)
{
item.IsNodeExpanded = true;
return item;
}
else
{
List<Vorlageneintrag> temp = item.Items.ToList();
foreach (Vorlageneintrag tmp_item in temp)
{
if (Loopandfilter(tmp_item).Items.Count == 0)
{
if (!tmp_item.erw_Ausgabe.ToLower().Contains(tb_qicksearch.Text.ToLower()))
{
item.Items.Remove(tmp_item);
}
else
{
item.IsNodeExpanded = true;
}
}
}
}
return item;
}
当您调用 MemberwiseClone()
时,您正在创建一个 shallow 副本,这意味着 Items
集合项目不会被复制,而是引用到集合本身将被复制,因此原始和克隆的 Vorlageneintrag
对象将引用相同的 Items
集合。要制作独立于原始对象的深拷贝,您可以这样做:
public class Vorlageneintrag : INotifyPropertyChanged
{
public string erw_Ausgabe { get; set; }
public string erw_Pfad { get; set; }
public string erw_Mailadresse { get; set; }
public ObservableCollection<Vorlageneintrag> Items { get; set; }
public Vorlageneintrag Copy()
{
var shallowClone = (Vorlageneintrag)MemberwiseClone();
if (shallowClone.Items == null)
return shallowClone;
var cloneItems = new ObservableCollection<Vorlageneintrag>();
foreach (var item in Items)
cloneItems.Add(item.Copy());
shallowClone.Items = cloneItems;
return shallowClone;
}
public event PropertyChangedEventHandler PropertyChanged;
}
此外,由于您可能绑定到 Items
,因此您可能应该在 Items
setter 中调用 PropertyChanged
事件,尽管这取决于您的其余部分代码。
我有一个 Class 像这样绑定到 wpf 树视图
public class Vorlageneintrag : INotifyPropertyChanged
{
public string erw_Ausgabe { get; set; }
public string erw_Pfad { get; set; }
public string erw_Mailadresse { get; set; }
...
public ObservableCollection<Vorlageneintrag> Items {get; set;}
...
public Vorlageneintrag Copy()
{
return (Vorlageneintrag)this.MemberwiseClone();
}
}
工作正常,现在我添加了一个过滤器功能,它删除了所有不匹配“Vorlageneintrag”的过滤器。按我的意愿工作。但现在我的问题: 如何让我恢复原始列表以重置我的过滤器? 通过使用副本中的 .toList() 或 .MemberwiseClone() 重置 Itemssource 仅适用于第一个“层”,但不适用于 Vorlageneintrag.Items 或 Vorlageneintrag.Items[0]。Items 和更深的(可以有“无限的“子子子......树)。还是 create/fill 一个全新的对象会更好?
编辑: 筛选:
List<Vorlageneintrag> temp =((ObservableCollection<Vorlageneintrag>)tree_vorlagen.ItemsSource).ToList();
ObservableCollection<Vorlageneintrag> orglist=((ObservableCollection<Vorlageneintrag>)tree_vorlagen.ItemsSource);
foreach (var item in temp)
{
if(Loopandfilter(item).Items.Count==0)
{
orglist.Remove(item);
}
else
{
item.IsNodeExpanded = true;
}
}
private Vorlageneintrag Loopandfilter(Vorlageneintrag item)
{
if(item.Items.Count==0)
{
item.IsNodeExpanded = true;
return item;
}
else
{
List<Vorlageneintrag> temp = item.Items.ToList();
foreach (Vorlageneintrag tmp_item in temp)
{
if (Loopandfilter(tmp_item).Items.Count == 0)
{
if (!tmp_item.erw_Ausgabe.ToLower().Contains(tb_qicksearch.Text.ToLower()))
{
item.Items.Remove(tmp_item);
}
else
{
item.IsNodeExpanded = true;
}
}
}
}
return item;
}
当您调用 MemberwiseClone()
时,您正在创建一个 shallow 副本,这意味着 Items
集合项目不会被复制,而是引用到集合本身将被复制,因此原始和克隆的 Vorlageneintrag
对象将引用相同的 Items
集合。要制作独立于原始对象的深拷贝,您可以这样做:
public class Vorlageneintrag : INotifyPropertyChanged
{
public string erw_Ausgabe { get; set; }
public string erw_Pfad { get; set; }
public string erw_Mailadresse { get; set; }
public ObservableCollection<Vorlageneintrag> Items { get; set; }
public Vorlageneintrag Copy()
{
var shallowClone = (Vorlageneintrag)MemberwiseClone();
if (shallowClone.Items == null)
return shallowClone;
var cloneItems = new ObservableCollection<Vorlageneintrag>();
foreach (var item in Items)
cloneItems.Add(item.Copy());
shallowClone.Items = cloneItems;
return shallowClone;
}
public event PropertyChangedEventHandler PropertyChanged;
}
此外,由于您可能绑定到 Items
,因此您可能应该在 Items
setter 中调用 PropertyChanged
事件,尽管这取决于您的其余部分代码。