C#动态创建Lambda表达式

Fab*_*one 3 c# linq lambda expression func


我与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; }
}
Run Code Online (Sandbox Code Playgroud)

这是查询:

SELECT * FROM "PAGE" INNER JOIN "USER" ON "PAGE"."IdUserCreatedPage" = "USER"."IdUser"
Run Code Online (Sandbox Code Playgroud)

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

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();  
}
Run Code Online (Sandbox Code Playgroud)

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

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

Fab*_*one 6

最终,我找到了做自己想要的方法。
这是Generate函数的代码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();
}
Run Code Online (Sandbox Code Playgroud)

这是“ GetPage”方法:

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