zoukankan      html  css  js  c++  java
  • 谈谈.NET MVC QMVC高级开发

        自从吾修主页上发布了QMVC1.0,非常感兴趣,用了半月的时间学习,真的感觉收益非浅,在此声明非常感谢吾修大哥的分享!

        1、轻快简单,框架就几个类,简单,当然代码少也就运行快!单纯的MVC,使的如果你想扩展框架,可以轻易的在QMVC上增加和减少功能,也就是说更容易的去修改和读懂源码。

        2、可以与webform框架融合,也就是说你用webform和mvc共同在同一个项目中运行。

        3、QMVC APP开发,QMVC APP可以轻易实现多个QMVC项目合并到一个项目中运行,也可以轻易将其分离开独立运行,比如一个网站常有论坛、文章系统、新闻系统等,如果你的程序用qmvc app,哪么可以轻易实现多个系统的融合。

        本文主题:“QMVC高级开发”,所谓高级我可不认为就是学不会,搞不懂的忽悠者是高级!而是高与初级的层面上知识,所以在此,我假设你己经懂了QMVC的基础级应用,如果想了解QMVC的初级应用,可以参考我写的一篇文章《高性能.NET MVC之QMVC》,或QMVC官网(www.wuxiu.org)。

        怎样的框架是最适合你的?哪么豪无疑问,最适合你的框架,什么是最适合自己的?肯定是量身定制的!就像是如果说qmvc,他的代码非常少,层次简单,逻辑清晰,简单。你可以通过基于QMVC轻易扩展成适合自己系统的框架!本文就谈谈如何扩展自己的MVC APP框架、谈谈QMVC源码结构!

        1、QMVC源码分析

        首先QMVC源码中存在两种MVC模式,一种是QMVC基本框架,一种是QMVC APP框架,QMVC APP的代码是基于QMVC基本框架实现的,源码中APPS目录中存的是关于QMVC APP框架的实现代码。根目录下是QMVC基本框架,Router为QMVC的路由功能,用HTTP请求地址的分析。

        ControllerBase类    所有控制器的基类

        IMVCModel  模型调用接口,当然你的Model也可以不继承于此接口,如果想在Model中调用QMVCRazorPage类,哪么可以实现接口中获取。

        MVCRazorPageBase类  是关于View模板的基类

         QMVCCore类          QMVC框架的核心类,控制qmvc运行的支持者。

        ViewResult类        是Controller传值给QMVCCore去调用Razor模板时的信息类,Controller把要显示的信息给ViewResult,QMVCCore通过ViewResult加载Razor模板。

        Router.IRouterProvider接口   如果你想重写QMVC的路由功能,哪么可以实现Router.IRouterProvider接口实现。

        至于APPS中的类于以上的类类似,并且实现了固定的Router。

        如果你想重写模板显示路径规则可以重写ControllerBase类中的getTemplateUrl方法,这无疑对你编写自己的框架起到了最大控制的空间,原方法代码:

    protected virtual string getTemplateUrl(string themeName,string controllerName,string viewName)
    {
          return "~/" + themeName + "Views/" + controllerName + "/" + viewName + ".cshtml";
    }

    DefaultRouter类

    using System;
    using System.Collections.Generic;
    using System.Collections.Specialized;
    using System.Linq;
    using System.Reflection;
    using System.Text;
    using System.Text.RegularExpressions;
    using System.Web;
    
    namespace wuxiu.QMVC.Router
    {
        public class DefaultRouter : IRouterProvider
        {
        
            string DefaultController;
            string DefaultAction;
            RouterControllerCollection controllers = new RouterControllerCollection();
            public DefaultRouter(Assembly asm, string defaultController, string defaultAction)
            {
                DefaultController = defaultController;
                DefaultAction = defaultAction;
                //
                controllers.LoadAssembly(asm);
            }
    
            public ControllerBase LoadUrl(HttpContextBase context, string apprawurl)
            {
                string controllerName = DefaultController;
                string action = DefaultAction;
                string pageurl;
                if (apprawurl.IndexOf('?') != -1)
                {
                    pageurl = apprawurl.Substring(0, apprawurl.IndexOf('?'));
                }
                else pageurl = apprawurl;
                string[] urls= Utils.ClearStrArrayEmpty(pageurl.Split('/'));
                string[] urldata;
                if(urls.Length>0)
                {
                    controllerName=urls[0];
                }else controllerName=DefaultController;
                if(urls.Length>1)
                {
                    action=urls[1];
                    if (!action.EndsWith(".action",StringComparison.OrdinalIgnoreCase))
                    {
                        return null;
                    } action = action.Substring(0,action.Length - ".action".Length);
                }else action=DefaultAction;
    
                if(urls.Length>2){
                    urldata=new string[urls.Length-2];
                    for(int i=2;i<urls.Length;i++)
                        urldata[i-2]=urls[i];
                }else urldata=new string[]{};
    
                Type ControllerType = controllers.GetController(controllerName);
                if (ControllerType != null)
                {
                    ControllerBase controller = Activator.CreateInstance(ControllerType) as ControllerBase;
                    controller.init(context, controllerName, action, urldata);
                    return controller;
                }
                return null;
            }
    
        }
    }

    QMVC默认路由器类,没有实现正则表达式,其实我觉得这是一个好事,因为使用字符分析更准确性能更佳。DefaultRouter类中实现了IRouterProvider接口,在LoadUrl函数中分析了http请求地址,并且通过反射的方式调用的控制器中的方法。

       2、QMVC APP介绍

      QMVC APP的实现目的是为了实现多个项目融合提出的解决方案,当然如果你的程序按照QMVC APP模式开发,可以很容易的实现与另一个APP之间融合到一个站点中运行。   如果你的项目按照QMVC APP模式开发,哪么Controller是继承自wuxiu.QMVC.APPS.APPController基类,每一个应用中必须实现APPProvider类。每个应用中存放app.config文件。

        QMVC APP目录结构

        APPS

         APPName

            DefaultViews

               ControllerName

                  ActionName.cshtml

               ...

            App.config

      Controllers

      Models

      APPProvider.cs

      Global.asax

      Web.config

      以上树中,APPS目录是必须目录,用于包含当前项目中所有的应用文件夹,APPName是应用的文件夹,其中包括了Views和app.config,Views里存放了应用所用的视图,app.config中定义了应用的配置信息。

      APPProvider.cs是APPS模式中所有应用实现的一个类,他继承于wuxiu.QMVC.APPS.APPProvider类,该类在QMVC初始化时调用,用于初始化APP。   Models和Controllers文件为当前应用的相关控制器和模型存放目录。

       QMVC APP项目配置

      若建一个QMVC APP模式的项目,必须以6.2节中定义的目录结构外,需要修改一些配置文件,让ASP.NET运行框架支持QMVC APP运行。

      Web.config中,添加或修改以下配置项:

      

    <configuration>
      <system.web>
        <compilation debug="true" targetFramework="4.5">
          <assemblies>
            <add assembly="System.Web.WebPages.Razor2, Version=0.0.0.0, Culture=neutral, PublicKeyToken=14679ed9c77dd5f5" />
            <add assembly="System.Web.WebPages2, Version=0.0.0.0, Culture=neutral, PublicKeyToken=14679ed9c77dd5f5" />
            <add assembly="System.Web.Razor2, Version=0.0.0.0, Culture=neutral, PublicKeyToken=14679ed9c77dd5f5" />
          </assemblies>
          <buildProviders>
            <remove extension=".cshtml"/>
            <add
               extension=".cshtml"
               type="System.Web.WebPages.Razor.RazorBuildProvider, System.Web.WebPages.Razor2"/>
          </buildProviders>
        </compilation>
      </system.web>
      <system.webServer>
        <modules runAllManagedModulesForAllRequests="true"/>
      </system.webServer>
      </configuration>

      APPS/web.config中,添加或修改以下配置项:

    <configuration>
      <configSections>
        <sectionGroup name="system.web.webPages.razor" type="System.Web.WebPages.Razor.Configuration.RazorWebSectionGroup, System.Web.WebPages.Razor2">
          <section name="host" type="System.Web.WebPages.Razor.Configuration.HostSection, System.Web.WebPages.Razor2" requirePermission="false" />
          <section name="pages" type="System.Web.WebPages.Razor.Configuration.RazorPagesSection, System.Web.WebPages.Razor2" requirePermission="false" />
        </sectionGroup>
      </configSections>
        <system.web>
        </system.web>
      <system.web.webPages.razor>
        <pages pageBaseType="wuxiu.QMVC.MVCRzorPageBase">
          <namespaces>
            <add namespace="wuxiu.QMVC" />
          </namespaces>
        </pages>
      </system.web.webPages.razor>
    </configuration>

    APPS/APPName/APPProvider.config

     

     <?xml version="1.0" encoding="utf-8" ?>
      <app 
        name="贴吧系统"
        version="1.0"
        copyright="bobby"
        urlFolder="wuxiu"
        appProviderClass="wuxiu.tiebaAPP.TiebaAPPProvider"
        assembly="bin:wuxiu.tiebaAPP"
        enable="true"
        >
        <appSettings>
               <add key="" value=""/>
      ... 
      </appSettings>
      <customElements>
      </customElements>  
      </app>

      如果您创建一个QMVC APP,哪么必须写一个app.config描述您的应用信息,QMVC会按照您的配置去加载应用,并且把您的配置信息传给应用。

      APP节点属性

    Name 应用的名称 Version                 当前应用的版本号 appProviderClass 表示实现wuxiu.QMVC.APPS.APPProvider类的命名空间和类名,用于初始化APP的类。 assembly         当前应用的APPProvider所在的类库,值有两种书写形式:

                            1、file冒号开头,标识app/bin/目录下的dll文件名,如:“file:mydll.dll”。

                            2、bin:默认为程序跟目录Bin目录下的程序集名称,如:“bin:mydll”。

    说明:推荐使用bin前缀,因为file虽然可以独立自己的程序目录,确存在着运行中独占的问题,可能导致程序运行页dll不允许移动和删除。 Enable 标识当前程序是否可用,值有两个:true或false。

    以上章节是翻译自吾修主页的QMVC Doumentation章节中关于QMVC APP章节!     

    实现一个自己的AppController

    public class HomeController:wuxiu.QMVC.APPS.APPController{
       public wuxiu.QMVC.ViewResult Index(){
          return View();
       }
    }

    实现一个IMVCModel类

    当在模型类中可以获取到MVCRazorPage对象无疑将Model的功能增加,他可以更灵活的提供些程序逻辑,减少了Razor模板中的代码,毕竟Razor是给美工操作的层面代码。

        public class MyModel:wuxiu.QMVC.IQMVCModel
        {
            public string LoginPageUrl { get { return page.APPUrlStart + "/user/login.action"; } }
    
            wuxiu.QMVC.APPS.APPRazorPage page;
            public void Init(QMVCRzorPageBase page)
            {
                this.page = (wuxiu.QMVC.APPS.APPRazorPage)page;
            }
        }
  • 相关阅读:
    MySQL学习——SQL查询语句(连接查询&子查询)(三)
    MySQL学习——SQL查询语句(使用集合函数查询)(二)
    MySQL学习——SQL查询语句(一)
    MySQL学习——操作数据库(增删改查)(二)
    MySQL学习——操作数据库(增删改查)(一)
    MySQL学习——MySQL数据库概述与基础
    Python学习之——Http协议简介
    Python学习之——Socket套接字(UDP连接)
    第04组 Beta版本演示
    第04组 Beta冲刺(4/5)
  • 原文地址:https://www.cnblogs.com/jsmblog/p/qmvc_app.html
Copyright © 2011-2022 走看看