ホームページ >バックエンド開発 >PHPチュートリアル >PHPのmvcフレームワークの進化過程を詳しく解説_PHPチュートリアル
1) /********** 追加、削除、変更、チェック機能を完了するための php と html の混合**************************** ****/
1. デザインアイデア:
関数の追加、削除、変更、確認の通常の方法、つまりデータベースへの接続操作を php ファイルで完了することに基づいています
そしてphpファイル内のhtmlコードを表示します。 HTML は、処理と保存のために現在のページの PHP 部分に送信されます。
$_SERVER['SCRIPT_FILENAME'] には、現在のスクリプトのパスが含まれます。
これは、ページがそれ自体を指す必要がある場合に便利です。
️定数には、現在のスクリプト(インクルード ファイルなど)の完全なパスとファイル名が含まれます。
最初のクラス: 製品ファイルの追加: addProduct.php
if(isset($_POST['submit'])){
$name= $_POST['name'];
$price= $_POST['価格'];
$description= $_POST['説明'];
//データベースへの接続操作
mysql_connect('localhost','root','master');
mysql_select_db('テスト');
mysql_query('set names gbk');
$query= "商品値に挿入(null,'$name','$price','$description')";
mysql_query($query);
mysql_close();
}
?>
製品名
商品価格
製品説明
アイデア: 両方ともデータベースの操作と同じファイル (現在のファイル内) での表示:
Mysql_fetch_assoc は配列を返します。通常、2 次元配列を形成するには空の配列に配置する必要があり、mysql_num_rows は行数を返します。
通常、データのクエリに使用されます:
Php
$result = Mysql_query(“select * fromproduct order by ID desc”);
$data = 配列();
while ($row = Mysql_fecth_assoc($result)){
$data[]= $row;
}
HTML:
…
クエリ製品ファイル: listProduct.php
//データベース接続操作
mysql_connect('localhost','root','master');
mysql_select_db('テスト');
mysql_query('set names gbk');
$query = "ID 説明による製品注文から * を選択";
$result = mysql_query($query);
$data = 配列();
while($row = mysql_fetch_assoc($result)){
$data[]= $row;
}
?>
foreach($dataas$row):
?>
表>
アイデア: 次に、削除操作と更新操作が実行され、削除操作と更新操作を処理するために同じリクエストが新しい php ファイルに対して行われます。新しい php ファイルを作成します: delProduct.php
delProduct.php:
//データベース接続操作
mysql_connect('localhost','root','master');
mysql_select_db('テスト');
mysql_query('set names gbk');
$id = $_GET['id'];
$query = "id = '$id' の製品から削除";
mysql_query($query);
mysql_close();
//同時に listProduct にジャンプして結果を表示します
header('location:listProduct.php');
更新操作:
updateProduct.php (以下のクラスは表示専用です。変更の送信にはphpファイルも必要です。別のファイルを追加しないように、このページに変更内容を記載し、表示と送信を区別するためのパラメータをアクションに追加します)の変更)
//データベース接続操作
mysql_connect('localhost','root','master');
mysql_select_db('テスト');
mysql_query('set names gbk');
$id = $_GET['id'];
$query = "select * from product where id = '$id'";
$result = mysql_query($query);
$row = mysql_fetch_row($result);
?>
製品名
商品価格
製品説明
修正内容は以下の通りです
//データベース接続操作
mysql_connect('localhost','root','master');
mysql_select_db('テスト');
mysql_query('set names gbk');
if(isset($_REQUEST['flag'])){
$id= $_POST['id'];
$name= $_POST['name'];
$price= $_POST['価格'];
$description= $_POST['説明'];
$query= "製品セット名 = '$name'、価格 = '$price'、説明 = '$description' where id='$id'";
mysql_query($query);
mysql_close();
header('location:listProduct.php');
}その他{
$id= $_GET['id'];
$query= "select * from product where id = '$id'";
$result= mysql_query($query);
$row= mysql_fetch_row($result);
}
?>
製品名
商品価格
製品説明
/**************DB クラスを追加してデータベースの操作をカプセル化します*******************/
この時点で、製品の追加、削除、変更、確認の基本的な機能は完了しましたが、データベース操作などの冗長なコードが多数あります。この時点で、データ操作の抽出を検討できます。
DB.class.php:
/**
* データベース操作クラス
* @作者ヘヨンジア
*
*/
クラスDB{
private$host ='localhost';
private$username ='root';
private$password ='マスター';
private$dbname ='テスト';
private$setCoding ='set names gbk';//挿入が失敗する場合があるため、エンコード設定に注意してください
private$conn;// リソースを接続する
private$result;//結果セット
パブリック関数connect(){
$this->conn =mysql_connect($this->ホスト,$this->ユーザー名,$this->パスワード);
mysql_select_db($this->dbname);
mysql_query($this->setCoding);
}
パブリック関数クエリ($クエリ){
$this->result = mysql_query($query);
}
パブリック関数close(){
mysql_close();
}
}
その後、addProduct.php、delProduct.php、listProduct.php、updateProduct.phpのデータベース操作部分をそれぞれ修正し、各ファイル内の接続データとクエリ部分をdatabase.class.phpに抽出します。次のように:
Product.phpを追加
if(isset($_POST['submit'])){
$name= $_POST['name'];
$price= $_POST['価格'];
$description= $_POST['説明'];
$query= "商品値に挿入(null,'$name','$price','$description')";
$db =newDB();
$db->connect();
$db->クエリ($クエリ);
$db->close();
}
delProduct.php:
「class/DB.class.php」をインクルードします;
$id = $_GET['id'];
$query = "id = '$id' の製品から削除";
$db = newDB();
$db->connect();
$db->クエリ($クエリ);
$db->close();
//同時に listProduct にジャンプして結果を表示します
header('location:listProduct.php');
listProduct.php:
//データベース接続操作
「class/DB.class.php」をインクルードします;
$query = "ID 説明による製品注文から * を選択";
$db = newDB();
$db->connect();
$db->クエリ($クエリ);
$data = 配列();
while($row = $db->fetch_assoc()){
$data[]= $row;
}
$db->close();
?>
/*********** データ転送用の製品オブジェクト、つまりデータモデルを参照します **********************/
アイデア: 追加、削除、変更、チェックの操作はビジネス ロジックに属し、カプセル化する必要があります。
クラス Product.class.php にカプセル化されます
クラス製品{
プライベート$ID;
プライベート$name;
プライベート$価格;
プライベート$説明;
public function__set($name,$value){
$this->$name= $value;
}
パブリック関数__get($name){
return$this->$name;
}
パブリック関数add(){
$query= "商品値に挿入(null,'$this->名前','$this->価格','$this->説明')";
$db= newDB();
$db->connect();
$db->query($query);
$db->close();
}
}
次に、このクラスを addProduct.php に導入し、操作データベースのアクションを変更します。
$product= newProduct();
$product->name = $name;
$product->price = $price;
$product->description = $description;
$product->add();
データベースを操作する他の方法も同様です。
アイデア: addProduct.php、delProduct.php、listProduct.php、updateProduct.php がすべて類似したコードを持っていることを観察すると、これらを統合してすべてのモジュールの関連リクエストを完了できるでしょうか?
新しい product.php ファイルを作成し、addProduct.php と delProduct.php 内の関連する操作データベース コードを product.php ファイルにコピーし、product.php のアクション パラメーターを使用してさまざまなリクエストを区別します。
//関連するすべての製品操作を完了するためのリクエスト
「class/DB.class.php」をインクルードします;
「class/Product.class.php」をインクルードします;
$アクション = $_GET['アクション'];
if($action =='追加'){
//追加
$name= $_POST['name'];
$price= $_POST['価格'];
$description= $_POST['説明'];
$product= newProduct();
$product->name = $name;
$product->price = $price;
$product->description = $description;
$product->add();
}
if($action =='del'){
//削除
$id= $_GET['id'];
$product= newProduct();
$product->id = $id;
$product->del();
//同時に listProduct にジャンプして結果を表示します
header('location:listProduct.php');
}
この時点で、addProduct.php の php コード部分を削除し、html 部分を addProduct.html ファイルに分離できます。
同時に、delProduct.php ファイルも削除できます。これら 2 つの PHP ファイルの PHP コードは product.php に移動されているためです。
/********************Smarty シミュレーション クラス ライブラリに参加**************************** *** ***/
まず上記の 2 つの機能を完了してください。
1. addProduct.html ファイルにアクセスし、追加操作を完了して、addProduct.html ページに戻ります。 listProduct.php ファイルにアクセスして、追加された結果を表示します。
2. listProduct.php ファイルにアクセスし、リンクを削除するためのリクエスト ポイントを変更します。同時に削除機能も完了します。
この時のproduct.phpのコードは以下の通りです:
//関連するすべての製品操作を完了するためのリクエスト
「class/DB.class.php」をインクルードします;
「class/Product.class.php」をインクルードします;
$アクション = $_GET['アクション'];
if($action =='追加'){
//追加
$name= $_POST['name'];
$price= $_POST['価格'];
$description= $_POST['説明'];
$product= newProduct();
$product->name = $name;
$product->price = $price;
$product->description = $description;
$product->add();
header('location:addProduct.html');
}
if($action =='del'){
//削除
$id= $_GET['id'];
$product= newProduct();
$product->id = $id;
$product->del();
//同時に listProduct にジャンプして結果を表示します
header('location:listProduct.php');
}
/**************smarty クラス ライブラリを追加*************************/
アイデア: 現時点では、クエリと更新の操作に加えて、追加と削除の機能が product.php に統合されています。listProduct.php にはデータのリストを表示する機能が含まれているため、表示するのは簡単ではありません。これを行うにはページを含めてください) 完了します。)この時点ではsmartyでできるので合理的です。
現時点では、表示のために addProduct.html を product.php に入れて、次のように変更することを検討できます。
include 'class/Template.class.php';//smarty をシミュレートする実際のクラス ファイルであるこのファイルのインポートに注意してください。 (クエリ関数も使用し、複数のデータを表示する必要があることを考慮して、Smartyをシミュレートするクラスファイルは修飾されていないため、表示としてsmartyを直接導入しています)
/***********************smarty クラス ライブラリに参加**************************/
このとき、Smartyクラスから自分で書いたTemplatesクラスを継承することで、Smartyの機能を利用したり、機能を拡張したりすることができます。
Template.class.php の内容は次のとおりです:
/**
* Smartyクラスを継承したテンプレートクラス
* @作者ヘヨンジア
*
*/
クラステンプレート拡張Smarty{
/**
* 親クラスのコンストラクターを呼び出して、親クラスのコンストラクターが実行されたことを確認します
* サブクラスにコンストラクターがある場合、親クラスのコンストラクターは実行されないためです
*/
function__construct(){
親::__construct();
//テンプレートディレクトリを初期化します
$this->setTemplateDir('tpl');
}
}
この時点で、CRUD 関数は product.php ファイルに統合されています。これは、次のようなすべてのリクエストがこのファイルを通過することを意味します。
Localhost/test/mvc/product.php?action=add/list/update およびその他のリクエスト。この時点では、product.php がコントローラーの役割に相当します。
/**************** エントリーファイルを追加 ***************************** **、
私たちのプログラムでは、管理を容易にするために、エントリ ファイルを定義します。つまり、すべてのリクエストはこのエントリ ファイルを経由し、このエントリを通じて各コントローラに分配されます。
index.php ファイルを追加します:
コントローラーへの配信の場合:
リクエストされたURL:
Localhost/test/mvc/index.php?module=product&action=add/list/update
index.php ファイルは次のようになります:
$module = $_GET['モジュール'];
$アクション = $_GET['アクション'];
$file = $module.'Control.php?action='.$action;
header('location:'.$file);
この時点で、上記のエントリーファイルを介して異なるコントローラー(モジュール)上の異なる操作(関数)に振り分けることができますが、ヘッダージャンプを使用しているため、ブラウザ上のアドレスバーがジャンプになります。つまり、エントリ ファイルから分離されます。
それでは、どうすればそれを達成できるのでしょうか?
product.phpを実装するクラスに変更します。 product.php を ProductControl.class.php に変換し、コントロール フォルダーに移動します。内容は次のとおりです。
クラス ProductControl{
パブリック関数addok(){
//追加
$name= $_POST['name'];
$price= $_POST['価格'];
$description= $_POST['description'];
$product= newProduct();
$product->name = $name;
$product->price = $price;
$product->description = $description;
$product->add();
header('location:product.php?action=list');
}
パブリック関数add(){
$smarty= newTemplate();
//$tpl = 新しいテンプレート();
//$tpl->display('addProduct.html');
$smarty->display('addProduct.html');
}
パブリック functiondel(){
//削除
$id= $_GET['id'];
$product= newProduct();
$product->id = $id;
$product->del();
//同時にリスト商品展示结果
header('location:product.php?action=list');
}
public functionupdate(){
$smarty= newTemplate();
$id= $_GET['id'];
$product= newProduct();
$product->id = $id;
$row= $product->getRow();
$smarty->assign('id',$row['id']);
$smarty->assign('name',$row['name']);
$smarty->assign('price',$row['price']);
$smarty->assign('description',$row['description']);
$smarty->display('updateProduct.html');
}
public functionupdateok(){
$id= $_POST['id'];
$name= $_POST['name'];
$price= $_POST['価格'];
$description= $_POST['description'];
$product= newProduct();
$product->id = $id;
$product->name = $name;
$product->price = $price;
$product->description = $description;
$product->update();
header('location:product.php?action=list');
}
}
今回は、この種のメソッドをどのように使用するのですか?index.php でこのオブジェクトの使用を例示します。
//関連するすべての製品操作の要求を完了するために使用されます
「class/DB.class.php」をインクルードします;
「class/Product.class.php」をインクルードします;
「smarty/Smarty.class.php」をインクルードします;
「class/Template.class.php」をインクルードします;
$module = $_GET['モジュール'];
$module = ucfirst($module);
$アクション = $_GET['アクション'];
$className = $module.'コントロール';
include 'control/'.$className.'.class.php';
$クラス = 新しい$クラス名; // によるモ块名引用类文件实例化该对オブジェクト
$class->$action(); // 対応するアクション メソッドを調整します。
修正用のすべての経路:index.php?module=product&action=xxx
Product.html の追加UpdateProduct.htmllistProduct.html
改良: 一般に、index.php ファイル内に過剰なコードを書き込むことはできませんが、コードを移動配置ファイル内に組み込むことができ、index.php にすぐに組み込むことができます。
通常、全体の工程会には、分発行制御器を処理するために使用される 1 つのメイン制御器グループがあります。Application.class.php:
クラスアプリケーション{
static functionrun(){
グローバル$モジュール; //関数数内で使用する外部の变量
グローバル$アクション;
$className= $module.'Control';
include'control/'.$className.'.class.php';
$class= new$className;
$class->$action();
}
}
init.php:
//関連するすべての製品操作の要求を完了するために使用されます
「class/DB.class.php」をインクルードします;
「class/Product.class.php」をインクルードします;
「smarty/Smarty.class.php」をインクルードします;
「class/Template.class.php」をインクルードします;
「control/Application.class.php」をインクルードします;
$module = $_GET['モジュール'];
$module = ucfirst($module);
$アクション = $_GET['アクション'];
Index.php:
「config/init.php」を含める;
アプリケーション::run();
現時点でのアクセスURLはindex.php?module=product&action=xxxです
ただし、index.php に直接アクセスすると、エラーが報告されます:
注意: 未定義のインデックス: 10 行目の F:ampapachestudydocstestphp_jsmvcconfiginit.php のモジュール
注意: 未定義のインデックス: 12 行目の F:ampapachestudydocstestphp_jsmvcconfiginit.php のアクション
これは、モジュールとアクションが init.php で初期化されていないためです。
そして、config フォルダーの下の conig.php ファイルに値を設定します。
product.php ファイル (実際にはモデル クラス) にはデータベースを操作するための同様のコードが多数あるため、コードのこの部分は親クラスの実装にカプセル化でき、モデル クラスを継承するだけで済みます。
/******************コアファイル core をカプセル化************************* ***
新しいフォルダー core を作成し、DB.class.php と Application.php を core に移動します。これらは公開ファイルであり、機能するファイルではないためです。
新しい基本クラス ファイル (モデル基本クラス) を作成してデータベースのインスタンス化をカプセル化し、サブクラス モデルはデータベース オブジェクトのインスタンスを取得するために継承するだけで済みます。
Model.class.php:
クラスメデル{
protected$db;
パブリック関数__construct(){
$this->db =newDB();
}
}
新しいフォルダー モデルを作成し、product.php をモデル フォルダーに移動し、名前を ProductModel.class.php に変更します。
クラス ProductModelextendsModel {
プライベート$ID;
プライベート$name;
プライベート$価格;
プライベート$説明;
public function__set($name,$value){
$this->$name= $value;
}
パブリック関数__get($name){
return$this->$name;
}
パブリック関数add(){
$query= "商品値に挿入(null,'$this->名前','$this->価格','$this->説明')";
$this->db->connect();
$this->db->query($query);
$this->db->close();
}
パブリック functiondel(){
$query= "id = '$this->id' の製品から削除";
$this->db->connect();
$this->db->query($query);
$this->db->close();
}
パブリック関数select(){
$query= "iddesc による製品注文から * を選択";
$this->db->connect();
$this->db->query($query);
$data= array();
while($row = $this->db->fetch_assoc()){
$data[]= $row;
}
$this->db->close();
return$data;
}
パブリック関数update(){
$query= "製品セット名 = '$this->name'、価格 = '$this->price'、説明 = '$this->description' where id='$this->id 「」;
$this->db->connect();
$this->db->query($query);
$this->db->close();
}
パブリック関数getRow(){
$query= "select * from product where id = '$this->id'";
$this->db->connect();
$this->db->query($query);
$row= $this->db->fetch_assoc();
$this->db->close();
return$row;
}
}
/***********************カプセル化の表示*************************** ***
Template クラスはコントローラー内で毎回インスタンス化され、表示メソッドが呼び出される必要があります。
Template.class.php をコア フォルダーに移動し、名前を View.class.php:
に変更します。
これを行う利点は、各コントローラーがこのクラス (Control.class.php) を継承している限り、各コントロールがこのクラスから継承することです (コントローラーの基本クラス Control.class.php を構築し、クラス内にビュー インスタンスを構築します)。 )、つまり、Smarty インスタンスがあります。
/**
* 1. Smartyクラスを継承したテンプレートクラス
*2. Smartyを継承する必要はなく、smartyのインスタンスです
* @作者ヘヨンジア
*
*/
クラスビュー拡張Smarty{
}
これを行う利点は、各コントローラーがこのクラス (Control.class.php) を継承している限り、各コントロールがこのクラスから継承することです (コントローラーの基本クラス Control.class.php を構築し、クラス内にビュー インスタンスを構築します)。 )、つまり、Smarty インスタンスがあります。
/**
*コントローラー基本クラス
* @作者ヘヨンジア
*
*/
クラス Controlextends View{
protected$view;
パブリック関数__construct(){
$this->view = new View();
$this->view->setTemplateDir('tpl');
}
}
最適化では、多くのモデルが出現する可能性があるため、モデルを init.php に直接含めるのは間違いです。現時点では、これを実現するために、オンボードでの自動読み込みを検討できます。 ※自動読み込み機能はsmarty3.0でも使用しているため、登録が必要です。
関数 autoload1($classname){
$filename= 'model/'.$classname.'.class.php';
if(is_file($filename)){
「$filename」を含める;
}
}
spl_autoload_register('autoload1');
関数addslashes_func(&$str){
$str= ラッシュを追加します($str);
}
if(!get_magic_quotes_gpc()){
array_walk_recursive($_POST,'addslashes_func');
array_walk_recursive($_GET,'addslashes_func');
}
この時点で、単純な MVC フレームワークがほぼ完成しました。
より直感的にわかるように、画像でまとめてみましょう: