0

好的,这是我的问题。我有数据库表BloodCenters,其中有latitudelongitude字段。我想使用 Linq Lambda 语法通过检索相对于某个点在给定半径内的 BloodCenters 来查询这些表。

我正在使用这段代码,但它仍然给我错误(在 Where 谓词内):

public static List<BloodCenter> GetAllNearby(int latitude, int longitude, int radius, int limit){
    List<BloodCenter> res = null;
    res = new DBEntities().BloodCenters.Where(b =>
        Util.distance(latitude, longitude, b.Latitude, b.Longitude, "K") <= radius).Take(limit);
    return res;
}

Util.distance 是一个返回纬度和经度 2 点之间距离的函数。 编辑 所以,我不能把我自己的函数放在 Where 谓词中。但是我可以使用 SQL 语法计算距离吗?这是我的Util.Distance实现:

public static double distance(double lat1, double longi1, double lat2, double longi2, string unit){
    var theta = longi1 - longi2;
    var dist = Math.Sin(DegreeToRadian(lat1)) * Math.Sin(DegreeToRadian(lat2)) +
        Math.Cos(DegreeToRadian(lat1)) * Math.Cos(DegreeToRadian(lat2)) * Math.Cos(DegreeToRadian(theta));
    dist = Math.Acos(dist);
    dist = RadianToDegree(dist);
    var miles = dist * 60 * 1.1515;
    unit = unit.ToUpper();
    if (unit.Equals("K"))
        return miles * 1.609344;
    else if (unit.Equals("N"))
        return miles * 0.8684;
    else
        return miles;
}
4

2 回答 2

1

这将不起作用,因为Util.distance在您的 C# 代码中实现,并且查询表面不知道如何将其转换为 SQL(或者即使这可能)。

一个直接的解决方法是从数据库中获取所有记录并将谓词应用于实例,如下所示:

res = new DBEntities().BloodCenters.ToList()
          .Where(b => Util.distance(latitude, longitude, b.Latitude, b.Longitude, "K") <= radius)
          .Take(limit);
return res;

当然,获取所有记录也不是最好的主意,但有时根本没有其他办法。

您可以尝试的是内联谓词Util.distance内部的逻辑。Where如果逻辑可以仅使用“简单”操作(如基本数学)来表达,则查询表面将能够将其转换为 SQL,并且一切都会顺利进行。

以下是可以自动转换为 SQL 的所有方法的列表。

于 2012-05-10T12:39:07.120 回答
0

您必须转换DBEntities().BloodCenters为本地存储的IEnumerableorList或其他内容。Linq to SQL 将 C# 转换为 SQL - 因此您只能使用 SQL 服务器提供的函数。

另一种方法是用可翻译的语句模拟函数或在数据库中实现它(存储过程或类似的)

于 2012-05-10T12:44:26.633 回答