3分鐘短文:說說Laravel模型關聯關係最單純的“一對一”

程式設計師小助手發表於2020-10-12

引言

關係型資料庫提供了非常好用的資料關聯繫結模式,使用SQL語句可以方便地進行關聯資料查詢和操作。

如果所有關聯關係放到資料庫層面操作,勢必非常不便。

img

所以,laravel模型提供了關聯關係,本文就來梳理梳理那些用法。

程式碼時間

我們不要PPT似的念稿子,羅列出所有的關係模型,那樣不直觀也不是高效學習的方式。

還是從示例觸發,看看關聯關係到底解決的是什麼問題,以及如何使用。

首先是資料庫的準備,假設有如下兩個表的欄位對應關係:

pic

使用命令列建立一個Profile模型,並同時建立遷移檔案:

php artisan make:model Profile --migration

來看生成的 app/Profile.php 檔案的內容:

namespace App;
use Illuminate\Database\Eloquent\Model;
class Profile extends Model {}

因為使用了 –migration 選項,laravel自動生成了 database/migrations/2020_10_11_015236_create_profiles_table.php 檔案,

用於生成對應的資料庫 profiles 表。

還是老規矩,先實現資料庫遷移使用的 up 方法:

public function up()
{
    Schema::create('profiles', function(Blueprint $table)
    {
        $table->increments('id');
        $table->integer('user_id')->unsigned();
        $table->foreign('user_id')->references('id')->on('users');
        $table->string('url');
        $table->string('telephone');
        $table->timestamps();
    });
}

用於回滾的方法就不列出來了,僅僅是表的刪除。下面手動執行遷移指令:

php artisan migrate

輸出內容如下:

Migrated: 2020_10_11_015236_create_profiles_table.php

關聯關係

一切準備就緒,我們開始使用關聯關係來處理資料的一致性。一個user對應一個profile,所以這是一對一的關係,

在User模型裡新增如下宣告:

class User extends Model {
    public function profile()
    {
        return $this->hasOne('App\Profile');
    }
}

一旦定義了上述方法,就可以這樣鏈式呼叫了:

$user = User::find(1)->profile->telephone;

我們拆解開做解釋。首先使用 User::find($id) 返回的是一個 User 模型物件的例項。

該例項有一個 profile 方法,就是上面這段關係宣告。

呼叫 profile 返回的是一個 Profile 物件的例項,所以可以繼續呼叫Profile的屬性,也就是 telephone 的由來了。

需要特別注意的是,類似下面的寫法,返回結果是不同的:

$user = User::find($id);
$user->profile; // 返回 Profile 物件
$user->profile(); // 返回 hasOne 關聯關係物件

大家在使用的時候,一定不要迷惑,要分清楚使用方法。

有了關聯查詢,自然就有關聯更新,用法如下:

$profile = new Profile;
$profile->telephone = '12345678';
$user = User::find(1);
$user->profile()->save($profile);

有了關聯更新這種寫操作,自然就有了關聯刪除,模型方法的呼叫而已:

$user = User::find($id);
$user->profile()->delete();

由於是嚴格一對一的關係,也就是一個user只有一個profile,如果某個user被刪除了,

就會剩下一個孤零零的profile無所依附。

除了在程式上下文的一致性保證外,還可以使用資料庫的外來鍵,在刪除user時將profile關聯刪除。那麼只要修改遷移檔案,

並新增如下內容:

$table->integer('user_id')->unsigned();
$table->foreign('user_id')->references('id')->on('users')->onDelete('cascade');

寫在最後

本文介紹了laravel模型關聯的最簡單的“一對一”,我們從程式角度和資料庫角度講解了

如何在刪除資源時的一致性刪除。更復雜的關係,在程式設計層面是有意義的,我們下一章介紹更多的關聯關係。

Happy coding :-)

我是@程式設計師小助手,專注程式設計知識,圈子動態的IT領域原創作者

本作品採用《CC 協議》,轉載必須註明作者和本文連結
write-less-do-more-make-you-out-of-door

相關文章