Laravel Database——查询构造器与语法编译器源码分析 (下)

insert 语句

insert 语句也是我们经常使用的数据库操作,它的源码如下:

public function insert(array $values)

{

    if (empty($values)) {

        return true;

    }

    if (! is_array(reset($values))) {

        $values = [$values];

    }

    else {

        foreach ($values as $key => $value) {

            ksort($value);

            $values[$key] = $value;

        }

    }

    return $this->connection->insert(

        $this->grammar->compileInsert($this, $values),

        $this->cleanBindings(Arr::flatten($values, 1))

    );

}

laravel 的 insert 是允许批量插入的,方法如下:

DB::table('users')->insert([['email' => 'foo', 'name' => 'taylor'], ['email' => 'bar', 'name' => 'dayle']]);

一个语句可以向数据库插入两条记录。sql 语句为:

insert into users (`email`,`name`) values ('foo', 'taylor'), ('bar', 'dayle');

因此,laravel 在处理 insert 的时候,首先会判断当前的参数是单条插入还是批量插入。

if (! is_array(reset($values))) {

    $values = [$values];

}

reset 会返回 values 的第一个元素。如果是批量插入的话,第一个元素必然也是数组。如果的单条插入的话,第一个元素是列名与列值。因此如果是单条插入的话,会在最外层再套一个数组,统一插入的格式。

如果是批量插入的话,首先需要把插入的各个字段进行排序,保证插入时各个记录的列顺序一致。

compileInsert

对 insert 的编译也是按照批量插入的标准来进行的:

public function compileInsert(Builder $query, array $values)

{

    $table = $this->wrapTable($query->from);

    if (! is_array(reset($values))) {

        $values = [$values];

    }

    $columns = $this->columnize(array_keys(reset($values)));

    $parameters = collect($values)->map(function ($record) {

        return '('.$this->parameterize($record).')';

    })->implode(', ');

    return "insert into $table ($columns) values $parameters";

}

首先对插入的列名进行 columnze 函数处理,之后对每个记录的插入都调用 parameterize 函数来对列值进行处理,并用 () 包围起来。


update 语句

public function update(array $values)

{

    $sql = $this->grammar->compileUpdate($this, $values);

    return $this->connection->update($sql, $this->cleanBindings(

        $this->grammar->prepareBindingsForUpdate($this->bindings, $values)

    ));

}

与插入语句相比,更新语句更加复杂,因为更新语句必然带有 where 条件,有时还会有 join 条件:

public function compileUpdate(Builder $query, $values)

{

    $table = $this->wrapTable($query->from);

    $columns = collect($values)->map(function ($value, $key) {

        return $this->wrap($key).' = '.$this->parameter($value);

    })->implode(', ');

    $joins = '';

    if (isset($query->joins)) {

        $joins = ' '.$this->compileJoins($query, $query->joins);

    }

    $wheres = $this->compileWheres($query);

    return trim("update {$table}{$joins} set $columns $wheres");

}


updateOrInsert 语句

updateOrInsert 语句会先根据 attributes 条件查询,如果查询失败,就会合并 attributes 与 values 两个数组,并插入新的记录。如果查询成功,就会利用 values 更新数据。

public function updateOrInsert(array $attributes, array $values = [])

{

    if (! $this->where($attributes)->exists()) {

        return $this->insert(array_merge($attributes, $values));

    }

    return (bool) $this->take(1)->update($values);

}


delete 语句

删除语句比较简单,参数仅仅需要 id 即可,delete 语句会添加 id 的 where 条件:

public function delete($id = null)

{

    if (! is_null($id)) {

        $this->where($this->from.'.id', '=', $id);

    }

    return $this->connection->delete(

        $this->grammar->compileDelete($this), $this->getBindings()

    );

}

删除语句的编译需要先编译 where 条件:

public function compileDelete(Builder $query)

{

    $wheres = is_array($query->wheres) ? $this->compileWheres($query) : '';

    return trim("delete from {$this->wrapTable($query->from)} $wheres");

}


动态 where

laravel 有一个有趣的功能:动态 where。

DB::table('users')->whereFooBarAndBazOrQux('corge', 'waldo', 'fred')

这个语句会生成下面的 sql 语句:

select * from users where foo_bar = 'corge' and baz = 'waldo' or qux = 'fred';

也就是说,动态 where 将函数名解析为列名与连接条件,将参数作为搜索的值。

我们先看源码:

public function dynamicWhere($method, $parameters)

{

    $finder = substr($method, 5);

    $segments = preg_split(

        '/(And|Or)(?=[A-Z])/', $finder, -1, PREG_SPLIT_DELIM_CAPTURE

    );

    $connector = 'and';

    $index = 0;

    foreach ($segments as $segment) {

        if ($segment !== 'And' && $segment !== 'Or') {

            $this->addDynamic($segment, $connector, $parameters, $index);

            $index++;

        }

        else {

            $connector = $segment;

        }

    }

    return $this;

}

protected function addDynamic($segment, $connector, $parameters, $index)

{

    $bool = strtolower($connector);

    $this->where(Str::snake($segment), '=', $parameters[$index], $bool);

}

首先,程序会提取函数名 whereFooBarAndBazOrQux,删除前 5 个字符,FooBarAndBazOrQux。

正则判断,根据 And 或 Or 对函数名进行切割:FooBar、And、Baz、Or、Qux。

添加 where 条件,将驼峰命名改为蛇型命名。

————————————————

原文作者:leoyang

转自链接:https://learnku.com/articles/6291/laravel-database-query-constructor-and-syntax-compiler-source-code-analysis-below

版权声明:著作权归作者所有。商业转载请联系作者获得授权,非商业转载请保留以上作者信息和原文链接。

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

推荐阅读更多精彩内容