c# - 组合两个类的成员选择器表达式

标签 c# inheritance multiple-inheritance expression-trees

我有两个具有一些共同属性的类(或更确切地说是模型)。例如:

public class Model1
{
    public int Common1 { get; set; }
    public int Common2 { get; set; }
    public int Prop11 { get; set; }
    public int Prop12 { get; set; }
}

public class Model2
{
    public int Common1 { get; set; }
    public int Common2 { get; set; }
    public int Prop21 { get; set; }
    public int Prop22 { get; set; }
}

我需要编写一个可以接受成员选择器表达式的方法,其中要选择的成员可以来自两个模型中的任何一个。这就是我的意思:

public List<T> GetAnon<T>(Expression<Func<??, T>> selector) // <-- intriguing part
{
    // codes to extract property names from selector and create the return object of type T
}

// And this is an example usage of the method
// At this point, I don't know which model Common1, Prop11 or Prop21 belongs to
var anonObj = GetAnon(m => new { m.Common1, m.Prop11, m.Prop21 });
// anonObj = { Common1 = VALUE_1, Prop11 = VALUE_2, Prop21 = VALUE_3 }

请注意,传递的选择器既选择了Model1,也选择了Model2

我目前的解决方案: 创建一个具有共同属性的模型,比如 Model,有 Model1Model2继承 Model 以及各个模型中的其余属性。然后制作另一个模型,比如 TempModel,继承 Model,并向其添加非公共(public)属性。像这样:

public class Model
{
    public int Common1 { get; set; }
    public int Common2 { get; set; }
}

public class Model1 : Model
{
    public int Prop11 { get; set; }
    public int Prop12 { get; set; }
}

public class Model2
{
    public int Prop21 { get; set; }
    public int Prop22 { get; set; }
}

public class TempModel : Model
{
    public int Prop11 { get; set; }
    public int Prop12 { get; set; }
    public int Prop21 { get; set; }
    public int Prop22 { get; set; }
}

// Finally, first type parameter can be TempModel
public List<T> GetAnon<T>(Expression<Func<TempModel, T>> selector)
{
    // codes to extract property names from selector and create the return object of type T
}

这种方法的问题是,如果 Model1Model2 很容易发生变化,我将不得不记住对 TempModel 进行更改 也是,这是我想避免的事情。另外,我想避免将属性名称作为字符串传递,这样我就不会丢失智能感知。

我怎样才能做到这一点?显然,多重继承在 C# 中是不可能的,否则我的问题就变得微不足道了。

编辑:

我已经编辑了我的问题以反射(reflect)我的实际需要(尽管我不确定这是否真的有帮助)。我实际上需要在方法 GetAnon() 中创建并返回一个匿名类型 T 的对象。要创建对象,我需要属性的名称,以便我可以执行检索(使用 SQL)。我不想通过将名称作为字符串传递来失去智能。另外,在调用方法时,我不知道哪个属性来自哪个模型,我只知道属性名称。

最佳答案

我认为您可能将选择器函数的使用过于复杂化了。我假设您想在那里使用 lambda,因为您不想在模型中更改字符串中的属性名称时担心更改它。这正是 nameof 函数的设计目的。

public List<string> GetMemberNames<T1, T2>(IEnumerable<string> propNames)
{
    var commonProps = (from pn in propNames
                       where
                       typeof(T1).GetProperties(BindingFlags.Instance | BindingFlags.Public)
                                 .Any(p => p.Name == pn) ||
                       typeof(T2).GetProperties(BindingFlags.Instance | BindingFlags.Public)
                                 .Any(p => p.Name == pn)
                       select pn).ToList();

    return commonProps;
}

用法:

var cp = GetMemberNames<Model1, Model2>(new[] 
                        { nameof(Model1.Common1),   
                          nameof(Model2.Common2), 
                          nameof(Model1.Prop11), 
                          nameof(Model2.Prop21),
                          "SomeUnknownProperty", 
                        });

请注意,现在您无需拥有任一模型的实例即可获取属性名称列表。

结果:

enter image description here

关于c# - 组合两个类的成员选择器表达式,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/48975317/

相关文章:

c# - SQLite .NET 是否在 CreateTable 方法中接受接口(interface)

python - 当您从 Python 中的模块而不是类继承时会发生什么?

inheritance - Kotlin 属性不能被子接口(interface)覆盖

iOS - 继承是如何工作的?

c++ - 使用多重继承来区分使用角色?

c++ - 多重继承模糊函数

java - 一个接口(interface)可以在Java中扩展多个接口(interface)吗?

c# - 改变谓词的表达式树以针对另一种类型

c# - 从图像动态创建 XNA Sprite

c# - MicrosoftAppId : '' is unauthorized to post to connector 的安全 token