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

4 回答 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);

        }
    }
}

创建此代码时有一些假设,因此不要“按原样”使用它。

于 2013-10-06T21:34:28.213 回答
6

我同意 Bojan 的观点,除非您使用的是实体框架?

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

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

但是,除非您使用 EF,否则我建议您仅使用 SqlGeometry 作为

  1. 没有演员表/转换
  2. SqlGeometry 具有比 DbGeometry 更多的功能。

希望有帮助。

于 2013-05-29T10:53:42.563 回答
3

SqlSpatialFunctions.MakeValid是一种特定于 SQL Server 的方法来执行此操作。

如果您对几何图形无效的原因感兴趣,可以询问 SQL Server:

SELECT @gm.IsValidDetailed()

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

于 2013-05-20T00:24:30.080 回答
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;
      }

代码的任何部分我都可以调用这个函数来使几何有效

于 2013-05-26T09:25:32.407 回答