zoukankan      html  css  js  c++  java
  • C# 泛型使用笔记

    泛型的基本概念我就不在这重复了,不了解的同学请自行百度。

    我主要写下我在项目中要到的泛型实例。献丑了.....有什么不好或不对的地方大家尽可评论留言。

    为什么要用泛型?

    通过使用泛型,我们可以极大地提高代码的重用度,同时还可以获得强类型的支持,避免了隐式的装箱、拆箱,在一定程度上提升了应用程序的性能。

    作为一个想法的程序猿,不能仅仅只会"复制 粘贴"。

    代码的封装、重用是可以大大提高你的效率,对于之后代码的修改及二次开发很有帮助。

    当然使你"看起来"很NB,增加自己的成就感。

    使用泛型实例

    我在项目用到泛型的地方主要是用在封装获取数据的时候。

    之前获取一个Model  PR_Form 的数据我是这样写的。

                public partial class PR_Form
                {
                    public PR_Form()
                    {}
                    public long FM_ID {get;set;}
                    public string PR_NO {get;set;}                        
                }
    //根据FM_ID获取PR_Form数据
            public PR_Form GetPRForms(long FM_ID)
            {
                //构造 where条件
                string searchCriteria = "where FM_ID=@FM_ID";
                //参数
                SqlParameter[] sqlParams = { new SqlParameter("@FM_ID", FM_ID) };
                PR_Form _form = new PR_Form();
                //拼接sql语句
                StringBuilder sqlBuilder = new StringBuilder();
                sqlBuilder.Append(@"select  [FM_ID],
                                            [PR_NO]
                    
        
                                            from PR_Form ");
                sqlBuilder.Append(searchCriteria);
                //查询数据   注:这里用dataset不是很好,哈哈
                DataSet ds = DbHelperSQL.Query(sqlBuilder.ToString(), sqlParams);
                //如果查到数据给_form 赋值
                if (ds.Tables[0].Rows.Count > 0)
                {
              //将记录映射为实例
    for (int i = 0; i < ds.Tables[0].Rows.Count; i++) { if (null != ds.Tables[0].Rows[i]["FM_ID"]) { _form.FM_ID = long.Parse(ds.Tables[0].Rows[i]["FM_ID"].ToString()); } if (null != ds.Tables[0].Rows[i]["PR_NO"]) { _form.PR_NO = ds.Tables[0].Rows[i]["PR_NO"].ToString(); } } } return _form; }

     上面代码可以优化的地方很多。

    首先我这边去掉了很多 PR_Form 属性,实际应用中PR_Form 的数据不止这几个。

    那么我每次在读取到数据的时候 “ 将记录映射为实例”是不是可以封装一下?方便之后重用?

    当然可以

     我们封装一个通用的 DataRow返回相应类的方法

    // 将记录映射为实例
            private T ConvertToModel<T>(DataRow dr) where T : class, new()//泛型的约束。class:使用此方法的必须是一个类,new():必须有空参数的构造函数
            {
                T o = new T();
                //获取指定类型的公共属性
                PropertyInfo[] propertys = o.GetType().GetProperties();
                foreach (PropertyInfo p in propertys)
                {
                    try
                    {
                        // 判断此属性是否有Setter
                        if (!p.CanWrite) 
                            continue;
    
                        object value = dr[p.Name];
                        if (value != DBNull.Value)
                            p.SetValue(o, dr[p.Name], null);
    
                    }
                    catch{}
                }
    
                return o;
            }

    然后我们的代码就变成这样了。

    //根据FM_ID获取PR_Form数据
            public PR_Form GetPRForms(long FM_ID)
            {
                //构造 where条件
                string searchCriteria = "where FM_ID=@FM_ID";
                //参数
                SqlParameter[] sqlParams = { new SqlParameter("@FM_ID", FM_ID) };
                PR_Form _form = new PR_Form();
                //拼接sql语句
                StringBuilder sqlBuilder = new StringBuilder();
                sqlBuilder.Append(@"select  [FM_ID],
                                            [PR_NO]
                                            from PR_Form ");
                sqlBuilder.Append(searchCriteria);
                //查询数据   注:这里用dataset不是很好,哈哈
                DataSet ds = DbHelperSQL.Query(sqlBuilder.ToString(), sqlParams);
                //如果查到数据给_form 复制
                if (ds.Tables[0].Rows.Count > 0)
                {
                    _form = ConvertToModel(ds.Tables[0].Rows[0]);
                }
    
                return _form;
            }

    博主:这样写之后我们不管 DataRow增加任何列都不用修改代码。

    博友:博主你在扯淡么? DataRow的列都是sql语句决定,如果数据库增加列,我还是要修改Model,和sql查询语句!

    博主:哈哈,是这样的。那我们继续封装.....接下来我们通过类的属性封装一下查询语句。

    博友:别别别。。。一步到位可以么?  如果我要查询其他Model的数据呢?这个也一起封装了岂不是更好?

    博主:OK。长话短说

    既然我们要通过类来生成Sql语句,那就要对Model类进行一些配置,比如说类对应的表名、表的主键、要查询的字段等。。。

    我们根据相应的配置来生成sql语句。

    首先我们要对model 类进行修改。给model类添加自定义属性

       [Table(TableName = "PR_Form", PrimaryKey = "FM_ID ", SortType = 0, SearchFields = "*")]
                //TableName:表示对应数据库的表名,PrimaryKey:表示表的主键,SearchFields:查询字段(如id,name....),我这里省事直接写了*
                public partial class PR_Form
                {
                    public PR_Form()
                    {}
                    public long FM_ID {get;set;}
                    public string PR_NO {get;set;}                        
                }

    添加接收自定义属性的类TableAttribute

     /// <summary>
        /// 数据库表属性
        /// </summary>
        public class TableAttribute : Attribute
        {
            public TableAttribute()
            {
            }
    
            /// <summary>
            /// 实例化
            /// </summary>
            /// <param name="tablename">表名或视图名 视图名要有对应类存在</param>
            public TableAttribute(string tablename)
            {
                _tablename = tablename;
            }
    
            /// <summary>
            /// 实例化
            /// </summary>
            /// <param name="tablename">表名或视图名 视图名要有对应类存在</param>
            /// <param name="primaryKey">主键 默认值:Id</param>
            public TableAttribute(string tablename, string primaryKey)
            {
                _tablename = tablename;
                _primaryKey = primaryKey;
            }
    
    
            private string _primaryKey = "Id";
            /// <summary>
            /// 主键 默认值:Id
            /// </summary>
            public string PrimaryKey { get { return _primaryKey; } set { _primaryKey = value; } }
    
            private string _tablename = "";
            /// <summary>
            /// 表名或视图名 视图名要有对应类存在
            /// </summary>
            public string TableName { get { return _tablename; } set { _tablename = value; } }
    
            private int _sortType = 0;
            /// <summary>
            /// 排序类型 查询时用到 默认值:-1(默认排序), 0(ASC) ,1(DESC)
            /// </summary>
            public int SortType { get { return _sortType; } set { _sortType = value; } }
    
            private string _sortFields = "Id";
            /// <summary>
            /// 排序字段 查询时用到 SortType>0时生效 默认值:Id
            /// </summary>
            public string SortFields { get { return _sortFields; } set { _sortFields = value; } }
    
            private string _SearchFields = "*";
            /// <summary>
            /// SearchFileds 查询时用到 默认值:*
            /// </summary>
            public string SearchFields { get { return _SearchFields; } set { _SearchFields = value; } }
    
    
            private int _Top = 0;
            /// <summary>
            /// Top多少数据 查询时用到 默认值:0 无限制
            /// </summary>
            public int Top { get { return _Top; } set { _Top = value; } }
    
    
    
            private string _ConnectionString = "ConnectionString";
            /// <summary>
            /// 连接字符串 默认为:ConnectionString
            /// </summary>
            public string ConnectionString { get { return _ConnectionString; } set { _ConnectionString = value; } }
    
    
        }

    获取model  PR_Form的自定义属性这样写。

    object[] objAttrs = new PR_Form().GetType().GetCustomAttributes(typeof(TableAttribute), true);

    OK,准备工作就绪。

    最后结果

         public T GetModel<T>(long FM_ID) where T : class, new()
            {
                T obj = new T();
                //获取 Model 配置
                object[] objAttrs = new T().GetType().GetCustomAttributes(typeof(TableAttribute), true);
                if (objAttrs.Length > 0)
                {
                    TableAttribute attr = objAttrs[0] as TableAttribute;
                    if (attr != null)
                    {
                        //参数
                        SqlParameter[] sqlParams = { new SqlParameter("@FM_ID", FM_ID) };
                        //构造 sql条件
                        string strSQL = "select  {0} from {1} where {2}=@FM_ID" ";
                        strSQL = string.Format(strSQL,  attr.SearchFields, attr.TableName, attr.PrimaryKey);
    
                        //查询数据   注:这里用dataset不是很好,哈哈
                        DataSet ds = DbHelperSQL.Query(strSQL, sqlParams);
                        if (ds.Tables[0] == null || ds.Tables[0].Rows.Count < 1)
                            return null;
    //将记录映射为实例 obj
    =ConvertToModel<T>(dt.Rows[0]); return obj; } } return null; }

    调用代码

    var _Form= bllbase.GetModel<PR_Form>( FormId);

     

    献丑了。这是我第一次在项目中使用泛型,大神莫笑我啊。

    那里不对的地方,请打加指出。

  • 相关阅读:
    vue 防抖 节流
    数组取最小数据长度,确定长度截取,看是否全等 ,全等通过不等提示,需要拆分
    数组去重取不重复的数据
    vue
    vue2.0 子组件获取父组件值 使用v-model可渲染不能更改
    使用mpvue 开发小程序 遇到的坑
    ztree 样式更改
    vue 跨域请求
    记录 vue2.0 再使用过程中遇到的问题
    bug
  • 原文地址:https://www.cnblogs.com/178mz/p/4618267.html
Copyright © 2011-2022 走看看