为什么Laravel保存方法更改updated_at列和其他时间戳列?

时间:2018-01-29 16:07:02

标签: laravel laravel-5.5

我在数据库中创建了这样的表。

Schema::create('packages', function (Blueprint $table) {
    $table->increments('id');
    $table->unsignedInteger('count');
    $table->timestamp('expire_date');
    $table->timestamps();
});

现在想象一下,我们在数据库中有以下记录:

+----+-------+---------------------+---------------------+---------------------+
| id | count | expire_date         | created_at          | updated_at          |
+----+-------+---------------------+---------------------+---------------------+
|  1 |   120 | 2018-01-31 19:22:45 | 2018-01-29 17:12:01 | 2018-01-29 19:22:45 |
|  4 |   140 | 2018-02-14 19:05:01 | 2018-01-29 17:20:16 | 2018-01-29 19:05:01 |
| 27 |    10 | 2018-02-17 19:01:10 | 2018-01-29 19:05:01 | 2018-01-29 19:05:01 |
+----+-------+---------------------+---------------------+---------------------+

当我尝试使用laravel eloquent编辑记录时:

$package = \App\Package::find(1);
$package->count = 100;
$package->save();

save方法将expire_date和updated_at列更改为当前时间!

+----+-------+---------------------+---------------------+---------------------+
| id | count | expire_date         | created_at          | updated_at          |
+----+-------+---------------------+---------------------+---------------------+
|  1 |   100 | 2018-01-29 19:32:24 | 2018-01-29 17:12:01 | 2018-01-29 19:32:24 |
|  4 |   140 | 2018-02-14 19:05:01 | 2018-01-29 17:20:16 | 2018-01-29 19:05:01 |
| 27 |    10 | 2018-02-17 19:01:10 | 2018-01-29 19:05:01 | 2018-01-29 19:05:01 |
+----+-------+---------------------+---------------------+---------------------+

为什么expire_date值在没有触及的情况下会发生变化?!!!

2 个答案:

答案 0 :(得分:1)

来自the docs

  

save方法也可用于更新数据库中已存在的模型。要更新模型,您应该检索它,设置要更新的任何属性,然后调用save方法。同样,updated_at时间戳将自动更新,因此无需手动设置其值

因此,Eloquent仅更新updated_at。如果expire_date也被更新,则意味着在应用程序中使用了Eloquent事件或类似的事情发生(例如DB updates)。

答案 1 :(得分:1)

看看您的表DDL,是

expire_date timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP
created_at timestamp NULL DEFAULT NULL,
updated_at timestamp NULL DEFAULT NULL,

expire_date的默认值为当前时间戳。
您需要做的就是将ON UPDATE CURRENT_TIMESTAMP更改为null。

ALTER TABLE table ALTER COLUMN expire_date DEFAULT NULL

或在创建之前:

https://github.com/laravel/framework/blob/be09eac00a2f750dc2d2209078cf61ae2eee85a1/src/Illuminate/Database/Schema/Grammars/MySqlGrammar.php#L647

https://github.com/laravel/framework/blob/d598a6630208b054376527e41848c106f4297b2b/src/Illuminate/Database/Schema/ColumnDefinition.php#L26

$table->timestamp('expire_date')->useCurrent(false);