Laravel framework - Detailed introduction to the advanced parts of EloquentORM

黄舟
Release: 2023-03-06 19:32:01
Original
1618 people have browsed it

Association

One To One
Assume that Usermodel is associated with the Phone model. To define such an association, you need to define a phone method in the User model, which returns An association defined by the hasOne method

<?phpnamespace App;use Illuminate\Database\Eloquent\Model;class User extends Model{
    /**
     * Get the phone record associated with the user.
     */
    public function phone()
    {
        return $this->hasOne(&#39;App\Phone&#39;);
    }
}
Copy after login

The first parameter of the hasOne method is the model to be associated. After it is defined, you can use the following syntax to query the associated attributes

$phone = User::find(1)->phone;
Copy after login

Eloquent will assume the association The foreign key is based on the model name, so the Phone model will automatically use the user_id field as the foreign key, which can be overridden using the second and third parameters

return $this->hasOne(&#39;App\Phone&#39;, &#39;foreign_key&#39;);return $this->hasOne(&#39;App\Phone&#39;, &#39;foreign_key&#39;, &#39;local_key&#39;);
Copy after login

Define the reverse relationship

After defining the above model, you can use the User model to obtain the Phone model. Of course, you can also obtain the corresponding User through the Phone model. This uses the belongsTo method

<?phpnamespace App;use Illuminate\Database\Eloquent\Model;class Phone extends Model{
    /**
     * Get the user that owns the phone.
     */
    public function user()
    {
        return $this->belongsTo(&#39;App\User&#39;);        // return $this->belongsTo(&#39;App\User&#39;, &#39;foreign_key&#39;);
        // return $this->belongsTo(&#39;App\User&#39;, &#39;foreign_key&#39;, &#39;other_key&#39;);

    }
}
Copy after login

One To Many

Suppose there is a post with a lot of associated comment information. In this case, a one-to-many association should be used, using the hasMany method

<?phpnamespace App;use Illuminate\Database\Eloquent\Model;class Post extends Model{
    /**
     * Get the comments for the blog post.
     */
    public function comments()
    {
        return $this->hasMany(&#39;App\Comment&#39;);
    }
}
Copy after login

Query operation

$comments = App\Post::find(1)->comments;
foreach ($comments as $comment) {    //}

$comments = App\Post::find(1)->comments()->where(&#39;title&#39;, &#39;foo&#39;)->first();
Copy after login

Define reverse association

Reverse association also uses the belongsTo method, refer to the One To One section.

$comment = App\Comment::find(1);
echo $comment->post->title;
Copy after login

Many To Many

Many-to-many association is more complicated to implement than hasOne and hasMany because there is an additional intermediate table.

Consider a scenario where a user can belong to multiple roles, and a role can also belong to multiple users. This introduces three tables: users, roles, role_user. The role_user table is a related table and contains two fields user_id and role_id.

Many-to-many association requires the belongsToMany method

<?phpnamespace App;use Illuminate\Database\Eloquent\Model;class User extends Model{
    /**
     * The roles that belong to the user.
     */
    public function roles()
    {
        // 指定关联表
        // return $this->belongsToMany(&#39;App\Role&#39;, &#39;role_user&#39;);
        // 指定关联表,关联字段
        // return $this->belongsToMany(&#39;App\Role&#39;, &#39;role_user&#39;, &#39;user_id&#39;, &#39;role_id&#39;);

        return $this->belongsToMany(&#39;App\Role&#39;);
    }
}
Copy after login

The above defines that a user belongs to multiple roles. Once the relationship is established, you can query it

user = App\User::find(1);
foreach ($user->roles as $role) {    //}$roles = App\User::find(1)->roles()->orderBy(&#39;name&#39;)->get();
Copy after login

Reverse association Relationship

The reverse relationship is implemented the same as the forward relationship

<?phpnamespace App;use Illuminate\Database\Eloquent\Model;class Role extends Model{
    /**
     * The users that belong to the role.
     */
    public function users()
    {
        return $this->belongsToMany(&#39;App\User&#39;);
    }
}
Copy after login

Retrieve the column value of the intermediate table

For many-to-many relationships, an intermediate table is introduced, so There needs to be a way to query the column values ​​of the intermediate table, such as the time when the relationship was established, etc. Use the pivot attribute to query the intermediate table

$user = App\User::find(1);

foreach ($user->roles as $role) {
    echo $role->pivot->created_at;
}
Copy after login

The above code accesses the created_at field of the intermediate table.

Note that by default, the keys of the subsequent models can be accessed through the pivotobject. If the intermediate table contains additional attributes, you need to use the withPivot method when specifying the association. Explicitly specify the column name

return $this->belongsToMany(&#39;App\Role&#39;)->withPivot(&#39;column1&#39;, &#39;column2&#39;);
Copy after login

Has Many Through

This relationship is relatively powerful. Suppose there is a scenario: the Country model contains multiple User models, and each User model contains Multiple Post models, that is to say, there are many users in a country, and these users have many posts. We want to query all posts in a certain country. How to achieve this? This uses the Has Many Through relationship

countries    id - integer
    name - stringusers    id - integer
    country_id - integer
    name - stringposts    id - integer
    user_id - integer
    title - string
Copy after login

As you can see, the posts table does not directly contain country_id, but it establishes a relationship with the countries table through the users table.

Using the Has Many Through relationship

namespace App;

use Illuminate\Database\Eloquent\Model;class Country extends Model{    /**
     * Get all of the posts for the country.
     */
    public function posts()
    {        // return $this->hasManyThrough(&#39;App\Post&#39;, &#39;App\User&#39;, &#39;country_id&#39;, &#39;user_id&#39;);

        return $this->hasManyThrough(&#39;App\Post&#39;, &#39;App\User&#39;);
    }
}
Copy after login

The first parameter of the method hasManyThrough is the name of the model we wish to access, and the second parameter is the intermediate model name.

HasManyThrough hasManyThrough( 
    string $related, 
    string $through, 
    string|null $firstKey = null, 
    string|null $secondKey = null, 
    string|null $localKey = null)
Copy after login

Polymorphic Relations

Polymorphic relations allow the same model to belong to multiple different models using one association. Assume such a scenario, we have a post table and In a comment table, users can like both posts and comments. How to deal with this situation?

The table structure is as follows

posts    id - integer
    title - string
    body - textcomments    id - integer
    post_id - integer
    body - textlikes    id - integer
    likeable_id - integer
    likeable_type - string
Copy after login

As you can see, we use the likeable_type field in the likes table to determine whether the record likes a post or a comment. With the table structure in place, it is time to define the model.

<?php

namespace App;

use Illuminate\Database\Eloquent\Model;class Like extends Model{    /**
     * Get all of the owning likeable models.
     */
    public function likeable()
    {        return $this->morphTo();
    }
}class Post extends Model{    /**
     * Get all of the product&#39;s likes.
     */
    public function likes()
    {        return $this->morphMany(&#39;App\Like&#39;, &#39;likeable&#39;);
    }
}class Comment extends Model{    /**
     * Get all of the comment&#39;s likes.
     */
    public function likes()
    {        return $this->morphMany(&#39;App\Like&#39;, &#39;likeable&#39;);
    }
}
Copy after login

By default, the type of likeable_type is the complete name of the associated model, such as App\Post and App\Comment here.

Normally we may use a custom value to identify the associated table name. Therefore, we need to customize this value. We need to register the association in the boot method of the project's service provider object. , for example, in the boot method of AppServiceProvider

use Illuminate\Database\Eloquent\Relations\Relation;Relation::morphMap([    &#39;posts&#39; => App\Post::class,
    &#39;likes&#39; => App\Like::class,]);
Copy after login

Retrieve polymorphic relationships

Access all likes of a post

$post = App\Post::find(1);  
foreach ($post->likes as $like) {    //}
Copy after login

Access a liked post or comment

$like = App\Like::find(1);   
$likeable = $like->likeable;
Copy after login

In the above example, the returned likeable will return posts or comments based on the type of the record.

Many-to-many polymorphic association

Many-to-many association uses the methods morphToMany and morphedByMany, there will be no more nonsense here

Association relationship query

In Eloquent, all relationships are defined using functions , and related instances can be obtained without executing related queries. Suppose we have a blog system, and the User model is associated with many Post models:

/**
 * Get all of the posts for the user.
 */public function posts()
{   return $this->hasMany(&#39;App\Post&#39;);
}
Copy after login

You can query the association as follows and add additional constraints

$user = App\User::find(1);$user->posts()->where(&#39;active&#39;, 1)->get();
Copy after login

If not needed Add constraints to the associated attributes, which can be accessed directly as attributes of the model. For example, in the above example, we can use the following method to access User's Post

$user = App\User::find(1);foreach ($user->posts as $post) {    //}
Copy after login

动态的属性都是延迟加载的,它们只有在被访问的时候才会去查询数据库,与之对应的是预加载,预加载可以使用关联查询出所有数据,减少执行sql的数量。

查询关系存在性

使用has方法可以基于关系的存在性返回结果

// 检索至少有一个评论的所有帖子...$posts = App\Post::has(&#39;comments&#39;)->get();
// Retrieve all posts that have three or more comments...$posts = Post::has(&#39;comments&#39;, &#39;>=&#39;, 3)->get();
// Retrieve all posts that have at least one comment with votes...$posts = Post::has(&#39;comments.votes&#39;)->get();
Copy after login

如果需要更加强大的功能,可以使用whereHas和orWhereHas方法,把where条件放到has语句中。

// 检索所有至少存在一个匹配foo%的评论的帖子$posts = Post::whereHas(&#39;comments&#39;, function ($query) {    
$query->where(&#39;content&#39;, &#39;like&#39;, &#39;foo%&#39;);
})->get();
Copy after login

预加载

在访问Eloquent模型的时候,默认情况下所有的关联关系都是延迟加载的,在使用的时候才会开始加载,这就造成了需要执行大量的sql的问题,使用预加载功能可以使用关联查询出所有结果

<?phpnamespace App;use Illuminate\Database\Eloquent\Model;class Book extends Model{
    /**
     * Get the author that wrote the book.
     */
    public function author()
    {
        return $this->belongsTo(&#39;App\Author&#39;);
    }
}
Copy after login

接下来我们检索所有的书和他们的作者

$books = App\Book::all();
foreach ($books as $book) {    
echo $book->author->name;
}
Copy after login

上面的查询将会执行一个查询查询出所有的书,然后在遍历的时候再执行N个查询查询出作者信息,显然这样做是非常低效的,幸好我们还有预加载功能,可以将这N+1个查询减少到2个查询,在查询的时候,可以使用with方法指定哪个关系需要预加载。

$books = App\Book::with(&#39;author&#39;)->get();
foreach ($books as $book) {
    echo $book->author->name;
}
Copy after login

对于该操作,会执行下列两个sql

select * from books
select * from authors where id in (1, 2, 3, 4, 5, ...)
Copy after login

预加载多个关系

$books = App\Book::with(&#39;author&#39;, &#39;publisher&#39;)->get();
Copy after login

嵌套的预加载

$books = App\Book::with(&#39;author.contacts&#39;)->get();
Copy after login

带约束的预加载

$users = App\User::with([&#39;posts&#39; => function ($query) {
    $query->where(&#39;title&#39;, &#39;like&#39;, &#39;%first%&#39;);
}])->get();$users = App\User::with([&#39;posts&#39; => function ($query) {
    $query->orderBy(&#39;created_at&#39;, &#39;desc&#39;);
}])->get();
Copy after login

延迟预加载#

有时候,在上级模型已经检索出来之后,可能会需要预加载关联数据,可以使用load方法

$books = App\Book::all();if ($someCondition) {    $books->load(&#39;author&#39;, &#39;publisher&#39;);
}$books->load([&#39;author&#39; => function ($query) {
    $query->orderBy(&#39;published_date&#39;, &#39;asc&#39;);
}]);
Copy after login

关联模型插入

save方法

保存单个关联模型

$comment = new App\Comment([&#39;message&#39; => &#39;A new comment.&#39;]);
$post = App\Post::find(1);$post->comments()->save($comment);
Copy after login

保存多个关联模型

$post = App\Post::find(1); 
$post->comments()->saveMany([    
new App\Comment([&#39;message&#39; => &#39;A new comment.&#39;]),    
new App\Comment([&#39;message&#39; => &#39;Another comment.&#39;]),
]);
Copy after login

save方法和多对多关联

多对多关联可以为save的第二个参数指定关联表中的属性

App\User::find(1)->roles()->save($role, [&#39;expires&#39; => $expires]);
Copy after login

上述代码会更新中间表的expires字段。

create方法

使用create方法与save方法的不同在于它是使用数组的形式创建关联模型的

$post = App\Post::find(1);$comment = $post->comments()->create([
    &#39;message&#39; => &#39;A new comment.&#39;,]);
Copy after login

更新 “Belongs To” 关系

更新belongsTo关系的时候,可以使用associate方法,该方法会设置子模型的外键

$account = App\Account::find(10);
$user->account()->associate($account);
$user->save();
Copy after login

要移除belongsTo关系的话,使用dissociate方法

$user->account()->dissociate();$user->save();
Copy after login

Many to Many 关系

中间表查询条件#

当查询时需要对使用中间表作为查询条件时,可以使用wherePivotwherePivotInorWherePivotorWherePivotIn添加查询条件。

$enterprise->with([&#39;favorites&#39; => function($query) {    
$query->wherePivot(&#39;enterprise_id&#39;, &#39;=&#39;, 12)->select(&#39;id&#39;);
}]);
Copy after login

Attaching / Detaching#

$user = App\User::find(1);
// 为用户添加角色
$user->roles()->attach($roleId);
// 为用户添加角色,更新中间表的expires字段
$user->roles()->attach($roleId, [&#39;expires&#39; => $expires]);
// 移除用户的单个角色
$user->roles()->detach($roleId);
// 移除用户的所有角色
$user->roles()->detach();
Copy after login

attach和detach方法支持数组参数,同时添加和移除多个

$user = App\User::find(1);
$user->roles()->detach([1, 2, 3]);
$user->roles()->attach([1 => [&#39;expires&#39; => $expires], 2, 3]);
Copy after login

更新中间表(关联表)字段

使用updateExistingPivot方法更新中间表

$user = App\User::find(1);$user->roles()->updateExistingPivot($roleId, $attributes);
Copy after login

同步中间表(同步关联关系)#

使用sync方法,可以指定两个模型之间只存在指定的关联关系

$user->roles()->sync([1, 2, 3]);
$user->roles()->sync([1 => [&#39;expires&#39; => true], 2, 3]);
Copy after login

上述两个方法都会让用户只存在1,2,3三个角色,如果用户之前存在其他角色,则会被删除。

更新父模型的时间戳#

假设场景如下,我们为一个帖子增加了一个新的评论,我们希望这个时候帖子的更新时间会相应的改变,这种行为在Eloquent中是非常容易实现的。

在子模型中使用$touches属性实现该功能

<?php

namespace App;

use Illuminate\Database\Eloquent\Model;class Comment extends Model{    /**
     * All of the relationships to be touched.
     *
     * @var array
     */
    protected $touches = [&#39;post&#39;];    /**
     * Get the post that the comment belongs to.
     */
    public function post()
    {        return $this->belongsTo(&#39;App\Post&#39;);
    }
}
Copy after login

现在,更新评论的时候,帖子的updated_at字段也会被更新

$comment = App\Comment::find(1);$comment->text = &#39;Edit to this comment!&#39;;$comment->save();
Copy after login

The above is the detailed content of Laravel framework - Detailed introduction to the advanced parts of EloquentORM. For more information, please follow other related articles on the PHP Chinese website!

Related labels:
source:php.cn
Statement of this Website
The content of this article is voluntarily contributed by netizens, and the copyright belongs to the original author. This site does not assume corresponding legal responsibility. If you find any content suspected of plagiarism or infringement, please contact admin@php.cn
Popular Tutorials
More>
Latest Downloads
More>
Web Effects
Website Source Code
Website Materials
Front End Template
About us Disclaimer Sitemap
php.cn:Public welfare online PHP training,Help PHP learners grow quickly!