含有大量数据的DataTable 设置了主键后,使用 Select() 查询效率显著提高

本文通过一个具体的案例,对比了在C#中使用foreach循环、LINQ查询和DataTable.Select方法在DataTable上进行数据检索的性能差异。通过对20万条记录的数据表进行相同条件的查询,实测了不同方法的执行时间和返回结果的数量。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

public class DataTableTest
{
private int dataCount = 10;
private DataTable dt = new DataTable();

    public DataTableTest(int initCount = 10)
    {
        dataCount = initCount;
        dt.Columns.Add("Id", typeof(int));
        dt.Columns.Add("CardNo", typeof(string));

        dt.PrimaryKey = new DataColumn[]
        {
            dt.Columns["CardNo"], //设置主键,可以显著提高Select效率
            //dt.Columns["Id"], //可以设置其他主键
        };


        for (int i = 0; i < dataCount; i++)
        {
            var dr = dt.NewRow();
            dr["id"] = i + 1;
            dr["CardNo"] = i + "A";
            dt.Rows.Add(dr);
        }
    }


    public int GetByForeach()
    {
        var count = 0;
        foreach ( DataRow item in dt.Rows)
        {
            //if ((int) item["Id"] > 10) count++;
            if (item["CardNo"].ToString() == "10000A") count++;
            
        }

        return count;
    }

    public int GetByLinq()
    {
        var list = (from item in dt.AsEnumerable()
                        //where item.Field<int>("Id") > 10
                        where item.Field<string>("CardNo") == "10000A"
                    select item).ToList();

        return list?.Count ?? 0;
    }

    public int GetBySelect()
    {
        //var ret = dt.Select("Id > 10");
        var cardNo = "10000A";

        var ret = dt.Select("CardNo ='" + cardNo + "'");
        return ret.Length;
    }

static void Main(string[] args)
{

        var ret = 0;
        Stopwatch sw = new Stopwatch();

        var dtTest = new DataTableTest(200000);

        
        sw.Start();
        ret = dtTest.GetByForeach();
        sw.Stop();
        Console.WriteLine("GetByForeach: ret=" + ret + ", " + sw.Elapsed.TotalMilliseconds);

        sw.Reset();
        sw.Start();
        ret = dtTest.GetByLinq();
        sw.Stop();
        Console.WriteLine("GetByLinq: ret=" + ret + ", " + sw.Elapsed.TotalMilliseconds);
        
        sw.Reset();
        sw.Start();
        ret = dtTest.GetBySelect();
        sw.Stop();
        Console.WriteLine("GetBySelect: ret=" + ret + ", " + sw.Elapsed.TotalMilliseconds);
        

        Console.Read();
    }

三种方式运行结果

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值