资讯专栏INFORMATION COLUMN

20 个 Laravel Eloquent 必备的实用技巧

clasnake / 2972人阅读

摘要:看起来是一个简单的机制,但是在底层,有很多半隐藏的函数和鲜为人知的方式来实现更多功能。在这篇文章中,我将演示几个小技巧。另外,在里也有些和时间相关的预定义方法通过关系排序一个复杂一点的技巧。幸运的是,确实有这样的方法。

Eloquent ORM 看起来是一个简单的机制,但是在底层,有很多半隐藏的函数和鲜为人知的方式来实现更多功能。在这篇文章中,我将演示几个小技巧。

1. 递增和递减

要代替以下实现:

</>复制代码

  1. $article = Article::find($article_id);
  2. $article->read_count++;
  3. $article->save();

你可以这样做:

</>复制代码

  1. $article = Article::find($article_id);
  2. $article->increment("read_count");

以下这些方法也可以实现:

</>复制代码

  1. Article::find($article_id)->increment("read_count");
  2. Article::find($article_id)->increment("read_count", 10); // +10
  3. Product::find($produce_id)->decrement("stock"); // -1

*

2. 先执行 X 方法,X 方法执行不成功则执行 Y 方法

Eloquent 有相当一部分函数可以把两个方法结合在一起使用, 例如 『 请先执行 X 方法, X 方法执行不成功则执行 Y 方法 』。

实例 1 -- findOrFail():

要替代以下代码的实现:

</>复制代码

  1. $user = User::find($id);
  2. if (!$user) { abort (404); }

你可以这样写:

</>复制代码

  1. $user = User::findOrFail($id);

实例 2 -- firstOrCreate():

要替代以下代码的实现:

</>复制代码

  1. $user = User::where("email", $email)->first();
  2. if (!$user) {
  3. User::create([
  4. "email" => $email
  5. ]);
  6. }

这样写就可以了:

</>复制代码

  1. $user = User::firstOrCreate(["email" => $email]);

*

3. 模型的 boot() 方法

在一个 Eloquent 模型中,有个神奇的地方,叫 boot(),在那里,你可以覆盖默认的行为:

</>复制代码

  1. class User extends Model
  2. {
  3. public static function boot()
  4. {
  5. parent::boot();
  6. static::updating(function($model)
  7. {
  8. // 写点日志啥的
  9. // 覆盖一些属性,类似这样 $model->something = transform($something);
  10. });
  11. }
  12. }

在创建模型对象时设置某些字段的值,大概是最受欢迎的例子之一了。 一起来看看在创建模型对象时,你想要生成 UUID 字段 该怎么做。

</>复制代码

  1. public static function boot()
  2. {
  3. parent::boot();
  4. self::creating(function ($model) {
  5. $model->uuid = (string)Uuid::generate();
  6. });
  7. }

*

4. 带条件与排序的关联关系

定义关联关系的一般方式:

</>复制代码

  1. public function users() {
  2. return $this->hasMany("AppUser");
  3. }

你知道吗?也可以在上面的基础上增加 where 或者 orderBy?
举个例子,如果你想关联某些类型的用户,同时使用 email 字段排序,你可以这样做:

</>复制代码

  1. public function approvedUsers() {
  2. return $this->hasMany("AppUser")->where("approved", 1)->orderBy("email");
  3. }

*

5. 模型特性:时间、追加等

Eloquent模型有些参数,使用类的属性形式。最常用是:

</>复制代码

  1. class User extends Model {
  2. protected $table = "users";
  3. protected $fillable = ["email", "password"]; // 可以被批量赋值字段,如 User::create() 新增时,可使用字段
  4. protected $dates = ["created_at", "deleted_at"]; // 需要被Carbon维护的字段名
  5. protected $appends = ["field1", "field2"]; // json返回时,附加的字段
  6. }

不只这些,还有:

</>复制代码

  1. protected $primaryKey = "uuid"; // 更换主键
  2. public $incrementing = false; // 设置 不自增长
  3. protected $perPage = 25; // 定义分页每页显示数量(默认15)
  4. const CREATED_AT = "created_at";
  5. const UPDATED_AT = "updated_at"; //重写 时间字段名
  6. public $timestamps = false; // 设置不需要维护时间字段

还有更多,我只列出了一些有意思的特性,具体参考文档 abstract Model class 了解所有特性.

*

6. 通过 ID 查询多条记录

所有人都知道 find() 方法,对吧?

</>复制代码

  1. $user = User::find(1);

我十分意外竟然很少人知道这个方法可以接受多个 ID 的数组作为参数:

</>复制代码

  1. $users = User::find([1,2,3]);

*

7. WhereX

有一种优雅的方式能将这种代码:

</>复制代码

  1. $users = User::where("approved", 1)->get();

转换成这种:

</>复制代码

  1. $users = User::whereApproved(1)->get();

对,你没有看错,使用字段名作为后缀添加到 where 后面,它就能通过魔术方法运行了。

另外,在 Eloquent 里也有些和时间相关的预定义方法:

</>复制代码

  1. User::whereDate("created_at", date("Y-m-d"));
  2. User::whereDay("created_at", date("d"));
  3. User::whereMonth("created_at", date("m"));
  4. User::whereYear("created_at", date("Y"));

*

8. 通过关系排序

一个复杂一点的「技巧」。你想对论坛话题按最新发布的帖子来排序?论坛中最新更新的主题在最前面是很常见的需求,对吧?

首先,为主题的最新帖子定义一个多带带的关系:

</>复制代码

  1. public function latestPost()
  2. {
  3. return $this->hasOne(AppPost::class)->latest();
  4. }

然后,在控制器中,我们可以实现这个「魔法」:

</>复制代码

  1. $users = Topic::with("latestPost")->get()->sortByDesc("latestPost.created_at");

*

9. Eloquent::when() -- 不再使用 if-else

很多人都喜欢使用"if-else"来写查询条件,像这样:

</>复制代码

  1. if (request("filter_by") == "likes") {
  2. $query->where("likes", ">", request("likes_amount", 0));
  3. }
  4. if (request("filter_by") == "date") {
  5. $query->orderBy("created_at", request("ordering_rule", "desc"));
  6. }

有一种更好的方法 -- 使用 when()

</>复制代码

  1. $query = Author::query();
  2. $query->when(request("filter_by") == "likes", function ($q) {
  3. return $q->where("likes", ">", request("likes_amount", 0));
  4. });
  5. $query->when(request("filter_by") == "date", function ($q) {
  6. return $q->orderBy("created_at", request("ordering_rule", "desc"));
  7. });

它可能看上去不是很优雅,但它强大的功能是传递参数:

</>复制代码

  1. $query = User::query();
  2. $query->when(request("role", false), function ($q, $role) {
  3. return $q->where("role_id", $role);
  4. });
  5. $authors = $query->get();

*

10. 一对多返回默认模型对象

假设现在有种情况是要显示文章的作者,然后模板代码是:

</>复制代码

  1. {{ $post->author->name }}

但是如果作者的信息被删除或者因为某些原因没有被设置。代码会返回一个错误,诸如 "property of non-object"。

当然你可以这样处理:

</>复制代码

  1. {{ $post->author->name ?? "" }}

你可以通过 Eloquent 关系这样做:

</>复制代码

  1. public function author()
  2. {
  3. return $this->belongsTo("AppAuthor")->withDefault();
  4. }

在此示例中,如果文字没有作者的信息, author() 会返回一个空的 AppAuthor 模型对象。

再者,我们也可以给默认的模型对象里面的属性赋默认值。

</>复制代码

  1. public function author()
  2. {
  3. return $this->belongsTo("AppAuthor")->withDefault([
  4. "name" => "Guest Author"
  5. ]);
  6. }

*

11. 通过赋值函数进行排序

想象一下你有这样的代码:

</>复制代码

  1. function getFullNameAttribute()
  2. {
  3. return $this->attributes["first_name"] . " " . $this->attributes["last_name"];
  4. }

现在,你想要通过 "full_name" 进行排序? 发现是没有效果的:

</>复制代码

  1. $clients = Client::orderBy("full_name")->get(); //没有效果

解决办法很简单.我们需要在获取结果后对结果进行排序.

</>复制代码

  1. $clients = Client::get()->sortBy("full_name"); // 成功!

注意的是方法名称是不相同的 -- 它不是orderBy,而是sortBy

*

12. 全局作用域下的默认排序

如果你想要 User::all() 总是按照 name 字段来排序呢? 你可以给它分配一个全局作用域。让我们回到 boot() 这个我们在上文提到过的方法:

</>复制代码

  1. protected static function boot()
  2. {
  3. parent::boot();
  4. // 按照 name 正序排序
  5. static::addGlobalScope("order", function (Builder $builder) {
  6. $builder->orderBy("name", "asc");
  7. });
  8. }

扩展阅读 查询作用域 。

*

13. 原生查询方法

有时候,我们需要在 Eloquent 语句中添加原生查询。 幸运的是,确实有这样的方法。

</>复制代码

  1. // whereRaw
  2. $orders = DB::table("orders")
  3. ->whereRaw("price > IF(state = "TX", ?, 100)", [200])
  4. ->get();
  5. // havingRaw
  6. Product::groupBy("category_id")->havingRaw("COUNT(*) > 1")->get();
  7. // orderByRaw
  8. User::where("created_at", ">", "2016-01-01")
  9. ->orderByRaw("(updated_at - created_at) desc")
  10. ->get();

*

14. 复制:复制一行的副本

很简单。说明不是很深入,下面是复制数据库实体(一条数据)的最佳方法:

</>复制代码

  1. $task = Tasks::find(1);
  2. $newTask = $task->replicate();
  3. $newTask->save();

*

15. Chunk() 方法之大块数据

与 Eloquent 不完全相关,它更多的关于 Collection (集合),但是对于处理大数据集合,仍然是很有用的。你可以使用 chunk() 将这些数据分割成小数据块

修改前:

</>复制代码

  1. $users = User::all();
  2. foreach ($users as $user) {
  3. // ...

你可以这样做:

</>复制代码

  1. User::chunk(100, function ($users) {
  2. foreach ($users as $user) {
  3. // ...
  4. }
  5. });

*

16. 创建模型时创建额外的东西

我们都知道Artisan命令:

</>复制代码

  1. php artisan make:model Company

但是,你知道有三个有用的标记可以为模型生成相关文件吗?

</>复制代码

  1. php artisan make:model Company -mcr

-m 将创建一个迁移文件

-c 将创建一个控制器

-r 表示控制器应该是一个资源控制器

*

17. 调用 save 方法的时候指定 updated_at

你知道  ->save() 方法可以接受参数吗? 我们可以通过传入参数阻止它的默认行为:更新 updated_at 的值为当前时间戳。

</>复制代码

  1. $product = Product::find($id);
  2. $product->updated_at = "2019-01-01 10:00:00";
  3. $product->save(["timestamps" => false]);

这样,我们成功在 save 时指定了 updated_at 的值。

*

18. update() 的结果是什么?

你是否想知道这段代码实际上返回什么?

</>复制代码

  1. $result = $products->whereNull("category_id")->update(["category_id" => 2]);

我是说,更新操作是在数据库中执行的,但 $result 会包含什么?

答案是受影响的行。 因此如果你想检查多少行受影响, 你不需要额外调用其他任何内容 -- update() 方法会给你返回此数字。

*

19. 把括号转换成 Eloquent 查询

如果你有个 andor 混合的 SQL 查询,像这样子的:

</>复制代码

  1. ... WHERE (gender = "Male" and age >= 18) or (gender = "Female" and age >= 65)

怎么用 Eloquent 来翻译它呢? 下面是一种错误的方式:

</>复制代码

  1. $q->where("gender", "Male");
  2. $q->orWhere("age", ">=", 18);
  3. $q->where("gender", "Female");
  4. $q->orWhere("age", ">=", 65);

顺序就没对。正确的打开方式稍微复杂点,使用闭包作为子查询:

</>复制代码

  1. $q->where(function ($query) {
  2. $query->where("gender", "Male")
  3. ->where("age", ">=", 18);
  4. })->orWhere(function($query) {
  5. $query->where("gender", "Female")
  6. ->where("age", ">=", 65);
  7. })

*

20. 复数参数的 orWhere

终于,你可以传递阵列参数给 orWhere()。平常的方式:

</>复制代码

  1. $q->where("a", 1);
  2. $q->orWhere("b", 2);
  3. $q->orWhere("c", 3);

你可以这样做:

</>复制代码

  1. $q->where("a", 1);
  2. $q->orWhere(["b" => 2, "c" => 3]);

*

我很确定还有更多隐藏的秘诀,但我希望至少上面的其中一些对你来说是新的。

</>复制代码

  1. 更多现代化 PHP 知识,请前往 Laravel / PHP 知识社区

文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。

转载请注明本文地址:https://www.ucloud.cn/yun/28610.html

相关文章

  • 人整理, 阅读过好文章 (每天随时更新)

    摘要:大家有好的文章可以在评论下面分享出来共同进步本文链接数组使用之道程序员进阶学习书籍参考指南教你在不使用框架的情况下也能写出现代化代码巧用数组函数框架中间件实现没错,这就是面向对象编程设计模式需要遵循的个基本原则令人困惑的在中使用协程实现多任 大家有好的文章,可以在评论下面分享出来, 共同进步! 本文github链接 php PHP 数组使用之道 PHP程序员进阶学习书籍参考指南 教你...

    Chiclaim 评论0 收藏0
  • 下载量最高 100 Laravel 扩展包推荐

    摘要:本文经授权转自社区,后续更新将以帖子内容和内容为准。说明另一个令人喜欢的地方,是拥有活跃的开发者社区,而活跃的开发者社区带来的,是繁华的扩展包生态。本文对上打了标签的扩展包进行整理,截止到现在年月号,有超过个扩展包,以下是下载量最大的个。 本文经授权转自 PHPHub 社区,后续更新将以 PHPHub 帖子内容 和 GitHub 内容 为准。 说明 Laravel 另一个令人喜欢的地方...

    Tychio 评论0 收藏0
  • laravel 数据迁移与 Eloquent ORM

    摘要:同时使用数据迁移管理数据库,可以与团队进行共享以及编辑。实际项目根据需求进行记录,以及选择存储方式。使用命令可以很方便的创建模型以及数据迁移。,参数在创建模型的同时也创建了数据迁移文件。参考资料数据库操作迁移快速入门。 导语 数据库可以说是后端开发最常用,也是最重要的部分。laravel 提供了很实用的 Eloquent ORM 模型类,简单、直观的与数据库进行交互。同时使用数据迁移管...

    tulayang 评论0 收藏0
  • 分享 10 你可能不知道 Laravel Eloquent技巧

    摘要:是一个功能丰富的框架。但是,你无法从官方文档中找到所有可用的功能。例数据库又插入一条为的数据。也很乐意听到你对此的看法和想法。你可以在上找到我。 showImg(https://segmentfault.com/img/remote/1460000017973901?w=800&h=511); Laravel 是一个功能丰富的框架。但是,你无法从官方文档中找到所有可用的功能。以下是一些...

    Simon_Zhou 评论0 收藏0
  • 人人必备10 Laravel 4 扩展包

    摘要:更多扩展包中有丰富的扩展包来帮你完成几乎任何你想实现的功能。我们不能把所有的扩展包都整理出来,然而,这里还是列出了一些很有用的。总之,你几乎总是能够找到一个扩展包可以解决你当前的问题。 Laravel 是一个非常流行且简单易用的PHP框架,它提供了很多基础的工具(如 RESTful 路由、内置的ORM、模版等)使你能够快速的创建应用。这意味着你可以花费更少的时间来建立应用程序的模版,给...

    darkbug 评论0 收藏0

发表评论

0条评论

最新活动
阅读需要支付1元查看
<