Laravel Eloquent 分表方法并使用模型关联

众所周知Laravel是PHP开发项目最优美的框架之一,尤其是Eloquent对数据库的操作提供了特别多的便利。
在实际开发中我们经常涉及到分库分表场景,那么怎样才能继续配合Eloquent优雅的使用Model模型呢,接下来给大家分享下我在实际开发中所遇到的问题。(备注:此方法来源 Stack OverFlow原文地址找不到了,配合我们实际项目更能清晰表述)
1、假设我们有一万本书籍,每本书籍有两千章节,我们创建数据库时的表结构是书籍信息表:books;以及章节信息表:chapters,前面说到书籍越多章节数也就越多解决方案是将章节表分成十个形式为chapters_0、chapters_1、......chapters_9表后缀规则是书籍ID与10取余,这样所有的书籍章节会分散在这10个chapters中。
2、表建好后开始创建model模型,按照惯例所有的模型都将写在App\Models下;首先我们先创建一个类名为Model的模型并继承Illuminate\Database\Eloquent\Model

<?php

namespace App\Models;

use Illuminate\Database\Eloquent\Model as EloquentModel;

class Model extends EloquentModel
{
    protected $suffix = null;

    // 设置表后缀
    public function setSuffix($suffix)
    {
        $this->suffix = $suffix;
        if ($suffix !== null) {
            $this->table = $this->getTable() . '_' . $suffix;
        }
    }

    // 提供一个静态方法设置表后缀
    public static function suffix($suffix)
    {
        $instance = new static;
        $instance->setSuffix($suffix);

        return $instance->newQuery();
    }

    // 创建新的"chapters_{$suffix}"的模型实例并返回
    public function newInstance($attributes = [], $exists = false)
    {
        $model = parent::newInstance($attributes, $exists);
        $model->setSuffix($this->suffix);

        return $model;
    }
}

2、其他模型全都继承以上的Model而不是继承Illuminate\Database\Eloquent\Model,获取某本书的章节controller

<?php

namespace App\Http\Controllers;

use App\Models\{Book, Chapter};

class ChaptersController extends Controller
{
    public function chapter (Book $book)
    {
        // 章节列表(普通查询)
        $list = Chapter::lists($book->id);

        // 章节列表(使用模型关联)
        $list = $book->chapters()->oldest('id')->get();
    }
}

3、chapter模型(普通查询)

<?php

namespace App\Models;

class Chapter extends Model
{
    public static function lists ($bookId)
    {
        $suffix = $bookId % 10;
       /*
        * 例如 $sufiix = 1; 我要要获取的就是:chapters_1的模型实例
        * 使用Model类中提供的静态方法创建该表的模型实例
        * 返回指定书籍的章节
        */
        return self::suffix($suffix)->where('book_id', $bookId)->get();
    }
}

3、好了,我们章节的分表模型已经完成了。那么如何使用模型关联呢?我们来看Book模型如何关联Chapter

<?php

namespace App\Models;

use Illuminate\Database\Eloquent\Relations\HasMany;

class Book extends Model
{
    public function chapters ()
    {
        /*
        * books表的id和chapters表中的book_id关联
        * 一对多关系(一本书对应多条章节)
        */
        $instance = new Chapter();
        $instance->setSuffix($this->id % 10);

        $foreignKey = $instance->getTable . '.' . $this->getForeignKey();
        $localKey = $this->getKeyName();

        return new HasMany($instance->newQuery(), $this, $foreignKey, $localKey);
    }
}

到此model发表查询及model关联就完成了,如果有其他更好的方式,请大家不吝赐教。第一次发表文章,如有不对的地方希望大家多多指教!!

本作品采用《CC 协议》,转载必须注明作者和本文链接
本帖由系统于 4年前 自动加精
Herbie
《L04 微信小程序从零到发布》
从小程序个人账户申请开始,带你一步步进行开发一个微信小程序,直到提交微信控制台上线发布。
《G01 Go 实战入门》
从零开始带你一步步开发一个 Go 博客项目,让你在最短的时间内学会使用 Go 进行编码。项目结构很大程度上参考了 Laravel。
讨论数量: 18

不明觉厉

4年前 评论

直接重写 getTable 方法会好点?

4年前 评论
lufeijun1234

厉害厉害,涨知识咯

4年前 评论
幽弥狂

那如果这十张表的存储还不够用,怎么解决呢?

面试的时候被问到过,每天写5000w的数据表,该怎么处理~

4年前 评论

楼主老哥,请问这个方法是不是少了点儿什么?需要加这个return $this;

// 设置表后缀
    public function setSuffix($suffix)
    {
        $this->suffix = $suffix;
        if ($suffix !== null) {
            $this->table = $this->getTable() . '_' . $suffix;
        }
    }
4年前 评论
Herbie (楼主) 3年前

作者整的太复杂了,明天我上传一个简单的,简单一个单例就搞定了

3年前 评论
mrchenzifan 3年前
jackven 2年前

简单的查询大家可能都明白,但是如果是新增和修改呢?如果按照普通model的方式又该如何去写?麻烦给点思路。

3年前 评论
Herbie (楼主) 3年前
游离不2

能支持 with ?

3年前 评论
Herbie (楼主) 3年前
游离不2 (作者) 3年前
Jeremy_zym 2年前

讨论应以学习和精进为目的。请勿发布不友善或者负能量的内容,与人为善,比聪明更重要!