ChatGPT解决这个技术问题 Extra ChatGPT

Laravel Eloquent 更新,如果已经进行了更改

是否有任何方法可以使用 eloquent 模型更新 Laravel 中的记录,只要对记录进行了更改?我不希望任何用户一遍又一遍地无缘无故地请求数据库,只需点击按钮即可保存更改。我有一个 javascript 功能,可以根据页面中是否发生更改来启用和禁用保存按钮,但我想知道是否可以确保在服务器端也执行这种功能。我知道我可以自己完成它(意思是:不诉诸框架的内部功能)只需检查记录是否有变化,但在这样做之前,我想知道 Laravel 雄辩模型是否已经处理那,所以我不需要重新发明轮子。

这是我用来更新记录的方式:

$product = Product::find($data["id"]);
$product->title = $data["title"];
$product->description = $data["description"];
$product->price = $data["price"];
//etc (string values were previously sanitized for xss attacks)
$product->save();
为什么不启用数据库日志记录,然后检查日志以查看 Eloquent 在保存时实际执行的查询:您可能会感到惊喜
请注意,Laravel 模型包含一个 dirty 标志,用于确定是否实际需要对数据库进行更新,并且该标志是通过将原始 find 列值与执行保存时的值进行比较来设置的

l
lukasgeiter

你已经在做!

save() 将检查模型中的某些内容是否已更改。如果没有,它将不会运行数据库查询。

以下是 Illuminate\Database\Eloquent\Model@performUpdate 中代码的相关部分:

protected function performUpdate(Builder $query, array $options = [])
{
    $dirty = $this->getDirty();

    if (count($dirty) > 0)
    {
        // runs update query
    }

    return true;
}

getDirty() 方法只是将当前属性与创建模型时保存在 original 中的副本进行比较。这是在 syncOriginal() 方法中完成的:

public function __construct(array $attributes = array())
{
    $this->bootIfNotBooted();

    $this->syncOriginal();

    $this->fill($attributes);
}

public function syncOriginal()
{
    $this->original = $this->attributes;

    return $this;
}

如果您想检查模型是否脏,只需调用 isDirty()

if($product->isDirty()){
    // changes have been made
}

或者,如果您想检查某个属性:

if($product->isDirty('price')){
    // price has changed
}

那太棒了。谢谢你。我现在想知道如何访问该脏标志以了解是否在不进行任何更改的情况下进行了更新尝试?我的意思是:那个动作的回报?
对于阅读本文的人,请务必在使用 isDirty() 之前查看 this answer
getChanges()方法。检查我的答案stackoverflow.com/a/54132163/1090395
仅供参考,如果您没有正确转换属性,isDirty() 将是 true。我有一个与数据库中完全相同的浮点数,但是因为我使用字符串设置浮点数(例如“10.50”而不是 10.50),它会将其检测为更改并执行更新。
M
Mladen Janjetovic

即使在持久化之后,您也可以在 Eloquent 模型上使用 $product->getChanges()。检查文档here


A
Ahmad Yousef

我喜欢添加此方法,如果您使用的是编辑表单,则可以使用此代码将更改保存在 update(Request $request, $id) 函数中:

$post = Post::find($id);    
$post->fill($request->input())->save();

请记住,您必须使用相同的列名来命名您的输入。 fill() 函数将为您完成所有工作:)


这实际上是我正在寻找的答案,我忘记了名称 fill,以及如何将请求大量传递给它。谢谢!我不需要传递 ID,因为我正在更新,所以它已经存在 $request->id