收集一些模型的使用技巧

# 1. 嵌套作用域查询

Laravel 支持将查用的查询封装为作用域(opens new window)

一bbs网站,首页的列表排序功能,可按照发布时间和回复时间排序

Controller中

    public function index(Request $request, Topic $topic, User $user)
    {
        // scopeWithOrder
        $topics = $topic->withOrder($request->order)
                        ->with('user', 'category')  // 预加载防止 N+1 问题
                        ->paginate(20);


        return view('topics.index', compact('topics'));
    }
1
2
3
4
5
6
7
8
9
10

其中 withOrder 是 本地作用域

Model 中 scopeWithOrder 又包含了两个小作用域

    public function scopeWithOrder($query, $order)
    {
        // 不同的排序,使用不同的数据读取逻辑
        switch ($order) {
            case 'recent':
                $query->recent();
                break;

            default:
                $query->recentReplied();
                break;
        }
    }
    
    public function scopeRecentReplied($query)
    {
        // 当话题有新回复时,我们将编写逻辑来更新话题模型的 reply_count 属性,
        // 此时会自动触发框架对数据模型 updated_at 时间戳的更新
        return $query->orderBy('updated_at', 'desc');
    }

    public function scopeRecent($query)
    {
        // 按照创建时间排序
        return $query->orderBy('created_at', 'desc');
    }
    
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27

# 2. 在 find 方法中指定属性

User::find(1, ['name', 'email']);
User::findOrFail(1, ['name', 'email']);
1
2

# 3. Clone 一个 Model

$user = User::find(1);
$newUser = $user->replicate();
$newUser->save();
1
2
3

# 4. 判断两个 Model 是否相同

$user = User::find(1);
$sameUser = User::find(1);
$diffUser = User::find(2);
$user->is($sameUser); // true
$user->is($diffUser); // false;
1
2
3
4
5

# 5. 重新加载一个 Model

$user = User::find(1);
$user->name; // 'Peter'
// 如果 name 更新过,比如由 peter 更新为 John
$user->refresh();
$user->name; // John
1
2
3
4
5

# 6. 加载新的 Model

$user = App\User::first();
$user->name;    // John
//
$updatedUser = $user->fresh(); 
$updatedUser->name;  // Peter
$user->name;    // John
1
2
3
4
5
6

# 7. 更新带关联的 Model

在更新关联的时候,使用 push 方法可以更新所有 Model

class User extends Model
{
 public function phone()
 {
  return $this->hasOne('App\Phone');
 }
}
$user = User::first();
$user->name = "Peter";
$user->phone->number = '1234567890';
$user->save(); // 只更新 User Model
$user->push(); // 更新 User 和 Phone Model
1
2
3
4
5
6
7
8
9
10
11
12

# 8. 自定义软删除字段

Laravel 默认使用 deleted_at 作为软删除字段,我们通过以下方式将 deleted_at 改成 is_deleted

class User extends Model
{
 use SoftDeletes;
  * deleted_at 字段.
  *
  * @var string
  */
 const DELETED_AT = 'is_deleted';
}
1
2
3
4
5
6
7
8
9

或者使用访问器

class User extends Model
{
 use SoftDeletes;
  
 public function getDeletedAtColumn(){
  return 'is_deleted';
 }
}
1
2
3
4
5
6
7
8

# 9. 查询 Model 更改的属性

$user = User::first();
$user->name; // John
$user->name = 'Peter';
$user->save();
 
dd($user->getChanges());
// 输出:
[
 'name' => 'John',
 'updated_at' => '...'
]
1
2
3
4
5
6
7
8
9
10
11

# 10. 查询 Model 是否已更改

$user = User::first();
$user->name;    // John
$user->isDirty();  // false 
$user->name = 'Peter'; 
$user->isDirty();  // true
$user->getDirty();  // ['name' => 'Peter']
$user->save();   
$user->isDirty();  // false
1
2
3
4
5
6
7
8

getChanges() 与 getDirty() 的区别

getChanges() 方法用在 save() 方法之后输出结果集

getDirty() 方法用在 save() 方法之前输出结果集

# 11. 查询修改前的 Model 信息

$user = App\User::first();
$user->name;     //John
$user->name = "Peter";   //Peter
$user->getOriginal('name'); //John
$user->getOriginal();   //Original $user record
1
2
3
4
5

# 12. 使用 withDefault 保持返回格式统一

public function _city()
{
    return $this->hasOne(City::class, 'id', 'city_id');
}
1
2
3
4

比如student和city是一对一关系,如果一个student表中city字段为空,返回的结果可能是

{name: "jack", _city: null}

这样会造成的问题是前端如果使用了student._city.name会造成undefined等错误。 为了避免可以改为

public function _city()
{
    return $this->hasOne(City::class, 'id', 'city_id')->withDefault([
      'name' => '',
    ]);
}
1
2
3
4
5
6

这样即使找不到也不会报错 返回的结果是: {name: "jack", _city: {id: null, name: ""}}

上次更新: 10/29/2020, 3:06:15 PM