深入理解 Laravel Eloquent(三)——模型間關系(關聯)


Eloquent是什么

Eloquent 是一個 ORM,全稱為 Object Relational Mapping,翻譯為 “對象關系映射”(如果只把它當成 Database Abstraction Layer 數組庫抽象層那就太小看它了)。所謂 “對象”,就是本文所說的 “模型(Model)”;對象關系映射,即為模型間關系。中文文檔: http://laravel-china.org/docs/eloquent#relationships

下面我們開始一個一個地學習。

一對一關系

顧名思義,這描述的是兩個模型之間一對一的關系。這種關系是不需要中間表的。

假如我們有兩個模型:User 和 Account,分別對應注冊用戶和消費者,他們是一對一的關系,那么如果我們要使用 Eloquent 提供的一對一關系方法,表結構應該是這樣的:

user: id ... ... account_id

account: id ... ... user_id

假設我們需要在 User 模型中查詢對應的 Account 表的信息,那么代碼應該是這樣的。 `/app/models/User.php`:

<?php

class User extends Eloquent {

  

  protected $table = 'users';

  public function hasOneAccount()

  {

      return $this->hasOne('Account', 'user_id', 'id');

  }

}

然后,當我們需要用到這種關系的時候,該如何使用呢?如下:

$account = User::find(10)->hasOneAccount;

此時得到的 `$account` 即為 `Account` 類的一個實例。


這里最難的地方在於后面的兩個 foreign_key 和 local_key 的設置,大家可以就此記住:在 User 類中,無論 hasOne 誰,第二個參數都是 `user_id`,第三個參數一般都是 `id`。由於前面的 `find(10)` 已經鎖定了 id = 10,所以這段函數對應的 SQL 為: `select * from account where user_id=10`。


這段代碼除了展示了一對一關系該如何使用之外,還傳達了三點信息,也是我對於大家使用 Eloquent 時候的建議:

1. 每一個 Model 中都指定表名

2. has one account 這樣的關系寫成 `hasOneAccount()` 而不是簡單的 `account()`

3. 每次使用模型間關系的時候都寫全參數,不要省略

相應的,如果使用 belongsTo() 關系,應該這么寫:

<?php

class Account extends Eloquent {

  protected $table = 'accounts';

  

  public function belongsToUser()

  {

    return $this->belongsTo('User', 'user_id', 'id');

  }

}

一對多關系

學會了前面使用一對一關系的基礎方法,后面的幾種關系就簡單多了。

我們引入一個新的Model:Pay,付款記錄。表結構應該是這樣的:

user: id ... ...

pay: id ... ... user_id

User 和 Pay 具有一對多關系,換句話說就是一個 User 可以有多個 Pay,這樣的話,只在 Pay 表中存在一個 `user_id` 字段即可。 `/app/models/User.php`:

<?php

class User extends Eloquent {

  

  protected $table = 'users';

  public function hasManyPays()

  {

    return $this->hasMany('Pay', 'user_id', 'id');

  }

}

然后,當我們需要用到這種關系的時候,該如何使用呢?如下:

$accounts = User::find(10)->hasManyPays()->get();

此時得到的 `$accounts` 即為 `Illuminate\Database\Eloquent\Collection` 類的一個實例。大家應該也已經注意到了,這里不是簡單的 `-> hasOneAccount` 而是 `->hasManyPays()->get()`,為什么呢?因為這里是 `hasMany`,操作的是一個對象集合。

相應的 belongsTo() 的用法跟上面一對一關系一樣:

<?php

class Pay extends Eloquent {

  protected $table = 'pays';

  

  public function belongsToUser()

  {

    return $this->belongsTo('User', 'user_id', 'id');

  }

}

多對多關系

多對多關系和之前的關系完全不一樣,因為多對多關系可能出現很多冗余數據,用之前自帶的表存不下了。

我們定義兩個模型:Article 和 Tag,分別表示文章和標簽,他們是多對多的關系。表結構應該是這樣的:

article: id ... ...

tag: id ... ...

article_tag: article_id tag_id

在 Model 中使用:

<?php

class Tag extends Eloquent {

  protected $table = 'tags';

  

  public function belongsToManyArticle()

  {

    return $this->belongsToMany('Article', 'article_tag', 'tag_id', 'article_id');

  }

}

需要注意的是,第三個參數是本類的 id,第四個參數是第一個參數那個類的 id。

使用跟 hasMany 一樣:

$tagsWithArticles = Tag::take(10)->get()->belongsToManyArticle()->get();

這里會得到一個非常復雜的對象,可以自行 `var_dump()`。跟大家說一個訣竅,`var_dump()` 以后,用 Chrome 右鍵 “查看源代碼”,就可以看到非常整齊的對象/數組展開了。

在這里給大家展示一個少見用法(奇技淫巧):

public function parent_video()

{

    return $this->belongsToMany($this, 'video_hierarchy', 'video_id', 'video_parent_id');

}

public function children_video()

{

    return $this->belongsToMany($this, 'video_hierarchy', 'video_parent_id', 'video_id');

}

對,你沒有看錯,可以 belongsToMany 自己。

其他關系

Eloquent 還提供 “遠層一對多關聯”、“多態關聯” 和 “多態的多對多關聯” 這另外三種用法,經過上面的學習,我們已經掌握了 Eloquent 模型間關系的基本概念和使用方法,剩下的幾種不常用的方法就留到我們用到的時候再自己探索吧。

重要技巧:關系預載入

你也許已經發現了,在一對一關系中,如果我們需要一次性查詢出10個 User 並帶上對應的 Account 的話,那么就需要給數據庫打 1 + 10 條 SQL,這樣性能是很差的。我們可以使用一個重要的特性,關系預載入:http://laravel-china.org/docs/eloquent#eager-loading

直接上代碼:

$users = User::with('hasOneAccount')->take(10)->get()

這樣生成的 SQL 就是這個樣子的:

select * from account where id in (1, 2, 3, ... ...)

這樣 1 + 10 條 SQL 就變成了 1 + 1 條,性能大增。

 


至此,深入理解 Laravel Eloquent 系列文章到此結束。推薦繼續了解 軟刪除轉換成數組/JSON

END


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM