泛型的基本概念我就不在這重復了,不了解的同學請自行百度。
我主要寫下我在項目中要到的泛型實例。獻丑了.....有什么不好或不對的地方大家盡可評論留言。
為什么要用泛型?
通過使用泛型,我們可以極大地提高代碼的重用度,同時還可以獲得強類型的支持,避免了隱式的裝箱、拆箱,在一定程度上提升了應用程序的性能。
作為一個想法的程序猿,不能僅僅只會"復制 粘貼"。
代碼的封裝、重用是可以大大提高你的效率,對於之后代碼的修改及二次開發很有幫助。
當然使你"看起來"很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);
獻丑了。這是我第一次在項目中使用泛型,大神莫笑我啊。
那里不對的地方,請打加指出。