찾다
백엔드 개발PHP 튜토리얼理解PHP中的MVC编程之MVC框架简介_PHP

理解PHP中的MVC编程之MVC框架简介_PHP

Jun 01, 2016 pm 12:29 PM
public액자이해하다소개프로그램 작성

MVC

【什么是MVC?】

  MVC是一个可以让你把“三个部分(即MVC的全称,Model、View、Controller)”谐调地组成一个复杂应用程序的概念。一辆汽车就是一个在现实生活中非常好的MVC例子。我们看车都看两个View(显示)部分:内部和外部。而这两个都离不开一个Controller(控制者):司机。刹车系统、方向盘和其他操控系统代表了Model(模型):他们从司机(Controller)那里取得控制方法然后应用到内部和外观(View)。

  【网络上的MVC】

  MVC框架所涵盖的概念相当简单并且极度灵活。基本的概念就是,你有一个单独的控制器(如index.php)用来控制所有建立在参数请求基础上的框架内应用程序。这个控制器通常包含了(最小程度上)一个定义模型的参数、一个事件和一个GET参数。这样控制器就能确认所有的请求然后运行相应的事件。打个比方来说,一个像这样/index.php?module=foo&event=bar的请求很有可能就是用来载入一个名叫foo的类,然后运行foo::bar()[就是其中的bar()函数]。这样做的好处有:

  一个对应所有应用程序的接口

  同时维护一个应用程序内无数的代码非常麻烦,因为每一段代码都有自己的相对路径、数据库链接、验证等等。而这样做就免除你在这方面的烦恼,允许你合并并重复使用代码

  【为什么要创建作者自己的MVC框架?】

  迄今为止,我没有见到过太多用PHP写的MVC框架。事实上我仅仅知道一个-Solar,是完全用PHP5写的。另外一个是Cake,一个试图成为PHP的RoR(Ruby on Rails-一个Ruby语言开源网络框架)。我自己对这两个框架都有一些不满意的地方:它们都没有利用到PEAR,Smarty等所包含的现有代码;现在的Cake还比较紊乱;最后,Solar是一个绝大部分由一个人写的作品(我无意说其作者Paul不是一个好人或者好程序员)。这些问题可能并不会让你否认它们,而且很有可能你根本不关心这些问题。但是正因为如此,我请各位尽可能地审视它们。

  【老方式】

  如果回到2001看自己写的代码,作者有可能找到一个叫template.txt的文件,它看起来像这样:www.phpv.net 转载请注明出处

<?php
 require_once('config.php'); // Other requires, DB info, etc.

 $APP_DB = 'mydb';
 $APP_REQUIRE_LOGIN = false; // Set to true if script requires login
 $APP_TEMPLATE_FILE = 'foo.php'; // Smarty template
 $APP_TITLE = 'My Application';

 if ($APP_REQUIRE_LOGIN == true) {
  if (!isset($_SESSION['userID'])) {
   header("Location: /path/to/login.php");
   exit();
  }
 }

 $db = DB::connect('mysql://'.$DB_USER.':'.$DB_PASS.'@localhost/'.$APP_DB);
 if (!PEAR::isError($db)) {
  $db->setFetchMode(DB_FETCHMODE_ASSOC);
 } else {
  die($db->getMessage());
 }

 // Put your logic here
 // Output the template

 include_once(APP_TEMPLATE_PATH.'/header.php');
 include_once(APP_TEMPLATE_PATH.'/'.$APP_TEMPLATE_FILE);
 include_once(APP_TEMPLATE_PATH.'/footer.php');
?>

  天哪,只是看这些代码都会让我有退缩的欲望。这段代码的概念就是确保每一个应用程序都能适用于这个处理方法,比如我可以简单地将template.txt拷进myapp.php,改变一些变量,瞧,它就能运行起来了。尽管如此,这个组织严密的处理方法存在一些严重的缺点:

  如果我的老板想让作者用myapp.php在一些情况下输出PDF、一些情况下输出HTML、一些情况下(直接提交的XML请求)SOAP,我该怎么办?

  如果这个应用程序需要IMAP或LDAP验证,我该怎么办?

  我该如何处理各种不同的代码(包括编辑、升级和删除)?

  我该如何处理多级验证(管理员 vs. 非管理员)?
我该如何启用输出缓存?www.phpv.net 转载请注明出处

  【新方式】

  将所有东西都扔进这个MVC框架,你会发现生活是如此简单。请对比以下代码:

<?php
 class myapp extends FR_Auth_User
 {
  public function __construct()
  {
   parent::__construct();
  }

 public function __default()
 {
  // Do something here
 }

 public function delete()
 { }

 public function __destruct()
 {
  parent::__destruct();
 }
}

?>


  注意这段代码显然不是用来链接到一个数据库、判断一个用户是否已经登陆、或者输出任何其他信息。控制器掌握了所有的一切。

  如果我想验证LDAP,我可以建立FR_Auth_LDAP。控制器可以识别某些输出方法(比如$_GET['output'])并可以随时转换成PDF或者SOAP。事件处理delete,只负责删除,其他的它都不管。因为这个模块拥有一个FR_User类的实例,它可以简单地判断一个用户是否已经登陆等等。Smarty,作为模板引擎控制缓存是理所当然的,但是控制器同样可以控制一部分缓存。

  从前面讲的老方式到MVC方式对于很多人来讲可能是一个全新、陌生的概念,但是一旦你转换到了这样一个概念,那么要转回去将是件相当困难的事情。

  【建立底层】

  我是一个PEAR尤其是PEAR_Error类的爱好者。PHP5引入了一个新的内建类“Exception”取代了PEAR_Error。但是PEAR_Error拥有一些比Exception还要实用的特性。所以,在此系列文章中的MVC框架实例将用到它来做错误处理。无论如何,我还是要用到Exception获得从构造器中的错误,因为它们本身不能传回错误。

  设计这些基础类的目的有如下几点:

  利用PEAR快速添加功能到基础类

  建立小巧、可反复实用的抽象类以便让使用者在此框架中快速开发出应用程序

  用phpDocumentor给所有的基础类生成文档

  类的层次看起来会像这样:

  -FR_Object将会提供基础的功能以供其他所有对象使用(包括logging,一般的setFrom(),toArray())

  -FR_Object_DB是一个小层面,给子类提供数据库链接等功能

  -FR_Module是所有应用(又称模块、模型等等)的底层类

  -FR_Auth是所有验证机制的底层类

   ·FR_Auth_User是一个验证类,用来验证所有需要验证用户是否登陆的模块

   ·FR_Auth_No是所有不需要验证的模块的“假验证类”

  -FR_Presenter是所有用来处理载入和显示应用的底层类

  -FR_Presenter_Smarty是包含了载入不同驱动器能力的显示层。Smarty是一个非常好的模板类,它拥有内建的缓存机制以及一个活跃的开发团体(译者注:这分明就是打广告嘛~)

   ·FR_Presenter_debug是调试部分的显示层。依靠它,开发者能够调试应用程序并给他们除错

   ·FR_Presenter_rest是一个可以让开发者能够以XML方式输出应用程序的REST显示层

  从以上的基础类结构上,你应该可以看到这个MVC框架的不同部分。FR_Module提供所有模块所需要的东西,而FR_Presenter则提供不同的显示方法。在此系列文章中的下一篇中,我将创建控制器将这上面所有的基础类结合在一块。

  【代码标准】

  在你正式编写代码之前,应该坐下来跟你的合伙人(或者你自己)好好讨论(或思考)一下代码标准。MVC编程的整体思想围绕着两点:代码的可再利用性(减少偶合)和代码的标准化。我推荐至少应该考虑到如下几点:

  首先要考虑的是变量命名和缩写标准。不要因为这个跟你的合作伙伴大吵一通,但是一旦定下来的标准,就要自始至终地遵从,尤其是写底层代码(基础类)的时候。

  定制一个标准前缀,用在所有的函数、类和全局变量上。不走运的是,PHP不支持“namespace(命名空间)”。所以要想避免混淆变量名和发生的冲突,用一个前缀是个明智的做法。我在整篇文章中将使用“FR_”作为这样的前缀。

  【编写底层】

  文件层次规划很重要。基本的层次规划很简单且在一定程度上是严格定义的:

/
config.php
index.php
includes/
Auth.php
Auth/
No.php
User.php
Module.php
Object.php
Object/
DB.php
Presenter.php
Presenter/
common.php
debug.php
smarty.php
Smarty/
modules/
example/
config.php
example.php
tpl/
example.tpl
tpl/
default/
cache/
config/
templates/
templates_c/

  你可能会想这样的文件层次肯定代表了很多的代码!没错,但是你能够完成它的。在整个系列结束后,你会发现你的编程将会变得更简单并且开发速度会得到很大的提升。

  在文件层次里面,所有的基础类都在includes文件夹内。每一个功能模块,都用一个配置文件,至少一个模块文件和一个模板文件。所有的模块包含在modules文件夹内。我已经习惯了将模板文件放在单独的外部文件夹内,也就是tpl文件夹。

  config.php-中枢配置文件,包含所有的全局配置变量。

  index.php-控制器,在接下来的一篇文章中会详细叙述。

  object.php-所有基础类的底层类,提供绝大部分类需要的功能。FR_Object_DB继承这个类并提供数据库链接。

  结构的基本概念就是,让所有的子类都继承一个中枢类以便它们都共享一些共同的特性。你完全可以把链接数据库的功能放进FR_Object,但是并不是所有类都需要这个功能的,所以FR_Object_DB就有了存在的理由,作者会稍后做出讨论它。

<?php
 require_once('Log.php');

 /**
 * FR_Object
 *
 * The base object class for most of the classes that we use in our framework.
 * Provides basic logging and set/get functionality.
 *
 * @author Joe Stump <joe@joestump.net>
 * @package Framework
 */

 abstract class FR_Object
 {
  /**
  * $log
  *
  * @var mixed $log Instance of PEAR Log
  */

  protected $log;
  /**
  * $me
  *
  * @var mixed $me Instance of ReflectionClass
  */

  protected $me;
  /**
  * __construct
  *
  * @author Joe Stump <joe@joestump.net>
  * @access public
  */

  public function __construct()
  {
   $this->log = Log::factory('file',FR_LOG_FILE);
   $this->me = new ReflectionClass($this);
  }

  /**
  * setFrom
  *
  * @author Joe Stump <joe@joestump.net>
  * @access public
  * @param mixed $data Array of variables to assign to instance
  * @return void
  */

 public function setFrom($data)
 {
  if (is_array($data) && count($data)) {
   $valid = get_class_vars(get_class($this));
   foreach ($valid as $var => $val) {
    if (isset($data[$var])) {
     $this->$var = $data[$var];
    }
   }
  }
 }

 /**
 * toArray
 *
 * @author Joe Stump <joe@joestump.net>
 * @access public
 * @return mixed Array of member variables keyed by variable name
 */

 public function toArray()
 {
  $defaults = $this->me->getDefaultProperties();
  $return = array();
  foreach ($defaults as $var => $val) {
   if ($this->$var instanceof FR_Object) {
    $return[$var] = $this->$var->toArray();
   } else {
    $return[$var] = $this->$var;
   }
  }

  return $return;
 }

 /**
 * __destruct
 *
 * @author Joe Stump <joe@joestump.net>
 * @access public
 * @return void
 */

 public function __destruct()
 {
  if ($this->log instanceof Log) {
   $this->log->close();
  }
 }
}

?>

   auth.php-这是所有验证功能的底层类。它是从FR_Module里面延伸出来的,主要功能是定义一个基本的验证类如何工作。

  跟FR_Module的道理一样,有些类不需要链接到数据库,那么同理,FR_Auth_No就可以被创建应用到不需要验证功能的类上。

<?php
 abstract class FR_Auth extends FR_Module
 {
  // {{{ __construct()
  function __construct()
  {
   parent::__construct();
  }
  // }}}
  // {{{ authenticate()
   abstract function authenticate();
  // }}}

  // {{{ __destruct()

   function __destruct()
   {
    parent::__destruct();
   }
  // }}}
 }

?>

module.php-所有模块的心脏

<?php
 abstract class FR_Module extends FR_Object_Web
 {
  // {{{ properties
  /**
  * $presenter
  *
  * Used in FR_Presenter::factory() to determine which presentation (view)
  * class should be used for the module.
  *
  * @author Joe Stump <joe@joestump.net>
  * @var string $presenter
  * @see FR_Presenter, FR_Presenter_common, FR_Presenter_smarty
  */
  public $presenter = 'smarty';
  /**
  * $data
  *
  * Data set by the module that will eventually be passed to the view.
  *
  * @author Joe Stump <joe@joestump.net>
  * @var mixed $data Module data
  * @see FR_Module::set(), FR_Module::getData()
  */

  protected $data = array();

  /**
  * $name
  *
  * @author Joe Stump <joe@joestump.net>
  * @var string $name Name of module class
  */

  public $name;

  /**
  * $tplFile
  *
  * @author Joe Stump <joe@joestump.net>
  * @var string $tplFile Name of template file
  * @see FR_Presenter_smarty
  */

  public $tplFile;

  /**
  * $moduleName
  *
  * @author Joe Stump <joe@joestump.net>
  * @var string $moduleName Name of requested module
  * @see FR_Presenter_smarty
  */

  public $moduleName = null;
  /**
  * $pageTemplateFile
  *
  * @author Joe Stump <joe@joestump.net>
  * @var string $pageTemplateFile Name of outer page template
  */

  public $pageTemplateFile = null;
  // }}}

  // {{{ __construct()
  /**
  * __construct
  *
  * @author Joe Stump <joe@joestump.net>
  */

  public function __construct()
  {
   parent::__construct();
   $this->name = $this->me->getName();
   $this->tplFile = $this->name.'.tpl';
  }

  // }}}
  // {{{ __default()

  /**
  * __default
  *
  * This function is ran by the controller if an event is not specified
  * in the user's request.
  *
  * @author Joe Stump <joe@joestump.net>
  */

  abstract public function __default();
  // }}}
  // {{{ set($var,$val)

  /**
  * set
  *
  * Set data for your module. This will eventually be passed toe the
  * presenter class via FR_Module::getData().
  *
  * @author Joe Stump <joe@joestump.net>
  * @param string $var Name of variable
  * @param mixed $val Value of variable
  * @return void
  * @see FR_Module::getData()
  */

  protected function set($var,$val) {
   $this->data[$var] = $val;
  }
  // }}}
  // {{{ getData()

  /**
  * getData
  *
  * Returns module's data.
  *
  * @author Joe Stump <joe@joestump.net>
  * @return mixed
  * @see FR_Presenter_common
  */

  public function getData()
  {
   return $this->data;
  }
  // }}}
  // {{{ isValid($module)

  /**
  * isValid
  *
  * Determines if $module is a valid framework module. This is used by
  * the controller to determine if the module fits into our framework's
  * mold. If it extends from both FR_Module and FR_Auth then it should be
  * good to run.
  *
  * @author Joe Stump <joe@joestump.net>
  * @static
  * @param mixed $module
  * @return bool
  */

  public static function isValid($module)
  {
   return (is_object($module) &&
   $module instanceof FR_Module &&
   $module instanceof FR_Auth);
  }
  // }}}
  // {{{ __destruct()

  public function __destruct()
  {
   parent::__destruct();
  }
  // }}}
 }
?>

presenter.php-表述层的核心。

<?php
 class FR_Presenter
 {
  // {{{ factory($type,FR_Module $module)
  /**
  * factory  
  *
  * @author Joe Stump <joe@joestump.net>
  * @access public
  * @param string $type Presentation type (our view)
  * @param mixed $module Our module, which the presenter will display
  * @return mixed PEAR_Error on failure or a valid presenter
  * @static
  */

  static public function factory($type,FR_Module $module)
  {
   $file = FR_BASE_PATH.'/includes/Presenter/'.$type.'.php';
   if (include($file)) {
    $class = 'FR_Presenter_'.$type;
    if (class_exists($class)) {
     $presenter = new $class($module);
     if ($presenter instanceof FR_Presenter_common) {
      return $presenter;
     }
     return PEAR::raiseError('Invalid presentation class: '.$type);
    }
    return PEAR::raiseError('Presentation class not found: '.$type);
   }
   return PEAR::raiseError('Presenter file not found: '.$type);
  }
  // }}}
 }

?>

  下一篇里,我将介绍控制器(MVC中的Controller,本文的index.php)的构造。第三篇里,我将介绍表述层(MVC里面的View)。第四篇里,我将用具体模块为例建立一个应用(MVC里面的Module或Model)。
성명
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.
PHP vs. Python : 차이점 이해PHP vs. Python : 차이점 이해Apr 11, 2025 am 12:15 AM

PHP와 Python은 각각 고유 한 장점이 있으며 선택은 프로젝트 요구 사항을 기반으로해야합니다. 1.PHP는 간단한 구문과 높은 실행 효율로 웹 개발에 적합합니다. 2. Python은 간결한 구문 및 풍부한 라이브러리를 갖춘 데이터 과학 및 기계 학습에 적합합니다.

PHP : 죽어 가거나 단순히 적응하고 있습니까?PHP : 죽어 가거나 단순히 적응하고 있습니까?Apr 11, 2025 am 12:13 AM

PHP는 죽지 않고 끊임없이 적응하고 진화합니다. 1) PHP는 1994 년부터 새로운 기술 트렌드에 적응하기 위해 여러 버전 반복을 겪었습니다. 2) 현재 전자 상거래, 컨텐츠 관리 시스템 및 기타 분야에서 널리 사용됩니다. 3) PHP8은 성능과 현대화를 개선하기 위해 JIT 컴파일러 및 기타 기능을 소개합니다. 4) Opcache를 사용하고 PSR-12 표준을 따라 성능 및 코드 품질을 최적화하십시오.

PHP의 미래 : 적응 및 혁신PHP의 미래 : 적응 및 혁신Apr 11, 2025 am 12:01 AM

PHP의 미래는 새로운 기술 트렌드에 적응하고 혁신적인 기능을 도입함으로써 달성 될 것입니다. 1) 클라우드 컴퓨팅, 컨테이너화 및 마이크로 서비스 아키텍처에 적응, Docker 및 Kubernetes 지원; 2) 성능 및 데이터 처리 효율을 향상시키기 위해 JIT 컴파일러 및 열거 유형을 도입합니다. 3) 지속적으로 성능을 최적화하고 모범 사례를 홍보합니다.

PHP의 초록 클래스 또는 인터페이스에 대한 특성과 언제 특성을 사용 하시겠습니까?PHP의 초록 클래스 또는 인터페이스에 대한 특성과 언제 특성을 사용 하시겠습니까?Apr 10, 2025 am 09:39 AM

PHP에서, 특성은 방법 재사용이 필요하지만 상속에 적합하지 않은 상황에 적합합니다. 1) 특성은 클래스에서 다중 상속의 복잡성을 피할 수 있도록 수많은 방법을 허용합니다. 2) 특성을 사용할 때는 대안과 키워드를 통해 해결할 수있는 방법 충돌에주의를 기울여야합니다. 3) 성능을 최적화하고 코드 유지 보수성을 향상시키기 위해 특성을 과도하게 사용해야하며 단일 책임을 유지해야합니다.

DIC (Dependency Injection Container) 란 무엇이며 PHP에서 사용하는 이유는 무엇입니까?DIC (Dependency Injection Container) 란 무엇이며 PHP에서 사용하는 이유는 무엇입니까?Apr 10, 2025 am 09:38 AM

의존성 주입 컨테이너 (DIC)는 PHP 프로젝트에 사용하기위한 객체 종속성을 관리하고 제공하는 도구입니다. DIC의 주요 이점에는 다음이 포함됩니다. 1. 디커플링, 구성 요소 독립적 인 코드는 유지 관리 및 테스트가 쉽습니다. 2. 유연성, 의존성을 교체 또는 수정하기 쉽습니다. 3. 테스트 가능성, 단위 테스트를 위해 모의 객체를 주입하기에 편리합니다.

SPL SplfixedArray 및 일반 PHP 어레이에 비해 성능 특성을 설명하십시오.SPL SplfixedArray 및 일반 PHP 어레이에 비해 성능 특성을 설명하십시오.Apr 10, 2025 am 09:37 AM

SplfixedArray는 PHP의 고정 크기 배열로, 고성능 및 메모리 사용이 필요한 시나리오에 적합합니다. 1) 동적 조정으로 인한 오버 헤드를 피하기 위해 생성 할 때 크기를 지정해야합니다. 2) C 언어 배열을 기반으로 메모리 및 빠른 액세스 속도를 직접 작동합니다. 3) 대규모 데이터 처리 및 메모리에 민감한 환경에 적합하지만 크기가 고정되어 있으므로주의해서 사용해야합니다.

PHP는 파일 업로드를 어떻게 단단히 처리합니까?PHP는 파일 업로드를 어떻게 단단히 처리합니까?Apr 10, 2025 am 09:37 AM

PHP는 $ \ _ 파일 변수를 통해 파일 업로드를 처리합니다. 보안을 보장하는 방법에는 다음이 포함됩니다. 1. 오류 확인 확인, 2. 파일 유형 및 크기 확인, 3 파일 덮어 쓰기 방지, 4. 파일을 영구 저장소 위치로 이동하십시오.

Null Coalescing 연산자 (??) 및 Null Coalescing 할당 연산자 (?? =)은 무엇입니까?Null Coalescing 연산자 (??) 및 Null Coalescing 할당 연산자 (?? =)은 무엇입니까?Apr 10, 2025 am 09:33 AM

JavaScript에서는 NullCoalescingOperator (??) 및 NullCoalescingAssignmentOperator (?? =)를 사용할 수 있습니다. 1. 2. ??= 변수를 오른쪽 피연산자의 값에 할당하지만 변수가 무효 또는 정의되지 않은 경우에만. 이 연산자는 코드 로직을 단순화하고 가독성과 성능을 향상시킵니다.

See all articles

핫 AI 도구

Undresser.AI Undress

Undresser.AI Undress

사실적인 누드 사진을 만들기 위한 AI 기반 앱

AI Clothes Remover

AI Clothes Remover

사진에서 옷을 제거하는 온라인 AI 도구입니다.

Undress AI Tool

Undress AI Tool

무료로 이미지를 벗다

Clothoff.io

Clothoff.io

AI 옷 제거제

AI Hentai Generator

AI Hentai Generator

AI Hentai를 무료로 생성하십시오.

인기 기사

R.E.P.O. 에너지 결정과 그들이하는 일 (노란색 크리스탈)
3 몇 주 전By尊渡假赌尊渡假赌尊渡假赌
R.E.P.O. 최고의 그래픽 설정
3 몇 주 전By尊渡假赌尊渡假赌尊渡假赌
R.E.P.O. 아무도들을 수없는 경우 오디오를 수정하는 방법
3 몇 주 전By尊渡假赌尊渡假赌尊渡假赌
WWE 2K25 : Myrise에서 모든 것을 잠금 해제하는 방법
3 몇 주 전By尊渡假赌尊渡假赌尊渡假赌

뜨거운 도구

에디트플러스 중국어 크랙 버전

에디트플러스 중국어 크랙 버전

작은 크기, 구문 강조, 코드 프롬프트 기능을 지원하지 않음

SublimeText3 Linux 새 버전

SublimeText3 Linux 새 버전

SublimeText3 Linux 최신 버전

WebStorm Mac 버전

WebStorm Mac 버전

유용한 JavaScript 개발 도구

스튜디오 13.0.1 보내기

스튜디오 13.0.1 보내기

강력한 PHP 통합 개발 환경

Atom Editor Mac 버전 다운로드

Atom Editor Mac 버전 다운로드

가장 인기 있는 오픈 소스 편집기