惯性聚合 高效追踪和阅读你感兴趣的博客、新闻、科技资讯
阅读原文 在惯性聚合中打开

推荐订阅源

酷 壳 – CoolShell
酷 壳 – CoolShell
H
Hacker News: Front Page
P
Palo Alto Networks Blog
T
ThreatConnect
Apple Machine Learning Research
Apple Machine Learning Research
博客园_首页
T
True Tiger Recordings
P
Privacy & Cybersecurity Law Blog
B
Blog
IT之家
IT之家
Last Week in AI
Last Week in AI
F
Full Disclosure
Hacker News: Ask HN
Hacker News: Ask HN
C
Comments on: Blog
Microsoft Azure Blog
Microsoft Azure Blog
C
Cybersecurity and Infrastructure Security Agency CISA
Microsoft Security Blog
Microsoft Security Blog
博客园 - 【当耐特】
N
News and Events Feed by Topic
NISL@THU
NISL@THU
腾讯CDC
雷峰网
雷峰网
Security Latest
Security Latest
李成银的技术随笔
M
Microsoft Research Blog - Microsoft Research
L
LangChain Blog
L
Lohrmann on Cybersecurity
cs.CL updates on arXiv.org
cs.CL updates on arXiv.org
C
Check Point Blog
Y
Y Combinator Blog
Recent Announcements
Recent Announcements
博客园 - Franky
N
News | PayPal Newsroom
V
V2EX
A
About on SuperTechFans
The Register - Security
The Register - Security
月光博客
月光博客
奇客Solidot–传递最新科技情报
奇客Solidot–传递最新科技情报
Google Online Security Blog
Google Online Security Blog
MyScale Blog
MyScale Blog
Cisco Talos Blog
Cisco Talos Blog
Vercel News
Vercel News
WordPress大学
WordPress大学
C
Cyber Attacks, Cyber Crime and Cyber Security
The Hacker News
The Hacker News
IntelliJ IDEA : IntelliJ IDEA – the Leading IDE for Professional Development in Java and Kotlin | The JetBrains Blog
IntelliJ IDEA : IntelliJ IDEA – the Leading IDE for Professional Development in Java and Kotlin | The JetBrains Blog
爱范儿
爱范儿
A
Arctic Wolf
L
LINUX DO - 最新话题
freeCodeCamp Programming Tutorials: Python, JavaScript, Git & More

博客园 - BloggerSb

Swagger 文档设置api版本 .Net Core Routing Demo .net Core读取配置比较 ASP.NET Core CRUD API 创建 UserController,实现 Get, Post, Put, Delete 方法,使用 EF Core 访问数据库。 (编程题) 大文件单词统计 (编程题) 设计模式落地:Repository + UnitOfWork + CQRS 完整实现 (编程题) 给定百万级订单表,实现高效分页 + 动态条件查询 + 导出 Excel(避免内存爆炸) (编程题) 实现一个带 CorrelationId、请求日志、异常统一处理的中间件链 (编程题) 异步限流器实现(编程题) 编程题,记录所有接口的执行耗时 .net面试题目 (问答题) 面试高频简答题 Aspose最新Slides破解 HttpContext.User.Identity.IsAuthenticated 为false 关于Cannot resolve scoped service from root provider解决方案 MongoDB用户权限管理,设置密码并连接 mongodb连接字符串 mongodb 使用 MongoDB Compass 创建账号,角色 安装mongodb bootstrap popover 设置悬浮框宽度 div contenteditable="true" 添加placehoder效果 光标自动定位到起始位置contenteditable="true" ,v-html绑定内容,div可编辑时,光标移到最前面
简化版DbExecutor,将DataTable映射到T属性(支持Dapper风格的匿名参数)。(编程题)
BloggerSb · 2026-04-17 · via 博客园 - BloggerSb

手写一个简化版DbExecutor,支持Query<T>(string sql, object parameters),将DataTable映射到T属性(支持Dapper风格的匿名参数)。

· 考察点:反射/Emit/表达式树性能对比、缓存映射逻辑、IDbCommand参数化注入。

public class DbExecutor
{
    private readonly string _connectionString;
    // 缓存类型映射委托,避免频繁反射
    private readonly ConcurrentDictionary<Type, Func<IDataRecord, object>> _mapperCache;

    public DbExecutor(string connectionString)
    {
        _connectionString = connectionString;
        _mapperCache = new ConcurrentDictionary<Type, Func<IDataRecord, object>>();
    }

    // 执行查询,返回T类型列表
    public async Task<List<T>> QueryAsync<T>(string sql, object parameters = null)
    {
        var result = new List<T>();
        // 获取类型映射委托(缓存,避免重复反射)
        var mapper = _mapperCache.GetOrAdd(typeof(T), CreateMapper<T>);

        using var connection = new SqlConnection(_connectionString);
        await connection.OpenAsync();

        // 处理参数
        using var command = new SqlCommand(sql, connection);
        if (parameters != null)
        {
            // 反射获取参数对象的属性,添加到SqlCommand
            var paramProperties = parameters.GetType().GetProperties();
            foreach (var prop in paramProperties)
            {
                var value = prop.GetValue(parameters) ?? DBNull.Value;
                command.Parameters.AddWithValue($"@{prop.Name}", value);
            }
        }

        // 执行查询,映射结果
        using var reader = await command.ExecuteReaderAsync();
        while (await reader.ReadAsync())
        {
            var item = (T)mapper(reader);
            result.Add(item);
        }

        return result;
    }

    // 创建类型映射委托(使用Expression树,提升性能)
    private Func<IDataRecord, object> CreateMapper<T>(Type type)
    {
        // 创建参数(IDataRecord)
        var parameter = Expression.Parameter(typeof(IDataRecord), "record");
        var properties = type.GetProperties(BindingFlags.Public | BindingFlags.Instance);
        var bindings = new List<MemberBinding>();

        foreach (var prop in properties)
        {
            // 获取列名(支持[Column]特性,无特性则使用属性名)
            var columnAttr = prop.GetCustomAttribute<ColumnAttribute>();
            var columnName = columnAttr?.Name ?? prop.Name;

            // 表达式:record[columnName]
            var indexExpr = Expression.Property(parameter, "Item", Expression.Constant(columnName));
            // 转换类型(将object转换为属性类型)
            var convertExpr = Expression.Convert(indexExpr, prop.PropertyType);
            // 绑定到属性
            var binding = Expression.Bind(prop, convertExpr);
            bindings.Add(binding);
        }

        // 表达式:new T { Prop1 = ..., Prop2 = ... }
        var newExpr = Expression.New(type);
        var initExpr = Expression.MemberInit(newExpr, bindings);
        // 转换为object
        var convertToObject = Expression.Convert(initExpr, typeof(object));
        // 编译为委托
        return Expression.Lambda<Func<IDataRecord, object>>(convertToObject, parameter).Compile();
    }
}

// 辅助特性:用于指定数据库列名
[AttributeUsage(AttributeTargets.Property)]
public class ColumnAttribute : Attribute
{
    public string Name { get; }

    public ColumnAttribute(string name)
    {
        Name = name;
    }
}