国产xxxx99真实实拍_久久不雅视频_高清韩国a级特黄毛片_嗯老师别我我受不了了小说

資訊專欄INFORMATION COLUMN

用PHP搭建自己的MVC框架

cyrils / 1997人閱讀

摘要:但是模型中數據的變化一般會通過一種刷新機制被公布。它處理事件并作出響應。事件包括用戶的行為和數據模型上的改變。三開始開發自己的框架在開始開發前,讓我們先來把項目建立好,假設我們建立的項目為,那么接下來的第一步就是把目錄結構先設置好。

現如今市面上有許多PHP框架,像 ThinkPHP、YII、Laravel,那么如何自己搭建一個PHP框架呢?這里有一篇博文寫的非常好,特意轉載過來,供朋友們研究,原文請戳詳情。

一、什么是MVC

MVC 模式(Model-View-Controller)是軟件工程中的一種軟件架構模式,把軟件系統分為三個基本部分:模型(Model)、視圖(View)和控制器(Controller)。

MVC 模式的目的是實現一種動態的程序設計,使后續對程序的修改和擴展簡化,并且使程序某一部分的重復利用成為可能。除此之外,此模式通過對復雜度的簡化,使程序結構更加直觀。軟件系統通過對自身基本部份分離的同時也賦予了各個基本部分應有的功能。專業人員可以通過自身的專長分組:

(控制器Controller)- 負責轉發請求,對請求進行處理。

(視圖View) – 界面設計人員進行圖形界面設計。

(模型Model) – 程序員編寫程序應有的功能(實現算法等等)、數據庫專家進行數據管理和數據庫設計(可以實現具體的功能)。

模型(Model): “數據模型”(Model)用于封裝與應用程序的業務邏輯相關的數據以及對數據的處理方法。“模型”有對數據直接訪問的權力,例如對數據庫的訪問。“模型”不依賴“視圖”和“控制器”,也就是說,模型不關心它會被如何顯示或是如何被操作。但是模型中數據的變化一般會通過一種刷新機制被公布。為了實現這種機制,那些用于監視此模型的視圖必須事先在此模型上注冊,從而,視圖可以了解在數據模型上發生的改變。

視圖(View): 視圖層能夠實現數據有目的的顯示(理論上,這不是必需的)。在視圖中一般沒有程序上的邏輯。為了實現視圖上的刷新功能,視圖需要訪問它監視的數據模型(Model),因此應該事先在被它監視的數據那里注冊。

控制器(Controller): 控制器起到不同層面間的組織作用,用于控制應用程序的流程。它處理事件并作出響應。“事件”包括用戶的行為和數據模型上的改變。

二、為什么要自己開發MVC框架

網絡上有大量優秀的 MVC 框架可供使用,本教程并不是為了開發一個全面的、終極的 MVC 框架解決方案,而是將它看作是一個很好的從內部學習 PHP 的機會,在此過程中,你將學習面向對象編程和設計模式,并學習到開放中的一些注意事項。

更重要的是,你可以完全控制你的框架,并將你的想法融入到你開發的框架中。雖然不一定是做好的,但是你可以按照你的方式去開發功能和模塊。

三、開始開發自己的MVC框架

在開始開發前,讓我們先來把項目建立好,假設我們建立的項目為 Frame,那么接下來的第一步就是把目錄結構先設置好。

雖然在這個教程中不會使用到上面的所有的目錄,但是為了以后程序的可拓展性,在一開始就把程序目錄設置好使非常必要的。下面就具體說說每個目錄的作用:

application – 存放程序代碼
config – 存放程序配置或數據庫配置
db – 用來存放數據庫備份內容
library – 存放框架代碼
public – 存放靜態文件
scripts – 存放命令行工具
tmp – 存放臨時數據
在目錄設置好以后,我們接下來就要來頂一下一些代碼的規范:

MySQL的表名需小寫并采用復數形式,如items,cars
模塊名(Models)需首字母大寫,并采用單數模式,如Item,Car
控制器(Controllers)需首字母大寫,采用復數形式并在名稱中添加“Controller”,如 ItemsController, CarsController
視圖(Views)采用復數形式,并在后面添加行為作為文件,如:items/view.php, cars/buy.php

上述的一些規則是為了能在程序鐘更好的進行互相的調用。接下來就開始真正的編碼了。

第一步將所有的的請求都重定向到 public 目錄下,解決方案是在 Frame 文件下添加一個.htaccesss文件,文件內容為:


RewriteEngine on
RewriteRule    ^$    public/    [L]
RewriteRule    (.*) public/$1    [L]

在我們把所有的請求都重定向到 public 目錄下以后,我們就需要將所有的數據請求都再重定向到 public 下的 index.php 文件,于是就需要在public文件夾下也新建一個.htaccess文件,文件內容為:


RewriteEngine On
#如果文件存在就直接訪問目錄不進行RewriteRule
RewriteCond %{REQUEST_FILENAME} !-f
#如果目錄存在就直接訪問目錄不進行RewriteRule
RewriteCond %{REQUEST_FILENAME} !-d
#將所有其他URL重寫到 index.php/URL
RewriteRule ^(.*)$ index.php?url=$1 [PT,L]

這么做的主要原因有:

可以使程序有一個單一的入口,將所有除靜態程序以外的程序都重定向到index.php上;

可以用來生成利于SEO的URL,想要更好的配置URL,后期可能會需要URL路由,這里先不做介紹了。

做完上面的操作,就應該知道我們需要做什么了,沒錯!在public目錄下添加 index.php 文件,文件內容為:


注意上面的PHP代碼中,并沒有添加PHP結束符號”?>”,這么做的主要原因是:對于只包含PHP代碼的文件,結束標志(“?>”)最好不存在,PHP自身并不需要結束符號,不添加結束符號可以很大程度上防止末尾被添加額外的注入內容,讓程序更加安全。

在index.php中,我們對 library 文件夾下的 bootstrap.php 發起了請求,那么bootstrap.php 這個啟動文件中到底會包含哪些內容呢?


以上文件都可以直接在index.php文件中引用,我們這么做的原因是為了在后期管理和拓展中更加的方便,所以把需要在一開始的時候就加載運行的程序統一放到一個多帶帶的文件中引用。

先來看看config文件下的 config .php 文件,該文件的主要作用是設置一些程序的配置項及數據庫連接等,主要內容為:


應該說config.php涉及到的內容并不多,不過是一些基礎數據的一些設置,再來看看library下的共用文件 shared.php 應該怎么寫。

的method屬性為post的時候應該用$_POST["user_name"]和$_POST["user_pass"]) 
 * 當register_globals=On的時候,下一個程序可以直接使用$user_name和$user_pass來接受值。
 * 顧名思義,register_globals的意思就是注冊為全局變量,所以當On的時候,傳遞過來的值會被直接的注冊為全局變量直接使用,而Off的
 * 時候,我們需要到特定的數組里去得到它。所以,碰到上邊那些無法得到值的問題的朋友應該首先檢查一下你的register_globals的設置和
 * 你獲取值的方法是否匹配。
 *
 * 那我們為什么要使用Off呢?
 * 原因有2: 
 * 1、php以后的新版本默認都用Off,雖然你可以設置它為On,但是當你無法控制服務器的時候,你的代碼的兼容性就成為一個大問題,所以
 *,你最好從現在就開始用Off的風格開始編程 2、這里有兩篇文章介紹為什么要Off而不用On 
 */
function unregisterGlobals()
{
    if(ini_get("register_globals"))
    {
         $array = array("_SESSION","_POST","_GET","_COOKIE","_REQUEST","_SERVER","_ENV","_FILES");
           foreach ($array as $value)
           {
                 // $GLOBALS 可以打印所有的全局變量
               foreach ($GLOBALS[$value] as $key => $var)
               {
                  if ($var === $GLOBALS[$key])
                  {
                      unset($GLOBALS[$key]);
                  }
               }
           }
    }

}

/**
 * 請求方法,主要目的拆分URL請求
 *
 * 相關學習:
 * 調用回調函數,并把一個數組參數作為回調函數的參數
 * call_user_func_array(array($obj, $method_name), $params);
 * call_user_method_array ( string $method_name , object &$obj , array $params )
 */
function callHook()
{
    global $url;

    $urlArray   = array();
    $urlArray   = explode("/", $url);

    // 獲取控制器 controller
    $controller = $urlArray[0];

    // 刪除第一個元素
    array_shift($urlArray);

    // 獲取動作 Action
    $action = $urlArray[0];

    array_shift($urlArray);

    $queryString = $urlArray;
    $controllerName = $controller;
    $controller = ucwords($controller);
    $model = rtrim($controller, "s");

    $controller .= "Controller";

    $dispatch = new $controller($model, $controllerName, $action);
    if((int)method_exists($controller, $action))  // 檢查類的方法是否存在(可以傳遞類名或對象名)
    {
        call_user_func_array(array($dispatch, $action), $queryString);
    }
    else
    {
        /* 生成錯誤代碼 */
        exit("該控制器不存在:". $controller);
    }
}

/**
 * 自動加載控制器和模型
 */
function __autoload($className)
{
    
    if(file_exists(ROOT . DS . "library" . DS . strtolower($className) . ".class.php"))
    {
        // 加載核心庫 library
        require_once(ROOT . DS . "library" . DS . strtolower($className) . ".class.php");
    } 
    else if(file_exists(ROOT . DS . "application" . DS . "controllers" . DS . strtolower($className) . ".php"))
    {
        // 加載控制器 application 目錄下
         require_once(ROOT . DS . "application" . DS . "controllers" . DS . strtolower($className) . ".php");
    }
    else if(file_exists(ROOT . DS . "application" . DS . "models" . DS . strtolower($className) . ".php"))
    {
        // 加載控制器 application 目錄下
         require_once(ROOT . DS . "application" . DS . "models" . DS . strtolower($className) . ".php");
    }
    else
    {
         /* 生成錯誤代碼 */
        exit("類名不存在:" . $className);
    }

}


 setReporting();
 removeMagicQuotes();
 unregisterGlobals();
 callHook();

接下來的操作就是在library中建立程序所需要的基類,包括控制器、模型和視圖的基類

新建控制器基類為controller.class.php,控制器的主要功能就是總調度,具體具體內容如下:

_controller =  $controller;
            $this->_action     =  $action;
            $this->_model      =  $model;
            $this->$model      =& new $model;  // php默認為傳值傳遞,這里默認為地址傳遞,即引用同一個對象
            $this->_template   =& new Template($controller, $action);
        }

        function set($name,$value)
        {
            $this->_template->set($name,$value);
        }

        function __destruct()
        {
            // 放在析構函數中是為了在釋放實例化的時候執行下面的方法
            $this->_template->render();
        }
    }

新建控制器基類為 model.class.php,考慮到模型需要對數據庫進行處理,所以可以新建一個數據庫基類 sqlquery.class.php,模型去繼承 sqlquery.class.php

新建 sqlquery.class.php,代碼如下:

_dbHandle = @mysql_connect($address, $account, $pwd);

            if ($this->_dbHandle != 0)
            {
               if (mysql_select_db($name, $this->_dbHandle))
               {
                   return 1;
               }
               else
               {
                   return 0;
               }
            }
            else
            {
               return 0;
            }
         }

        /** 中斷數據庫連接 **/
        function disconnect()
        {
             if (@mysql_close($this->_dbHandle) != 0)
             {
                 return 1;
             }
             else
             {
                 return 0;
             }
        }

        /** 查詢所有數據表內容 **/
        function selectAll()
        {
            $query = "select * from `".$this->_table."`";
            return $this->query($query);
        }

        /** 查詢數據表指定列內容 **/
        function select($id)
        {
            $query = "select * from `".$this->_table."` where `id` = "".mysql_real_escape_string($id).""";
            return $this->query($query, 1);
        }

        /** 自定義SQL查詢語句 **/
        function query($query, $singleResult = 0)
        {
            $this->_result = mysql_query($query, $this->_dbHandle);
            if (preg_match("/select/i", $query))
            {
                 $result = array();
                 $table  = array();
                 $field  = array();
                 $tempResults = array();
                 $numOfFields = mysql_num_fields($this->_result);

                 for ($i = 0; $i < $numOfFields; ++$i)
                 {
                      array_push($table, mysql_field_table($this->_result, $i));
                      array_push($field, mysql_field_name($this->_result, $i));
                 }
                 while ($row = mysql_fetch_row($this->_result))
                 {
                     for ($i = 0;$i < $numOfFields; ++$i) {
                        $table[$i] = trim(ucfirst($table[$i]),"s");
                        $tempResults[$table[$i]][$field[$i]] = $row[$i];
                     }
                     if ($singleResult == 1) {
                         mysql_free_result($this->_result);
                         return $tempResults;
                     }
                     array_push($result,$tempResults);
                 }
                 mysql_free_result($this->_result);
                 return($result);
             }
          }

         /** 返回結果集行數 **/
        function getNumRows()
        {
            return mysql_num_rows($this->_result);
        }

        /** 釋放結果集內存 **/
        function freeResult()
        {
            mysql_free_result($this->_result);
        }

       /** 返回MySQL操作錯誤信息 **/
       function getError()
       {
           return mysql_error($this->_dbHandle);
       }
    }

新建 model.class.php,代碼如下:

connect(DB_HOST, DB_USER, DB_PASSWORD, DB_NAME);
            $this->_model = get_class($this);
            $this->_table = strtolower($this->_model)."s";
        }

        function __destruct()
        {

        }
    }

新建視圖基類為template.class.php,具體代碼如下:

_controller = $controller;
           $this->_action =$action;
       }

       /* 設置變量 */
       function set($name,$value)
       {
            $this->variables[$name] = $value;
       }

       /* 顯示模板 */
       function render()
       {
           extract($this->variables);
           if (file_exists(ROOT.DS. "application" .DS. "views" .DS. $this->_controller .DS. "header.php")) {
               include(ROOT.DS. "application" .DS. "views" .DS. $this->_controller .DS. "header.php");
           } else {
               include(ROOT.DS. "application" .DS. "views" .DS. "header.php");
           }
           include (ROOT.DS. "application" .DS. "views" .DS. $this->_controller .DS. $this->_action . ".php");
           if (file_exists(ROOT.DS. "application" .DS. "views" .DS. $this->_controller .DS. "footer.php")) {
               include (ROOT.DS. "application" .DS. "views" .DS. $this->_controller .DS. "footer.php");
           } else {
               include (ROOT.DS. "application" .DS. "views" .DS. "footer.php");
           }
        }
    }
四、框架應用

做完了以上這么多操作,基本上整個MVC框架已經出來了,下面就該制作我們的站點了。我們要做的站點其實很簡單,一個ToDo程序。

首先是在我們的 /application/controller/ 目錄下面新建一個站點控制器類為ItemsController,命名為itemscontroller.php,內容為:

set("title",$name." - My Todo List App");
           $this->set("todo",$this->Item->select($id));
       }
       function viewall() {
           $this->set("title","All Items - My Todo List App");
           $this->set("todo",$this->Item->selectAll());
       }
       function add() {
           $todo = $_POST["todo"];
           $this->set("title","Success - My Todo List App");
           $this->set("todo",$this->Item->query("insert into items (item_name) values ("".mysql_real_escape_string($todo)."")"));
       }
       function delete($id) {
           $this->set("title","Success - My Todo List App");
           $this->set("todo",$this->Item->query("delete from items where id = "".mysql_real_escape_string($id)."""));
       }
    }

接下來就是先建站點的模型,在我們的/application/model/ 目錄下面先建一個站點模型類為Item,內容直接繼承Model,代碼如下:


最后一步是設置我們站點的視圖部分,我們現在/application/views/目錄下新建一個items的文件夾,再在items文件夾下建立與控制器重Action相同的文件,分別為view.php,viewall.php,add.php,delete.php,考慮到這么頁面中可能需要共用頁首和頁尾,所以再新建兩個文件,命名為header.php,footer.php,每個文件的代碼如下:

view.php文件:查看單條待處理事務

"> Delete this item

viewall.php文件:查看所有待處理事務



/">

add.php文件:添加待處理事務

Todo successfully added. Click here to go back.

delete.php文件:刪除事務

Todo successfully deleted. Click here to go back.

header.php:頁首文件



<?php echo $title?>



My Todo-List App

footer.php:頁尾文件


當然還有一個必不可少的操作就是在數據中中建立一張表,具體代碼如下:

CREATE TABLE IF NOT EXISTS `items` (
    `id` int(11) NOT NULL AUTO_INCREMENT,
    `item_name` varchar(255) NOT NULL,
    PRIMARY KEY (`id`)
) ENGINE=MyISAM DEFAULT CHARSET=latin1 AUTO_INCREMENT=17 ;

至此一個使用MVC開發的網站就開發完成了,你現在可以通過訪問http://localhost/Frame/items/... 查看新建的站點。

框架核心小結:

1、單一入口原則

在項目Frame 下寫.htaccess 文件指定到 pulibc 目錄, 然后在 public 目錄下重寫入口模塊 .htaccess 指定到 index.php 文件

這樣只要訪問該項目,就會指定到 Frame/public/index.php 目錄下

2、加載初始配置文件

定義根路徑,分隔符,等常量
define("DEBUG", TRUE);

3、封裝核心類
sql.php
model.php (該類繼承sql.php)
controller.php

4、其他的類都繼承核心類


注:本文為轉載,原文地址:請戳這里

文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。

轉載請注明本文地址:http://specialneedsforspecialkids.com/yun/23045.html

相關文章

  • 后端API從入門到放棄指北

    摘要:菜鳥教程框架中文手冊入門目標使用搭建通過對數據增刪查改沒了純粹占行用的拜 后端API入門學習指北 了解一下一下概念. RESTful API標準] 所有的API都遵循[RESTful API標準]. 建議大家都簡單了解一下HTTP協議和RESTful API相關資料. 阮一峰:理解RESTful架構 阮一峰:RESTful API 設計指南 RESTful API指南 依賴注入 D...

    Jeffrrey 評論0 收藏0
  • 后端API從入門到放棄指北

    摘要:菜鳥教程框架中文手冊入門目標使用搭建通過對數據增刪查改沒了純粹占行用的拜 后端API入門學習指北 了解一下一下概念. RESTful API標準] 所有的API都遵循[RESTful API標準]. 建議大家都簡單了解一下HTTP協議和RESTful API相關資料. 阮一峰:理解RESTful架構 阮一峰:RESTful API 設計指南 RESTful API指南 依賴注入 D...

    sf190404 評論0 收藏0
  • 后端API從入門到放棄指北

    摘要:菜鳥教程框架中文手冊入門目標使用搭建通過對數據增刪查改沒了純粹占行用的拜 后端API入門學習指北 了解一下一下概念. RESTful API標準] 所有的API都遵循[RESTful API標準]. 建議大家都簡單了解一下HTTP協議和RESTful API相關資料. 阮一峰:理解RESTful架構 阮一峰:RESTful API 設計指南 RESTful API指南 依賴注入 D...

    Airmusic 評論0 收藏0
  • 合格PHP工程師知識結構

    摘要:不是原創的,主要是看了覺得有些幫助,就分享給大家看看。對于的知識要求我覺得以鳥哥的私房菜這本書作為要求吧。前端大部分使用,對靜態資源使用可以加載更快。其實編程使用什么語言不重要,最重要的是思想,編程的思路。 不是原創的,主要是看了覺得有些幫助,就分享給大家看看。 PHP的運行環境 連環境都搞不起來,就是你有多么喜歡PHP,那也是白搭,開始我們大多會使用集成環境軟件例如xampp,wam...

    KaltZK 評論0 收藏0
  • Symfony2中創建數據模型

    摘要:今天我在這里給大家分享一下在中如何創建數據模型和基于的搭建。對象與數據庫的關系通常在電腦的世界里創建的數據模型會轉換為數據庫里的一條條記錄。 Symfony是一款優秀的基于MVC架構的PHP框架。今天我在這里給大家分享一下在Symfony中如何創建數據模型和基于RESTful api的搭建。重點是如何創建數據模型哦! 本教程使用的當前Symfony的LTS版本(Symfony 2.8...

    rainyang 評論0 收藏0

發表評論

0條評論

最新活動
閱讀需要支付1元查看
<