如何写一个属于自己的数据库封装(1) - 基本思路(修订版)

下一期 如何写一个属于自己的数据库封装(2) - 数据库连接

写在前头

  1. 依然在前进的菜鸟一只,错误什么的请轻喷指出
  2. 对于数据库连接由于较浅的工作经验所以无法完全覆盖,较复杂的query可能会有意想不到的bug
  3. 所以本系列只提供思路,成熟稳定的数据库封装有请自行搜索
  4. 编写该系列的初衷就是为了抛砖引玉, 在每一节的后端,我都会提出一些个人疑问(或无),希望能引出大神为我解答迷惑
  5. 使用的php版本为7.0+,该系列并不向下兼容,还在5.6版本的童鞋们快过来玩呀
  6. 数据库封装参考了laravel的源代码, 如有雷同, 不是巧合

最终效果

假设我们有一个表,名'Actor', 经过简单设置, 可以直接如下调用

$a = Actor::select('first_name', 'last_name')
        ->where('first_name', 'NICK')
        ->where('last_name', 'WAHLBERG')
        ->first()
var_dump($a);

返回格式

object(Actor)[11]
  public 'first_name' => string 'NICK' (length=4)
  public 'last_name' => string 'WAHLBERG' (length=8)

返回的数据依然可操作(update/delete)

$a->update(...);
$a->delete();

更多例子可以查看第4期 - 如何写一个属于自己的数据库封装(4) - 查询 - 入门篇用法

本期知识点

【PHP】PDO操作数据库

常见的数据库连接写法

对php有一定了解的人都知道,相比已被淘汰的mysql或取而代之的mysqli, pdo 可以避免SQL注入式攻击, 更安全, 而且面向对象, 所以请看下方pdo例子

<?php
$driver='mysql'; //数据库类型

$host='localhost'; //数据库主机名

$db = 'sakila'; //数据库名称

$username='root'; //数据库连接用户名

$password=''; //数据库密码

$dsn="$driver:host=$host;dbname=$db";

try {
    $pdo= new PDO($dsn, $username, $password); //初始化一个PDO对象

    $sql = "select * from actor";

    $res = $pdo->query($sql); // 从actor中获取所有数据

    foreach($res as $row)
        echo $row['first_name']."<br>";
} catch (PDOException $e) {
    die($e->getMessage());
}

例子中的代码看起来非常简洁, 然而, 这么做真的好吗?

很多菜鸟因为经验或自身的原因, 在实现业务逻辑时并未考虑过维护代码, 要不重复性代码全写在一块, 要不随便写个函数概括进去, 这一类代码, 是导致经手人辞职的最大元凶

代码太烂了我看不下去

因此, 考虑到一个项目长远开发, 我们需要一个封装类来减少代码的重复性, 就像平时自行写的一些辅助函数, 不过封装类略微进阶而已。

基本思路

那么,应该怎么写呢?
平时我们要简略一些代码, 都会自行编写一些辅助函数,比如我个人有个不好的习惯,不喜欢用编辑器的调试功能(画外音:我用的是sublime text 3),那么debug的时候一开始都是这么写的

$a = "is bug";
die(var_dump($a)); // 输出变量并且停止运行之后的代码
/***从xdebug得知bug出现在这一行,所以查看上一行的逻辑与数据 ***/

die(var_dump())的出场率略高, 因此我将它们写成了一个辅助函数

function dd($var) {
    die(var_dump($var));
}

同样道理,在数据库读取的时候,select的出场率极高, 所以简单的包裹一下

function select($pdo, $table, $require, $where = []) {
    $w = [];

    // 将搜索条件转化为数据库命令能接受的格式
    foreach ($where as $key => &$val)
        $w[] = "$key = '$val'";

    // 有时候并不需要条件,仅仅将所有数据取出,因此$where默认为空
    if(!empty($w))
        $w = "where ".implode(', ', $w);
    else
        $w = "";

    // 生成sql query
    $sql = "select $require from $table $w";

    // 将已生成的query带入pdo实例
    $res = $pdo->query($sql);

    // 返回结果
    return $res->fetchAll();
}

例子

select($pdo, 'actor', '*', ['first_name'=>'PENELOPE'])
/**
  * 生成sql query => select * from actor where first_name = 'PENELOPE'
**/

总结一下, 数据库封装浅显的形容就是将参数带入辅助函数,让其自动生成SQL命令。

数据库封装的架构

主要分成四个文件

  • Connector.php - 负责与数据库的通信, 请求与返回数据库数据
  • Model.php - 入口文件,作为一个模型, 定义表的各种属性(如主键, 表名等等), 并接受请求, 返回数据库数据
  • Grammar.php - 将Builder存储的请求转化为SQL语句
  • Builder.php - 核心文件, 存储Model的请求,调用Grammar编译SQL语句, 与参数变量一并送往Connector以获取数据库数据,再返还给Model

架构可能解释得不太好,但没关系, 接下来会慢慢一个个的深入解释

大致如此, 如果可以的话点个赞,或在下方评论区讨论, 只有反馈才能推动一个懒癌晚期病患继续前行。

下一期 如何写一个属于自己的数据库封装(2) - 数据库连接

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,928评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,192评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,468评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,186评论 1 286
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,295评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,374评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,403评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,186评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,610评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,906评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,075评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,755评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,393评论 3 320
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,079评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,313评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,934评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,963评论 2 351

推荐阅读更多精彩内容