zoukankan      html  css  js  c++  java
  • Linq行转列

    通过企业微信返回的打卡数据,数据是那种打一次卡,就有一次记录的格式。

    表结构为

    CREATE TABLE t_Checkindata
    (
    Cid int identity(1,1) primary key,
    userid varchar(10), --用户id
    groupname varchar(50), --打卡规则名称
    checkin_type varchar(20), --打卡类型。字符串,目前有:上班打卡,下班打卡,外出打卡
    exception_type varchar(100), --异常类型,字符串,包括:时间异常,地点异常,未打卡,wifi异常,非常用设备。如果有多个异常,以分号间隔
    checkin_time int --打卡时间戳
    )
    表结构非常简单,但是在前端展示数据时,反馈不清晰。根据对userid分组,使员工每天上下班打卡的时间能够同行显示。

    一、Linq 子查询行转列

              var query = from c in db.t_Checkindatas
                                        where (c.checkin_time < EndTime && c.checkin_time > StartTime && (c.t_Users.UserDepart == depart || c.t_Users.CorpName == depart))
                                        orderby c.checkin_time descending
                                        group c by c.userid into g
                                        select new
                                        {
                                            userid = g.Key,
                                            groupname = g.Select(u => u.groupname).FirstOrDefault(),
                                            exception_type = g.Select(u => u.exception_type).FirstOrDefault(),
                                            上班打卡 = (from d in g
                                                    where d.checkin_type == "上班打卡"
                                                    select d.checkin_time).Min(),
                                            下班打卡 = (from d in g
                                                    where d.checkin_type == "下班打卡"
                                                    select d.checkin_time).Max()

     

    对于代码中的where条件可以忽略,我这里是根据所选的部门和日期来筛选哪天的数据。

    返回exception_type的值处理有点问题,因为同一个人的打卡数据上午的打卡记录是正常的,下午因为忘记了而未打卡。

    此时两条数据,上午正常打卡返回的值为空,而下午返回的值为未打卡。所以对整天的数据进行降序排列,利用FirstOrDefault返回第一个序列的值。

    二、Linq 实现DataTable行转列

    因水平有限,我原始数据为List,这个解决方法是查找此类问题时,找到一位前辈多年前写过的博客,依葫芦画瓢,照搬下来解决的。

    所以怎么办呢,我得先把我List数据转为DataTable,头又疼了,这怎么转呢?请手动打开百度,搜索此标题。。。留下了没有技术的眼泪.jpg

            /// <summary>
            /// 将类型为list的打卡数据行转列后返回datatable
            /// </summary>
            /// <typeparam name="T"></typeparam>
            /// <param name="list">原始打卡数据</param>
            /// <returns></returns>
            public static DataTable ToDatatable3<T>(List<T> list)
            {
                using (MVCDBAdmin db = new MVCDBAdmin())
                {
                    DataTable source = ToDataTable2(list);//List转为DataTable
    
                    DataTable dt = new DataTable();
                    dt.Columns.Add("userid");
                    dt.Columns.Add("groupname");
                    dt.Columns.Add("exception_type");
    
                    var columns = (from x in source.Rows.Cast<DataRow>() select x[3].ToString()).Distinct();
                    foreach (var item in columns)
                    {
                        dt.Columns.Add(item).DefaultValue = 0;
                    }
                    var data = from x in source.Rows.Cast<DataRow>()
                               group x by x[0] into g
                               select new { Key = g.Key.ToString(), Items = g };
                    data.ToList().ForEach(x =>
                    {
                        string[] array = new string[dt.Columns.Count];
                        array[0] = x.Key;
                        array[1] = x.Items.ToList<DataRow>()[0]["groupname"].ToString();
                        array[2] = x.Items.ToList<DataRow>()[0]["exception_type"].ToString();
                        for (int i = 3; i < dt.Columns.Count; i++)
                        {
                            array[i] = (from y in x.Items
                                        where y[3].ToString() == dt.Columns[i].ToString()
                                        select y[4].ToString()
                                       ).FirstOrDefault(); //SingleOrDefault 若返回多个元素会发生异常
                            //如上班卡迟到,因为忘记打卡而申请补打,此时会有2条上班卡的记录,将引起报错
                            //暂使用FirstOrDefault返回打卡时间最早的那条记录,补卡申请完成后,系统会自动设置上班卡为标准起始时间
                        }
                        dt.Rows.Add(array);
                    }); 
                    return dt;
                }
            } 

    行转列完成后,我又将DataTable转换成List返回给前台。这绕来绕去的,有点无奈。

    参考博客地址:

    Linq DataTable行转列 https://www.cnblogs.com/li-peng/archive/2012/02/27/2370213.html#commentform

    关于List和DataTable 互相转换 https://www.cnblogs.com/shiyh/p/7478241.html

     

  • 相关阅读:
    动态库 DLL 封装二:dll封装方法
    动态库 DLL 封装一:dll分类
    C++读取配置文件ini
    WCHAR 字符串拼接
    关于web桌面应用的集成解决方案
    CSS Grid网格布局(转)
    使用 antd 的 form 组件来自定义提交的数据格式
    syslogd日志的一些作用
    我的三年感悟——避免无尽的自我内耗
    《被讨厌的勇气》读后感
  • 原文地址:https://www.cnblogs.com/qinsikao/p/12462041.html
Copyright © 2011-2022 走看看