با مرور سه بخش از فصل ۱۲ با انواع روابط و تکنیکهای دسترسی به دادههای آنها دست پیدا کردید. اما ممکن است این سوال برای شما پیش آمده باشد که چگونه باید اطلاعات دو جدول که مرتبط به یکدیگر هستند را ذخیره، حذف و یا بروزرسانی کنیم. در ادامه با ما همراه باشید تا به توضیح این مبحث مهم بپردازیم.
برای مقدمه با یک مثال شروع میکنیم. فرض کنید شما دو جدول به نامهای posts و comments دارید. به ازای هر پست چندین کامنت وجود دارد و بنابراین یک رابطهی یک به چند ایجاد میشود. حال یک کلید خارجی در جدول comments حضور خواهد داشت که با ستون post_id مشخص شده است که این کلید اجباری است و مقدار null نمیپذیرد. بنابراین برای ذخیرهی یا حذف یا بروزرسانی یک نظر هموراه باید id آن پست به صورت خودکار به ستون اضافه شود.
Eloquent ابزار و متدی را جهت ذخیرهسازی یک مدل جدید یا به عبارتی یک رکورد جدید در اختیار ما قرار میدهد. این متد save نام دارد. فرض کنید میخواهیم یک رکورد جدید در جدول comments ثبت کنیم. بنابراین داریم:
$comment = new App\Comment(['message' => 'A new comment.']); $post = App\Post::find(1); $post->comments()->save($comment);
با این اجرای دستورهای فوق یک مدل جدید به نام comment میسازیم. سپس post موردنظر را با متد find پیدا میکنیم و در نهایت برای متد comments که در مدل Post.php تعریف شده است از دستور save استفاده کرده و مقدار متغییر comment$ را به عنوان آرگومان ورودی به آن ارسال خواهیم کرد.
همچنین برای ذخیره سازی تعداد بیشتری از نظرات میتوان از متد saveMany استفاده کرد:
$post = App\Post::find(1); $post->comments()->saveMany([ new App\Comment(['message' => 'A new comment.']), new App\Comment(['message' => 'Another comment.']), ]);
متد create
این متد دقیقا مشابه متد save و saveMany میباشد با این تفاوت که متد create آرایه خام PHP را هم به عنوان ورودی میپذیرد ولی متدهای save , saveMany همواره یک نمونهی کامل از شیء Eloquent را دریافت میکنند:
$post = App\Post::find(1); $comment = $post->comments()->create([ 'message' => 'A new comment.', ]);
در نظر داشته باشید که قبل از استفاده متد create باید همواره از صفت mass assignment در مدل خود استفاده کرده باشید. برای آگاهی از این صفت مقاله زیر را مطالعه بفرمایید:
روابط BelongsTo
هنگامیکه یک رابطهی belongsTo بروزرسانی میشود، باید و باید از متد associate استفاده شود. این متد یک کلید خارجی در مدل فرزند تنظیم میکند:
$account = App\Account::find(10); $user->account()->associate($account); $user->save();
هنگام حذف یک رابطهی belongsTo نیز باید از متد dissociate بهره ببریم. این متد کلید خارجی رابطه را به null تغییر میدهد:
$user->account()->dissociate(); $user->save();
مباحث مربوط به ذخیره سازی اطلاعات برای روابط چند به چند متفاوت تر است. و خبری از دستورهای save یا create و متدهای associate یا dissociate نیست.
اضافه کردن یک رکورد جدید
پس از اینکه روابط مشخصی بین مدلهای چند به چند برقرار شد باید به نحوهی ذخیره کردن اطلاعات رو بروزرسانی آنها در کنترلر بپردازیم. به عنوان مثال میخواهیم یک نقش برای کاربر تعریف کنیم. برای اینکار کاربر موردنظر را انتخاب کرده و با استفاده از دستور attach آن نقش را به کاربر ضمیمه یا پیوست میکنیم:
$user = User::find($user_id); $user->roles()->attach($role_id);
در نتیجه با اعمال دستور فوق در کنترلر یک ردیف یا رکورد به جدول role_user اضافه خواهد شد که مقدار ستونهای user_id و role_id آن معادل مقادیر متغییرهای user_id$ و role_id$ است.
حذف کردن یک رکورد
برای حذف کردن یک رکورد نیز باید از متد detach استفاده کرد. این متد نقشی که به کاربر پیوست شده است، را جدا میکند. برای استفاده از این متد به صورت زیر عمل میکنیم:
$user->roles()->detach($role_id);
همچنین برای حذف تمام نقشهای کاربران در جدول میانی role_user باید از دستور زیر استفاده کنیم:
$user->roles()->detach();
در نظر داشته باشید که متدهای attach و detach همواره مقادیر متفاوتی را به عنوان آرایه میپذیرند:
$user->roles()->attach([123, 456, 789]); $user->roles()->detach([321, 654, 987]);
بروزرسانی یک رکورد
و اما مهم ترین دستور که برای بروزرسانی این جداول مورد استفاده قرار میگیرد، دستور بروزرسانی است که با متد sync صورت میپذیرد. بنابراین برای بروزرسانی دستهای چند رکورد در جدول میانی باید از آرایه استفاده کرد:
$role->users()->sync([1, 2, 3]);
با اعمال دستور فوق به صورت دستهای نقشهای کاربران با idهای ۱ و ۲ و ۳ تغییر کرده و بروزرسانی میشود.
در صورتیکه بخواهیم یک رکورد را بدون detach کردن آی دی های مشخص بروزرسانی کنیم باید از دستور syncWithoutDetaching استفاده کنیم:
$user->roles()->syncWithoutDetaching([1, 2, 3]);
ذخیره اطلاعات اضافه در یک جدول میانی یا Pivot Table
همانطور که در مثال ذکر کردیم میتوان ستونهای مختلفی به جدول میانی یا pivot table با استفاده از متد withPivot اضافه کرد. مثلا فرض کنید به جدول میانی role_user یک ستون دیگر به نام registerDay اضافه کنیم. آنگاه تمام متدهای attach و detach و sync برای این پارامتر اضافه نیز اعمال میشود:
$user->roles()->attach(1, 'registerDay' => '12 bahman');
بروزرسانی یک رکورد در جدول میانی یا Pivot Table
درصورتیکه بخواهیم یک رکورد از جدول میانی یا Pivot Table را بروزرسانی کنیم باید از متد iupdateExistingPivot استفاده کنیم. این متد کلید خارجی یک رکورد میانی را به همراه آرایهای از صفات دریافت کرده و بروزرسانی میکند:
$user = App\User::find(1); $user->roles()->updateExistingPivot($roleId, $attributes);
گاها برای ما پیش آمده است که میخواهیم هنگامیکه ستون updated_at در یک مدل فرزند بروزرسانی شد، تاثیر آن روی مدل والد نیز مشخص شود و ستون updated_at در مدل و جدول والد نیز بروزرسانی شود. برای اینکه این کار به صورت اتوماتیک صورت پذیرد باید متغییر touches$
را به صورت زیر تعریف کرد و مدل یا مدلهای والد را درون آرایهای به آن انتساب داد:
<?php namespace App; use Illuminate\Database\Eloquent\Model; class Comment extends Model { /** * All of the relationships to be touched. * * @var array */ protected $touches = ['post']; /** * Get the post that the comment belongs to. */ public function post() { return $this->belongsTo('App\Post'); } }
بسیار عالیست! با مطالعه این بخش تمام روشهای بروزرسانی حذف یا اضافه کردن اطلاعات که دارای روابط جداول هستند را فرا گرفتید. فصل ۱۲ با ۴ بخش کلی به اتمام رسید و هم اکنون میتوانید به صورت کامل و بدون هیچگونه مشکل سیستم دیتابیس خود را طراحی و اجرا کنید.
در این قسمت، به پرسشهای تخصصی شما دربارهی محتوای مقاله پاسخ داده نمیشود. سوالات خود را اینجا بپرسید.