这个问题与this one有关,但是我简化了我的实际问题。因此,我创建了一个单独的问题:
我有两个桌子。一个保持间隔(以天为单位)与bimonthly = 15
之类的名称相关联,另一个保持间隔列出周期性动作并与间隔相关联。
我想列出在纯SQL中容易实现的下一个周期性操作:
SELECT *, DATE_ADD(updated_at, INTERVAL days DAY) AS next_action
FROM periodic_actions
INNER JOIN intervals ON interval_id = intervals.id
HAVING next_action > NOW()
这是我的桌子:
CREATE TABLE `intervals` (
`id` INT NOT NULL AUTO_INCREMENT,
`name` VARCHAR(45) NULL,
`days` INT NULL,
PRIMARY KEY (`id`));
CREATE TABLE `periodic_actions` (
`id` INT NOT NULL AUTO_INCREMENT,
`interval_id` INT NOT NULL,
`name` VARCHAR(45) NULL,
`updated_at` TIMESTAMP NOT NULL,
PRIMARY KEY (`id`));
间隔的内容:
+----+----------+------+
| id | name | days |
+----+----------+------+
| 1 | daily | 1 |
| 2 | monthly | 30 |
| 3 | annually | 365 |
+----+----------+------+
我想用Eloquent构建相同的查询,在其中我可以在虚拟列where
上使用next_action
:
Action::orderBy('next_action', 'asc')
->where('next_action', '>', Carbon::now())
->get()
我无法隐藏到模型中的最接近的是:
Action::join('intervals', 'interval_id', '=', 'intervals.id')
->select(['*', DB::RAW('DATE_ADD(updated_at, INTERVAL days DAY) AS next_action')])
->havingRaw(DB::RAW('next_action > NOW()'))->get();
我当然有这个基本模型:
class Action extends Model
{
protected $table = 'periodic_actions';
public function interval()
{
return $this->belongsTo(Interval::class);
}
}
答案 0 :(得分:2)
解决方案1:使用全局范围将该字段添加到您的选择语句中
关于虚拟列的一个问题是Eloquent,如果您不阻止,则会尝试更新它们。您可以将其从$ fillable数组中排除,并在模型中使用global scope添加虚拟列,例如:
protected static function boot()
{
parent::boot();
static::addGlobalScope('next_action', function (Builder $builder) {
$builder->addSelect(DB::raw('DATE_ADD(updated_at, INTERVAL days DAY) AS next_action'));
});
}
您必须记住,这会将您的Model耦合到MySQL,因为您使用的日期函数在sqlite之类的数据库中不起作用。
解决方案2:使用MySQL视图
通常,当我有多个计算字段时,我要做的是创建一个MySQL视图。使用所需的所有计算字段创建视图,然后可以为该视图创建新的Eloquent模型,而不必担心查询范围。
一个警告是,您当然不能在此模型上使用create / update / delete,但是如果您仅将模型用于显示目的,那应该没问题。