从零开始编写自己的C#框架(19)——Web层后端权限模块

  不知不觉本系统写了快三个月了,最近写页面的具体功能时感觉到有点吃力,很多地方如果张嘴来讲的话可以说得很细,很全面,可写成文字的话,就不太会写了,有些地方想讲得清晰的话,得用多几倍的文字+实例+变化中的图片才能表达得清楚,而写这些又太费时间了,近段时间又特忙,所以只能是尽力而为,希望大家自行研究,如果有什么地方不明白的,发发评论或邮件给我,我再重新详细讲解。

  说回正题,对于页面访问权限以及每个按键的权限控制,很久以前用过好几种不同的方法,比如为每个控件分配名称或编码,然后在写代码时绑定这些值,又比如用XML来配置制权限等方法。这些方法都比较麻烦,而且由于都是使用编码方法,开发时需要一个个进行绑定,容易出错。经过后来不断的完善,最后完成了本系统所采用的页面控件注册管理来绑定控件权限(如有雷同,纯属巧合,哈哈...),首先创建菜单,并绑定好对应的文件(页面),然后将系统要用到的名称添加到公用标识库中,跟着在页面控件权限管理页面对各个页面控件进行绑定(只需要点击鼠标即可),通过职位(角色)来赋于不同的操作权限,只需要设置好管理员的职位,那么该管理员就拥有他所绑定的角色的全部权限了。

  如下面的说明

  首先在后端注册菜单(菜单绑定页面)

  创建公用页面权限标识,待用

  为需要绑定页面控件权限的菜单页面绑定操作控件(左列为上图录入的公共控件名称,右列为已绑定的页面控件,只需要点击鼠标就可以轻松绑定)

  创建好部门

  在不同部门创建相应的职位(角色)

  为不同角色设置菜单与页面控件操作权限

  开发说明:(主要讲讲与上一章中不同的内容)

  1、PagePowerSignPublicList.aspx.cs 公用页面控件权限标识列表管理文件

  这个页面是比较经典的普通列表页面,比较代码,有以下一些地方和菜单页面不一样的

 1         #region 加载数据
 2         /// <summary>读取数据</summary>
 3         public override void LoadData()
 4         {
 5             //设置排序
 6             if (sortList == null)
 7             {
 8                 Sort(null);
 9             }
10
11             //绑定Grid表格
12             bll.BindGrid(Grid1, Grid1.PageIndex + 1, Grid1.PageSize, null, sortList);
13         }
14
15         #endregion

  对于排序函数的调用,菜单页面是有层次感列表,所以要自定义排序函数,而对于正常的列表,我们直接调用父类的Sort(null)函数即可

  绑定Grid表格也与菜单页面调用的不一样,bll.BindGrid()函数使用的是参数中需要添加当前在第几页,每个页面显示多少行,null是条件参数,后面会详细解释它怎么使用,最后一个参数是排序规则参数

  另外,Delete()删除函数也使用批量删除功能,不过会在删除前进行检查,发现指定记录不能删除时,则弹出提示是那个Id的记录无法删除,大家自行比较一下两个列表文件就明白了。

  2、PagePowerSignPublicEdit.aspx.cs公用页面控件权限标识编辑文件

  这个文件要注意的地方是Save()函数

 1         #region 保存
 2         /// <summary>
 3         /// 数据保存
 4         /// </summary>
 5         /// <returns></returns>
 6         public override string Save()
 7         {
 8             string result = string.Empty;
 9             int id = ConvertHelper.Cint0(hidId.Text);
10
11             try
12             {
13                 #region 数据验证
14
15                 if (string.IsNullOrEmpty(txtCName.Text.Trim()))
16                 {
17                     return txtCName.Label + "不能为空!";
18                 }
19                 var sName = StringHelper.Left(txtCName.Text, 20);
20                 if (PagePowerSignPublicBll.GetInstence().Exist(x => x.CName == sName && x.Id != id))
21                 {
22                     return txtCName.Label + "已存在!请重新输入!";
23                 }
24                 if (string.IsNullOrEmpty(txtEName.Text.Trim()))
25                 {
26                     return txtEName.Label + "不能为空!";
27                 }
28                 var sEname = StringHelper.Left(txtEName.Text, 50);
29                 if (PagePowerSignPublicBll.GetInstence().Exist(x => x.EName == sEname && x.Id != id))
30                 {
31                     return txtEName.Label + "已存在!请重新输入!";
32                 }
33
34                 #endregion
35
36                 #region 赋值
37                 //定义是否更新标识——即当前记录的名称是否改变了
38                 bool isUpdate = false;
39
40                 //获取实体
41                 var model = new PagePowerSignPublic(x => x.Id == id);
42
43                 //判断是否有改变名称
44                 if (id > 0 && (sName != model.CName || sEname != model.EName))
45                 {
46                     isUpdate = true;
47                 }
48
49                 //设置名称
50                 model.CName = sName;
51                 //设置英文名称
52                 model.EName = sEname;
53                 #endregion
54
55                 //----------------------------------------------------------
56                 //存储到数据库
57                 PagePowerSignPublicBll.GetInstence().Save(this, model);
58
59                 //判断是否需要同步更新关联表字段
60                 if (isUpdate)
61                 {
62                     //调用更新函数,同步更新对应的所有记录
63                     PagePowerSignBll.GetInstence().UpdateValue_For_PagePowerSignPublic_Id(this, model.Id, PagePowerSignTable.CName, model.CName, PagePowerSignTable.EName, model.EName);
64                 }
65             }
66             catch (Exception e)
67             {
68                 result = "保存失败!";
69
70                 //出现异常,保存出错日志信息
71                 CommonBll.WriteLog(result, e);
72             }
73
74             return result;
75         }
76         #endregion

  在函数中,大家可以找到下面这些代码

 1   //定义是否更新标识——即当前记录的名称是否改变了
 2   bool isUpdate = false;
 3
 4   //判断是否有改变名称
 5   if (id > 0 && (sName != model.CName || sEname != model.EName))
 6   {
 7     isUpdate = true;
 8   }
 9
10   //判断是否需要同步更新关联表字段
11   if (isUpdate)
12   {
13     //调用更新函数,同步更新对应的所有记录
14     PagePowerSignBll.GetInstence().UpdateValue_For_PagePowerSignPublic_Id(this, model.Id, PagePowerSignTable.CName, model.CName, PagePowerSignTable.EName, model.EName);
15   }

  这是用于我们修改A表记录的名称时,同步修改其他关联表引用了这个记录名称字段的所有记录

  为什么要这么处理呢?大家在看数据字典的数据库结构时,会发现很多表与其他表关联时,不单将其他表的Id引用了过来,还将这个Id对应的名称也引用了,这种操作方式使我们在编写查询语句时,几乎可以做到不用多表关联,因为我们想要显示的内容已经在查询的表中存在了,这样处理后,我们在修改相关表的名称时,就必须同步修改关联表中的名称,对于这些关联表名称的修改,我们的T4模板也生成了相应的函数,如:UpdateValue_For_表名_Id()函数,然后直接按上面代码编写方法实现就可以了。

  3、UseLogList.aspx.cs用户操作日志管理文件

  这个文件要注意的是InquiryCondition()函数

 1         /// <summary>
 2         /// 查询条件
 3         /// </summary>
 4         /// <returns></returns>
 5         private List<ConditionHelper.SqlqueryCondition> InquiryCondition()
 6         {
 7             var wheres = new List<ConditionHelper.SqlqueryCondition>();
 8
 9             //如果Id有值时,即表示查询的是指定管理员的操作日志
10             if (_id != 0)
11             {
12                 wheres.Add(new ConditionHelper.SqlqueryCondition(ConstraintType.Where, LoginLogTable.Manager_Id, Comparison.Equals, _id));
13             }
14
15             //起始时间
16             if (!string.IsNullOrEmpty(dpStart.Text.Trim()))
17             {
18                 wheres.Add(new ConditionHelper.SqlqueryCondition(ConstraintType.And, LoginLogTable.AddDate, Comparison.GreaterOrEquals, StringHelper.FilterSql(dpStart.Text)));
19                 //终止时间
20                 if (!string.IsNullOrEmpty(dpEnd.Text.Trim()))
21                 {
22                     wheres.Add(new ConditionHelper.SqlqueryCondition(ConstraintType.And, LoginLogTable.AddDate, Comparison.LessOrEquals, StringHelper.FilterSql(dpEnd.Text)));
23                 }
24             }
25
26             //ip地址
27             if (!string.IsNullOrEmpty(txtIp.Text.Trim()))
28             {
29                 wheres.Add(new ConditionHelper.SqlqueryCondition(ConstraintType.And, LoginLogTable.Ip, Comparison.Equals, StringHelper.FilterSql(txtIp.Text)));
30             }
31             //登录备注信息
32             if (!string.IsNullOrEmpty(txtloginfo.Text.Trim()))
33             {
34                 wheres.Add(new ConditionHelper.SqlqueryCondition(ConstraintType.And, LoginLogTable.Notes, Comparison.Like, "%" + StringHelper.FilterSql(txtloginfo.Text) + "%"));
35             }
36
37             return wheres;
38         }

  这是查询条件函数,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

1   var wheres = new List<ConditionHelper.SqlqueryCondition>();
2
3   wheres.Add(new ConditionHelper.SqlqueryCondition(ConstraintType.Where, "A", Comparison.GreaterThan, 2, true));
4
5   wheres.Add(new ConditionHelper.SqlqueryCondition(ConstraintType.Or, "A", Comparison.LessOrEquals, 10));
6
7   wheres.Add(new ConditionHelper.SqlqueryCondition());
8
9   wheres.Add(new ConditionHelper.SqlqueryCondition(ConstraintType.And, "B", Comparison.Equals, 100));

  例子2:

  A == 2 And B in arr And C like ‘abc%‘ (in查询必须加左右括号)

1   var wheres = new List<ConditionHelper.SqlqueryCondition>();
2
3   wheres.Add(new ConditionHelper.SqlqueryCondition(ConstraintType.Where, "A", Comparison.Equals, 2));
4   //加左括号
5   wheres.Add(new ConditionHelper.SqlqueryCondition(ConstraintType.And, "B", Comparison.In, arr, true));
6   //加右括号
7   wheres.Add(new ConditionHelper.SqlqueryCondition());
8
9   wheres.Add(new ConditionHelper.SqlqueryCondition(ConstraintType.And, "C", Comparison.Like, "abc%"));

  这个封装类最终会生成SubSonic3.0底层调用的条件参数,经过测试发现,SubSonic3.0底层调用的条件参数只支持单括号,不支持括号的嵌套,多重嵌套后最终生成的也只是单括号且会出错,所以多重嵌套括号时,最好使用其它方法来实现,比如Linq、存储过程、SQL语句拼接等。

  4、小结

  本次更新的代码功能比如多,已将部门、职位、公用页面控件权限标识、页面控件权限管理、在线用户列表、用户登陆日志、用户操作日志、错误日志等功能都已完成了,同时也对在线用户相关类和函数进行了优化处理。

  内容看起来好像挺多了,其实都是对之前所写的模板函数以及其他公共函数的调用,这么多页面查看cs代码时可以发现,里面的内容都差不多,都是很格式化的东西,如果你尝试过用这些已完成的模板页面添加新功能,就可以发现很简单,开发也很迅速。

  至于每个页面如何去实现,接下来就不必再像之前那样讲得很细致了,会针对一些特殊功能或函数的调用进行解说,往细讲也只能从逻辑层的各个函数应用来说起,所以就不再罗嗦一一细说,代码中有大量的注释,大家慢慢研究吧。

  最好附上几张页面的效果图

Web层后端权限模块.rar

 版权声明:

  本文由AllEmpty原创并发布于博客园,欢迎转载,未经本人同意必须保留此段声明,且在文章页面明显位置给出原文链接,否则保留追究法律责任的权利。如有问题,可以通过[email protected] 联系我,非常感谢。

  发表本编内容,只要主为了和大家共同学习共同进步,有兴趣的朋友可以加加Q群:327360708 ,大家一起探讨。

  更多内容,敬请观注博客:http://www.cnblogs.com/EmptyFS/

从零开始编写自己的C#框架(19)——Web层后端权限模块

时间: 2025-01-02 05:10:59

从零开始编写自己的C#框架(19)——Web层后端权限模块的相关文章

从零开始编写自己的C#框架(18)——Web层后端权限模块——菜单管理

从本章开始,主要讲解的是页面中对框架相关功能的调用方法,比如列表页面(又分为有层次感列表和普通列表).编辑页面.多标签页面等,只要熟悉了这些函数的使用方法,那么开发起来就会很便捷了. 1.如图先创建菜单列表与编辑页面 MenuInfoList.aspx 1 <%@ Page Language="C#" AutoEventWireup="true" CodeBehind="MenuInfoList.aspx.cs" Inherits=&quo

从零开始编写自己的C#框架(1)——前言

原文:从零开始编写自己的C#框架(1)--前言 记得十五年前自学编程时,拿着C语言厚厚的书,想要上机都不知道要用什么编译器来执行书中的例子.十二年前在大学自学ASP时,由于身边没有一位同学和朋友学习这种语言,也只能整天混在图收馆里拼命的啃书.而再后来也差不多,自学了很多不同的知识,都一直只能自己默默的克服一个又一个困难.所以这几年带一些应届生或只有一两年经验的新人时,都会同他们讲:你们现在太幸福了,有问题可以找度娘,还可以找我来解决.做为过来人,能深深的体会到刚入门的时候,没有系统的介绍和老师指

从零开始编写自己的C#框架(15)——Web层后端登陆功能

对于一个后端管理系统,最重要内容之一的就是登陆页了,无论是安全验证.用户在线记录.相关日志记录.单用户或多用户使用帐号控制等,都是在这个页面进行处理的. 1.在解决方案中创建一个Web项目,并将它设置为启动项 2.添加引用 3.添加WebManage文件夹与Login.aspx文件 4.添加登陆页面HTML代码 1 <%@ Page Language="C#" AutoEventWireup="true" CodeBehind="Login.aspx

从零开始编写自己的C#框架(17)——Web层后端首页

后端首页是管理员登陆后进入的第一个页面,主要是显示当前登陆用户信息.在线人数.菜单树列表.相关功能按键和系统介绍.让管理员能更方便的找到息想要的内容. 根据不同系统的需要,首页会显示不同的内容,比如显示公司公告.公司新闻.内部短消息.个人事务.各种业务提醒......等各种内容,这些大家可以需要去进行呈现. 先上代码 Main.aspx 1 <%@ Page Language="C#" AutoEventWireup="true" CodeBehind=&qu

从零开始编写自己的C#框架(14)——T4模板在逻辑层中的应用(三)

原本关于T4模板原想分5个章节详细解说的,不过因为最近比较忙,也不想将整个系列时间拉得太长,所以就将它们整合在一块了,可能会有很多细节没有讲到,希望大家自己对着代码与模板去研究. 本章代码量会比较大,基本将Web层要使用到的大部分函数都用模板生成了出来,而模板中的函数,很多也是互相关联调用的.另外在DotNet.Utilities(公共函数项目)中也添加与修改了一些类和函数. 需要特别说明的是,在逻辑层添加了July大神编写的超强上传类,具体怎么使用功能怎么强大,在后面调用到时会用一个章节详细说

从零开始编写自己的C#框架(8)——后台管理系统功能设计

原文:从零开始编写自己的C#框架(8)--后台管理系统功能设计 还是老规矩先吐下槽,在规范的开发过程中,这个时候应该是编写总体设计(概要设计)的时候,不过对于中小型项目来说,过于规范的遵守软件工程,编写太多文档也会拉长进度,一般会将它与详细设计合并到一起来处理,所以本文档看起来会怪怪的,不是很符合规范,它只是从实用角度出发来编写,以指导后面功能的设计与开发. 从零开始编写自己的C#框架 后台管理系统功能设计文档 文件状态: [√] 草稿 [  ] 正式发布 [  ] 正在修改 文件标识: C#框

从零开始编写自己的C#框架(21)——添加分类类型页面

页面权限与页面控件权限经过简单的调试后,终于启用起来了,以后大家添加新页面时,就必须按照本章介绍的方法,将你新增的页面注册到系统中,这样才能访问与进行相关操作. 下面讲讲如何创建一个分类类型的页面. 分类类型,顾名思义指的是按照一定规律.特点进行归类划分,放到一块的集合.我们开发时这些分类类型,经常用下拉列表来表现,如果有多级分类时,采用的是下拉树列表方式显示. 普通下拉列表 下拉树列表 下面将介绍如何从创建数据表.修改文件到权限绑定逐个步骤进行说明. 首先,我们先要创建好数据表 我们打开数据字

从零开始编写自己的C#框架(24)——测试

导航 1.前言 2.不堪回首的开发往事 3.测试推动开发的成长——将Bug消灭在自测中 4.关于软件测试 5.制定测试计划 6.编写测试用例 7.执行测试用例 8.发现并提交Bug 9.开发人员修复Bug 10.对已修复Bug进行返测 11.将修复完成的Bug关闭,对未修复的Bug重新激活 12.灵活使用压力测试工具 13.测试与版本控制 14.小结 15.附件下载 1.前言 对于测试,很多公司并不看重,接触过不少朋友或客户,打开网站随便点击一下,就可以很容易发现爆黄页.404.UI变型(浏览器

从零开始编写自己的C#框架(5)——三层架构介绍

原文:从零开始编写自己的C#框架(5)--三层架构介绍 三层架构对于开发人员来说,已经是司空见惯了,除了大型与超小型项目外,大多都是这种架构来进行开发. 在这里为初学者们简单介绍一下三层架构: (下面内容摘自<趣味理解:三层架构与养猪-<.NET深入体验与实战精要>>,这是以前看到的关于三层架构介绍,觉得挺经典的,大家有时间的话认真看看) 对比以上两图,我们可以看出: 1)数据库好比猪圈 ,所有的猪有序地按区域或编号,存放在不同的猪栏里. 2)DAL 好比是屠宰场 ,把猪从猪圈取出