Membership三步曲之进阶篇 - 深入剖析Provider Model

Membership 三步曲之进阶篇 - 深入剖析Provider Model

本文的目标是让每一个人都知道Provider Model 是什么,并且能灵活的在自己的项目中使用它。

  1. Membership三步曲之入门篇 - Membership 基础示例
  2. Membership三步曲之进阶篇 - 深入剖析Provider Model
  3. Membership三步曲之高级篇

  在入门篇中我们已经从0开始将Membership集成到一个空的MVC站点中,并且与ASP.NET的权限管理体系相结合。本篇(进阶篇)将剖析Membership的设计理念以及它的结构。别忘了我们还有高级篇我们将会扩展自己的MembershipProvider和RolesProvider,目地是直接利用我们老系统中已经存在的用户表和角色表,也就是用我们已经存在的数据库去集成Membership。

  我们了解到Membership的重要组成部分是MembershipProviders。 要理解Membership的架构设计,我们首先要理解Provider。Provider 的全称是 Provider Model (中文是提供程序模型),它早就已经不是什么新鲜事了,它是在ASP.NET 1.1的时候被 Rob Howard 设计出来的,从ASP.NET 2.0时代,它就已经开始大行其道了。到现在的MVC中,各种Provider已经随处可见了, 貌似微软对它特别青睐。下面我们就来好好看一看Provider到底是个神马东西,Membership是如何利用它的!

目录

  1. 随处可见的Provider
  2. Provider并不是一种设计模式
    1. 策略模式
    2. 工厂方法
    3. 单件模式
    4. 外观模式
  3. Provider Model 的配置及初始化框架
  4. 利用Provider Model实现记日志组件
  5. 参考引用

随处可见的Provider

  各位Provider们,先来和大家混个眼熟吧,相信下面的几种Provider大家并不会觉得陌生:

  前面两种Provider我们在Membership入门篇里面已经接触到了,后面两种我想也不用说了吧? 而在MVC中的下面几个兄弟,你们都见识过吧?

  那么Provider到底是起了一个什么样的作用呢?为什么微软要如此广泛的使用它?又或者我们是不是可以在自己的项目中使用这种设计呢?

  首先,一个Provider可以是一个类,或者好几个类共同组成的一个模块,它们提供了一些特定的功能,这些特定的功能要么用接口中的方法或者抽象类的抽象方法暴露出来(.NET框架中一般使用抽象类)。这就是ProvderModel的第一要素:具要良好定义的公有API。比如说我们的MembershipProvider提供了GetUser, CreateUser, ChangePassword等与用户管理息息相关的功能,这些API并不是越全越好,而是越精越好,只定义最核心的功能,其它的可以添加自己的扩展去实现,正是小巧又精悍呀。

  其次,要有一种配置机制。这种配置机制能够将具体的Provider与我们定义的功能集绑定起来。比如说我们有SqlMembershipProvider,还有ActiveDirectoryMemberProvider,他们都是抽象类MembershipProvider的子类。我还可以自己去添加其它的子类以不同的方式实现同样的功能,那么我就需要一种灵活的方法将我的子类加入到程序中。ASP.NET是通过web.config里面的配置结点实现的。

  最后,我们还需要有一种加载机制,通过这种加载机制(我们下面会探讨这种加载机制),我们可以读取配置并创建Provider的具体实例传递给我们的功能API。.NET已经为我们提供好了一些类来帮助我们达到简单配置和初始化Providers的目地,我们马上就会说到。

  那么Provider model能给我们带来哪些好处呢?

  • 提高灵活性和可扩展性
  • 隔绝功能代码和具体的数据访问代码
  • 让扩展变得简单。既可以从抽象类继承,也可以从其它具体Provider继承,只实现不一样的地方,最后只需要轻松配置就可以搞定。

  拿Membership来举例,核心功能通过Membership调用完成。而Membership并不负责具体的实现,它定义了公开良好的接口在MembershipProvider中。 所以Membership实际上并不处理业务代码,大量的功能是调用具体的Provider来实现的。这就起到了隔绝的作用,用户信息可以保存在SQL数据库中,还可以保存在域控制器中,这些Membership不用关心,只需要交给具体的Provider就可以了。同理,如果我们想让用户保存到其它地方,只需要添加自己的Provider来实现就可以了。

  

  Membership.GetUser的代码

1
2
3
4
5
public static MembershipUser GetUser(string username, bool userIsOnline)
{
    SecUtility.CheckParameter(ref username, true, false, true, 0, "username");
    return Provider.GetUser(username, userIsOnline);
}

  所以最后怎么样去实现,实现上还是看我们具体配置的Provider如何去做。  

Provider并不是一种设计模式

  Provider在最开始(ASP.NET 1.1)的时候其实是一种设计模式,全称是 Provider Model Design Pattern,但是到了ASP.NET2.0 的时候其实就直接叫Provider Model 了(具体叫不叫设计模式,是不明确的,还有的地方叫Provider Pattern,在这里我们就不做争辩了)。 但是实际上我们会在Provider Model身上发现好几种设计模式的影子,比如说:

  • 策略模式
  • 工厂方法模式
  • 单一模式
  • 外观模式

  本文不会详细讨论这四种设计模式,但是为了让大家更好的理解Provider Model,我们来简单的结合Provider Model的结构提一下每个设计模式的定义。

2.1 策略模式

  关于策略模式有一个故事,话说老王有时候开车特别快,开的快有时候就会被警察逮住。但是有时候遇到的是比较好说话的警察可能不会扣他的分,只是口头警告一下,有时候会遇到比较严格又正直的警察一定要扣分。但是在没有被逮住之前,我们都不知道那个会是一个好说话的警察还是一个严格又正直的警察。

  

  有没有发现这个类图和我们上面的Membership的类图很像? 简单一句话概括策略模式:将一个特性的主要功能抽象出来,允许不同的实现方式,在运行时可以任意切换到不同的实现。MembershipProvider抽象类为我们定义了一组具体的API, 而Membership类则负责调用这些API,但是Membership并不在乎现在是哪一个实现。

2.2 工厂方法模式

  工厂方法(如果想详细了解工厂方法的同学,可以参考Terry Lee的这篇博客)的关键在于: 使用者不需要知道也不用知道哪一个具体对象会被创建。对于Membership类来说,它会调用具体的Provider相关的方法而不用知道当前是哪一个Provider在工作。 在Membership内部有一个名叫Provider的属性,它的类型是基类- 抽象类MembershipProvider。而它的初始化是借助于.NET为我们提供的一套配置和初始化的基础框架来实现的,这两套框架我们马上就会说到了。

2.3 单件模式

  单件模式简单的说即在整个应用程序的生命周期内,某个类的实例至始至终都是同一个。在Membership的内部,维护着一个Provider的属性,这就是singleton的实现。

2.4 外观模式

  对外观模式浅显易懂的定义是“为一组分布在不同的子系统或者不同地方的接口提供一个统一的入口点”。大家可以对比一下,所有的和Membership相关的功能都是通过调用Membership完成的,即使里面有一些功能是由MembershipUser和其它的类来实现,但是没有关系,Membership给这些所有的功能作了一个统一。这就是传说中的外观模式。

Provider Model的配置及初始化框架

  Provider Model之所以能在.NET的世界里面大放光彩,不仅仅是因为它提供了我们上述的功能及特性,而是它还提供了这样的一套配置及初始化框架,从而使得我们在.NET中使用Provider Model中使用Provider Model是那么的容易。下面我们就来看看它是如何做到的。

 

  我们在上一篇中列举了Membership中使用到的一些类型,但是用到起到读取配置作用的实际上是MembershipSection这个类,它会帮助我们把配置文件中membership结点信息加载到一个MembershipSection的实例中。我们可以来看看Membership中的Initialize方法:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
private static MembershipProvider s_Provider;
private static MembershipProviderCollection s_Providers;
 
private static void Initialize()
{
    MembershipSection membershipSection = null;
 
    if (!_initialized)
    {
        // 这里是加载配置信息
        membershipSection = (MembershipSection)ConfigurationManager.GetSection("membership");
 
        if (membershipSection == null)
            throw new Exception("没有找到Membership的配置");
 
        s_Providers = new MembershipProviderCollection();
 
        // 这里是创建Provider实例
        ProvidersHelper.InstantiateProviders(
            membershipSection.Providers,
            s_Providers,
            typeof(MembershipProvider));
         
        // 将当前Provider指向默认的Provider
        s_Provider = s_Providers[membershipSection.DefaultProvider];
 
        _initialized = true;
    }
}
 
public static MembershipProvider Provider
{
    get
    {
        Initialize();
        return s_Provider;
    }
}

  而初始化是由System.Web.Configuration.ProvidersHelper来完成的,它里面只包含两个公有的静态方法: InstantiateProviders和InstantiateProvider。大家可以看到前者只不过是一个包装方法,其实是通过遍历调用后者来加载所有的provider实例的。后者的代码我就不贴了,其实就是用了反射去创建对象而已,还记得我们已经将具体类型配置到config文件去了么?

1
2
3
4
5
6
7
8
9
10
public static void InstantiateProviders(
    ProviderSettingsCollection configProviders,
    ProviderCollection providers,
    Type providerType)
{
    foreach (ProviderSettings settings in configProviders)
    {
        providers.Add(InstantiateProvider(settings, providerType));
    }
}

  有了这一套配置和初始化框架,才有了我们在config文件中的简单操作。

利用Provider Model实现记日志组件

  记日志组件也已经是被炒了很多次的话题,既然都炒了那么多次了,也不在乎我再炒一次吧? 其实.NET本身提供的一些事件记录器也是基于Provider Model来实现的,既然我们学习了Provider Model,那就来自己动手实现一个简单的日志组件小小实战一下吧。

  我们要实现的功能很简单,只有能够写入消息到一个文件就可以了,但是我们要求可以按不同的格式写入,可以是纯文本的,可以是XML的,当然你还可以扩展成任意其它你想要的格式。 对于调用者来说,它只需要轻松调用就可以了。并且我们要实现可以在config文件中灵活配置使用哪一种格式来记录我们的日志。

1
2
3
4
5
public ActionResult Index()
{
    MyLog.LogManager.Log(MyLog.LogType.Info, "我要开始记日志了");
    return View();
}

  我们的LogManager是对外暴露的唯一对象,里面的方法和属性都是静态的,和Membership类一样,它不负责具体的实现,所有的实现都是转交给具体的Provider的。  

 LogManager.Log的代码

1
2
3
4
5
6
7
8
9
10
11
12
13
public static void Log(LogType logType, string Message)
{
    Provider.WriteLog(logType, Message);
}
 
public static LogProvider Provider
{
    get
    {
        Initialize();
        return _provider;
    }
}

  初始化Provider的代码

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
private static LogProvider _provider;
private static LogProviderCollection _providers;
 
// 是否已经初始化
private static bool _initialized = false;
 
private static void Initialize()
{
    LogProviderConfigurationSection myLogSection = null;
    if (!_initialized)
    {
        // 读取myLog结点的配置
        myLogSection = (LogProviderConfigurationSection)ConfigurationManager.GetSection("myLog");
 
        if (myLogSection == null)
            throw new Exception("没有找到myLog的配置");
         
        // 初始化providers
        _providers = new LogProviderCollection();
        ProvidersHelper.InstantiateProviders(myLogSection.Providers, _providers, typeof(LogProvider));
 
        _provider = _providers[myLogSection.DefaultProvider];
        _initialized = true;
    }
}

  TextLogProvider.WriteLog的代码 

1
2
3
4
5
6
7
8
9
10
11
12
public override void WriteLog(LogType logType, string message)
{
    var dir = Path.GetDirectoryName(FilePath);
    if (!Directory.Exists(dir))
        Directory.CreateDirectory(dir);
 
    using (var sw = new StreamWriter(FilePath, true))
    {
        string s = string.Format("{0}, {1}, {2}", DateTime.Now, logType.ToString(), message);
        sw.WriteLine(s);
    }
}

  XmlLogProvider我就不实现了,如果大家有兴趣可以自己动手尝试一下,还是老样子附上整个项目的源码。Membership三步曲最后一篇,我们将实现自己的MembershipProvider利用EF来完成的用户的管理。下周见!

  下载源码:http://pan.baidu.com/s/1jGI9dhs (觉得有用的同学就帮助点个推荐吧,您的满意,我的动力!)

参考引用

  • http://msdn.microsoft.com/en-us/library/aa479046.aspx
  • http://msdn.microsoft.com/en-us/library/ms972319.aspx   
  • http://msdn.microsoft.com/en-us/library/aa479020.aspx
作者:Jesse 出处:http://jesse2013.cnblogs.com/

本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,否则保留追究法律责任的权利。 
如果觉得还有帮助的话,可以点一下右下角的【推荐】,希望能够持续的为大家带来好的技术文章! 
我会持续为大家带来我自己的或翻译一些国外的好的文章,想跟我一起进步么?那就【关注】我吧。
 

深入剖析Provider Model的更多相关文章

  1. Membership三步曲之进阶篇 - 深入剖析Provider Model

    Membership 三步曲之进阶篇 - 深入剖析Provider Model 本文的目标是让每一个人都知道Provider Model 是什么,并且能灵活的在自己的项目中使用它. Membershi ...

  2. 从Membership 到 .NET4.5 之 ASP.NET Identity

    我们前面已经讨论过了如何在一个网站中集成最基本的Membership功能,然后深入学习了Membership的架构设计.正所谓从实践从来,到实践从去,在我们把Membership的结构吃透之后,我们要 ...

  3. Membership三步曲之入门篇 - Membership基础示例

    Membership 三步曲之入门篇 - Membership基础示例 Membership三步曲之入门篇 -  Membership基础示例 Membership三步曲之进阶篇 -  深入剖析Pro ...

  4. 跌倒了,再爬起来:ASP.NET 5 Identity

    "跌倒了"指的是这一篇博文:爱与恨的抉择:ASP.NET 5+EntityFramework 7 如果想了解 ASP.NET Identity 的"历史"及&q ...

  5. [转]Membership 到 .NET4.5 之 ASP.NET Identity

    本文转自:http://www.cnblogs.com/jesse2013/p/membership-part3.html 我们前面已经讨论过了如何在一个网站中集成最基本的Membership功能,然 ...

  6. [转]Membership三步曲之入门篇 - Membership基础示例

    本文转自:http://www.cnblogs.com/jesse2013/p/membership.html Membership三步曲之入门篇 - Membership基础示例   Members ...

  7. ASP.NET 5 Identity

    ASP.NET 5 Identity   “跌倒了”指的是这一篇博文:爱与恨的抉择:ASP.NET 5+EntityFramework 7 如果想了解 ASP.NET Identity 的“历史”及“ ...

  8. MVC的Membership

    摘自:http://stackoverflow.com/a/16734651/1616023 See the summaries below each quote for a quick answer ...

  9. provider 设计模式

    相关介绍文章: Provider Model Design Pattern and Specification, Part 1 (old but detailed). The ASP.NET 2.0 ...

随机推荐

  1. Android发展简报

    Android这个词的本义是指“机器人”.同时它是Google至2007年11月5根据公布Linux台的开源手机操作系统的名称,该平台由操作系统.中间件.用户界面和应用软件组成.号称是首个为移动终端打 ...

  2. hdu 4911 Inversion(找到的倒数)

    主题链接:http://acm.hdu.edu.cn/showproblem.php?pid=4911 Inversion Time Limit: 2000/1000 MS (Java/Others) ...

  3. 每天进步一点点-->功能fseek() 使用方法

    在阅读代码时,遇到了非常早之前用过的fseek(),非常久没实用了.有点陌生,写出来以便下次查阅. 函数功能是把文件指针指向文件的开头.须要包括头文件stdio.h fseek   函数名: fsee ...

  4. Javascript学习1 - Javascript中的类型对象

    原文:Javascript学习1 - Javascript中的类型对象 1.1关于Numbers对象. 常用的方法:number.toString() 不用具体介绍,把数字转换为字符串,相应的还有一个 ...

  5. HDU 2616 Kill the monster (暴力搜索 || 终极全阵列暴力)

    主题链接:HDU 2616 Kill the monster 意甲冠军:有N技能比赛HP有M怪物,技能(A,M),能伤害为A.当怪兽HP<=M时伤害为2*A. 求打死怪兽(HP<=0)用的 ...

  6. Matlab基于学习------------------函数微分学

    <span style="font-size:18px;">% 函数微分学 % 函数微分学难比功能区,中的积分函数的性质整体叙述性说明.在某些时候差描述叙事的斜率功能 ...

  7. Java设计模式偷跑系列(十二)组合模式建模和实现

    转载请注明出处:http://blog.csdn.net/lhy_ycu/article/details/39828653 组合模式(Composite):组合模式有时又叫部分-总体模式.将对象组合成 ...

  8. MVC4 学习笔记01

    1 . ASP.NET MVC 中 ActionResult 和 ViewResult 在使用上的区别是什么?要注意什么吗? ActionResult 是一个抽象(abstract)类,ViewRes ...

  9. 错 &#39;Cannot run program &quot;/home/uv/IDE/adt/sdk/platform-tools/adb&quot;: error=2, No such file or directory

    为linux平台搭建android开发环境的人,您可能会遇到问题,如下面有: 64位置linux安装64位置eclipse和64位置jdk开场后eclipse错误后 ""Canno ...

  10. Backbone入门

    Backbone入门讲解 Backbone是一个实现了web前端mvc模式的js框架. 一种解决问题的通用方法,我们叫做模式. 设计模式:工厂模式,适配器模式,观察者模式等,推荐js设计模式这本书.设 ...