MySQL&Laravel:如何在where子句中包含我的自定义派生列?

huangapple go评论70阅读模式
英文:

MySQL & Laravel: How to include my custom derived column in where clauses?

问题

我正在处理一个根据不同标准筛选用户的函数。其中一个筛选条件是基于用户净资产。在数据库中,我们目前只存储净资产和净债务,计算净资产只是从资产中减去债务。

我的筛选函数需要能够找到低于目标数字的用户、高于目标数字的用户或在两个目标数字之间的用户。

在 Eloquent 代码中,这是我所拥有的:

//更多代码...
// 筛选净资产
->when(isset($request_filter_params['minNetWorth']) 
    || isset($request_filter_params['maxNetWorth']), fn ($query) => 
    $query->whereHas('net_worth_estimate_calc_response', fn ($q) => 
        $q->select(DB::raw('(net_worth_estimate_asset - net_worth_estimate_debt) as net_worth'))
            ->when(isset($request_filter_params['minNetWorth']), fn ($qq) => 
                $qq->where('net_worth', '>=', $request_filter_params['minNetWorth'])
            )
            ->when(isset($request_filter_params['maxNetWorth']), fn ($qqq) => 
                $qqq->where('net_worth', '<=', $request_filter_params['maxNetWorth'])
            )
    )
)
//更多代码...

它生成的 SQL 如下:

select
	*
from
	`users`
where
	and `is_active` = 1
	and `is_groupadmin` = 0
	and exists (
	select
		(net_worth_estimate_asset - net_worth_estimate_debt) as net_worth
	from
		`user_net_worth_estimate_calculator_responses`
	where
		`users`.`id` = `user_net_worth_estimate_calculator_responses`.`user_id`
		and `net_worth` >= 2500
		and `net_worth` <= 10000)
order by
	`updated_at` desc

这个 SQL 产生错误 Unknown column 'net_worth' in 'where clause'

显然是由于我自定义的 DB::raw('(net_worth_estimate_asset - net_worth_estimate_debt) as net_worth') 调用引起的。有没有办法允许基于这个派生值在 where 子句中进行筛选?还是 MySQL 无法做到这一点,我必须将其移到 PHP 层(我很想避免这样做)?

编辑:'user_net_worth_estimate_calculator_responses' 只包含 user_id (bigint)net_worth_estimate_asset (double)net_worth_estimate_debt (double)、一个主键 id 和一些时间戳。模型文件是一个完全标准的 Eloquent 模型。

英文:

I am working on a function that filters users based on varied criteria. One of the filters is based on user net worth. In the database, we currently only store net assets, and net debt, and calculating net worth is simply subtracting debt from assets.

My filter function needs to be able to find users under a target number, over a target number, or between two target numbers.

In Eloquent code, here is what I have.

//more code before...
// filter net worth
        -&gt;when(isset($request_filter_params[&#39;minNetWorth&#39;]) 
            || isset($request_filter_params[&#39;maxNetWorth&#39;]), fn ($query) =&gt; 
            $query-&gt;whereHas(&#39;net_worth_estimate_calc_response&#39;, fn ($q) =&gt; 
                $q-&gt;select(DB::raw(&#39;(net_worth_estimate_asset - net_worth_estimate_debt) as net_worth&#39;))
                    -&gt;when(isset($request_filter_params[&#39;minNetWorth&#39;]), fn ($qq) =&gt; 
                        $qq-&gt;where(&#39;net_worth&#39;, &#39;&gt;=&#39;, $request_filter_params[&#39;minNetWorth&#39;])
                    )
                    -&gt;when(isset($request_filter_params[&#39;maxNetWorth&#39;]), fn ($qqq) =&gt; 
                        $qqq-&gt;where(&#39;net_worth&#39;, &#39;&lt;=&#39;, $request_filter_params[&#39;maxNetWorth&#39;])
                    )
            )
        )
//more code after...

The SQL it produces:

select
	*
from
	`users`
where
	and `is_active` = 1
	and `is_groupadmin` = 0
	and exists (
	select
		(net_worth_estimate_asset - net_worth_estimate_debt) as net_worth
	from
		`user_net_worth_estimate_calculator_responses`
	where
		`users`.`id` = `user_net_worth_estimate_calculator_responses`.`user_id`
		and `net_worth` &gt;= 2500
		and `net_worth` &lt;= 10000)
order by
	`updated_at` desc

That sql produces the error Unknown column &#39;net_worth&#39; in &#39;where clause&#39;.

Clearly caused by my custom DB::raw(&#39;(net_worth_estimate_asset - net_worth_estimate_debt) as net_worth&#39;) call, is there any way to allow filtering in the where clauses based on this derived value? Or is this something MySQL cant do and I have to move this into the PHP layer (which I would love to avoid)?

Edit: 'user_net_worth_estimate_calculator_responses' only has user_id (bigint), net_worth_estimate_asset (double), net_worth_estimate_debt (double, a PK id and a couple of timestamps. The Model file is a completely standard eloquent model.

答案1

得分: 0

你可以尝试使用 having

$qq->having('net_worth', '>=', $request_filter_params['minNetWorth'])

或者使用 whereRaw(这 应该 会有更好的性能):

$qq->whereRaw('net_worth_estimate_asset - net_worth_estimate_debt >= ?', $request_filter_params['minNetWorth'])
英文:

You can try with having:

$qq-&gt;having(&#39;net_worth&#39;, &#39;&gt;=&#39;, $request_filter_params[&#39;minNetWorth&#39;])

or alternatively with whereRaw (this should have better performance)

$qq-&gt;whereRaw(&#39;net_worth_estimate_asset - net_worth_estimate_debt &gt;= ?&#39;, $request_filter_params[&#39;minNetWorth&#39;])

huangapple
  • 本文由 发表于 2023年8月4日 07:16:15
  • 转载请务必保留本文链接:https://go.coder-hub.com/76832101.html
匿名

发表评论

匿名网友

:?: :razz: :sad: :evil: :!: :smile: :oops: :grin: :eek: :shock: :???: :cool: :lol: :mad: :twisted: :roll: :wink: :idea: :arrow: :neutral: :cry: :mrgreen:

确定