模型更新经过修改器、自动完成及模型事件等 不等同于数据库更新 更新方法和新增方法使用同个方法 系统自动判断新增还是更新数据。

查找并更新

取出数据后 更改字段内容 用save方法更新数据。是最佳的更新方式

$user = User::get(1);
$user->name  = 'think';
$user->email = 'think@qq.com';
$user->save();

save返回影响的记录数 并只有当before_update事件返回false的时候返回false

复杂的查询条件 可用查询构造器来查询数据并更新

$user = User::where('status',1)->where('name','liuchen')->find();
$user->name     = 'think';
$user->email    = 'think@qq.com';
$user->save();

save更新数据 只会更新变化的数据 没有变化的数据不会重新更新。
如果要强制更新数据 使用

$user = User::get(1);
$user->name  = 'think';
$user->email = 'think@qq.com';
$user->force()->save();

无论修改后的数据是否和之前一样都会强制更新该字段的值

SQL函数

$user = User::get(1);
$user->name  = 'think';
$user->email = 'think@qq.com';
$user->score=  Db::raw('score+1');
$user->save();

只是字段的增加/减少 直接用inc/dec方式

$user = User::get(1);
$user->name  = 'think';
$user->email = 'think@qq.com';
$user->score= ['inc', 1];
$user->save();

直接更新数据

可直接带更新条件更新数据

$user = new User;// save方法第二个参数为更新条件
$user->save(['name'=> 'think','email' => 'think@qq.com'],['id' => 1]);

如果需要过滤非数据表字段的数据

$user = new User;// 过滤post数组中的非数据表字段数据
$user->allowField(true)->save($_POST,['id' => 1]);

如果通过外部提交赋值给模型 希望指定某些字段写入

$user = new User();// post数组中只有name和email字段会写入
$user->allowField(['name','email'])->save($_POST, ['id' => 1]);

建议传入模型数据之前进行过滤

$user = new User();// post数组中只有name和email字段会写入
$data = Request::only(['name','email']);
$user->save($data, ['id' => 1]);

批量更新数据

saveAll方法批量更新数据 只需在批量更新的数据中包含主键即可

$user = new User;
$list = [
    ['id'=>1, 'name'=>'thinkphp', 'email'=>'thinkphp@qq.com'],
    ['id'=>2, 'name'=>'onethink', 'email'=>'onethink@qq.com']
];
$user->saveAll($list);

批量更新方法返回的是数据集对象

批量更新仅能根据主键值进行更新 其它情况请自行处理

静态方法

模型支持调用数据库的方法直接更新数据

User::where('id',1)->update(['name'=>'think']);

数据库的update方法返回影响的记录数

或使模型的静态update方法更新

User::update(['id' => 1, 'name' => 'think']);

模型update方法返回模型的对象实例

区别
第一种使用的数据库的update方法
第二种使用的模型的update方法(支持模型的修改器、事件和自动完成)

自动识别

模型的新增和更新方法是save方法 系统默认的规则来识别当前的数据更新还是新增

  • 实例化模型后调用save方法表示新增

  • 查询数据后调用save方法表示更新

  • save传入更新条件后表示更新

如果数据操作复杂 用isUpdate方法显式的指定当前调用save是新增操作还是更新操作

显式更新数据:

// 实例化模型
$user = new User;// 显式指定更新数据操作
$user->isUpdate(true) ->save(['id' => 1, 'name' => 'thinkphp']);

显式新增数据:

$user = User::get(1);
$user->name = 'think';// 显式指定当前操作为新增操作
$user->isUpdate(false)->save();

不在模型实例里面做多次更新 会导致部分重复数据不再更新
正确的方式应该是先查询后更新或使用模型类的update方法更新

如果调用save方法多次数据写入
第二次save方法的时候必须使用isUpdate(false) 否则会视为更新数据

最佳实践

最佳实践原则
如果使用模型事件 先查询后更新
如果不需要事件 直接使用静态的Update方法进行条件更新
如非必要 尽量不要使用批量更新