建了自己的博客后,还没有真正认认真真的写过文章,总是有种感觉思路整理的还是不够清楚,但路总是要走出来的,也希望各位能给与我更多的支持。
最近刚刚完成了一个项目,“基于B/S的资源平台的建设”。
首先介绍一下项目的背景,我们公司是一家专门多媒体教学行业为主的企业,而我所做的资源平台的任务就是将采集下载的视频文件,通过前台网站,供不同用户的 用户实现对资源的上传,下载,点播,直播等功能(通过javscript调用Activex组件实现);后台是分为基础管理、资源管理、用户管理、统计分 析四个主模块。基础管理分为:公告管理、系统参数、登陆日志、操作日志、友情连接;资源管理分为:资源分类、上传资源、资源添加编辑;用户管理分为:用户 添加、编辑、用户权限等;统计分析等分为:资源统计、用户统计等。
以上就是项目的简单背景,下面我将就标题的顺序,介绍这个项目我所用到的架构和技术。
1.模块的编排
当明确了这个项目的需求后,我首先想到的是如何安排工程的模块分布,才能是整个工程整洁有序,为后面的二次开发,提供更好的接口。
整个工程涉及到数据库操作类、Page基类、UserControl基类,实体类、公共类、各个模块的页面、用户控件、以及用到的js、CSS、images等;只有找到合理的模块编排的方式,才能对项目的后期维护和二次开发更为有效,清晰。
首先,将数据库操作类、Page基类、UserControl基类,实体类、公共类放在APP_CODE中,然后建立两个主文件夹,一个放置前台网站模 块,另一个放置后提台模块,然后根据设计好的模块分置各个的存储位置,如资源管理主模块,首先在后台模块文件夹下面建立名为“ResManage”文件 夹,然后在该文件夹下面建立“Page”、“UserControl”、“images”三个子文件夹,在Page中存放该模块的.aspx页 面,UsrControl中存放该模块的.ascx(命名一般根据Page中的.aspx页面),images中存放该模块所用到的图片。
至于一些公共的图片文件夹则直接放置在根目录下。这样整个工程的模块分布就很清晰了。
2.数据库的操作类
数据库的操作类是一个项目的基石,它掌管这我们与数据的直接的数据交互,通过对select、insert、update、delete不同方式的调用,插入,更新等操作采用Hash 算法,实现我们的所需要的功能。
在设计中我们基于SQLHelper框架,可操作Sql Server,Oracle,Access数据库,设计了dbOperate.cs、sqlOperate.cs、OleDbOperate.cs、OraOperate.cs 首先dbOperate.cs是操作的基类,设置数据库连接字符串,数据库类型,对数据的操作等如下:
namespace dbclass { //数据库的枚举类型 public enum DbType { sqlServer=0,//0 oracle=1, //1 oledb=2 } public class DbOperate { public DbOperate() { // // TODO: 在此处添加构造函数逻辑 // } private static DbType _DataBaseType; private static string _ConnectString = System.Configuration.ConfigurationSettings.AppSettings["con"].ToString(); //数据库类型 public static DbType DataBaseType { get { return _DataBaseType; } set { _DataBaseType = value; } } //连接字符串 public static string ConnectString { get { return _ConnectString; } set { _ConnectString = value; } }
通过在外部设定 DataBaseType,ConnectString属性值,实现对不同数据库的操作和连接字符串的属性。
然后利用重载是实现DataReader,ExecuteNonQuery,DataSet,ExecuteScale等功能,如 ExecuteNonQuery的实现:
public static int ExecuteNonQuery(Hashtable htParams, params string[] commandTexts) { int retval = -1; retval=ExecuteNonQuery(ConnectString, CommandType.Text, GetParameterSet(htParams), commandTexts); htParams.Clear(); htParams = null; return retval; } public static int ExecuteNonQuery(params string[] commandTexts) { return ExecuteNonQuery(ConnectString, CommandType.Text, (IDbDataParameter[])null,commandTexts); } public static int ExecuteNonQueryTrans(ArrayList commandTexts) { using (IDbConnection con = GetConnection()) {
con.Open(); IDbTransaction trans=con.BeginTransaction(); string strsql=""; IDbCommand cmd = GetCommand(); for(int i=0;i<commandTexts.Count;i++) { try {
strsql = commandTexts[i].ToString(); PrepareCommand(cmd, con, trans, CommandType.Text, strsql, (IDbDataParameter[])null); cmd.ExecuteNonQuery(); } catch (Exception e) { trans.Rollback(); ShowMessage("数据库执行出错,出错语句:" + strsql + ",出错描述:" + e.Message); HttpContext.Current.Response.Redirect("~/ErrorPage.aspx?errMessage=" + "数据库操作出错,出错语句:" + strsql + ",出错描述:" + HttpUtility.UrlEncode(e.Message)); break;
}
} trans.Commit(); } return 0; } public static int ExecuteNonQuery(CommandType commandType, params string[] commandTexts) { return ExecuteNonQuery(ConnectString,commandType, (IDbDataParameter[])null,commandTexts); } public static int ExecuteNonQuery(string connectionString, CommandType commandType, params string[] commandTexts) { return ExecuteNonQuery(connectionString, commandType, (IDbDataParameter [])null,commandTexts); } public static int ExecuteNonQuery(string connectionString, CommandType commandType,IDbDataParameter[] commandParameters,params string[] commandTexts) { using (IDbConnection cn = GetConnection(connectionString)) { cn.Open(); return ExecuteNonQuery(cn, commandType, commandParameters,commandTexts); } } public static int ExecuteNonQuery(IDbConnection connection, CommandType commandType, params string[] commandTexts) { return ExecuteNonQuery(connection, commandType, (IDbDataParameter[])null,commandTexts); } public static int ExecuteNonQuery(IDbConnection connection, CommandType commandType, IDbDataParameter[] commandParameters, params string[] commandTexts) { int retval = -1;
try { IDbCommand cmd = GetCommand(); PrepareCommand(cmd, connection, (IDbTransaction)null, commandType, GetCommandText(commandTexts), commandParameters); retval = cmd.ExecuteNonQuery(); cmd.Parameters.Clear(); connection.Close(); } catch (Exception e) { connection.Close(); ShowMessage("数据库操作出错,出错语句:" +GetCommandText(commandTexts) + ",出错描述:" + e.Message); HttpContext.Current.Trace.Warn("数据库操作出错,出错语句:" + GetCommandText(commandTexts)); HttpContext.Current.Trace.Warn("错误描述信息:" + e.Message); HttpContext.Current.Response.Redirect("~/ErrorPage.aspx?errMessage=" + "数据库操作出错,出错语句:" + GetCommandText(commandTexts) + ",出错描述:" + HttpUtility.UrlEncode(e.Message)); }
return retval; } public static int ExecuteNonQuery(CommandType commandType, string spName, Hashtable paramValue) { return ExecuteNonQuery((IDbTransaction)null, commandType, spName, paramValue); } public static int ExecuteNonQuery(IDbTransaction transaction, CommandType commandType, string spName, Hashtable paramsValue) { int retval = -1; if ((paramsValue != null) && (paramsValue.Count > 0)) { IDbDataParameter[] commandParameters =GetSpParameterSet(transaction.Connection.ConnectionString, spName); AssignParameterValues(commandParameters, paramsValue); retval = ExecuteNonQuery(transaction, commandType, commandParameters, new string[] { spName}); } else { retval = ExecuteNonQuery(transaction, commandType,(IDbDataParameter[])null, new string[] {spName}); } paramsValue.Clear(); paramsValue =null; return retval; } public static int ExecuteNonQuery(IDbTransaction transaction, CommandType commandType, params string[] commandTexts) { return ExecuteNonQuery(transaction, commandType, (IDbDataParameter[])null,commandTexts); } public static int ExecuteNonQuery(IDbTransaction transaction, CommandType commandType, IDbDataParameter[] commandParameters, params string[] commandTexts) { int retval = -1; try { IDbCommand cmd = GetCommand(); PrepareCommand(cmd, transaction.Connection, transaction, commandType, GetCommandText(commandTexts), commandParameters); retval = cmd.ExecuteNonQuery(); cmd.Parameters.Clear(); } catch (Exception e) { transaction.Rollback(); transaction.Connection.Close(); ShowMessage("数据库操作出错,出错语句:" + GetCommandText(commandTexts) + ",出错描述:" + e.Message); HttpContext.Current.Trace.Warn("数据库操作出错,出错语句:" + GetCommandText(commandTexts)); HttpContext.Current.Trace.Warn("错误描述信息:" + e.Message); HttpContext.Current.Response.Redirect("~/ErrorPage.aspx?errMessage=" + "数据库操作出错,出错语句:" + GetCommandText(commandTexts) + ",出错描述:" + HttpUtility.UrlEncode(e.Message)); } return retval; }
sqlOperate.cs、OleDbOperate.cs、OraOperate.cs提供给接口的特殊功能,如数据的分页,Hash表的实现。下面是sqlOperate.cs的实现。
public static SqlParameter[] GetParameterSet(Hashtable htParams) { List<SqlParameter> paramsList = new List<SqlParameter>(); foreach (object key in htParams.Keys) { object [] paramAttr=(object [])htParams[key]; SqlParameter param = new SqlParameter(); param.ParameterName = key.ToString(); int precision = 0; byte scale = 0; if (paramAttr[1].ToString() != "") { string[] sizeAry = paramAttr[1].ToString().Split(new char[] { '.' }, StringSplitOptions.RemoveEmptyEntries); precision = int.Parse(sizeAry[0]); if (sizeAry.Length==2) { scale = byte.Parse(sizeAry[1]); } } switch (paramAttr[0].ToString().Trim()) { case "c": { param.SqlDbType = SqlDbType.VarChar; if (precision != 0) { param.Size = precision; } break; } case "nc": { param.SqlDbType = SqlDbType.NVarChar; if (precision != 0) { param.Size = precision; } break; } case "i": { param.SqlDbType = SqlDbType.Int; break; } case "bi": { param.SqlDbType = SqlDbType.BigInt; break; } case "dt": { param.SqlDbType = SqlDbType.DateTime; break; } case "de": { param.SqlDbType = SqlDbType.Decimal; param.Precision = byte.Parse(precision.ToString()); param.Scale =scale; break; } case "txt": { param.SqlDbType = SqlDbType.Text; break; } } param.Value = paramAttr[2]; paramsList.Add(param); } return paramsList.ToArray();
} //分页读取数据 public static DataTable getPagerData(string strSql, string orderField, string uniqueValueField,int top, int pageSize, int pageIndex,int doCount,out int recordCount) { //pageIndex从1开始 //top取记录的数目 recordCount = 0; string strcmd = ""; string strAfterFrom = strSql.Substring(strSql.IndexOf("from"));//查询字符串中from之后内容 if (doCount != 0) { string strAfterFromBeforeOrder = strAfterFrom; if (strAfterFromBeforeOrder.IndexOf("order") > -1) { strAfterFromBeforeOrder = strAfterFromBeforeOrder.Substring(0, strAfterFromBeforeOrder.IndexOf("order")); } recordCount= (int)dbclass.DbOperate.ExecuteScalar("select count(*) " + strAfterFromBeforeOrder); } if (strSql.IndexOf("order") == -1) { if (orderField.Trim() != "") { strSql += " order by " + orderField; strAfterFrom += " order by " + orderField; } } int startRow = (pageIndex - 1) * pageSize + 1; int endRow = 0; if (top != 0) { endRow = top; } else { endRow = pageIndex * pageSize; }
strcmd = "declare @indexTable table(selfid bigint identity(1,1),uniqueValue nvarchar(50));insert into @indexTable(uniqueValue) select top 100 percent " + uniqueValueField + " " + strAfterFrom + ";"; string strAfterSelect = "";//select之后的字符串 strAfterSelect = strSql.Substring(strSql.IndexOf("select") + 6); strcmd += "select a.* from (select top 100 percent " + strAfterSelect + ") a,@indexTable t where t.uniqueValue=" + uniqueValueField + " and t.selfid>=" + startRow.ToString() + " and t.selfid<=" + endRow.ToString(); DataSet ds; ds=dbclass.DbOperate.ExecuteDataset(strcmd); return ds.Tables[0];
} }
通过先前做的几个项目相比较,现在这个数据库操作基类在这次开发中取得了很好的效果。
今天先发两个部分,后面我将陆续的写出Ajax.net、Athem.NET、javascript结合Activex组件实现多文件,文件夹上传、下载以及视频点播的功能。
posted @ 2008-10-12 18:39 张清扬 阅读(540) 评论(0)
忙碌的一天,不应该是结束,生活还是一片篮....
C++,MFC, 图像处理,Windows API, socket,H.264, C#,.net,数据库,架构,系统设计,需求分析==,充斥着我的大脑,当然,还有生活,家庭,我的未来,我的理想,我的爱好,我的欢乐,我的苦恼,我 的希望,突然,也不是在不经意间,感觉到许多的事情在上空盘旋,认识到每一件事情,技术方面,生活方面,理想未来等等的一切,都需要你认真的思考,虽然会 迷茫,但会理顺清楚的.
既然迷惑,又想清楚就从今天开始,生活总是随着你我的思考,一步步扎扎实实的向前攀登着,分享我的快乐,我的思考,得到我们生活的意义.
2008.8.27
今天终于将郁闷了几天的问题解决了(有取巧的嫌疑),对存储的VNC的视频在本地回放的时候出现的问题,用C++编写的DLL,用 Delphi编写的上层应用软件调用时,将视频在Panel控件中显示的时候总是会将Panel的位置移动,很是奇怪,在创建句柄,调用调色版的过程中, 分辨率的设定,等等方面都没有问题,因为整个项目就剩下我这个环节,最后这能被迫调整Panel位置了.
明天将问题详细描述,希望大家能帮忙解决!!