C# 动态创建 Lambda 表达式

3


我使用Dapper,尝试创建内连接的自动映射方法。
以下是模型示例:

public class User
{
    public long IdUser { get; set; }
    public string Email { get; set; }
} 

public class Page
{
    public long Id { get; set; }
    public string Name { get; set; }
    public long IdUserCreatedPage { get; set; }
    public User UserCreatedPage { get; set; }
}

这是查询:

SELECT * FROM "PAGE" INNER JOIN "USER" ON "PAGE"."IdUserCreatedPage" = "USER"."IdUser"

如果我手写代码,我会写成这样:

public List<Page> GetPage(IDbConnection dbConnection, string sql)
{
    return (List<Page>)dbConnection.Query<Page, User, Page>(sql, 
        (Page p, User u) =>
        {
            p.UserCreatedPage = u;
            return p;
        },
        splitOn: "IdUser").ToList();  
}

现在,我想要的是动态创建我需要用于映射对象的 Func<TFirst, TSecond, TOut>
有人能帮助我吗?非常感谢。

P.S. 我知道在这种情况下动态创建它没有意义,但这只是自动映射项目的简单版本。

这可能会有所帮助:https://github.com/kahanu/System.Linq.Dynamic/wiki/Dynamic-Expressions - Richard Friend
这是一个问答网站。如果您已经解决了问题,请添加并接受一个答案。在论坛网站的标题中加入[SOLVED]的做法是因为这些网站不是为像这个网站一样寻找问题答案而设计的。 - Jon Hanna
@JonHanna 好的,我做到了! - Fabrizio Pairone
2个回答

9

解决方案

最终我找到了实现想要的功能的方法。
这是生成 Func<TFirst, TSecond, TOut> 函数的代码:

public static Func<TFirst, TSecond, TFirst> MappingDynamicFunc<TFirst, TSecond>()
{
    ParameterExpression paramFirst = Expression.Parameter(typeof(TFirst), "paramFirst");
    ParameterExpression paramSecond = Expression.Parameter(typeof(TSecond), "paramSecond");

    MemberExpression memberExpression = Expression.PropertyOrField(paramFirst, "UserCreatedPage");
    BinaryExpression assign = Expression.Assign(memberExpression, paramSecond);

    LabelTarget labelTarget = Expression.Label(typeof(TFirst));
    GotoExpression returnExpression = Expression.Return(labelTarget, paramFirst, typeof(TFirst));
    LabelExpression labelExpression = Expression.Label(labelTarget, Expression.Default(typeof(TFirst)));

    BlockExpression block = Expression.Block(
        assign,
        returnExpression,
        labelExpression
    );

    return Expression.Lambda<Func<TFirst, TSecond, TFirst>>(block, new ParameterExpression[] { paramFirst, paramSecond }).Compile();
}

这是"GetPage"方法:

public List<Page> GetPage(IDbConnection dbConnection, string sql)
{
    return (List<Page>)dbConnection.Query<Page, User, Page>(sql, 
        MappingDynamicFunc<Page, User>(),
        splitOn: "IdUser").ToList();  
}

3
请查看PredicateBuilder。 http://www.albahari.com/nutshell/predicatebuilder.aspx 以下是一些伪代码。
var predicate = PredicateBuilder.True<SomeClass>();

if (SomeCondition)
{
     var inner = PredicateBuilder.False<SomeClass>();
         inner = inner.Or(p => p.Category == "WhatEver");
         inner = inner.Or(p => p.Category == "");

     predicate = predicate.And(inner);
}

...

var result = MyIEnumerable<SomeClass>.AsQueryable()
                                     .Where(predicate)
                                     .FirstOrDefault();

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接