开发自己PHP MVC框架(一)

本教程翻译自John Squibb 的Build a PHP MVC Framework in an Hour,但有所改动,原文地址:http://johnsquibb.com/tutorials

这个教程可以使大家掌握用mvc模式开发php应用的基本概念。此教程分为三个部分,现在这篇是第一部分。

现在市面上有很多流行的框架供大家使用,但是我们也可以自己动手开发一个mvc框架,采用mvc模式可以大大减少我们开发应用的时间,并且能够更好的组织项目源代码,而且其中的某些模块还可在其它项目中使用。现在我要教大家写一个简单的mvc框架。由于这个项目很简单,轻量,所以可能并不是最佳实践,也不具备安全性,还需要大家在实际应用中完善。

所用技术:php,面向对象开发方法。

开始

首先在网站根目录下建立三个文件夹

  • models
  • views
  • controllers

然后在根目录下新建一个文件:

  • index.php

现在项目结构应该像这样

§ 网站根目录

§ index.php

§ models/

§ views/

§ controllers/

index.php是整个web应用的入口点,所有的用户请求都会经过它。我们会写一些代码来把用户请求分派到相应的控制器中,这些控制器存放在controllers文件夹里。之后,我们就可以用下面的方式来实现页面跳转:

  • http://你的域名.com/index.php?page1
  • http://你的域名.com/index.php?page2
  • http://你的域名.com/index.php?page3

设置前端控制器index.php

首先在index.php中定义网站根目录和网站域名,以便在整个应用中访问。

[php] view plaincopy

  1. <?php
  2. //应用的根目录就是index.php的父目录
  3. define("SERVER_ROOT", dirname(__FILE__));
  4. //你的域名.comm 是你的服务器域名
  5. define(‘SITE_ROOT‘ , ‘http://你的域名.com‘);

定义了网站根目录后,在任何php文件中,都能很方便的引用其它目录的php文件,因为index.php是入口文件,这样就能够在整个应用中访问在它之中定义的这些变量。

设置路由器router.php(转发用户请求到相应控制器)

在controllers目录下新建一个文件,名字为“router.php",这个文件用来处理所有页面请求。想像一下你家里的路由器,它负责把internet路由到家中的每个电脑。router.php文件将会获取传入到index.php的页面请求,然后把请求分派给不同的控制器(controllers)。

route.php中的代码:

[php] view plaincopy

  1. <?php
  2. //获取所有请求
  3. $request = $_SERVER[‘QUERY_STRING‘];

这句代码会获取传入到应用中的请求参数。QUERY_STRING就是”?“后面的所有字符串。

  • http://你的域名.com/index.php?page1

上面的地址会在代码中得到”page1&action=login“,为了把page1和后面的参数分开,我们需要在route.php中继续加入下列代码:

[php] view plaincopy

  1. //解析$request变量,得到用户请求的页面(page1)和其它GET变量(&分隔的变量)如一个请求http://你的域名.com/index.php?page1&article=buildawebsite,则被解析为array("page1", "article=buildawebsite")
  2. $parsed = explode(‘&‘ , $request);
  3. //用户请求的页面,如上面的page1,为$parsed第一个变量,shift之后,数组为array("article=buildawebsite")
  4. $page = array_shift($parsed);
  5. //剩下的为GET变量,把它们解析出来
  6. $getVars = array();
  7. foreach ($parsed as $argument)
  8. {
  9. //用"="分隔字符串,左边为变量,右边为值
  10. list($variable , $value) = split(‘=‘ , $argument);
  11. $getVars[$variable] = $value;
  12. }
  13. //这是测试语句,一会儿会删除
  14. print "The page your requested is ‘$page‘";
  15. print ‘<br/>‘;
  16. $vars = print_r($getVars, TRUE);
  17. print "The following GET vars were passed to the page:<pre>".$vars."</pre>";

现在我们需要在index.php中引入route.php

[php] view plaincopy

  1. <?php
  2. /**
  3. * 定义文档路径
  4. */
  5. define("SERVER_ROOT", dirname(__FILE__));
  6. define(‘SITE_ROOT‘ , ‘http://你的域名.com‘);
  7. /**
  8. * 引入router.php
  9. */
  10. require_once(SERVER_ROOT . ‘/controllers/‘ . ‘router.php‘);
  11. ?>

如果顺利的话,你可以打开浏览器输入:

  • http://你的域名.com/index.php?news&article=howtobuildaframework

我们会看到如下输出

[html] view plaincopy

  1. The page you requested is ‘news‘
  2. The following GET vars were passed to the page:
  3. Array
  4. (
  5. [article] => howtobuildaframework
  6. )

如果没有上述输出,请检查你的服务器配置是否正确,并检查代码是否有错误。

现在来让我们添加一个页面到我们的网站里,这样就可以让router.php来产生一个页面,而不是直接输出上面的信息。

创建一个控制器(controller)

在controllers文件夹里新建一个文件名为“news.php",定义如下的类:

[php] view plaincopy

  1. <?php
  2. /**
  3. * 这个文件处理文章的查询,并提供文章
  4. */
  5. class News_Controller
  6. {
  7. /**
  8. * $template变量会保存与此控制器相关的"view(视图)"的文件名,不包括.php后缀
  9. */
  10. public $template = ‘news‘;
  11. /**
  12. * 此方法为route.php默认调用
  13. *
  14. * @param array $getVars 传入到index.php的GET变量数组
  15. */
  16. public function main(array $getVars)
  17. {
  18. //测试代码,以后会删除
  19. print "We are in news!";
  20. print ‘<br/>‘;
  21. $vars = print_r($getVars, TRUE);
  22. print
  23. (
  24. "The following GET vars were passed to this controller:" .
  25. "<pre>".$vars."</pre>"
  26. );
  27. }
  28. }


注意我们把route.php中的测试代码复制过来了,并做了一些修改,我们把它放置在main函数里。现在让我们来修改route.php中的代码:

[php] view plaincopy

  1. <?php
  2. /**
  3. * 此文件会把所有的传入参数分派到相应的控制器中
  4. */
  5. //获取请求参数
  6. $request = $_SERVER[‘QUERY_STRING‘];
  7. //解析请求页面和其它GET变量
  8. $parsed = explode(‘&‘ , $request);
  9. //页面是第一个元素
  10. $page = array_shift($parsed);
  11. //剩余的为GET变量,也把它们解析出来
  12. $getVars = array();
  13. foreach ($parsed as $argument)
  14. {
  15. //split GET vars along ‘=‘ symbol to separate variable, values
  16. list($variable , $value) = split(‘=‘ , $argument);
  17. $getVars[$variable] = $value;
  18. }
  19. //构成控制器文件路径
  20. $target = SERVER_ROOT . ‘/controllers/‘ . $page . ‘.php‘;
  21. //引入目标文件
  22. if (file_exists($target))
  23. {
  24. include_once($target);
  25. //修改page变量,以符合命名规范(如$page="news",我们的约定是首字母大写,控制器的话就在后面加上“<strong>_Controller”</strong>,即News_Controller)
  26. $class = ucfirst($page) . ‘_Controller‘;
  27. //初始化对应的类
  28. if (class_exists($class))
  29. {
  30. $controller = new $class;
  31. }
  32. else
  33. {
  34. //类的命名正确吗?
  35. die(‘class does not exist!‘);
  36. }
  37. }
  38. else
  39. {
  40. //不能在controllers找到此文件
  41. die(‘page does not exist!‘);
  42. }
  43. //一但初始化了控制器,就调用它的默认函数main();
  44. //把get变量传给它
  45. $controller->main($getVars);?>

再次访问http://你的域名.com/index.php?news&article=howtobuildaframework,你将会看到从News_Controller打印出来的信息。注意,我们现在用die()来处理错误,我们可以用其它更好的错误处理来规制它,但现在使用die()足够了,试试访问其它页面如http://你的域名.com/index.php?books,你会看到"page does not exist!"错误。创建一个Model(模型)完善News_Controller。假设我们有一些新闻片段来供读者阅读,那么就需要News_Controller这个控制器去调用一个模型来抓取相关的新闻片段,无论它们是存储在数据库还是文件里。在models文件夹里新建一个文件,“news.php”,代码如下:

[php] view plaincopy

  1. <?php
  2. /**
  3. * 新闻模型为新闻控制器做复杂的后台操作
  4. */
  5. class News_Model
  6. {
  7. public function __construct()
  8. {
  9. print ‘I am the news model‘;
  10. }
  11. }

现在,我们需要对新闻控制器稍做一些更改,打开controllers里的news.php,把News_Controller类的main函数的代码改为如下,这样,我们就会在“News_Model”初始化时,看到打印在屏幕上的信息:

[php] view plaincopy

  1. public function main(array $getVars)
  2. {
  3. $newsModel = new News_Model;
  4. }

现在刷新页面,你会看到:

[php] view plaincopy

  1. Fatal error: Class ‘News_Model‘ not found in /var/www/mvc/controllers/news.php on line xx

等一下,这不是我们想要的结果!我们正试图去加载一个不存在的类。那么原因就是我们并没有引入/models/news.php文件。为了解决这个问题,让们重新来看一下router.php,然后在它的顶部添加一些代码:

[php] view plaincopy

  1. //当类初始化时,自动引入相关文件
  2. function __autoload($className)
  3. {
  4. //解析文件名,得到文件的存放路径,如News_Model表示存放在models文件夹里的news.php(这里是作者的命名约定)
  5. list($filename , $suffix) = split(‘_‘ , $className);
  6. //构成文件路径
  7. $file = SERVER_ROOT . ‘/models/‘ . strtolower($filename) . ‘.php‘;
  8. //获取文件
  9. if (file_exists($file))
  10. {
  11. //引入文件
  12. include_once($file);
  13. }
  14. else
  15. {
  16. //文件不存在
  17. die("File ‘$filename‘ containing class ‘$className‘ not found.");
  18. }
  19. }

这个函数重载了PHP内置的autoload函数。当我们试图去初始化一个不存在的类时,这个‘魔术方法’允许我们拦截php所执行的动作。通过使用__autoload函数,我们能够告诉php寻找包含此类的文件的位置。假设你遵循了这篇文章中类和文件名的命名约定,那么每当你初始化一个类时,你就不必手动去引入包含此类的文件了!

保存route.php,再刷新一次浏览器,你会看到:

[html] view plaincopy

  1. I am the news model

让我们在新闻模型类里定义一些函数来提供文章。现在,我们只简单的定义了一个数组,并保存一些文章,然后提供一个函数,让控制器从中根据标题获取一篇文章。修改models/news.php:

[php] view plaincopy

  1. <?php
  2. /**
  3. * 新闻模型为新闻控制器做复杂的后台操作
  4. *
  5. */
  6. class News_Model
  7. {
  8. /**
  9. * 文章数组. key为文章标题, 值为相应的
  10. * 文章。
  11. */
  12. private $articles = array
  13. (
  14. //文章1
  15. ‘new‘ => array
  16. (
  17. ‘title‘ => ‘New Website‘ ,
  18. ‘content‘ => ‘Welcome to the site! We are glad to have you here.‘
  19. )
  20. ,
  21. //2
  22. ‘mvc‘ => array
  23. (
  24. ‘title‘ => ‘PHP MVC Frameworks are Awesome!‘ ,
  25. ‘content‘ => ‘It really is very easy. Take it from us!‘
  26. )
  27. ,
  28. //3
  29. ‘test‘ => array
  30. (
  31. ‘title‘ => ‘Testing‘ ,
  32. ‘content‘ => ‘This is just a measly test article.‘
  33. )
  34. );
  35. public function __construct()
  36. {
  37. }
  38. /**
  39. * 根据标题获取文章
  40. *
  41. * @param string $articleName
  42. *
  43. * @return array $article
  44. */
  45. public function get_article($articleName)
  46. {
  47. //从数组中获取文章
  48. $article = $this->articles[$articleName];
  49. return $article;
  50. }
  51. }?>

现在修改controllers/news.php中的main函数:

[php] view plaincopy

  1. public function main(array $getVars)
  2. {
  3. $newsModel = new News_Model;
  4. //获取一篇文章
  5. $article = $newsModel->get_article(‘test‘);
  6. print_r($article);
  7. }

现在我们并没有考虑过滤用户输入的问题,因为我们现在只是为了尽快让大家掌握PHP MVC的基本内容,所以我们现在不必太关心这些。

如果访问如下网址:

§ http://yourdomain.com/mvc/index.php?news&article=test

你会看到如下输出:

[html] view plaincopy

  1. Array ( [title] => Testing [content] => This is just a measly test article. )

创建视图(VIEW

现在我们已经有控制器和模型了,只差一个视图。视图是表现层,它是你的应用中,与用户接触最频繁的部分。之前我提到过,视图是提供与业务逻辑分离的用户接口,有很多方法可以做到这个。你可以使用模板引擎Smarty或其它类似的。你也可以写一个自己的模板引擎,但那肯定是相当艰巨的任务。最后,你可以使用原生php视图。

对于目前来说,php视图足够了。这个就像以前php与html代码混合编程一样,但是有一点不同是,我们的业务逻辑已经和视图分离了。看一下如下代码:

[php] view plaincopy

  1. <html>
  2. <head></head>
  3. <body>
  4. <h1>Welcome to Our Website!</h1>
  5. <hr/>
  6. <h2>News</h2>
  7. <h4><?=$data[‘title‘];?></h4>
  8. <p><?=$data[‘content‘];?></p>
  9. </body>
  10. </html>

注意,嵌入的php标签利用了PHP 快捷操作符。这样就能够把我们的内容直接输出到HTML里面了。在views文件夹里新建一个文件“news.php”,把上述代码拷贝进来。现在我们有了视图文件,但是我们需要一个与视图交互的方法。在models文件夹里新建一个文件“view.php”,添加如下代码:

[php] view plaincopy

  1. <?php
  2. /**
  3. * 在我们的MVC框架中,处理视图的功能
  4. */
  5. class View_Model
  6. {
  7. /**
  8. * 保存赋给视图模板的变量
  9. */
  10. private $data = array();
  11. /**
  12. * 保存视图渲染状态
  13. */
  14. private $render = FALSE;
  15. /**
  16. * 加载一个视图模板
  17. */
  18. public function __construct($template)
  19. {
  20. //构成完整文件路径
  21. $file = SERVER_ROOT . ‘/views/‘ . strtolower($template) . ‘.php‘;
  22. if (file_exists($file))
  23. {
  24. /**
  25. * 当模型对象销毁时才能渲染视图
  26. * 如果现在就渲染视图,那么我们就不能给视图模板赋予变量
  27. * 所以此处先保存要渲染的视图文件路径
  28. */
  29. $this->render = $file;
  30. }
  31. }
  32. /**
  33. * 接受从控制器赋予的变量,并保存在data数组中
  34. *
  35. * @param $variable
  36. * @param $value
  37. */
  38. public function assign($variable , $value)
  39. {
  40. $this->data[$variable] = $value;
  41. }
  42. public function __destruct()
  43. {
  44. //把类中的data数组变为该函数的局部变量,以方便在视图模板中使用
  45. $data = $this->data;
  46. //渲染视图
  47. include($this->render);
  48. }
  49. }

现在,最后一件要做的事就是从News_Controller里加载视图。修改controllers/news.php:

[php] view plaincopy

  1. <?php
  2. /*
  3. *这个文件处理文章的查询,并产生新闻文章*
  4. */
  5. class News_Controller{
  6. /**
  7. * $template变量会保存与此控制器相关的"view(视图)"的文件名,不包括.php后缀
  8. *
  9. */
  10. public $template = ‘news‘;
  11. /**
  12. * 此方法为route.php默认调用
  13. *
  14. * @param array $getVars 传入到index.php的GET变量数组
  15. */
  16. public function main(array $getVars) {
  17. $newsModel = new News_Model;
  18. //获取一片文章
  19. $article = $newsModel->get_article($getVars[‘article‘]);
  20. //创建一个视图,并传入该控制器的template变量
  21. $view = new View_Model($this->template);
  22. //把文章数据赋给视图模板
  23. $view->assign(‘title‘ , $article[‘title‘]);
  24. $view->assign(‘content‘ , $article[‘content‘]);
  25. }
  26. }
  27. ?>

再加载页面,你就能够看到你的视图模板中的变量,已经被正确的替换掉了。好了,你的简单的MVC框架已经搭建好了,下面我会继续讲《开发自己PHP MVC框架(二)》

时间: 2024-10-15 16:32:12

开发自己PHP MVC框架(一)的相关文章

web开发中的MVC框架与django框架的MTV模式

有一种程序设计模式叫MVC,核心思想:分层,解耦,分离了 数据处理 和 界面显示 的代码,使得一方代码修改了不会影响到另外一方,提高了程序的可扩展性和可维护性. MVC的全拼为Model-View-Controller,最早由TrygveReenskaug在1978年提出,是施乐帕罗奥多研究中心(Xerox PARC)在20世纪80年代为程序语言Smalltalk发明的一种软件设计模式,是为了将传统的输入(input).处理(processing).输出(output)任务运用到图形化用户交互模

手把手编写自己的PHP MVC框架实例教程

1 什么是MVC MVC模式(Model-View-Controller)是软件工程中的一种软件架构模式. MVC把软件系统分为三个基本部分:模型(Model).视图(View)和控制器(Controller). PHP中MVC模式也称Web MVC,从上世纪70年代进化而来. MVC的目的是实现一种动态的程序设计,便于后续对程序的修改和扩展简化,并且使程序某一部分的重复利用成为可能. 除此之外,此模式通过对复杂度的简化,使程序结构更加直观. MVC各部分的职能: 模型Model – 管理大部分

使用PHP搭建自己的MVC框架

一.什么是MVC MVC模式(Model-View-Controller)是软件工程中的一种软件架构模式,把软件系统分为三个基本部分:模型(Model).视图(View)和控制器(Controller). MVC模式的目的是实现一种动态的程序设计,使后续对程序的修改和扩展简化,并且使程序某一部分的重复利用成为可能.除此之外,此模式通过对复杂度的简化,使程序结构更加直观.软件系统通过对自身基本部份分离的同时也赋予了各个基本部分应有的功能.专业人员可以通过自身的专长分组: (控制器Controlle

编写自己的PHP MVC框架笔记

1.MVC MVC模式(Model-View-Controller)是软件工程中的一种软件架构模式,把软件系统分为三个基本部分:模型(Model).视图(View)和控制器(Controller). PHP中MVC模式也称Web MVC,从上世纪70年代进化而来.MVC的目的是实现一种动态的程序设计,便于后续对程序的修改和扩展简化,并且使程序某一部分的重复利用成为可能.除此之外,此模式通过对复杂度的简化,使程序结构更加直观.软件系统通过对自身基本部份分离的同时,也赋予了各个基本部分应有的功能.

开发自己的PHP MVC框架(一)

这个教程能够使大家掌握用mvc模式开发php应用的基本概念.此教程分为三个部分.如今这篇是第一部分. 如今市面上有非常多流行的框架供大家使用.可是我们也能够自己动手开发一个mvc框架.採用mvc模式能够大大降低我们开发应用的时间,并且能够更好的组织项目源码,并且当中的某些模块还可在其他项目中使用.如今我要教大家写一个简单的mvc框架.因为这个项目非常easy,轻量.所以可能并非最佳实践,也不具备安全性.还须要大家在实际应用中完好. 所用技术:php,面向对象开发方法. 開始 首先在站点根文件夹下

分享一实战性开源MVC框架&lt;Linux、Windows跨平台开发so easy&gt;

一.引子   开源地址 http://pan.baidu.com/s/1slCMspR .NET Core微软还在发力,但作为商用还有一段距离,很多开发库尚不能用于.NET Core,因此我们还没有看到一个大型的使用案例, 这时大家或许想到Mono部署ASP.NET MVC,但部署过的人必然知道:这不是一个容易的工作,需要各种工作,至今也没有几个人容易地完成这项工作. .NET Core作为微软的新星,但还不够成熟强大,在其发出耀眼光辉之前,我们分享一实战性强的开源MVC框架: Moon.Mvc

开发自己的PHP MVC框架

本教程翻译自John Squibb 的Build a PHP MVC Framework in an Hour,但有所改动,原文地址:http://johnsquibb.com/tutorials 这个教程可以使大家掌握用mvc模式开发php应用的基本概念.此教程分为三个部分,现在这篇是第一部分. 现在市面上有很多流行的框架供大家使用,但是我们也可以自己动手开发一个mvc框架,采用mvc模式可以大大减少我们开发应用的时间,并且能够更好的组织项目源代码,而且其中的某些模块还可在其它项目中使用.现在

unity3D 游戏开发之工程代码框架设计思路MVC

unity3D 游戏开发之工程代码框架设计思路MVC 设计目的 1.使工程结构更规范. 2.提高代码可读性,封装性,拓展性 3.提高工作效率. 正文内容: 1.Frame的组成结 (1)视图层(View) (2) 控制层(Control) (3)数据层(Model) 整个Frame是由这三个部分组成,每一层管理属于自己的逻辑,核心思想是游戏逻辑和UI 逻辑独立开.目前遇到的项目工程大多数View和Control逻辑都写在一起,这样后期修改 和维护效率会很低,因为耦合性很高而View又是经常要修改

框架开发之——AngularJS+MVC+Routing开发步骤总结——5.14

1.延续MVC的观念:包括路由映射的编写,Controller的内容,具体View页面js的分离. 2.结合AngularJS做前端,后端使用Node.Js的写法,引入MVC框架,进行快速的开发. 步骤总结如下: 1.view页面修改,文件夹名称,文件名称,名称的大小写规范,在base文件夹中的main.ejs文件中记得注册. 注意使用angular.js进行前端页面的编写,以及相应页面代码的编写,相关angular.Js用法的收集. 2.view对应js文件的编写,文件夹的名称,文件名称,名称