往期回顾:「七天自制PHP框架」第三天:PHP实现的设计模式,点击此处

原文地址:http://www.cnblogs.com/sweng/p/6624845.html,欢迎关注:编程老头

前阵子在网上关心一个话题:对于一个PHP程序员,或者Java程序员,或者C#程序员,怎么区分3年,5年,10年工作经验?工作经验是否和薪资成正比?

个人认为:无论擅长哪一种语言,都不要把自己绑在一种语言上,更不要做一个代码的搬运工。平时每写完一个项目,都留出充足的时间去思考“还有什么地方可以改进”,相信即使1年的工作经验,也会有3年工作经验的收获。

最近我在思考一个问题:平时做PHP项目常常要和“新闻发布”,“博客评论”打交道,而每一次写功能都会重写“相似”的代码,如果能够把这一块代码做好重用,以后只需修改几个参数就能用在另一个项目,就可以很短时间完成一个功能。

当然使用PHP框架,会让你工作效率得到成倍的提升,但是,你的学习成本也就跟着上去了。

最初使用Laravel框架的时候,觉得Eloquent的语法实现得很美,比如:

$comments = App\Post::find(1)->comments;

foreach ($comments as $comment) {
//
}

这里继续我们上一节讲的,怎么做模型关联。

模型,简单点说,就是把数据库中这么多表抽象成若干个对象,使得开发的过程中,不用再关心数据表的结构,而是专心类和对象的设计。

就拿在微信朋友圈发消息来说,这里涉及到了3个对象:消息(纯文字,图片,或者图文),点赞,评论。

根据这个图,我们设计三个类:

class MessageModel extends Model{
public static $data;
public static $name;
public $messageid;
public function __construct(){
parent::__construct();
$this::$name='message_list';
$this::$table='message';
}
} class LikeModel extends Model{
public static $data;
public static $name;
public $likeid;
public function __construct(){
parent::__construct();
$this::$name='like_list';
$this::$table='messagelike';
}
} class CommentModel extends Model{
public $commentid;
public function __construct(){
parent::__construct();
$this::$name='comment_list';
$this::$table='reply';
}
}

这是典型的“一对多”的模型,也就是一个Message对象对应了多个Like对象和Comment对象,而一个Like对象或者Comment只对应了一个Message对象。

有人说,为什么不用SQL中的where或者join来查询?

因为我实在厌倦了拼接SQL,实在太无趣了。

关键是查询完得到的多维数组,还需要写一段代码来组装成对象数组,让我不得不思考怎么避免这低效劳动。

我的方案是每一个Model都实现这样的接口,让你尽量少写select

	public static function get($id){
return self::where('id',$id);
} public static function where($condition,$value){
$sql=sprintf("select * from %s where %s='%s'",self::$table,$condition,$value);
return self::$db->Query($sql);
} public static function first($num){
$sql=sprintf("select * from %s limit %s",self::$table,$num);
return self::$db->Query($sql);
} public static function all(){
$sql=sprintf("select * from %s",self::$table);
return self::$db->Query($sql);
}

那如果要实现一对多,怎么办?Laravel使用了trait特性,让Model来use这个特性。

这里我们简单的做一个函数:

	public function HasMany(Model $model,$foreignkey){
for($i=0;$i<count($this::$data);++$i){
$this::$data[$i][$model::$name]=[];
for($j=0;$j<count($model::$data);++$j){
if($this::$data[$i][$foreignkey]==$model::$data[$j][$foreignkey]){
array_push($this::$data[$i][$model::$name],$model::$data[$j]);
}
}
}
}

对于三张数据表:Message,Like,Comment来说,Message的主键是msgid,而msgid同时也是Like和Comment这两张表格的外键,靠着外键,三张表形成了一对多的关系。

所以凭借这样的一个关联数组的操作,我们把Like和Comment数组作为一个关联数组的Value塞入Message数组中的一个元素。

最后我们测试一下效果:

$messageModel=new MessageModel();
$messageModel::$data=$messageModel::all(); $likeModel=new LikeModel();
$likeModel::$data=$likeModel::all(); $commentModel=new CommentModel();
$commentModel::$data=$commentModel::all(10); $messageModel->HasMany($likeModel,'msgid');
$messageModel->HasMany($commentModel,'msgid'); echo json_encode($messageModel::$data);

我们最后使用JSON格式输出,结构一目了然,给前端调用也很便利。

「七天自制PHP框架」第四天:模型关联的更多相关文章

  1. 「七天自制PHP框架」第二天:模型与数据库

    往期回顾:「七天自制PHP框架」第一天:路由与控制器,点击此处 什么是模型? 我们的WEB系统一定会和各种数据打交道,实际开发过程中,往往一个类对应了关系数据库的一张或多张数据表,这里就会出现两个问题 ...

  2. 「七天自制PHP框架」第三天:PHP实现的设计模式

    往期回顾:「七天自制PHP框架」第二天:模型与数据库,点击此处 原文地址:http://www.cnblogs.com/sweng/p/6624845.html,欢迎关注:编程老头 为什么要使用设计模 ...

  3. 「七天自制PHP框架」应用:JSON生成器

    刚刚开始学做一个WebAPP,数据查询的一般套路是通过一张PHP页面读取数据库,获得列表后“嵌写”在PHP页面中,虽然写法上丑陋至极,但也有“快糙猛”出效果的成就感,如图. 后来想想,不对啊,难道以后 ...

  4. 「七天自制PHP框架」第一天:路由与控制器

    我们为什么要使用路由? 原因1:一个更漂亮的URI 1.URI的改进 刚刚开始学PHP时,我们一定写过blog.php?id=1之类的URI,使用GET方式获取参数.这样的URI有两个缺点,一是容易被 ...

  5. 「七天自制PHP框架」应用:Model外键链接

    这里以行政区数据为例: 一级行政区数据范例: 二级行政区范例: 三级行政区范例: 在Model层建立三个Model class ProvinceModel extends Model{ public ...

  6. 「7天自制PHP框架」第一天:路由与控制器

    我们为什么要使用路由? 原因1:一个更漂亮的URI 1.URI的改进 刚刚开始学PHP时,我们一定写过blog.php?id=1之类的URI,使用GET方式获取参数.这样的URI有两个缺点,一是容易被 ...

  7. 「BUAA OO Unit 4 HW16」第四单元总结与课程回顾

    「BUAA OO Unit 4 HW16」第四单元总结与课程回顾 目录 「BUAA OO Unit 4 HW16」第四单元总结与课程回顾 Part 0 第四单元作业架构设计 架构设计概要 AppRun ...

  8. Web自动化必会知识:「Web基础、元素定位、元素操作、Selenium运行原理、项目实战+框架」

    1.web 基础-html.dom 对象.js 基本语法 Dom 对象里面涉及元素定位以及对元素的修改.因为对元素操作当中涉及的一些 js 操作,js 基本语法要会用.得要掌握前端的基本用法.为什么要 ...

  9. 「拥抱开源, 又见 .NET」系列第三次线下活动简报

    「拥抱开源, 又见 .NET」 随着 .NET Core的发布和开源,.NET又重新回到人们的视野. 自2016年 .NET Core 1.0 发布以来,其强大的生命力让越来越多技术爱好者对她的未来满 ...

随机推荐

  1. python中从文件中读取数据

    # average5.py def main(): fileName = input("What file are the numbers in?") infile = open( ...

  2. sublime设置 快捷键(自动换行)

    一.菜单view > word wrap选上就好了 二.如果让编辑器默认是自动换行的话把它保存到配置中 Preference > Settings-User插入以下一行配置 "w ...

  3. 如何在R中导入不同类型的数据

    这个表格是我在datacamp学习R导入文件的课程的归纳 遇到的问题及解决方法(环境: Rv3.2.5,win7,32位) 1. 使用gdata中的read.xls时提示找不到Perl路径 >l ...

  4. LISTCTRL控件方法

    以下未经说明,listctrl默认view风格为report --------------------------------------------------------------------- ...

  5. 使用cl编译C/C++

    每次写程序都是用VS下打开的,而且有智能提示,经常很容易看到自己哪里写错了,其实想联系自己写代码的能力,不应该要这些的,纯粹的不要智能提示 所以自己想用轻量级的编辑器写,然后就用了notepad++( ...

  6. CopyOnWriteArrayList实现原理及源码分析

    CopyOnWriteArrayList是Java并发包中提供的一个并发容器,它是个线程安全且读操作无锁的ArrayList,写操作则通过创建底层数组的新副本来实现,是一种读写分离的并发策略,我们也可 ...

  7. User-Agent详解

    User-Agent : 用户代理  用户在上网的时候会作为http 请求头的一部分传递给服务端 ,用于识别用户当前环境(如浏览器类型及版本号,以及操作系统信息 )  右键f12可以查看 下面是我的浏 ...

  8. Day5模块-random模块

    random:随机数 >>> import random>>> print(random.random()) #生成随机小数0.6906362176182085 & ...

  9. ArrayList源码解析(三)

    1.isEmpty()  如果此列表中没有元素,则返回 true /** * Returns <tt>true</tt> if this list contains no el ...

  10. java 上传1(使用java组件fileupload)

    使用fileupload要添加以下包