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{// 判断此属性是否有Setterif (!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);

 

 

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

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

 

转载于:https://www.cnblogs.com/178mz/p/4618267.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/459204.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

汇编题目:编写包含多个功能子程序的中断例程

安装一个新的int 7ch中断例程&#xff0c;为显示输出提供如下的功能子程序。 &#xff08;1&#xff09;清屏。&#xff08;2&#xff09;设置前景色。&#xff08;3&#xff09;设置背景色。&#xff08;4&#xff09;向上滚动一行 &#xff08;一&#xff09;用ah寄…

micropython web ws2812_MicroPython实例之TPYBoard v102炫彩跑马灯WS2812B

一、实验目的了解ws2812b的工作原理学习ws2812b的驱动方法二、实验器材TPYBoard v102 1块ws2812b RGB-Ring-8 1个micro USB数据线 1条杜邦线 若干三、WS2812B的介绍WS2812B是一个集控制电路与发光电路于一体的智能外控LED光源。 其外型与一个5050LED灯珠相同&#xff0c; 每个元…

工作面试经历

这几天去找了找工作&#xff0c;面试了两家IT公司&#xff0c;下面就说说关于这两次面试的经历以及感受。 第一家公司很小&#xff0c;在这就不说它的名字的&#xff0c;公司是一个三室两厅的感觉&#xff0c;客厅里摆了两排电脑&#xff0c;里面有一个员工&#xff0c;老板见我…

mysql数据库表复用_MySQL 数据库之表操作

一、创建表的完整语法create table 表(字段名1 类型 [(宽度) 约束条件]&#xff0c;字段名2 类型 [(宽度) 约束条件]&#xff0c;字段名3 类型 [(宽度) 约束条件])&#xff1b;1.类型&#xff1a;使用限制字段必须以什么样的数据类型传值约束条件&#xff1a;约束条件是在类型之…

ubuntu命令收集

软件操作&#xff1a; - sudo apt-get install xxx  安装软件 - sudo apt-get --purge remove XXX  卸载软件 - sudo apt-get -f install  修复安装 - sudo dpkg -i [fileName.deb] 安装deb结尾的文件(不会解决依赖) echo XXX  输出到终端 nohup XXX  不挂断地运行命…

你不必害怕,岁月有的是时间让你遇见更好的人(沈善书)

这是我单身生活的第四年。一个人的生活里&#xff0c;总会羡慕别人的爱情。时间久了&#xff0c;我就慢慢告诉自己&#xff0c;其实也不必羡慕别人的爱情&#xff0c;我也可以轰轰烈烈&#xff0c;只是上辈子欠了岁月一个人情&#xff0c;岁月要让我多等待&#xff0c;磨练我的…

postgis 导出 栅格_postgis常见的空间数据的导入导出

本片文章简单记录了postgis涉及的常用数据导入导出的使用方法。1.将osm数据导入postgisosm2pgsql -c -d osm -l -E 4326 -S /usr/share/osm2pgsql/default.style xxxxx.osm这里要注意几个参数&#xff0c;是选用-c还是-a要看你的应用方式,-S这个是在ubuntu下路径不同时使用&…

修复./mysql/proc

mysql数据库只能建不能删的错误提示及处理方法&#xff1a;mysql> drop database testmysqldatabase;ERROR 145 (HY000): Table ./mysql/proc is marked as crashed and should be repaired处理方法&#xff1a;直接在mysql数据库里面使用root帐号登录后&#xff0c;进行修复…

Android有道词典查询功能

有道词典 任务要求&#xff1a;完成查词等功能 因为需要申请API key&#xff0c;这里直接给出地址供使用&#xff1a;http://fanyi.youdao.com/openapi?pathdata-mode 1、activity_main.xml基本格局&#xff08;不做任何说明&#xff09; 代码如下&#xff1a; 1 <Relative…

C++的迭代器Interator

*************************************************** 更多精彩&#xff0c;欢迎进入&#xff1a;http://shop115376623.taobao.com STL视频教程&#xff1a; http://item.taobao.com/item.htm?spma1z10.5-c.w4002-9510581626.21.y9vLuz&id43055362725 ***************…

从零开始学Axure原型设计(高级篇)

如果你熟悉了Axure的部件库&#xff0c;那么你可以得心应手地画出心目中产品的线框图&#xff1b;如果你会用Axure的母版、动态面板功能&#xff0c;那么你应该能够画出一些简单网站的原型图&#xff1b;但只有你精通了Axure的条件逻辑、变量、函数等高级交互&#xff0c;才能将…

python vector 初始化_从零开始搭建机器学习算法框架(python)--计算框架

介绍今天开始一个新的系列&#xff0c;这个系列的目标是用python在不使用任何第三方库的情况下去实现各类机器学习或者深度学习的算法。之所以会有这种想法是因为每当我想提高编程技巧的时候&#xff0c;我总希望能够做一些简单又有趣的小项目练手。我一直对机器学习算法颇感兴…

windows phone 页面主题设计

达到如图这个效果&#xff1a; 1.保证状态栏背景色与主题栏颜色相同需设置状态栏的透明度&#xff0c;代码如下&#xff1a;shell:SystemTray.IsVisible"True" shell:SystemTray.Opacity"0.01" 2.顶部状态栏高度为25&#xff0c;字的上下要留部分空白3.内容…

C++ Vector 使用心得

*************************************************** 更多精彩&#xff0c;欢迎进入&#xff1a;http://shop115376623.taobao.com STL视频教程&#xff1a; http://item.taobao.com/item.htm?spma1z10.5-c.w4002-9510581626.21.y9vLuz&id43055362725 ***************…

JAVA如何才能导出这样的EXCEL?

2019独角兽企业重金招聘Python工程师标准>>> 最近的项目有个需求&#xff0c;需要做个报表&#xff0c;excel如上所示。没有很好的办法&#xff0c;求指教。 转载于:https://my.oschina.net/secret620/blog/611450

python数据分析、挖掘与可视化 慕课答案_Python数据分析、挖掘与可视化(慕课版)...

第1章 Python开发环境的搭建与编码规范 1n1&#xff0e;1 Python开发环境的搭建与使用 1n1&#xff0e;1&#xff0e;1 IDLE 2n1&#xff0e;1&#xff0e;2 Anaconda3 3n1&#xff0e;1&#xff0e;3 安装扩展库 4n1&#xff0e;2 Python编码规范 5n1&#xff0e;3…

Xcode 修改工程名以及注意事项

1、先把整个工程文件夹名改为新的工程名。 2、打开工程&#xff0c;单击&#xff0c;输入新的工程名,会出现&#xff0c;点击确定。 3、回到工程界面&#xff0c;在中选择 Manage Schemes,然后再弹出的对话框&#xff0c;把工程名改为新的名字。 4、最好在工程中&#xff0c;把…

C++语言中multiset的相关用法及扩展

*************************************************** 更多精彩&#xff0c;欢迎进入&#xff1a;http://shop115376623.taobao.com *************************************************** cpp语言中&#xff0c;multiset是<set>库中一个非常有用的类型&#xff0c;它可…

[20160201]db_link与子光标问题.txt

[20160201]db_link与子光标问题.txt --生产系统遇到一个关于db_link产生大量子光标问题&#xff0c;当cursor_sharingforce的情况下&#xff0c;通过测试说明。 --注&#xff1a;这个问题我的测试仅仅存在10.2.0.4,11.2.0.4没有这个问题。 1.环境&#xff1a; SCOTTtest> &a…

脚本命令配置mysql_MySQL 自动化部署脚本

一、环境说明操作系统&#xff1a;CentOS数据库版本&#xff1a;MySQL 5.7/8.0参数&#xff1a;buffer pool 会根据系统内存指定、默认双一、GTID、SlowLog脚本默认安装路径&#xff1a;/usr/local/mysql脚本默认数据路径&#xff1a;/data/mysql*(根据安装包版本适应 比如 5.7…