gpt4 book ai didi

带有 MVVM 的 C# 泛型,将 T 从 中拉出

转载 作者:太空宇宙 更新时间:2023-11-03 18:48:02 25 4
gpt4 key购买 nike

我的 Model是一个包含(例如)Value 的通用类可以是 int、float、string、bool 等的属性。所以自然地,此类表示为 Model<T>。 .为了收藏Model<T>实现接口(interface) IModel , 尽管 IModel本身没有任何内容。

我的 ViewModel包含 Model<T> 的实例并通过ViewModel传入的构造函数。我还想知道什么T在 ViewModel 中,所以当我公开 Model 时到 View我知道 Model 的数据类型埋了Value属性(property)。 ViewModel 的类最终如下所示:

class ViewModel<T>
{
private Model<T> _model;

public ViewModel(Model<T> model) { ....blah.... }

public T ModelsValue {get; set; }

}

这工作正常,但有限制。所以现在我需要公开 IModels 的集合随着变化 Ts到我的View ,所以我正在尝试设置一个 ObservableCollection<strong>new</strong> ViewModel<T>s更改列表 IModels .问题是,我不知道如何获得 T来自 Model<T>来自 IModel build ViewModel<T>(Model<T>)在运行时。

在 VS2010 调试器中,我可以将鼠标悬停在任何 IModel 上对象并查看其完整 Model<int>例如在运行时,所以我知道数据在那里。

有什么想法吗?

最佳答案

这是我用于 View 模型集合的内容:

前言:

您的 View 模型对象可以是弱类型的。给IModel一处特性object Value {get;}并在 ModelViewModel : ViewModel<IModel> 中公开它你用于所有 IModel对象(参见下面我的 ViewModel<T> 实现)。如果你有 ObservableCollection<IModel> 的各种组合, ICollection<Model<T>>等,此处显示的框架是救星。如果您仍然需要通用 View 模型,您可以派生一个 ModelViewModel<T> : ModelViewModel这需要 Model<T>在它的构造函数中。创建适当类型的逻辑将进入传递给 ViewModelCollection.Create 的转换器中以下。 请注意,此设计会带来性能损失。

ModelViewModel CreateModelViewModel(IModel model)
{
Type viewModelType = typeof(ModelViewModel<>).MakeGenericType(model.Type);
ModelViewModel viewModel = Activator.CreateInstance(viewModelType, model);
return viewModel;
}

示例用法:

public class CatalogViewModel : ViewModel<ICatalog>
{
public CatalogViewModel(ICatalog catalog)
: base(catalog)
{
Func<ICatalogProduct, ProductViewModel> viewModelFactory = CreateProductViewModel;

this.Products = ViewModelCollection.Create(catalog.Products, viewModelFactory);
}

public ICollection<ProductViewModel> Products
{
get;
private set;
}

private ProductViewModel CreateProductViewModel(ICatalogProduct product)
{
return new ProductViewModel(product, this);
}
}

好处:

  • 使用延迟实现以允许在树中进行高效甚至递归绑定(bind)。
  • View 模型集合仅实现 INotifyCollectionChanged如果基础模型集合实现 INotifyCollectionChanged .

类概述(链接到 github 的完整实现):

一些可能对您的 View 模型集合有用的额外类型:

ConcatCollection :与 ViewModelCollection 一样,这包括一个静态帮助程序来自动选择适当的实现。 ConcatCollection 通过直接绑定(bind)到源集合来连接集合。

这是我如何使用此类型公开 Children 的示例将属性添加到 View ,同时一直维护我的可观察集合,直到返回原始来源。

public class ProductViewModel : ViewModel<IProduct>
{
public ProductViewModel(IProduct product)
: base(product)
{
Func<IProduct, ProductViewModel> productViewModelFactory = CreateProductViewModel;
Func<IRelease, ReleaseViewModel> releaseViewModelFactory = CreateReleaseViewModel;

this.Products = ViewModelCollection.Create(product.Products, productViewModelFactory);
this.Releases = ViewModelCollection.Create(product.Releases, releaseViewModelFactory);
this.Children = ConcatCollection.Create<object>((ICollection)this.Products, (ICollection)this.Releases);
}

public IList<ProductViewModel> Products
{
get;
private set;
}

public IList<ReleaseViewModel> Releases
{
get;
private set;
}

public IEnumerable<object> Children
{
get;
private set;
}

private ProductViewModel CreateProductViewModel(IProduct product)
{
return new ProductViewModel(product);
}

private ReleaseViewModel CreateReleaseViewModel(IRelease release)
{
return new ReleaseViewModel(release);
}
}

关于带有 MVVM 的 C# 泛型,将 T 从 <T> 中拉出,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/2698910/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com