在运行时使用新的匿名类型创建lambda表达式

EPL*_*ens 4 c# reflection entity-framework anonymous-types expression-trees

我想调用一个需要这样的参数的方法:

Expression<Func<sometype, 'a>> expr
Run Code Online (Sandbox Code Playgroud)

我需要在运行时构造这个参数,因为我不知道之前的匿名类型是什么样的; 它可以有任何数量的字段:

x => new { a=x.a, b=x.b, c=x.c, etc... }
Run Code Online (Sandbox Code Playgroud)

我可以在运行时创建一个类型具有相同的'签名'(这是正确的单词吗?)作为所需的匿名类型,但问题是:如何在运行时从那个构造这个lambda表达式?特别是Expression.New是烦我的,因为我需要传递一个构造函数信息给它,我必须从一个现有的类型(它确实可以是一个匿名类型,但我不能在运行时创建一个匿名类型.或者在那里一种方法吗?).

更新(评论中要求的一些上下文)

我想调用的方法是:

DependentNavigationPropertyConfiguration.HasForeignKey<TKey>(Expression<Func<TDependentEntityType, TKey>> foreignKeyExpression)
Run Code Online (Sandbox Code Playgroud)

我想这样做的原因是自动为从某个基类继承的实体创建一个导航属性,包括该外键中该基类的键.因为实体可以有多个任何类型的键字段,所以TKey类型只在运行时才为我所知.

sir*_*lot 10

使用单独的方法:

public static void Main()
{
    var myExpression = Express(str => new { 
        String = str, 
        Length = str.Length 
    });

    // We can compile/use it as well...
    var compiledExpression = myExpression.Compile();
    var anonymousOutput = compiledExpression("Input String");

    Console.WriteLine(anonymousOutput.String); // Output: Input String
    Console.WriteLine(anonymousOutput.Length); // Output: 12

    Debug.WriteLine(myExpression); // Output: "str => new <>f__AnonymousType0`2(String = str, Length = str.Length)"
    Console.ReadLine();
}


static Expression<Func<String, T>> Express<T>(Expression<Func<String, T>> expression)
{
    return expression;
}
Run Code Online (Sandbox Code Playgroud)

但请注意,String必须事先知道起始类型(在我的示例中).

更新:

既然你想要做的是动态创建一个类型,我会给你一个如何做到这一点的简单例子.

public static void Main()
{
        // Create an anonymous type with two fields
    Type myAnonymousType = CreateNewType<String, Int32>();
    dynamic myAnon = Activator.CreateInstance(myAnonymousType);

    myAnon.FieldA = "A String";
    myAnon.FieldB = 1234;


    Console.WriteLine(myAnon.FieldA); // Output : "AString"
    Console.WriteLine(myAnon.FieldB); // Output : 1234
    Console.ReadLine();
}

public static Type CreateNewType<TFieldTypeA, TFieldTypeB>()
{
    // Let's start by creating a new assembly
    AssemblyName dynamicAssemblyName = new AssemblyName("MyAsm");
    AssemblyBuilder dynamicAssembly = AssemblyBuilder.DefineDynamicAssembly(dynamicAssemblyName, AssemblyBuilderAccess.Run);
    ModuleBuilder dynamicModule = dynamicAssembly.DefineDynamicModule("MyAsm");

    // Now let's build a new type
    TypeBuilder dynamicAnonymousType = dynamicModule.DefineType("MyAnon", TypeAttributes.Public);

    // Let's add some fields to the type.
    FieldInfo dynamicFieldA = dynamicAnonymousType.DefineField("FieldA", typeof(TFieldTypeA), FieldAttributes.Public);
    FieldInfo dynamicFieldB = dynamicAnonymousType.DefineField("FieldB", typeof(TFieldTypeB), FieldAttributes.Public);

    // Return the type to the caller
    return dynamicAnonymousType.CreateType();
}
Run Code Online (Sandbox Code Playgroud)

如您所见,这有点复杂.如果你想进一步研究这个话题,请务必参考Reflectoin.Emit.

  • 我的意思是我可以创建这个表达式: **x=&gt;new MyAnonymousType() {a=xa, b=xb}**,但我无法创建 **x=&gt;new {a=xa, b =xb}** 我尝试调用的方法需要第二种表达式(没有显式类型)。 (2认同)