从零开始编写自己的C#框架(19)——Web层后端权限模块
2014-06-23 23:04
495 查看
不知不觉本系统写了快三个月了,最近写页面的具体功能时感觉到有点吃力,很多地方如果张嘴来讲的话可以说得很细,很全面,可写成文字的话,就不太会写了,有些地方想讲得清晰的话,得用多几倍的文字+实例+变化中的图片才能表达得清楚,而写这些又太费时间了,近段时间又特忙,所以只能是尽力而为,希望大家自行研究,如果有什么地方不明白的,发发评论或邮件给我,我再重新详细讲解。
说回正题,对于页面访问权限以及每个按键的权限控制,很久以前用过好几种不同的方法,比如为每个控件分配名称或编码,然后在写代码时绑定这些值,又比如用XML来配置制权限等方法。这些方法都比较麻烦,而且由于都是使用编码方法,开发时需要一个个进行绑定,容易出错。经过后来不断的完善,最后完成了本系统所采用的页面控件注册管理来绑定控件权限(如有雷同,纯属巧合,哈哈...),首先创建菜单,并绑定好对应的文件(页面),然后将系统要用到的名称添加到公用标识库中,跟着在页面控件权限管理页面对各个页面控件进行绑定(只需要点击鼠标即可),通过职位(角色)来赋于不同的操作权限,只需要设置好管理员的职位,那么该管理员就拥有他所绑定的角色的全部权限了。
如下面的说明
首先在后端注册菜单(菜单绑定页面)
View Code
在函数中,大家可以找到下面这些代码
这是用于我们修改A表记录的名称时,同步修改其他关联表引用了这个记录名称字段的所有记录
为什么要这么处理呢?大家在看数据字典的数据库结构时,会发现很多表与其他表关联时,不单将其他表的Id引用了过来,还将这个Id对应的名称也引用了,这种操作方式使我们在编写查询语句时,几乎可以做到不用多表关联,因为我们想要显示的内容已经在查询的表中存在了,这样处理后,我们在修改相关表的名称时,就必须同步修改关联表中的名称,对于这些关联表名称的修改,我们的T4模板也生成了相应的函数,如:UpdateValue_For_表名_Id()函数,然后直接按上面代码编写方法实现就可以了。
3、UseLogList.aspx.cs用户操作日志管理文件
这个文件要注意的是InquiryCondition()函数
这是查询条件函数,ConditionHelper.SqlqueryCondition这个类是自定义封装条件类
它的构造函数(public SqlqueryCondition(ConstraintType ctype, string columnname, Comparison cparsion, object value, bool isParentheses = false))一共有5个参数,其中4个为必填参数
第一个参数ConstraintType ctype为查询的类型,主要是ConstraintType.Where、ConstraintType.And、ConstraintType.Or三种。
如果有多个条件时,ConstraintType.Where只能放在最前面,且只能有一个条件使用这个参数,一般都很少用它。
ConstraintType.And指的是当前参数与前面参数的关系是And关系
ConstraintType.Or通常情况下指的是当前参数与前面参数的关系是Or关系
第二个参数string columnname是条件列字段名称
第三个参数Comparison cparsion是表达式,使用Comparison.X来设置,根据需求设置==、>、>=、<、<=、like、in、not in......
第四个参数object value是条件值,如果是in与not in表达式时,条件值必须为数据类型,比如:string[]、int[]、object[]等
第五个参数bool isParentheses是加左括号,而右括号使用new ConditionHelper.SqlqueryCondition()或new ConditionHelper.SqlqueryCondition(Comparison.CloseParentheses)来添加,然后in或not in表达式时,必须紧跟着右括号
例子1:
(A<2 Or A>= 10) And B=100
例子2:
A == 2 And B in arr And C like 'abc%' (in查询必须加左右括号)
这个封装类最终会生成SubSonic3.0底层调用的条件参数,经过测试发现,SubSonic3.0底层调用的条件参数只支持单括号,不支持括号的嵌套,多重嵌套后最终生成的也只是单括号且会出错,所以多重嵌套括号时,最好使用其它方法来实现,比如Linq、存储过程、SQL语句拼接等。
4、小结
本次更新的代码功能比如多,已将部门、职位、公用页面控件权限标识、页面控件权限管理、在线用户列表、用户登陆日志、用户操作日志、错误日志等功能都已完成了,同时也对在线用户相关类和函数进行了优化处理。
内容看起来好像挺多了,其实都是对之前所写的模板函数以及其他公共函数的调用,这么多页面查看cs代码时可以发现,里面的内容都差不多,都是很格式化的东西,如果你尝试过用这些已完成的模板页面添加新功能,就可以发现很简单,开发也很迅速。
至于每个页面如何去实现,接下来就不必再像之前那样讲得很细致了,会针对一些特殊功能或函数的调用进行解说,往细讲也只能从逻辑层的各个函数应用来说起,所以就不再罗嗦一一细说,代码中有大量的注释,大家慢慢研究吧。
最好附上几张页面的效果图
![](http://images.cnitblog.com/i/129385/201406/232221190964458.jpg)
![](http://images.cnitblog.com/i/129385/201406/232221453467477.jpg)
![](http://images.cnitblog.com/i/129385/201406/232222078926365.jpg)
Web层后端权限模块.rar
版权声明:
本文由AllEmpty原创并发布于博客园,欢迎转载,未经本人同意必须保留此段声明,且在文章页面明显位置给出原文链接,[b]否则保留追究法律责任的权利。如有问题,可以通过1654937@qq.com 联系我,非常感谢。[/b]
发表本编内容,只要主为了和大家共同学习共同进步,有兴趣的朋友可以加加Q群:327360708 ,大家一起探讨。
更多内容,敬请观注博客:http://www.cnblogs.com/EmptyFS/
说回正题,对于页面访问权限以及每个按键的权限控制,很久以前用过好几种不同的方法,比如为每个控件分配名称或编码,然后在写代码时绑定这些值,又比如用XML来配置制权限等方法。这些方法都比较麻烦,而且由于都是使用编码方法,开发时需要一个个进行绑定,容易出错。经过后来不断的完善,最后完成了本系统所采用的页面控件注册管理来绑定控件权限(如有雷同,纯属巧合,哈哈...),首先创建菜单,并绑定好对应的文件(页面),然后将系统要用到的名称添加到公用标识库中,跟着在页面控件权限管理页面对各个页面控件进行绑定(只需要点击鼠标即可),通过职位(角色)来赋于不同的操作权限,只需要设置好管理员的职位,那么该管理员就拥有他所绑定的角色的全部权限了。
如下面的说明
首先在后端注册菜单(菜单绑定页面)
#region 保存 /// <summary> /// 数据保存 /// </summary> /// <returns></returns> public override string Save() { string result = string.Empty; int id = ConvertHelper.Cint0(hidId.Text); try { #region 数据验证 if (string.IsNullOrEmpty(txtCName.Text.Trim())) { return txtCName.Label + "不能为空!"; } var sName = StringHelper.Left(txtCName.Text, 20); if (PagePowerSignPublicBll.GetInstence().Exist(x => x.CName == sName && x.Id != id)) { return txtCName.Label + "已存在!请重新输入!"; } if (string.IsNullOrEmpty(txtEName.Text.Trim())) { return txtEName.Label + "不能为空!"; } var sEname = StringHelper.Left(txtEName.Text, 50); if (PagePowerSignPublicBll.GetInstence().Exist(x => x.EName == sEname && x.Id != id)) { return txtEName.Label + "已存在!请重新输入!"; } #endregion #region 赋值 //定义是否更新标识——即当前记录的名称是否改变了 bool isUpdate = false; //获取实体 var model = new PagePowerSignPublic(x => x.Id == id); //判断是否有改变名称 if (id > 0 && (sName != model.CName || sEname != model.EName)) { isUpdate = true; } //设置名称 model.CName = sName; //设置英文名称 model.EName = sEname; #endregion //---------------------------------------------------------- //存储到数据库 PagePowerSignPublicBll.GetInstence().Save(this, model); //判断是否需要同步更新关联表字段 if (isUpdate) { //调用更新函数,同步更新对应的所有记录 PagePowerSignBll.GetInstence().UpdateValue_For_PagePowerSignPublic_Id(this, model.Id, PagePowerSignTable.CName, model.CName, PagePowerSignTable.EName, model.EName); } } catch (Exception e) { result = "保存失败!"; //出现异常,保存出错日志信息 CommonBll.WriteLog(result, e); } return result; } #endregion
View Code
在函数中,大家可以找到下面这些代码
//定义是否更新标识——即当前记录的名称是否改变了 bool isUpdate = false; //判断是否有改变名称 if (id > 0 && (sName != model.CName || sEname != model.EName)) { isUpdate = true; } //判断是否需要同步更新关联表字段 if (isUpdate) { //调用更新函数,同步更新对应的所有记录 PagePowerSignBll.GetInstence().UpdateValue_For_PagePowerSignPublic_Id(this, model.Id, PagePowerSignTable.CName, model.CName, PagePowerSignTable.EName, model.EName); }
这是用于我们修改A表记录的名称时,同步修改其他关联表引用了这个记录名称字段的所有记录
为什么要这么处理呢?大家在看数据字典的数据库结构时,会发现很多表与其他表关联时,不单将其他表的Id引用了过来,还将这个Id对应的名称也引用了,这种操作方式使我们在编写查询语句时,几乎可以做到不用多表关联,因为我们想要显示的内容已经在查询的表中存在了,这样处理后,我们在修改相关表的名称时,就必须同步修改关联表中的名称,对于这些关联表名称的修改,我们的T4模板也生成了相应的函数,如:UpdateValue_For_表名_Id()函数,然后直接按上面代码编写方法实现就可以了。
3、UseLogList.aspx.cs用户操作日志管理文件
这个文件要注意的是InquiryCondition()函数
/// <summary> /// 查询条件 /// </summary> /// <returns></returns> private List<ConditionHelper.SqlqueryCondition> InquiryCondition() { var wheres = new List<ConditionHelper.SqlqueryCondition>(); //如果Id有值时,即表示查询的是指定管理员的操作日志 if (_id != 0) { wheres.Add(new ConditionHelper.SqlqueryCondition(ConstraintType.Where, LoginLogTable.Manager_Id, Comparison.Equals, _id)); } //起始时间 if (!string.IsNullOrEmpty(dpStart.Text.Trim())) { wheres.Add(new ConditionHelper.SqlqueryCondition(ConstraintType.And, LoginLogTable.AddDate, Comparison.GreaterOrEquals, StringHelper.FilterSql(dpStart.Text))); //终止时间 if (!string.IsNullOrEmpty(dpEnd.Text.Trim())) { wheres.Add(new ConditionHelper.SqlqueryCondition(ConstraintType.And, LoginLogTable.AddDate, Comparison.LessOrEquals, StringHelper.FilterSql(dpEnd.Text))); } } //ip地址 if (!string.IsNullOrEmpty(txtIp.Text.Trim())) { wheres.Add(new ConditionHelper.SqlqueryCondition(ConstraintType.And, LoginLogTable.Ip, Comparison.Equals, StringHelper.FilterSql(txtIp.Text))); } //登录备注信息 if (!string.IsNullOrEmpty(txtloginfo.Text.Trim())) { wheres.Add(new ConditionHelper.SqlqueryCondition(ConstraintType.And, LoginLogTable.Notes, Comparison.Like, "%" + StringHelper.FilterSql(txtloginfo.Text) + "%")); } return wheres; }
这是查询条件函数,ConditionHelper.SqlqueryCondition这个类是自定义封装条件类
它的构造函数(public SqlqueryCondition(ConstraintType ctype, string columnname, Comparison cparsion, object value, bool isParentheses = false))一共有5个参数,其中4个为必填参数
第一个参数ConstraintType ctype为查询的类型,主要是ConstraintType.Where、ConstraintType.And、ConstraintType.Or三种。
如果有多个条件时,ConstraintType.Where只能放在最前面,且只能有一个条件使用这个参数,一般都很少用它。
ConstraintType.And指的是当前参数与前面参数的关系是And关系
ConstraintType.Or通常情况下指的是当前参数与前面参数的关系是Or关系
第二个参数string columnname是条件列字段名称
第三个参数Comparison cparsion是表达式,使用Comparison.X来设置,根据需求设置==、>、>=、<、<=、like、in、not in......
第四个参数object value是条件值,如果是in与not in表达式时,条件值必须为数据类型,比如:string[]、int[]、object[]等
第五个参数bool isParentheses是加左括号,而右括号使用new ConditionHelper.SqlqueryCondition()或new ConditionHelper.SqlqueryCondition(Comparison.CloseParentheses)来添加,然后in或not in表达式时,必须紧跟着右括号
例子1:
(A<2 Or A>= 10) And B=100
var wheres = new List<ConditionHelper.SqlqueryCondition>(); wheres.Add(new ConditionHelper.SqlqueryCondition(ConstraintType.Where, "A", Comparison.LessThan, 2, true)); wheres.Add(new ConditionHelper.SqlqueryCondition(ConstraintType.Or, "A", Comparison.GreaterOrEquals, 10)); wheres.Add(new ConditionHelper.SqlqueryCondition()); wheres.Add(new ConditionHelper.SqlqueryCondition(ConstraintType.And, "B", Comparison.Equals, 100));
例子2:
A == 2 And B in arr And C like 'abc%' (in查询必须加左右括号)
var wheres = new List<ConditionHelper.SqlqueryCondition>(); wheres.Add(new ConditionHelper.SqlqueryCondition(ConstraintType.Where, "A", Comparison.Equals, 2)); //加左括号 wheres.Add(new ConditionHelper.SqlqueryCondition(ConstraintType.And, "B", Comparison.In, arr, true)); //加右括号 wheres.Add(new ConditionHelper.SqlqueryCondition()); wheres.Add(new ConditionHelper.SqlqueryCondition(ConstraintType.And, "C", Comparison.Like, "abc%"));
这个封装类最终会生成SubSonic3.0底层调用的条件参数,经过测试发现,SubSonic3.0底层调用的条件参数只支持单括号,不支持括号的嵌套,多重嵌套后最终生成的也只是单括号且会出错,所以多重嵌套括号时,最好使用其它方法来实现,比如Linq、存储过程、SQL语句拼接等。
4、小结
本次更新的代码功能比如多,已将部门、职位、公用页面控件权限标识、页面控件权限管理、在线用户列表、用户登陆日志、用户操作日志、错误日志等功能都已完成了,同时也对在线用户相关类和函数进行了优化处理。
内容看起来好像挺多了,其实都是对之前所写的模板函数以及其他公共函数的调用,这么多页面查看cs代码时可以发现,里面的内容都差不多,都是很格式化的东西,如果你尝试过用这些已完成的模板页面添加新功能,就可以发现很简单,开发也很迅速。
至于每个页面如何去实现,接下来就不必再像之前那样讲得很细致了,会针对一些特殊功能或函数的调用进行解说,往细讲也只能从逻辑层的各个函数应用来说起,所以就不再罗嗦一一细说,代码中有大量的注释,大家慢慢研究吧。
最好附上几张页面的效果图
![](http://images.cnitblog.com/i/129385/201406/232221190964458.jpg)
![](http://images.cnitblog.com/i/129385/201406/232221453467477.jpg)
![](http://images.cnitblog.com/i/129385/201406/232222078926365.jpg)
Web层后端权限模块.rar
版权声明:
本文由AllEmpty原创并发布于博客园,欢迎转载,未经本人同意必须保留此段声明,且在文章页面明显位置给出原文链接,[b]否则保留追究法律责任的权利。如有问题,可以通过1654937@qq.com 联系我,非常感谢。[/b]
发表本编内容,只要主为了和大家共同学习共同进步,有兴趣的朋友可以加加Q群:327360708 ,大家一起探讨。
更多内容,敬请观注博客:http://www.cnblogs.com/EmptyFS/
相关文章推荐
- 从零开始编写自己的C#框架(18)——Web层后端权限模块——菜单管理
- 从零开始编写自己的C#框架(17)——Web层后端首页
- 从零开始编写自己的C#框架(17)——Web层后端首页
- 从零开始编写自己的C#框架(15)——Web层后端登陆功能
- 从零开始编写自己的C#框架(9)——数据库设计与创建
- 从零开始编写自己的C#框架(10)——项目实施计划与甘特图
- 【转】从零开始编写自己的C#框架(7)——需求分析
- 从零开始编写自己的C#框架(10)——项目实施计划与甘特图
- 从零开始编写自己的C#框架(3)——开发规范
- 从零开始编写自己的C#框架——框架学习补充说明
- 从零开始编写自己的C#框架(4)——文档编写说明
- 从零开始编写自己的C#框架(5)——三层架构介绍
- 从零开始编写自己的C#框架(2)——开发前准备工作
- 从零开始编写自己的C#框架(1)——前言
- 从零开始编写自己的C#框架(12)——T4模板在逻辑层中的应用(一)(附源码)
- 从零开始编写自己的C#框架(8)——后台管理系统功能设计
- 从零开始编写自己的C#框架(8)——后台管理系统功能设计
- 从零开始编写自己的C#框架(4)——文档编写说明
- 从零开始编写自己的C#框架(9)——数据库设计与创建
- 从零开始编写自己的C#框架(13)——T4模板在逻辑层中的应用(二)