.NET 4.5中是否有类似于dbgeometry makevalid的东西?

4

我正在尝试计算这样的折线的面积

    string poly = "POLYGON ((637604.918432772 2230520.64934531,
637622.257266129 2230419.44632915, 637279.107128549 2230192.04910755, 636765.470527745 2230179.6468564, 636778.005055813 2229861.77192838, 636529.81646905 2229464.29327025, 635813.486592791 2229523.30345774, 636017.385069448 2229974.32341381, 636267.323659164 2230070.32127916, 637035.026966561 2230404.70764784, 637275.265066307 2230401.13408429, 637604.918432772 2230520.64934531, 637604.918432772 2230520.64934531))";
     DbGeometry gm = DbGeometry.FromText(poly, 32637);
double area= gm.Area.Value; // here I got the error Exception has been thrown by the target of an invocation.

后来我发现错误的原因是dbgeometry无效。我在ms sql 2012中尝试过这段代码,也出现了同样的错误,但当我像这样尝试时:

SELECT @gm.MakeValid().STArea()

我曾经在SQL中使用过这个功能,但我的问题是是否有一种方法能够使.NET中的几何形状有效。谢谢。

4个回答

9
你绝对不应该去数据库获取你想要的东西。 一个简单且快速的方法是通过将以下代码添加到你的项目中扩展DbGeography:
using System.Data.SqlTypes;
using Microsoft.SqlServer.Types;

namespace System.Data.Spatial
{
    public static class DbGeometryExtension
    {
        public static DbGeometry MakeValid(this DbGeometry geom)
        {
            if (geom.IsValid)
                return geom;

            return DbGeometry.FromText(SqlGeometry.STGeomFromText(new SqlChars(geom.AsText()), 4326).MakeValid().STAsText().ToSqlString().ToString(), 4326);

        }
    }
}

这段代码在创建时做出了一些假设,请不要直接使用它。

实际上我们就是这样做的。 - Khalid Omar
请注意,上述投影是“32637”而不是“4326”。 - jocull

6

我赞同Bojan的看法,除非您正在使用实体框架?

SqlGeometry 对象有一个 MakeValid() 函数,因此可以使用您的示例,允许在 DbGeography 和 SqlGeography 之间进行转换:

DbGeography geog;
SqlGeometry geom = SqlGeometry.STGeomFromWKB(new SqlBytes(geog.AsBinary()), 32637);

然而,除非您正在使用EF,否则我建议仅使用SqlGeometry作为:

  1. 不存在转换问题
  2. SqlGeometry拥有比DbGeometry更多的可用函数。

希望这能够帮助到您。


我正在使用实体框架 5,我能否使用 sqlgeometry,然后将其分配给 dbgeometry? - Khalid Omar
1
@Khalid,最终,是的,你在EF5中只能使用DBGeometry,但你可以从数据库中读取它,将其转换为SQLGeometry,执行所需操作,再将其转换回DBGeometry并更新数据库(如果需要)。然而,虽然单个转换的开销很小,但在处理大量对象时,这可能会迅速变得显著。因此,我想问你,为什么你一定要进行转换,有有效的理由,但我正在努力确保我能够给出最好的建议。 - Jon Bellamy

4

SqlSpatialFunctions.MakeValid是一种特定于SQL Server的方法。

如果您想知道为什么您的几何图形无效,可以向SQL Server提问:

SELECT @gm.IsValidDetailed()

此外,您可能需要考虑直接使用SQL Server的几何类型:SqlGeometry.MakeValid

如何在C#中使用SqlSpatialFunctions来验证dbgeometry的有效性。 - Khalid Omar
gm = SqlSpatialFunctions.MakeValid(gm); 给我这个函数只能从LINQ to Entities中调用。 - Khalid Omar
1
啊,抱歉,我的错 - 那似乎只是让LINQ调用SQL的MakeValid来处理数据库对象的一种方式。我已经更新了我的答案。 - Bojan Resnik

0

这里有一个解决该问题的中间方法,我可以像这样创建标量函数

CREATE FUNCTION [dbo].[GeomMakeValid](@geom as geometry)
RETURNS geometry
AS
BEGIN

DECLARE @gm as geometry;
set @gm = @geom.MakeValid();

return (@gm);
END

GO

然后我从模型中这样调用它

public static DbGeometry geoMakeValid(DbGeometry geom)
      {

          EntityConnection Connec = getConnection();

          DbCommand com = Connec.StoreConnection.CreateCommand();
          com.CommandText = "select dbo.GeomMakeValid(@geom)";
          com.CommandType = System.Data.CommandType.Text;
          com.Parameters.Add(new SqlParameter("@geom", geom.AsText()));
          if (com.Connection.State == ConnectionState.Closed) com.Connection.Open();
          try
          {
              var result = com.ExecuteScalar(); // should properly get your value
              Connec.Close();
              return DbGeometry.FromText( result.ToString(),geom.CoordinateSystemId);

          }
          catch (System.Exception e)
          {

          }
          Connec.Close();
          return geom;
      }

无论代码的哪个部分,我都可以调用此函数使几何图形有效。


由于您的代码已经是针对SQL Server的,因此在代码中使用SqlGeometry对象而不是DbGeometry会更加清晰和快速,而不必进行整个往返到数据库。 - Bojan Resnik

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接