DataTable

* DataTable的循环

1、DataTable的遍历

 获得的值都是object类型,都需要先判断是否是DBNull.Value然后再转换

1、linq 语句

var result = from dr in dt.AsEnumerable()
             where !dr.IsNull("BookId") && int.Parse(dr.ItemArray[0].ToString()) == 3
             select new
             {
                 BookId = Convert.ToInt32(dr["BookId"]),
                 Title = dr["Title"]
             };

2、for

for(int i=0; i<dt.Rows.Count; i++)
{
    var c1 = dt.Rows[i]["BookId"] == DBNull.Value ? 0 : Convert.ToInt32(dt.Rows[i]["BookId"]) ;
}
foreach(var item in dt.AsEnumerable())
{
     var c2 = item["BookId"];
     var c3 = item["Title"];
}

   可以用linq语法

 

 

 2、DataTable的计算

           var money = dt.AsEnumerable().Select(k => new          //第一次select得到值
            {
                f_money = Convert.ToDecimal(k["price"]),
                f_title = k["title"].ToString(),
                f_id = Convert.ToInt32(k["BookId"]),
                f_author = k["author"].ToString()
            }).GroupBy(k => k.f_author)                         //groupby进行了分组,得到的是一个key,对应一组value
            .Select(k => new {
                f_money = k.Sum(v => v.f_money),                //分组之后这里只能用聚合函数
                f_author = k.Key                               //得到key
            });
  //添加统计数据
            var allMoney = dtAllMoney.AsEnumerable().Select(k => new
            {

                f_name = isExcel == 1 ? LocalHelper.Lang("Font_meiritongji") : "<b><span style='color:blue'>" + LocalHelper.Lang("Font_meiritongji") + "</span></b>",
                f_bankId = "",
                f_money = Convert.ToDecimal(k["f_money"]),
                f_time = k["f_time"].ToString(),
                f_number = "",
                f_fee = dtAllMoney.Columns.Contains("f_fee") ? (k.IsNull("f_fee") ? 0 : Convert.ToDecimal(Math.Floor(Convert.ToDouble((Convert.ToDouble(k["f_fee"]) * 100).ToString())) / 100)) : 0
            }).GroupBy(k => k.f_time).Select(k => new
            {
                f_name = k.Min(v => v.f_name),
                f_allMoney = k.Sum(v => v.f_money),
                f_allFee = k.Sum(v => v.f_fee),
                f_time = k.Key
            }).GroupBy(k => k.f_name);


//进行循环
            List<StatisticsModel> statisticsList2 = new List<StatisticsModel>();
            foreach (var item in allMoney)
            {
                StatisticsModel tempeModel = new StatisticsModel();
                tempeModel = item.Select(k => new StatisticsModel
                {
                    f_sort = int.MinValue,
                    f_name = item.Min(v => v.f_name),
                    f_bankid = item.Key.ToString(),
                    f_number = 0,
                    f_allMoney = item.Sum(v => v.f_allMoney),
                    f_allFee = item.Sum(v => v.f_allFee)
                }).FirstOrDefault();
                decimal[] f_money = new decimal[2];
                decimal[] f_fee = new decimal[2];
                f_money[0] = item.Where(v => v.f_time == string.Concat(sYesteryear, "-", sYestermonths, sYesterday)).Sum(v => v.f_allMoney);
                f_money[1] = item.Where(v => v.f_time == string.Concat(year, "-", months, sThisDay)).Sum(v => v.f_allMoney);
                f_fee[0] = item.Where(v => v.f_time == string.Concat(sYesteryear, "-", sYestermonths, sYesterday)).Sum(v => v.f_allFee);
                f_fee[1] = item.Where(v => v.f_time == string.Concat(year, "-", months, sThisDay)).Sum(v => v.f_allFee);
                tempeModel.f_Money = f_money;
                tempeModel.f_Fee = f_fee;
                statisticsList2.Add(tempeModel);
            }
原创

* DataTable、List、Enumerable比较

1、DataTable

            DataTable dt = SqlHelper.ExecuteDataTable("select * from bookinfo", null);
            DataView dtview = dt.DefaultView;
            DataColumnCollection columns = dt.Columns;
            DataSet dset = dt.DataSet;
            dt.Merge(dt); //返回void类型
            DataRow[] dtRows = dt.Select("bookid = 1");

2、List实现了IEnumerable接口

List实现了IEnumerable,可以用Enumerable类中的方法

是否支持foreach变量,必须满足2个条件:

1、实现IEnumerable接口;

2、这个类有一个public 的GetEnumerator的实例方法,并且返回类型IEnmerator具有public的bool MoveNext()方法和public的Current属性

IEnumerable 可迭代的,声明式的接口,没有具体实现如果迭代,仅有返回IEnumerator类型的方法;自定义类要使用foreach循环,就必须实现IEnumerable接口

    [TypeDependencyAttribute("System.SZArrayHelper")]
    public interface IEnumerable<out T> : IEnumerable
    {
        //
        // 摘要:
        //     返回一个循环访问集合的枚举器。
        //
        // 返回结果:
        //     用于循环访问集合的枚举数。
        IEnumerator<T> GetEnumerator();
    }
IEnumerable接口

IEnumerator  迭代器,实现该接口就可以作为迭代器(iterator),包含了具体要实现的方法;

    //
    // 摘要:
    //     支持对非泛型集合的简单迭代。
    [ComVisible(true)]
    [Guid("496B0ABF-CDEE-11d3-88E8-00902754C43A")]
    public interface IEnumerator
    {
        //
        // 摘要:
        //     获取集合中位于枚举数当前位置的元素。
        //
        // 返回结果:
        //     集合中位于枚举数当前位置的元素。
        object Current { get; }

        //
        // 摘要:
        //     将枚举数推进到集合的下一个元素。
        //
        // 返回结果:
        //     如果枚举数已成功地推进到下一个元素,则为 true;如果枚举数传递到集合的末尾,则为 false。
        //
        // 异常:
        //   T:System.InvalidOperationException:
        //     创建枚举器后,已修改该集合。
        bool MoveNext();
        //
        // 摘要:
        //     将枚举数设置为其初始位置,该位置位于集合中第一个元素之前。
        //
        // 异常:
        //   T:System.InvalidOperationException:
        //     创建枚举器后,已修改该集合。
        void Reset();
    }
IEnumerator
            DataTable dt = SqlHelper.ExecuteDataTable("select * from bookinfo", null);

            List<BookInfo> list = ConvertHelper<BookInfo>.ConvertToList(dt);
            bool isContainElement = list.Any();   //不包含时为false
            var k3 = list.Average(w => w.Price);
            var list1 = list.AsEnumerable();
            var k2 = list1.Average(w => w.Price);
            var k4 = list1.Where(k => k.BookId == 1);
            var k1 = list1.Select(k => new { f_dd = k.Author });

3、linq语句

            dt.Merge(dt2);  //融合DataTable返回值为void
            var banks = from dr in dt.AsEnumerable()
                        where !dr.IsNull("f_id") && !dr.IsNull("f_ShowName")
                        select new { f_id = dr["f_id"].ToString(), f_name = dr["f_ShowName"].ToString() };
            return Json(banks);

4、DataTable转为List

    public static class ConvertHelper<T> where T : new()
    {
        /// <summary>
        /// DateTable 转List
        /// </summary>
        /// <param name="dt"></param>
        public static List<T> ConvertToList(DataTable dt)
        {
            List<T> list = new List<T>();

            Type type = typeof(T);

            string tempName = string.Empty;

            foreach (DataRow dr in dt.Rows)
            {
                T t = new T();

                //获得公共属性,利用反射获得属性
                PropertyInfo[] propertys = t.GetType().GetProperties();
                foreach (var it in propertys)
                {
                    tempName = it.Name;

                    //是否包含某一列
                    if (dt.Columns.Contains(tempName))
                    {
                        if (!it.CanWrite) continue;

                        object value = dr[tempName];
                        if (value != DBNull.Value)
                        {
                            //得到列的数据类型
                            string typ = it.GetGetMethod().ReturnType.Name.ToLower();
                            switch (typ)
                            {
                                case "int32":
                                    it.SetValue(t, Convert.ToInt32(value.ToString() == "" ? "0" : value) as object, null);
                                    break;
                                case "double":
                                    it.SetValue(t, Convert.ToDouble(value.ToString() == "" ? "0" : value) as object, null);
                                    break;
                                case "string":
                                    it.SetValue(t, value.ToString() as object, null);
                                    break;
                                case "decimal":
                                    it.SetValue(t, Convert.ToDecimal(value.ToString() == "" ? "0" : value) as object, null);
                                    break;
                                default:
                                    it.SetValue(t, value, null);
                                    break;
                            }
                        }
                    }
                }

                list.Add(t);
            }
            return list;
        }

    }
View Code

* 常用

        static void Main(string[] args)
        {
            DataTable dt = new DataTable();
            dt.Columns.Add("Id");           //添加列
            dt.Columns.Add("Name");
            dt.Rows.Add("1", "刘备");       //添加行
            dt.Rows.Add("2", "关羽");
            dt.Rows.Add("3", "张飞");
            dt.Rows.Add("4", "赵云");
            dt.Rows.Add("5", "黄忠");

            DataRow[] drArr = dt.Select("Id > 1 and Name <> '关羽'" ,"Id desc");    //条件支持and or like与SQL语句类似
            //DataRow[] drArr2 = dt.Select("Name like '关%'", "Id desc");    //like示例
            foreach(DataRow dr in drArr)
            {
                Console.WriteLine(dr["Name"]);  //输出 黄忠 赵云 张飞
            }

            Console.ReadKey();
        }
1、select
        static void Main(string[] args)
        {
            DataTable dt = new DataTable();
            dt.Columns.Add("Id");  
            dt.Columns.Add("Name");
            dt.Columns.Add("Age", typeof(Int32));       //注意要计算平均值要设置列类型为数字
            dt.Rows.Add("1", "刘备", 31); 
            dt.Rows.Add("2", "关羽", 29);
            dt.Rows.Add("3", "张飞", 28);
            dt.Rows.Add("4", "赵云", 27);
            dt.Rows.Add("5", "黄忠", 50);

            dt.DefaultView.Sort = "Age DESC";           //排序
            dt = dt.DefaultView.ToTable();                 //转换为DataTable

            foreach (DataRow dr in dt.Rows)
            {
                Console.WriteLine(dr["Name"]);
            }


            object obj = dt.Compute("Count(Id)", "Age>26");     //输出3 查询年龄大于26的人数
            Console.WriteLine(obj.ToString());

            object obj2 = dt.Compute("Avg(Age)", "true");     //输出33 查询所有人的平均年龄(要设置列为整型,否则不能计算)
            Console.WriteLine(obj2.ToString());

            object obj3 = dt.Compute("Sum(Age)", "true");     //输出165
            Console.WriteLine(obj3.ToString());

            Console.ReadKey();
        }
2、排序、聚合
       static void Main(string[] args)
        {
            DataTable dt = new DataTable();
            DataColumn dc = dt.Columns.Add("Id", Type.GetType("System.Int32"));
            dc.AutoIncrement = true;    //自动增加
            dc.AutoIncrementSeed = 1;   //起始为1
            dc.AutoIncrementStep = 1;   //步长为1
            dc.AllowDBNull = false;     //不允许为空
            
            dt.Columns.Add("Name", Type.GetType("System.String"));
            dt.Columns.Add("Age", typeof(Int32));       //注意要计算平均值要设置列类型为数字
            dt.Rows.Add(null,"刘备", 31);            //注意这次添加不用在手输Id了,但是要给个null
            dt.Rows.Add(null,"关羽", 29);
            dt.Rows.Add(null,"张飞", 28);
            dt.Rows.Add(null,"赵云", 27);
            dt.Rows.Add(null, "黄忠", 50);


            foreach (DataRow dr in dt.Rows)
            {
                Console.WriteLine(dr["Id"]);    //输出1 2 3 4 5
            }


            Console.ReadKey();
        }
3、自增列

1、System.DBNull

    null在.Net中表示无效的对象引用。 即空对象。

  DBNull是一个类(System.DBNull)。这个类直接继承于Object,它只有继承下来的属性与方法,只有一个自己的字段value。表示它自己。指数据库中数据为空(NULL)时,在.Net中的值。DBNull.Value表示一个对象在数据库中的值为空,或者说未初始化,DBNull.Value对象是指向有效的对象。

  但是为什么 DBNull 可以表示数据库中的字符串,数字,或日期呢?原因是.Net储存这些数据的类(DataRow等)都是以 object 的形式来储存数据的

  对于 DataRow , 它的 row["column"] 返回的值永远不为null , 要么就是具体的为column的类型的值。要么就是DBNull 。 所以 row[column].ToString() 这个写法永远不会在ToString那里发生NullReferenceException,但有可能抛下标越界的异常。

  DBNull 实现了 IConvertible 。 但是,除了 ToString 是正常的外,其他的ToXXX都会抛出不能转换的错误。

  在 IDbCommand(OleDbCommand,SqlCommand...) 的ExecuteScalar的返回值中,情况可以这样分析:

SELECT 1     --这样返回的object是1。
SELECT null --这样返回的是DBNull.Value。

  示例,假设在DataSet中我设置了一个int类型,但是在显示的时候,我想让为0的地方显示为空白该怎么实现呢?

        static void Main(string[] args)
        {
            DataTable dt = new DataTable();
            dt.Columns.Add("Name", typeof(String));
            dt.Columns.Add("Age",typeof(Int32));
            dt.Rows.Add("撼地神牛",0);
            dt.Rows.Add("刘备", 21);
            for (int i = 0; i < dt.Rows.Count; i++)
            {
                if (dt.Rows[i]["Age"].ToString() == "0") //注意DataRow返回的都是Object类型,需要自己转换为对应的类型
                {
                    //dt.Rows[i]["Age"] = null;     这样写会提示如下错误:不能将 Column“Age”设置为 null。请改用 DBNull。
                    dt.Rows[i]["Age"] = DBNull.Value; //用DBNull.Value; 在C#中显示空白,在数据库中显示null
                }
                Console.WriteLine(dt.Rows[i]["Name"] + ":" + dt.Rows[i]["Age"]);
            }

            Console.ReadKey();
        }

 

posted on 2018-06-29 13:17  莫伊筱筱  阅读(320)  评论(0编辑  收藏  举报