嘿伙计们怎么样? 提前谢谢。
我在 laravel 模型中编写了一个工作函数,该函数执行地理搜索并按距离返回搜索顺序。它工作得很好。
public static function searchByDistance($distance, $unit)
{
//spatial queries
// $circle_radius = 6371;
$circle_radius = 3959;
$max_distance = $distance;
$lat = Auth::user()->getLoc()->latitude;
$lng = Auth::user()->getLoc()->longitude;
//get all nearby places
$nearbyusers = DB::select(
'SELECT * FROM
(SELECT users.id, users.country, places.name, places.state_code, places.latitude, places.longitude, (' . $circle_radius . ' * acos(cos(radians(' . $lat . ')) * cos(radians(latitude)) *
cos(radians(longitude) - radians(' . $lng . ')) +
sin(radians(' . $lat . ')) * sin(radians(latitude))))
AS distance
FROM users
LEFT JOIN places ON users.suburb=places.name) AS distances
WHERE distance < ' . $max_distance . '
ORDER BY distance;
');
$nearbyusers = (object)$nearbyusers;
return $nearbyusers;
}
但是我有一个问题。我想调用该函数,返回查询并继续能够向其add ->where()
操作,直到我准备好使用get()
并返回结果。
我不能使用where() or get()
,我知道为什么。 它的格式不正确。 有没有办法帮助我修改/更改函数,以便在返回$nearbyusers
值后可以添加更多过滤器?
干杯
您需要查看底层的Fluent查询构建器系统。就目前而言,您的查询是独立的。这个特定的也可能需要保持独立。
为了以您想要的方式挂接到 Fluent,您需要传入查询(这应该得到自动解析(,添加到其中,然后返回查询。就您当前的函数而言,这可能需要一种与您拥有的方法完全不同的方法(尽管您拥有的是您想要最终得到的查询的良好模型(。
查看有关查询范围的文档,这些文档可能正在解决您尝试如何集成它。此外,请查看代码本身,例如用于 JoinClause 函数的代码,以了解如何构建代码。
基本元素是您需要获取并返回查询对象。这意味着它不能是静态方法(它需要对象实例(,并且根据您采用的方法,您将返回传入的$query
对象(范围(或$this
(它们的实例模型对象本身(。
您需要找到一种方法来将您拥有的查询封装到您需要的段中。它可能是这样的:
public function byDistance($distance, $unit) {
$lat = Auth::user()->getLoc()->latitude;
$lng = Auth::user()->getLoc()->longitude;
return $this->join(DB::select('SELECT users.id, users.country, places.name, places.state_code, places.latitude, places.longitude, (' . $circle_radius . ' * acos(cos(radians(' . $lat . ')) * cos(radians(latitude)) *
cos(radians(longitude) - radians(' . $lng . ')) +
sin(radians(' . $lat . ')) * sin(radians(latitude)))
AS distance'),
'users.id', 'distance.user_id = users.id')
->leftJoin('places', 'users.suburb = places.name')
->where('distance', '<', $distance)
->orderBy('distance');
}
请注意,查询构建部分中缺少实际select
(尽管嵌套选择(,并且缺少任何类型的实际获取。这样做是添加到当前查询中,然后在最终调用get
或其他获取函数时构建该查询。这为您提供了较大查询的离散段,然后可以在调用它时添加到该段。
查看原始表达式的官方文档 我认为这就是您要找的。无论如何,您必须将原始查询更改为"构建"查询,如下所示:
$nearbyusersquery = DB::table('users')
->select(DB::RAW('SELECT users.id, users.country, places.name, places.state_code, places.latitude, places.longitude, (' . $circle_radius . ' * acos(cos(radians(' . $lat . ')) * cos(radians(latitude)) *
cos(radians(longitude) - radians(' . $lng . ')) +
sin(radians(' . $lat . ')) * sin(radians(latitude))))
AS distance'))
->leftJoin('places', 'users.suburb', '=', 'places.name')
->where('distance', '<', $max_distance)
->orderBy('distance');
return $nearbyusersquery
这样,您的方法将返回一个可以与其他语句链接的Builder
对象。
你也可以用sql来做到这一点。例如:
$filtr = "";
$filtr2 = "";
If (something)
$filtr =" and field = var";
If (somethingelse)
$filtr2 = " and field2 = other_var";
// this can be in the elloquent query
$query = "select ....
from ....
Where 1=1
$filtr
$filtr2
";
编辑
//assuming the country Id is passed to your controller
// first initialize the filter as empty string
$countryFilter = "";
// if the country is not null or empty replace the filter
if ($country != null && $country != "")
$countryFilter = "and country = $countryFilter";
$nearbyusers = DB::select(
'SELECT * FROM
(SELECT users.id, users.country, places.name, places.state_code, places.latitude, places.longitude, (' . $circle_radius . ' * acos(cos(radians(' . $lat . ')) * cos(radians(latitude)) *
cos(radians(longitude) - radians(' . $lng . ')) +
sin(radians(' . $lat . ')) * sin(radians(latitude))))
AS distance
FROM users
LEFT JOIN places ON users.suburb=places.name) AS distances
WHERE distance < ' . $max_distance . '
-- if no country was selected then it will just be an empty line
$countryFilter
ORDER BY distance;
');
像往常一样,自己解决了。 问题就是这么简单。 您所要做的就是查找与结果匹配的用户对象并将它们推送到数组中。 然后返回该数组,您可以使用这些函数!因此,如果您想从原始查询中获取可传递的对象,这就是您要做的。再次感谢您的有用答案伙计们。这个问题真的太简单了。最后,我摆弄它来工作,而不是从互联网上得到任何答案。
//spatial queries
$circle_radius = 6371;
// $circle_radius = 3959;
$max_distance = request('distance');
$lat = Auth::user()->getLoc()->latitude;
$lng = Auth::user()->getLoc()->longitude;
//get all nearby places
$nearbyusers = DB::select(
'SELECT * FROM
(SELECT users.id, users.country, places.name, places.state_code, places.latitude, places.longitude, (' . $circle_radius . ' * acos(cos(radians(' . $lat . ')) * cos(radians(latitude)) *
cos(radians(longitude) - radians(' . $lng . ')) +
sin(radians(' . $lat . ')) * sin(radians(latitude))))
AS distance
FROM users
LEFT JOIN places ON users.suburb=places.name) AS distances
WHERE distance < ' . $max_distance . '
ORDER BY distance;
');
$users = [];
foreach($nearbyusers as $user)
{
array_push($users, User::where('id', $user->id)->first());
}
//$nearbyusers = (object)$nearbyusers;
return Response::json( array(
'searchsuccess' => (String) view('members.yessearch', compact('users'))
) );