3

我编写了一个小型应用程序,它从一系列 KML 文件中读取数据,然后Microsoft.SqlServer.Types.SqlGeography使用以下代码将它们转换为类型:

    private SqlGeography CreateGeographyFromKML( string kml, bool debug )
    {
        // use SqlGeographyBuilder to help create the SqlGeography type 
        var geographyBuilder = new SqlGeographyBuilder();

        // Get co-ordinates 
        var xml = XDocument.Parse(kml);
        var df = xml.Root.Name.Namespace;
        XElement coordinates = xml.Descendants(df + "coordinates").Single();

        // set the Spatial Reference Identifiers that will used to create the point 
        geographyBuilder.SetSrid(_srid);
        geographyBuilder.BeginGeography(OpenGisGeographyType.Polygon);

        var longLat = coordinates.Value.Split(' ').Select(c => new { Lat = Convert.ToDouble(c.Split(',')[1]), Long = Convert.ToDouble(c.Split(',')[0]) });

        Console.Write("Found {0} ", longLat.Count());

        foreach (var coord in longLat.Select((x, i) => new { Index = i, Value = x }))
        {
            if (coord.Index == 0)
            { // First point
                if ( debug ) Console.WriteLine("First point: {0},{1}", coord.Value.Lat, coord.Value.Long);
                geographyBuilder.BeginFigure(coord.Value.Lat, coord.Value.Long);
            }
            else
            { // Intermediate points
                if (debug) Console.WriteLine("Intermediate point: {0},{1}", coord.Value.Lat, coord.Value.Long);
                geographyBuilder.AddLine(coord.Value.Lat, coord.Value.Long);
            }

            if (coord.Index == longLat.Count() - 1 )
            { // Last point (Close polygon)     
                if (debug) Console.Write("Last point: ");

                // Check not already added
                if (longLat.Last().Lat == longLat.First().Lat && longLat.Last().Long == longLat.First().Long)
                {
                    if (debug) Console.Write("Already exists - not adding...");
                }
                else
                {
                    if (debug) Console.Write("{0},{1}", longLat.Last().Lat, longLat.Last().Long);
                    geographyBuilder.AddLine(longLat.Last().Lat, longLat.Last().Long);
                }

                geographyBuilder.EndFigure(); // End figure
            }
        }

        if (debug) Console.WriteLine();

        // close the figure and geography class 
        geographyBuilder.EndGeography();

        // get the geography builder to return the sqlgeography type 
        return geographyBuilder.ConstructedGeography;
    }

基本上,此代码从 KML 文件中检索纬度/经度列表,然后遍历它们以创建多边形。

但是,我正在导入的一些 KML 文件失败,但出现以下异常:

捕获 System.ArgumentException 消息 = 24200:指定的输入不代表有效的地理实例。

这发生在以下行:return geographyBuilder.ConstructedGeography;

我发现了一些对这个异常的引用,但是在我发现他们在 SQL Server 中遇到并处理这个异常的情况下,而不是在 C# 中。

4

2 回答 2

7

我有同样的错误,但结果证明是多边形环方向问题。一个简单的翻转坐标数组顺序的问题就解决了这个问题。

为了说明,这失败并出现上述错误:

 select geography::STGeomFromText ('Polygon  ( (10 10, 10 20, 20 20, 20 10, 10 10))',4326)

而这有效:

 select geography::STGeomFromText ('Polygon  ( (10 10, 20 10, 20 20, 10 20, 10 10))',4326)

请注意,我不是在一个点内翻转 x,y 对,而是翻转整个点数组的顺序(例如 {pt1, pt2, pt3, pt4, pt5} 变为 {pt5, pt4, pt3, pt2, pt1 }

于 2013-04-17T00:31:55.737 回答
2

我遇到了同样的问题,并使用名为 Sql Server Spatial Tools ( http://sqlspatialtools.codeplex.com ) 的项目解决了它。

它有(以及其他有趣的东西)这两种方法:

  • MakeValidGeographyFromGeography
  • MakeValidGeographyFromText(接收 WKT 字符串)

他们修改这些点,使其符合地理限制。

它真的很好用,我已经用了几个月了,没有任何问题。

于 2011-11-26T12:59:47.237 回答