严格说起来,自己并非一个合格的php程序员。第一次使用php开发,不过是因为游戏上线,需要一个统计管理后台和GM后台,因为招聘已经来不及,所以我就上前线了!
凭着对php语法的一点点记忆(大学的时候学习过),边看《php与Mysql Web开发》,就边上手了!刚开始写的时候,根本没有章法,html 和 php直接混在一起,后来边开发边琢磨,不能这样,起码要界面和逻辑分离一下吧。想起来大学的时候做过ASP.NET,使用了自己理解的三层构架,那么就把三层构架引入吧。可是实际的开发过程中,觉得中间逻辑层和数据层实在不想费脑筋分开,就索性用两层好了,于是就调整成了两层构架。
比如我们举一个例子:我们要做一下金币消耗的统计。
首先看下 coinsCustomView.php:
<table>
<tr>
<th>类型</th>
<th>数值</th>
</tr>
<?php
require_once('dataStaticsLogic.php');
$arr = array();
$arr = getCoinStatics();
foreach($arr as $key=>$value)
{
?>
<tr>
<td><?php echo $value["type"]; ?></td>
<td><?php echo $value["content"]; ?></td>
</tr>
<?php
}
?>
</table>
再看下 dataStaticsLogic.php的定义:
<?php
require_once ('../../include/common.inc.php');
require_once('../../include/biz/order.php');
require_once ('../comm/safety.php');
require_once ('../serialsModule/serialsMuduleBase.php');
//金币统计
function getCoinStatics()
{
global $mongoUrl;
global $moneyConfig;
global $mongoName;
$mongoConn = new Mongoclient($mongoUrl);
$mongoDB = $mongoConn->selectDB($mongoName);
$mongoCollection = $mongoDB->selectCollection("mmo.moneySerials");
$statics = array();
foreach ($moneyConfig as $key=>$value)
{
$query = array('type'=>$key);
$itemSum = 0;
$cursor = $mongoCollection->find($query);
while($cursor->hasNext())
{
$row = $cursor->getNext();
$itemSum += $row['value'];
}
$item["type"] = $value;
$item["content"] = $itemSum;
$statics[] = $item;
}
$mongoConn->close();
return $statics;
}
***View.php 是用来展现数据的代码,而***Logic.php是用来处理逻辑和获取数据的代码。就是套用这样一个简单的逻辑,基本是面向对象的思想(项目中基本没有使用php面向对象的思想),完成了第一个php统计后台和GM管理后台!我只能可能要被php大神耻笑了。大家还是体谅一下一个java程序写php的苦衷吧!
很多,我们迎来了第二个项目,第二个项目也需要一个统计后台。当开始着手做这个php统计后台的时候,我自己心中有两个声音一直在提醒我。一个是,把原来的项目拔拔,再改吧改吧,继续套用好了,反正上次也没有出现什么问题。第二个声音就是,专业一点,不要重复自己,使用现在流行的mvc模式,去做这个新的后台,这样也能完成自己对php的新的认知。很多朋友来提来意见,有的说,代码技术只是工具啦,做出来能用就好,而且还是一个内部使用工具,做那么专业干嘛,简单高效是准则! 有的说,直接套用php成熟的框架smarty或者 thinkphp啦。根本不用想太多!
最终最终还是,奔着绝不重复自己的原则,决定使用mvc模式,把数据,逻辑和界面分开!也趁机好好学习了一下mvc模式,因为毕业以来一直在做服务器的开发,所以对前段这些框架,虽然他们天天讲,可是自己始终不太理解。好吧,趁这次机会,提升自己一下啦。
那么我们就拿一个简单的例子来做参考,来理解一下mvc模式,顺便看看新的项目,是如何做模块的划分的,如何分离界面,数据和逻辑的!
首先给我们看下图:
首先在代码的组织上,就将业务逻辑和界面分开。关于MVC的示意图,网上是在是太多了,我就不在这里贴出来了,大家随便搜搜,大把的图片例子和文档。
我们用一个获取所有房间列表的例子来做参考好了!
首先看下Model层,就是所谓业务层,包含Room类的设计,还有需要说明的是,CurlRequest类,这里就不贴出来了,就是http请求获取数据,在这里相较于前一个项目,很多东西都是封装成了类!无论是http请求,还是数据库请求!
class Room {
private $roomId;
private $roomName;
private $roomUserCnt;
private $roomSpectCnt;
public function __construct($roomId,$roomName,$roomUserCnt,$roomSpectCnt) {
$this->roomId = $roomId;
$this->roomName = $roomName;
$this->roomUserCnt = $roomUserCnt;
$this->roomSpectCnt = $roomSpectCnt;
}
public function getRoomId(){
return $this->roomId;
}
public function getRoomName(){
return $this->roomName;
}
public function getRoomUserCnt(){
return $this->roomUserCnt;
}
public function getRoomSpectCnt(){
return $this->roomSpectCnt;
}
}
class RoomModel {
private $cmd = "php";
private $subCmd = "roomList";
public function fetchAllRooms(){
require_once dirname( dirname( dirname(__FILE__) ) ).'/base/CurlRequest.class.php';
require_once 'Room.class.php';
$allRoom = array();
$curl = new CurlRequest();
$result = $curl->fetchResponse($this->cmd, $this->subCmd);
$resultByJson = json_decode($result,true);
for($i = 0;$i<count($resultByJson);$i++){
$allRoom[] = new Room($resultByJson[$i]["roomId"],
$resultByJson[$i]["roomName"],
$resultByJson[$i]["roomUserCnt"],
$resultByJson[$i]["roomSpectCnt"]);
}
return $allRoom;
}
}
RoomModel的职责很简单,就是获得所有的房间数据,无论是通过Http请求获得,还是通过Mysql数据库获得,界面和逻辑层都无从得知。
下面是界面层:也就是RoomView.php的代码
<?php
echo "<table>";
echo "<tr><th>房间id</th><th>房间名称</th><th>参与人数</th><th>旁观人数</th></tr> ";
foreach ($allRoom as $room)
{
echo "<tr>";
echo "<td>".$room->getRoomId()."</td>";
echo "<td>".$room->getRoomName()."</td>";
echo "<td>".$room->getRoomUserCnt()."</td>";
echo "<td>".$room->getRoomSpectCnt()."</td>";
echo "</tr>";
}
echo "</table>"
?>
View在这里,就是控制界面,输出所要展现的数据。
很多人可能就想,有View和Model在这里,完全不需要Controller了吧,起始我觉得也是这样
下面看下RoomListController的代码
<?php
class RoomListController {
public function displayRoomList(){
require_once (dirname(dirname(__FILE__))."/model/RoomModel.class.php");
$roomModel = new RoomModel();
$allRoom = $roomModel->fetchAllRooms();
include_once dirname(dirname(__FILE__))."/view/RoomListView.php";
}
}
$roomController = new RoomListController();
$roomController->displayRoomList();
?>
Controller层,起始就是控制界面的输出的,属于Model和View两层之间的桥接。在这里提下我所理解的三层构架和MVC模式的区别---也许并不成熟,那就权当抛砖引玉好了,三层构架可以理解为VCM,即V冲在最前面,M起始就是用来采集数据或者更新数据,C呢,起到了桥接M和V的作用。而MVC,则是C冲在最前面,界面的展现与否,也要C来控制,当然也是桥接MV。
按照这样一个设计,前几个模块,都是这样套用mvc模式来做的,有了这样一个划分,代码整体看起来还是整洁了很多,规范了很多!就在游戏后端的开发中,我们基本上都会定义这样一个规范,以便于出了问题bug,能很快定位,那块出了问题。
写在最后
也许我对MVC的理解,还是不够准确,也许我对于php的开发,还是很菜鸟级别。但是由菜鸟向大神,由普通向高级的过程中,不重复自己,会让我走的更快。