利用.net的内部机制在asp.net中实现身份验证

简介:
  在.net内部存在一些默认机制,具体说来有如下:
    (1)如果类的实例字段没有赋初始值,那么它们就会被赋默认值,对于引用类型字段的默认值为null,对于简单值类型字段(即数值类型)默认值为对应形式的0(如sbyte、byte、short、ushort、int、uint、long 和 ulong都是0,而char类型字段的默认值值是’\0x0000’,float类型字段是0.0f,double类型字段是0.0d等等)。
    (2)如果类中赋初值的静态字段,那么自动生成一个静态构造函数,在静态函数中完成赋初值。
    (3)如果一个类没有明确指明派生自哪个类,默认会派生自Object类。
    (4)如果一个类没有定义任何构造函数,默认会给该类生成一个构造函数。
    (5)在调用子类的非静态构造函数的时候,默认会调用父类的构造函数。例如A继承B,B继承C,C没有明确指明继承哪个类(实际上是继承Object),那么在实例化A的时候,会首先调用B的构造函数,在调用的构造函数的时候又会调用C的构造函数,在调用C的构造函数的时候又会去调用Object的构造函数,因此实例化一个A对象时,在内部会经历一连串的构造函数调用,就上面的继承层次来说,构造函数的调用顺序从先到后依次为:object->C->B->A。
下面这部分代码可以演示这个演示这个过程:
using System;        
     
public  class C        
{        
C() C()        
        {        
                Console.WriteLine( "C");        
        }        
}        
public  class B:C        
{        
B() B()        
        {        
                Console.WriteLine( "B");        
        }        
}        
public  class A:B        
{        
A() A()        
        {        
                Console.WriteLine( "A");        
        }        
}        
     
public  class Demo:A        
{        
        static int d=5;        
Demo() Demo()        
        {        
        }        
static void Main() static void Main()        
        {        
                Demo d =  new Demo();        
                Console.ReadLine();        
        }        
}    
将上面程序编译成控制台应用程序,执行会得到如下结果:
如果将生成的Demo.exe文件用ILDASM.exe打开的话,会看到如下情形:
    上面说的这些都不是我今天要说的重点,不过我们可以它们来实现一些功能。在早期的时候我开发项目的时候,做后台身份控制的时候总是觉得比较麻烦,每页都会写检查用户是否登录的代码,后来asp.net2.0出来之后就使用了母板页,这样对用户的身份验证就可以在母板页中的Load事件中来控制。这样虽然能完成功能,不过由于母板页的Load事件是晚于内容页的Load事件的,所以这个时候验证用户身份可以看到页面上的部分内容了。
    利用上面的第(5)个机制,就可以比较好的解决这个问题。我们可以首先定义一个后台所有需要进行身份验证的页的基类,这个基类继承自System.Web.UI.Page类(因为在asp.net中所有的asp.net页所对应的类必须直接或者简介继承自System.Web.UI.Page类)。我们直到当访问一个页面的时候,肯定要实例化这个页面对应的类,由于内部机制必须要首先调用我们编写的基类中定义的构造方法(如果有的话,然后又鸡生蛋、蛋生鸡一样地追朔到object类,不过那些我们不用管)。我们要下手的话,就应该在这个我们自定义的基类的构造方法里面。这时候有两种选择:
    一是直接在基类的构造函数进行身份验证,比如在基类构造函数里检查Cookie或者Session集合中是否存在某个特定的名称的Session或者Cookie(用Session或者用Cookie或者二者兼用取决于具体项目要求),这个特定名称的Session或者Cookie是当用户登录了后台时才会设定的,因此只要存在这个Cookie或者Session就可以判断用户是登录过的。
    二是再利用一些机制来做,比如在基类的构造函数中再利用一个asp.net页面的生命周期中必经的事件来进行处理。比如这个在构造函数中定义基类的某个事件的处理方法,在这个处理方法中进行身份判断。
我记得有人说过这么一句话:“最直接的方法往往是最难的方法。”上面的办法中,第一种是实现不了的,因为构造函数调用的时候,很多信息刚刚被初始化,有很多还没有被初始化,在基类的构造方法中Session和Cookie都是没有被实例化的。
    看来只有用第二种办法了,看过《asp.net夜话之五:Page类和回调技术》之五的朋友应该有印象,所有的Page类都会经历一个加载事件,我们就选择在基类的构造函数中定义基类的加载事件的处理方法,在这个方法里检查用户是否登录。至于为什么要选择这个事件而不用其它事件,那是因为有些事件发生得还是太早,怕Session和Cookie还是不可用,有些事件又太晚,用加载事件是刚刚好(郭德钢有一首歌正好叫《刚刚好》,呵呵)。
    因为我们的这个基类没有任何显示部分代码,所以我们可以在项目中添加一个类文件,代码如下:
using System;        
using System.Data;        
using System.Configuration;        
using System.Web;        
using System.Web.Security;        
using System.Web.UI;        
using System.Web.UI.WebControls;        
using System.Web.UI.WebControls.WebParts;        
using System.Web.UI.HtmlControls;        
     
/// <summary>        
/// AdminPage是后台所有需要身份验证的页的基类        
/// </summary>        
public  class AdminPage:System.Web.UI.Page        
{        
AdminPage() AdminPage()        
        {        
                //处理加载事件        
                this.Load +=  new EventHandler(AdminPage_Load);        
        }        
     
        void AdminPage_Load(object sender, EventArgs e)        
        {        
                //假设用户登录成功之后会设置一个名为“UserInfo”的Session        
                //假设用户登录页面login.aspx位于网站根目录下的Admin文件下        
                 if (Session[ "UserInfo"] ==  null)        
                {        
                        //使用~来保证无论在什么目录层次进行身份验证都能跳转到后台登录页面        
                        Response.Redirect( "~/Admin/Login.aspx");        
                }        
        }        
}    

    这样,对于后台任何需要进行身份验证的页面,只需将它更改为继承自AdminPage类即可,这样这些类中就不需要编写任何身份验证的代码了。
    如下,就是一个类的代码,这里没有任何手动编写的代码,却已经能实现验证了,因为它继承自AdminPage类:
using System;        
using System.Data;        
using System.Configuration;        
using System.Collections;        
using System.Web;        
using System.Web.Security;        
using System.Web.UI;        
using System.Web.UI.WebControls;        
using System.Web.UI.WebControls.WebParts;        
using System.Web.UI.HtmlControls;        
     
public partial  class Admin_News_Index : AdminPage        
{        
void Page_Load() void Page_Load(object sender, EventArgs e)        
        {        
     
        }        
}
    对上面的处理方法还可以进一步处理,比如某些页面不光要求用户必须登录,还要求用户必须具有相关权限,读者朋友可以利用这种思想去实现。此外,还可以利用这种机制做一些其它的业务处理。















本文转自周金桥51CTO博客,原文链接: http://blog.51cto.com/zhoufoxcn/166380 ,如需转载请自行联系原作者









相关文章
|
6月前
|
存储 开发框架 前端开发
asp.net与asp.net优缺点及示例
asp.net与asp.net优缺点及示例
|
29天前
|
开发框架 前端开发 .NET
进入ASP .net mvc的世界
进入ASP .net mvc的世界
28 0
|
1月前
|
开发框架 中间件 .NET
C# .NET面试系列七:ASP.NET Core
## 第一部分:ASP.NET Core #### 1. 如何在 controller 中注入 service? 在.NET中,在ASP.NET Core应用程序中的Controller中注入服务通常使用<u>依赖注入(Dependency Injection)</u>来实现。以下是一些步骤,说明如何在Controller中注入服务: 1、创建服务 首先,确保你已经在应用程序中注册了服务。这通常在Startup.cs文件的ConfigureServices方法中完成。例如: ```c# services.AddScoped<IMyService, MyService>(); //
60 0
|
1月前
|
开发框架 前端开发 .NET
C# .NET面试系列六:ASP.NET MVC
<h2>ASP.NET MVC #### 1. MVC 中的 TempData\ViewBag\ViewData 区别? 在ASP.NET MVC中,TempData、ViewBag 和 ViewData 都是用于在控制器和视图之间传递数据的机制,但它们有一些区别。 <b>TempData:</b> 1、生命周期 ```c# TempData 的生命周期是短暂的,数据只在当前请求和下一次请求之间有效。一旦数据被读取,它就会被标记为已读,下一次请求时就会被清除。 ``` 2、用途 ```c# 主要用于在两个动作之间传递数据,例如在一个动作中设置 TempData,然后在重定向到另
95 5
|
3月前
|
开发框架 安全 搜索推荐
分享105个NET源码ASP源码,总有一款适合您
分享105个NET源码ASP源码,总有一款适合您
27 4
|
4月前
|
小程序 安全 JavaScript
.NET微信网页开发之通过UnionID机制解决多应用用户帐号统一问题
.NET微信网页开发之通过UnionID机制解决多应用用户帐号统一问题
.NET微信网页开发之通过UnionID机制解决多应用用户帐号统一问题
|
7月前
|
开发框架 前端开发 .NET
.NET 8 Release Candidate 1 (RC1)现已发布,包括许多针对ASP.NET Core的重要改进!
.NET 8 Release Candidate 1 (RC1)现已发布,包括许多针对ASP.NET Core的重要改进!
174 0
.NET 8 Release Candidate 1 (RC1)现已发布,包括许多针对ASP.NET Core的重要改进!
|
8月前
|
存储 开发框架 .NET
ASP.NET学生管理系统(.NET毕业设计)
ASP.NET学生管理系统(.NET毕业设计)
99 0
|
9月前
|
开发框架 前端开发 JavaScript
ASP .Net Core 中间件的使用(一):搭建静态文件服务器/访问指定文件
ASP .Net Core 中间件的使用(一):搭建静态文件服务器/访问指定文件
|
3月前
|
开发框架 前端开发 .NET
ASP.NET CORE 3.1 MVC“指定的网络名不再可用\企图在不存在的网络连接上进行操作”的问题解决过程
ASP.NET CORE 3.1 MVC“指定的网络名不再可用\企图在不存在的网络连接上进行操作”的问题解决过程
38 0