數據表中的數據實體類

在寫基於數據庫的程序中發現,通常關係數據庫中一個表可以抽象爲一個類,數據表中的每條數據都可以看做是該類的一個實例。
例如用戶表Users有三個字段:id(varchar2),name(Varchar2),age(number)。用戶理所當然要抽象成一個類(當然,這裏談的是面向對象設計),而該類的數據成員應該是Users表中的三個字段。至於用戶類的方法,總少不了增刪改查這些操作,例如添加新用戶,修改用戶信息,查詢用戶等等。
問題來了,如果我們有N個表,要對這N個表進行操作,寫N個類,他們的數據成員(其實就是數據表字段)不一樣,卻有着一些相同的方法(增刪改查),是不是要給每一個類一一地寫這些實現呢?如果N>10,你是否感覺很累呢?以後如果還有類似的項目,你又要重新再寫N個類,買噶的,我都要瘋了,不知道你有何感想?
我們是程序員,但不能傻傻的悶聲寫程序,那樣鍵盤敲破了都不會有進步。寫程序之前,想一想,我的代碼怎麼樣才能設計的更加精巧更加容易複用呢?
我在前面的博客中好像寫過利用抽象工廠模式實現不同數據庫的自由切換(?時間太長,不大記得了)。這裏再寫一個抽象的數據表數據實體類作爲基類,爲他的每個子類實現增刪改查這些數據庫操作,子類只要向基類傳入表名和列名就可以了。這樣,以後每個項目你都不用再染指關於數據庫的東西了。

首先是一個數據項類,該類把列名和數據綁定起來,方便操作。
  1. /// <summary>
  2.     /// 數據項類 包括數據庫字段名稱和值
  3.     /// </summary>
  4.     public class DataItem
  5.     {
  6.         private static IDatabaseFactory m_fac = SqlHelper.CreateDatabaseFactory(); //這個數據庫抽象工廠,會在後面列出來
  7.         private readonly string m_name;
  8.         private object m_value;
  9.         /// <summary>
  10.         /// 構造 數據項實例
  11.         /// </summary>
  12.         /// <param name="name">數據庫字段名稱</param>
  13.         public DataItem(string name)
  14.         {
  15.             this.m_name = name;
  16.         }
  17.         /// <summary>
  18.         /// 獲取 數據庫字段名稱
  19.         /// </summary>
  20.         public string Name
  21.         {
  22.             get { return m_name; }
  23.         }
  24.         /// <summary>
  25.         /// 獲取或設置 字段對應的值
  26.         /// </summary>
  27.         public object Value
  28.         {
  29.             get { return m_value; }
  30.             set { m_value = value; }
  31.         }
  32.         /// <summary>
  33.         /// 獲取 值在SQL語句中的字符串表示
  34.         /// </summary>
  35.         public string ValueSqlString
  36.         {
  37.             get
  38.             {
  39.                 if (m_value == null)
  40.                 {
  41.                     return "null";
  42.                 }
  43.                 else if (m_value.GetType() == typeof(string))
  44.                 {
  45.                     return "'" + m_value.ToString().Replace("'","''") + "'";
  46.                 }
  47.                 else if (m_value.GetType() == typeof(int))
  48.                 {
  49.                     return m_value.ToString();
  50.                 }
  51.                 else if (m_value.GetType() == typeof(DateTime))
  52.                 {
  53.                     return m_fac.ToDBTimeString(Convert.ToDateTime(m_value));
  54.                 }
  55.                 else
  56.                 {
  57.                     return "null";
  58.                 }
  59.             }
  60.         }
  61.         public override string ToString()
  62.         {
  63.             return m_name + "="+ValueSqlString;
  64.         }
  65.     }
然後是本文的核心,數據表實體對象類,比較長,呵呵
  1. /// <summary>
  2.     /// 與數據庫表緊密相關 的 實體類
  3.     /// </summary>
  4.     public class DataTableObject
  5.     {
  6.         protected readonly string m_tableName;
  7.         protected DataItem m_key;
  8.         protected IList<DataItem> m_col;
  9.         /// <summary>
  10.         /// 構造 一個數據表實體類
  11.         /// </summary>
  12.         /// <param name="tableName">表名</param>
  13.         /// <param name="keyName">主鍵名</param>
  14.         /// <param name="colName">列名</param>
  15.         protected DataTableObject(string tableName,string keyName,params string[] colName)
  16.         {
  17.             m_col = new List<DataItem>();
  18.             this.m_tableName = tableName;
  19.             this.m_key = new DataItem(keyName);
  20.             if (colName==nullreturn;
  21.             foreach (string col in colName)
  22.             {
  23.                 m_col.Add(new DataItem(col));
  24.             }
  25.         }
  26.         /// <summary>
  27.         /// 構造 一個數據表實體類
  28.         /// </summary>
  29.         /// <param name="key">主鍵值</param>
  30.         /// <param name="tableName">表名</param>
  31.         /// <param name="keyName">主鍵名</param>
  32.         /// <param name="colName">列名</param>
  33.         protected DataTableObject(object key,string tableName, string keyName,params string[] colName)
  34.             :this(tableName,keyName,colName)
  35.         {
  36.             m_key.Value = key;
  37.         }
  38.         /// <summary>
  39.         /// 獲取或設計 主鍵值
  40.         /// </summary>
  41.         public object Key
  42.         {
  43.             get
  44.             {
  45.                 return m_key.Value;
  46.             }
  47.             set
  48.             {
  49.                 this.m_key.Value = value;
  50.                 this.Load();
  51.             }
  52.         }
  53.         /// <summary>
  54.         /// 獲取 指定列值
  55.         /// </summary>
  56.         /// <param name="index">列索引 依照構造函數傳入的列順序 從0開始</param>
  57.         /// <returns></returns>
  58.         public object GetDataValue(int index)
  59.         {
  60.             if (index > m_col.Count - 1) return null;
  61.             if (m_col[index].Value == nullthis.Load();
  62.             return m_col[index].Value;
  63.         }
  64.         /// <summary>
  65.         /// 獲取 指定列值
  66.         /// </summary>
  67.         /// <param name="columnName">列名稱</param>
  68.         /// <returns></returns>
  69.         public object GetDataValue(string columnName)
  70.         {
  71.             foreach (DataItem di in m_col)
  72.             {
  73.                 if (di.Name.Equals(columnName, StringComparison.OrdinalIgnoreCase))
  74.                 {
  75.                     if (di.Value == nullthis.Load();
  76.                     return di.Value;
  77.                 }
  78.             }
  79.             return null;
  80.         }
  81.         /// <summary>
  82.         /// 設置 指定列值
  83.         /// </summary>
  84.         /// <param name="index">列索引 依照構造函數傳入的列順序 從0開始</param>
  85.         /// <param name="value">新的列值</param>
  86.         /// <returns>設置是否成功</returns>
  87.         public bool SetDataValue(int index, object value)
  88.         {
  89.             if (index > m_col.Count - 1) return false;
  90.             m_col[index].Value = value;
  91.             return true;
  92.         }
  93.         /// <summary>
  94.         /// 設置 指定列值
  95.         /// </summary>
  96.         /// <param name="columnName">列名</param>
  97.         /// <param name="value">新的列值</param>
  98.         /// <returns>設置是否成功</returns>
  99.         public bool SetDataValue(string columnName, object value)
  100.         {
  101.             foreach (DataItem di in m_col)
  102.             {
  103.                 if (di.Name.Equals(columnName, StringComparison.OrdinalIgnoreCase))
  104.                 {
  105.                     di.Value = value;
  106.                     return true;
  107.                 }
  108.             }
  109.             return false ;
  110.         }
  111.         /// <summary>
  112.         /// 根據主鍵值 重新從數據庫中加載數據
  113.         /// </summary>
  114.         /// <returns>加載是否成功</returns>
  115.         public virtual bool Load()
  116.         {
  117.             if (m_key.Value == null)
  118.             {
  119.                 Logger.LogText("不能讀取數據,主鍵值爲空.主鍵:"+m_key.Name);
  120.                 return false;
  121.             }
  122.             StringBuilder sbSql = new StringBuilder("select ");
  123.             foreach (DataItem di in m_col)
  124.             {
  125.                 sbSql.Append(di.Name);
  126.                 sbSql.Append(",");
  127.             }
  128.             int len = sbSql.Length;
  129.             sbSql.Remove(len - 1, 1);
  130.             sbSql.Append(" from ");
  131.             sbSql.Append(m_tableName);
  132.             sbSql.Append(" where ");
  133.             sbSql.Append(m_key.Name);
  134.             sbSql.Append("=");
  135.             sbSql.Append(m_key.ValueSqlString);
  136.             DataSet ds = SqlHelper.ExecuteQuery(sbSql.ToString());
  137.             if (ds == null)
  138.             {
  139.                 Logger.LogText("數據庫插入操作失敗.");
  140.                 return false;
  141.             }
  142.             if (ds.Tables[0].Rows.Count < 1)
  143.             {
  144.                 ds.Dispose();
  145.                 return false;
  146.             }
  147.             int index = 0;
  148.             DataRow dr = ds.Tables[0].Rows[0];
  149.             foreach (DataItem di in m_col)
  150.             {
  151.                 di.Value = dr[index++];
  152.             }
  153.             return true;
  154.         }
  155.         /// <summary>
  156.         /// 將本條記錄插入數據庫
  157.         /// </summary>
  158.         /// <returns>插入是否成功</returns>
  159.         public virtual bool Insert()
  160.         {
  161.             StringBuilder sbSql = new StringBuilder("insert into ");
  162.             sbSql.Append(m_tableName);
  163.             sbSql.Append(" (");
  164.             foreach (DataItem di in m_col) 
  165.             {
  166.                 sbSql.Append(di.Name);
  167.                 sbSql.Append(",");
  168.             }
  169.             sbSql.Remove(sbSql.Length-1,1);
  170.             sbSql.Append(") values (");
  171.             foreach (DataItem di in m_col)
  172.             {
  173.                 sbSql.Append(di.ValueSqlString);
  174.                 sbSql.Append(",");
  175.             }
  176.             sbSql.Remove(sbSql.Length - 1, 1);
  177.             sbSql.Append(")");
  178.             int result = SqlHelper.ExecuteNonQuery(sbSql.ToString());
  179.             if(result>0)
  180.                 return true;
  181.             return false;
  182.         }
  183.         /// <summary>
  184.         /// 將本條記錄的更新 寫入數據庫
  185.         /// </summary>
  186.         /// <returns>更新是否成功</returns>
  187.         public virtual bool Update()
  188.         {
  189.             StringBuilder sbSql = new StringBuilder("update ");
  190.             sbSql.Append(m_tableName);
  191.             sbSql.Append(" set ");
  192.             foreach (DataItem di in m_col)
  193.             {
  194.                 sbSql.Append(di.Name);
  195.                 sbSql.Append("=");
  196.                 sbSql.Append(di.ValueSqlString);
  197.                 sbSql.Append(",");
  198.             }
  199.             sbSql.Remove(sbSql.Length - 1, 1);
  200.             sbSql.Append(" where ");
  201.             sbSql.Append(m_key.Name);
  202.             sbSql.Append("=");
  203.             sbSql.Append(m_key.ValueSqlString);
  204.             int result = SqlHelper.ExecuteNonQuery(sbSql.ToString());
  205.             if (result > 0)
  206.                 return true;
  207.             return false;
  208.         }
  209.         /// <summary>
  210.         /// 將本條記錄從數據庫中刪除
  211.         /// </summary>
  212.         /// <returns>刪除是否成功</returns>
  213.         public virtual bool Delete()
  214.         {
  215.             StringBuilder sbSql = new StringBuilder("delete from ");
  216.             sbSql.Append(m_tableName);
  217.             sbSql.Append(" where ");
  218.             sbSql.Append(m_key.Name);
  219.             sbSql.Append("=");
  220.             sbSql.Append(m_key.ValueSqlString);
  221.             int result = SqlHelper.ExecuteNonQuery(sbSql.ToString());
  222.             if (result > 0)
  223.                 return true;
  224.             return false;
  225.         }
  226.         /// <summary>
  227.         /// 獲取 數據表中所有記錄的對象列表
  228.         /// </summary>
  229.         /// <param name="tabName">表名</param>
  230.         /// <param name="keyName">主鍵名</param>
  231.         /// <param name="colName">列名</param>
  232.         /// <returns>數據表中所有記錄的對象列表</returns>
  233.         protected static IList<DataTableObject> GetAll(string tabName,string keyName,params string[] colName)
  234.         { 
  235.             IList<DataTableObject> list = new List<DataTableObject>();
  236.             StringBuilder sbSql = new StringBuilder("select ");
  237.             sbSql.Append(keyName);
  238.             foreach(string name in colName)
  239.             {
  240.                 sbSql.Append(",");
  241.                 sbSql.Append(name);
  242.             }
  243.             sbSql.Append(" from ");
  244.             sbSql.Append(tabName);
  245.             DataSet ds = SqlHelper.ExecuteQuery(sbSql.ToString());
  246.             if (ds == null)
  247.             {
  248.                 Logger.LogText("數據庫查詢操作失敗.");
  249.                 return null;
  250.             }
  251.             int len = colName.Length;
  252.             foreach (DataRow dr in ds.Tables[0].Rows)
  253.             {
  254.                 DataTableObject obj = new DataTableObject(tabName, keyName, colName);
  255.                 obj.Key = dr[0];
  256.                 for (int i = 0; i <len; i++)
  257.                 {
  258.                     obj.SetDataValue(i, dr[i+1]);
  259.                 }
  260.                 list.Add(obj);
  261.             }
  262.             return list;
  263.         }
  264.     }
數據表實體對象類的應用列子,還以上面的用戶表做例子吧.
  1. public class User:DataTableObject
  2.     {
  3.         private const string tabName = "Users";
  4.         private const string keyName = "id";
  5.         private static readonly string[] colName = { "name","age"};
  6.         public User():base(tabName,keyName,colName) { }
  7.         public User(int id) : base(id, tabName, keyName, colName) { }
  8.         public int ID
  9.         {
  10.             get
  11.             {
  12.                 if (base.Key == nullreturn -1;//出錯
  13.                 return Convert.ToInt32(base.Key);
  14.             }
  15.             set
  16.             {
  17.                 base.Key = value;
  18.             }
  19.         }
  20.         public string Name
  21.         {
  22.             get
  23.             {
  24.                 if (base.GetDataValue(0) == nullbase.Load(); //如果爲空則重新載入
  25.                 return base.GetDataValue(0).ToString();
  26.             }
  27.             set
  28.             {
  29.                 base.GetDataValue(0) = value;
  30.             }
  31.         }
  32.         public int Age
  33.         {
  34.             get
  35.             {
  36.                 if (base.GetDataValue("age") == nullbase.Load(); //如果爲空則重新載入
  37.                 return Convert.ToInt32(base.GetDataValue("age"));
  38.             }
  39.             set
  40.             {
  41.                 base.GetDataValue("age") = value;
  42.             }
  43.         }
  44.         //增刪改查操作均繼承自父類,此處不需要重寫,如有需要也可以重寫
  45.     }





如果對前面的數據庫抽象工廠類感到疑惑,可以看看下面的兩個工廠類:
抽象工廠:
  1.   /// <summary>
  2.     /// 數據庫工廠接口
  3.     /// </summary>
  4.     public interface IDatabaseFactory
  5.     {
  6.         /// <summary>
  7.         /// 獲取 數據庫連接實例
  8.         /// </summary>
  9.         /// <returns>連接實例</returns>
  10.         IDbConnection GetConnection();
  11.         /// <summary>
  12.         /// 獲取 數據庫命令實例
  13.         /// </summary>
  14.         /// <returns>命令實例</returns>
  15.         IDbCommand GetCommand();
  16.         /// <summary>
  17.         /// 獲取 數據適配器實例
  18.         /// </summary>
  19.         /// <returns>數據適配器實例</returns>
  20.         IDbDataAdapter GetDataAdapter();
  21.         /// <summary>
  22.         /// 將指定時間轉換成SQL語句
  23.         /// </summary>
  24.         /// <param name="time">時間</param>
  25.         /// <returns>SQL語句</returns>
  26.         string ToDBTimeString(DateTime time);
  27.     }
具體的Oracle數據庫工廠示例:
  1.  /// <summary>
  2.     /// Oracle數據庫相關對象 工廠(同一對象關於多線程不安全)
  3.     /// </summary>
  4.     public class OracleFactory:IDatabaseFactory
  5.     {
  6.         //數據庫連接字符串
  7.         private readonly string m_strConn = Config.Instance.ConnenctionString;
  8.         private IDbConnection m_conn;
  9.         private IDbCommand m_cmd;
  10.         private IDbDataAdapter m_adapter;
  11.         /// <summary>
  12.         /// 獲取 數據庫連接實例
  13.         /// </summary>
  14.         /// <returns>數據庫連接實例</returns>
  15.         public IDbConnection GetConnection()
  16.         {
  17.             if (m_conn == null)
  18.             { 
  19.                 m_conn = new OracleConnection();
  20.                 m_conn.ConnectionString = m_strConn;
  21.             }
  22.             return m_conn;
  23.         }
  24.         /// <summary>
  25.         /// 獲取 數據庫命令實例
  26.         /// </summary>
  27.         /// <returns>數據庫命令實例</returns>
  28.         public IDbCommand GetCommand()
  29.         {
  30.             if (m_cmd == null)
  31.             {
  32.                 m_cmd = new OracleCommand();
  33.             }
  34.             return m_cmd;
  35.         }
  36.         /// <summary>
  37.         /// 獲取 數據庫數據適配器實例
  38.         /// </summary>
  39.         /// <returns>數據庫數據適配器實例</returns>
  40.         public IDbDataAdapter GetDataAdapter()
  41.         {
  42.             if (m_adapter == null)
  43.                 m_adapter = new OracleDataAdapter();
  44.             return m_adapter;
  45.         }
  46.         public string ToDBTimeString(DateTime time)
  47.         {
  48.             return String.Format("to_date('{0}','yyyy-mm-dd hh24:mi:ss')",time);
  49.         }
  50.     }


看了這麼多代碼,一定很暈吧,哈,可能是我的代碼寫的不夠完美,如果有什麼改進意見,請不吝賜教~

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章