分享六个能提高代码可读性的 Laravel Eloquent 小技巧

Laravel

Eloquent是Laravel默认使用的ORM。采用了活动记录模式。让你可以以一种更加轻松的方式与数据库进行交互。每个单独的模型都表示数据库中的一个表,你可以使用该表进行操作。在这篇文章中,我们将或多或少地向你展示你可能不知道的隐藏秘密、方法和属性,以改进你的代码。

蛇形命名 属性

蛇形命名属性是一个有趣的属性。让我们看看代码是怎么说的:

/**
 * 指示是否在数组上使用蛇形大小写属性。
 *
 * @var bool
 */
public static $snakeAttributes = true;

人们经常会错误地使用这个属性来更改访问属性的方式。许多人认为,如果更改这个属性,就可以使用驼峰式注释轻松访问属性。事实并非如此。我们强烈建议你不要使用它。当模型作为数组输出时,只需定义属性是驼峰式还是蛇形即可。

如果你想基于驼峰式命名,我们建议你查看 Kirk Bushell 的软件包 Eloquence by Kirk Bushell.

分页

如果你正在使用 Laravel 的 Eloquent ORM,那这对你是个好消息。 它提供了一种开箱即用的分页方法。你可能很熟悉这样的写法:

$comments = Comment::paginate(20);

使用此方法,你可以每页20个条目来为 comment model 分页。更改该值可以定义每页显示多少条数。如果不指定任何内容,则应用默认值,即15。

假设您希望在网站上的多个位置显示评论。每页总是有30条评论。如果必须在每个地方传递参数30,那就麻烦了。因此,您可以直接在模型上设置新的默认值。

protected $perPage = 30;

在model里面增加自定义值

Eloquent有一个称为“访问器”的强大功能。该功能允许你向模型或表中不存在的模型添加自定义字段。使用现有的值或定义全新的值都不重要,随时都可以回退。下面是访问器如何工作的示例。假设有一个名为user的模型,我们给他添加一个FullName的访问器:

function getFullNameAttribute() {
    return sprintf('%s %s', $this->first_name, $this->last_name);
}

现在你可以访问post模型上的full_name属性,如下:

User::latest()->first()->full_name;

如果返回对象(如集合),则此属性不会附加到用户模型。将protected$appends属性添加到模型中。它接受一个数组,其中包含一个或多个字段,从现在起应该自动追加这些字段。写成这样就可以了:

protected $appends = ['full_name'];

为不存在列增加一个变异器(属性设置器)

变异器与获取器相反。你可以用它做些很有意思的事情。 举个例子,转换不同类型的输入。 让我来给你详细说说。假设你想要保存一种类型的时间段。通常,你总是保存尽可能小的单位。在我们的案例中是秒。由于UX的原因,用户不想输入秒,例如在一个地方输入分钟,或者在另一个地方输入小时。这一切都可以很快解决。

class Video extends Model
{
    public function setDurationInMinutes($value)
    {
        $this->attributes['duration_in_seconds'] = $value * 60;
    }

    public function setDurationInHours($value)
    {
        $this->attributes['duration_in_seconds'] = $value * 60 * 60;
    }
}

上述代码意味着你可以使用一个不存在于数据表本身的字段。
model中使用的是duration_in_minutes字段,但是在后台,我们使用duration_in_seconds进行更新,也有可能使用一个不存在字段duration_in_hours。根据这个逻辑,我们在Controller如此调用:

class AnyController
{
    public function store()
    {
        $video->update([
            'title' => request('title'),
            'duration_in_minutes' => request('duration_in_minutes'),
        ]);
    }
}

这将节省你在控制器中进行计算的时间,你可以简单地使用不存在的列,并在执行某些计算时使用变异器将其结果映射到正确的字段上。

渴求式加载: with $with

让我们来谈谈关联关系。默认情况下,Laravel使用延迟加载。就关联关系而言,这意味着什么?延迟加载的好处是可以节省内存,因为并不是所有的数据都需要保留,我们可以在需要的时候加载数据。如下:

$comments = Comment::all();
foreach ($comments as $comment) {
    echo $comment->user->name;
}

在上面的例子中,我们会获取所有评论数据。然后遍历评论并显示每个评论的用户名。这段代码没什么问题,可以正常工作,但我们遇到了一个问题。延迟加载现在确保仅当我们想要输出用户名时才执行获取用户的查询。

欢迎来到你的第一个 N+1 问题。为什么是 N+1?N 始终是评论的数量,1 是获取评论的查询。例如,如果我们有500个评论,那么获取所有评论的查询被触发一次,然后一个查询获得相应的user-per评论。所以500+1查询。这意味着随着注释数量的增加,查询的数量也会增加。

为了防止出现这种情况,有一种称为渴求式加载的方法。

$comments = Comment::with('user')->get();
foreach ($comments as $comment) {
    echo $comment->user->name;
}

这会以两个查询结束。第一个查询获取所有注释,第二个查询立即获取所有关联用户。在后台,会发生以下情况(简化版SQL):

SELECT id, user_id, body FROM comments;
SELECT name FROM users WHERE user_id IN (1,2,3,4,5...);

不论是 10、500 还是 10000 条评论数据都不重要,我们都依旧只执行两次SQL查询。

好了,你现在已经看到如何使用渴求式加载了。但只限于如何手动使用。你还可以将整个过程自动化,以便某些关联关系总是自动通过渴求式方式加载。为此,需要给模型设定一个属性。

protected $with = [];

我们可以在Comment model简单设置 protected $with = ['user'];, 从现在起,user在任何时候都会自动加载。

我们还有很多种渴求式加载,有仅加载特定列、嵌套即时加载、多个即时加载等等。更多详情请Laravel文档或深入核心。

modelKeys 方法

有的时候需要查询所有的主键 ID, 查询是否复杂并不重要,大多数人可能会像这样做:

User::all()->pluck('id');

这个操作很 nice,但是返回的是一个集合,想要转换成数组的话可以使用 toArray()

User::all()->pluck('id')->toArray();

大多数情况下,上面的操作的可以简化成这样:

User::all()->modelKeys();

这种方式返回一个数组。重要的是这个方法并不会总是返回 id。 顾名思义,他是以数组的形式返回所有模型主键。主键默认是id,同时也可以在模型中定义主键名。

protected $primaryKey = 'id';
本文中的所有译文仅用于学习和交流目的,转载请务必注明文章译者、出处、和本文链接
我们的翻译工作遵照 CC 协议,如果我们的工作有侵犯到您的权益,请及时联系我们。

原文地址:https://laravel-news.com/6-eloquent-secr...

译文地址:https://learnku.com/laravel/t/42659

本帖已被设为精华帖!
本文为协同翻译文章,如您发现瑕疵请点击「改进」按钮提交优化建议
《L01 基础入门》
我们将带你从零开发一个项目并部署到线上,本课程教授 Web 开发中专业、实用的技能,如 Git 工作流、Laravel Mix 前端工作流等。
《L05 电商实战》
从零开发一个电商项目,功能包括电商后台、商品 & SKU 管理、购物车、订单管理、支付宝支付、微信支付、订单退款流程、优惠券等
讨论数量: 2

每次看小技巧分享都会有收获

3年前 评论
游离不2

不太建议 User::all()->modelKeys() 这种方式,建议使用 User::pluck('id')->all(),理由大家自己琢磨。这个故事告诉我们一个道理,有些技巧会有坑。

3年前 评论
sunrui318 3年前
游离不2 (作者) 3年前
JerryBool 3年前

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