【问题标题】:How to Calculate Centroid如何计算质心
【发布时间】:2012-04-06 15:01:13
【问题描述】:

我正在处理地理空间形状并在此处查看质心算法,

http://en.wikipedia.org/wiki/Centroid#Centroid_of_polygon

我已经像这样在 C# 中实现了代码(就是这样改编的),

Finding the centroid of a polygon?

class Program
{
    static void Main(string[] args)
    {
        List<Point> vertices = new List<Point>();

        vertices.Add(new Point() { X = 1, Y = 1 });
        vertices.Add(new Point() { X = 1, Y = 10 });
        vertices.Add(new Point() { X = 2, Y = 10 });
        vertices.Add(new Point() { X = 2, Y = 2 });
        vertices.Add(new Point() { X = 10, Y = 2 });
        vertices.Add(new Point() { X = 10, Y = 1 });
        vertices.Add(new Point() { X = 1, Y = 1 });

        Point centroid = Compute2DPolygonCentroid(vertices);
    }

    static Point Compute2DPolygonCentroid(List<Point> vertices)
    {
        Point centroid = new Point() { X = 0.0, Y = 0.0 };
        double signedArea = 0.0;
        double x0 = 0.0; // Current vertex X
        double y0 = 0.0; // Current vertex Y
        double x1 = 0.0; // Next vertex X
        double y1 = 0.0; // Next vertex Y
        double a = 0.0;  // Partial signed area

        // For all vertices except last
        int i=0;
        for (i = 0; i < vertices.Count - 1; ++i)
        {
            x0 = vertices[i].X;
            y0 = vertices[i].Y;
            x1 = vertices[i+1].X;
            y1 = vertices[i+1].Y;
            a = x0*y1 - x1*y0;
            signedArea += a;
            centroid.X += (x0 + x1)*a;
            centroid.Y += (y0 + y1)*a;
        }

        // Do last vertex
        x0 = vertices[i].X;
        y0 = vertices[i].Y;
        x1 = vertices[0].X;
        y1 = vertices[0].Y;
        a = x0*y1 - x1*y0;
        signedArea += a;
        centroid.X += (x0 + x1)*a;
        centroid.Y += (y0 + y1)*a;

        signedArea *= 0.5;
        centroid.X /= (6*signedArea);
        centroid.Y /= (6*signedArea);

        return centroid;
    }
}

public class Point
{
    public double X { get; set; }
    public double Y { get; set; }
}

问题是这个算法当我有这个形状(这是一个L形)时,

(1,1) (1,10) (2,10) (2,2) (10,2) (10,1) (1,1)

它给了我结果 (3.62, 3.62)。没关系,除了那个点在形状之外。是否有其他算法考虑到这一点?

基本上,一个人会在地图上绘制一个形状。这个形状可能跨越多条道路(所以可能是 L 形),我想计算出形状的中心。这样我就可以在那时计算出道路名称。如果他们画了一个又长又瘦的 L 形,我觉得它在形状之外是没有意义的。

【问题讨论】:

  • 多边形的质心不必在里面。这只保证适用于多边形。
  • 是的,我同意算法是正确的,但是是否有另一种算法可以确保计算多边形内的点?理想情况下,上述形状的结果是 (1.5, 1.5)。
  • 如果对您的问题有意义,您可以将道路表示为粗线甚至联合或矩形(通过引入轴的概念)。您的中心位于该轴的中点。
  • 不要用点来查找路名。无法保证该点与所描绘的道路足够接近。正确的做法是检索该区域中的道路线段,并找出多边形中哪条道路的长度最长。

标签: c# geospatial centroid


【解决方案1】:

这个答案的灵感来自 Jer2654 的答案和这个来源: http://coding-experiments.blogspot.com/2009/09/xna-quest-for-centroid-of-polygon.html

  /// <summary>
  /// Method to compute the centroid of a polygon. This does NOT work for a complex polygon.
  /// </summary>
  /// <param name="poly">points that define the polygon</param>
  /// <returns>centroid point, or PointF.Empty if something wrong</returns>
  public static PointF GetCentroid(List<PointF> poly)
  {
     float accumulatedArea = 0.0f;
     float centerX = 0.0f;
     float centerY = 0.0f;

     for (int i = 0, j = poly.Count - 1; i < poly.Count; j = i++)
     {
        float temp = poly[i].X * poly[j].Y - poly[j].X * poly[i].Y;
        accumulatedArea += temp;
        centerX += (poly[i].X + poly[j].X) * temp;
        centerY += (poly[i].Y + poly[j].Y) * temp;
     }

     if (Math.Abs(accumulatedArea) < 1E-7f)
        return PointF.Empty;  // Avoid division by zero

     accumulatedArea *= 3f;
     return new PointF(centerX / accumulatedArea, centerY / accumulatedArea);
  }

【讨论】:

  • if (Math.Abs​​(accumulatedArea)
  • @Koray 谢谢!不太了解这样做的必要性,但我认为您是对的,并且我已将您的建议包含在我的回答中 - 以及我自己对该代码的使用中,也是我在 Android 应用程序中使用的 Java 版本。
  • 您的代码可以正常工作,但是如果某些坐标为负数,则这是必需的。如果保证每个坐标都是正的,那就没有必要了。谢谢。
【解决方案2】:
public static Point GetCentroid( Point[ ] nodes, int count )
{
    int x = 0, y = 0, area = 0, k;
    Point a, b = nodes[ count - 1 ];

    for( int i = 0; i < count; i++ )
    {
        a = nodes[ i ];

        k = a.Y * b.X - a.X * b.Y;
        area += k;
        x += ( a.X + b.X ) * k;
        y += ( a.Y + b.Y ) * k;

        b = a;
    }
    area *= 3;

    return ( area == 0 ) ? Point.Empty : new Point( x /= area, y /= area );
}

【讨论】:

【解决方案3】:

您可以检查 .NET 4.5 DbSpatialServices 是否具有功能,例如 DbSpatialServices.GetCentroid

【讨论】:

    【解决方案4】:

    对于 3d Points,我在 C# 中创建了一个方法,希望对您有所帮助:

    public static double[] GetCentroid(List<double[]> listOfPoints)
        {
            // centroid[0] = X
            // centroid[1] = Y
            // centroid[2] = Z
            double[] centroid = new double[3];
    
            // List iteration
            // Link reference:
            // https://en.wikipedia.org/wiki/Centroid
            foreach (double[] point in listOfPoints)
            {
                centroid[0] += point[0];
                centroid[1] += point[1];
                centroid[2] += point[2];
            }
    
            centroid[0] /= listOfPoints.Count;
            centroid[1] /= listOfPoints.Count;
            centroid[2] /= listOfPoints.Count;
    
            return centroid;
        }
    

    【讨论】:

      【解决方案5】:

      我的实现:

      GpsCoordinates GetCentroid(ICollection<GpsCoordinates> polygonCorners)
      {
          return new GpsCoordinates(polygonCorners.Average(x => x.Latitude), polygonCorners.Average(x => x.Longitude));
      }
      
      public readonly struct GpsCoordinates
      {
          public GpsCoordinates(
              double latitude,
              double longitude
              )
          {
             
              Latitude = latitude;
              Longitude = longitude;
          }
      
          public double Latitude { get; }
          public double Longitude { get; }
      }
      

      灵感来自here

      【讨论】:

        猜你喜欢
        • 1970-01-01
        • 2015-06-04
        • 1970-01-01
        • 1970-01-01
        • 1970-01-01
        • 2019-05-12
        • 2022-11-21
        • 1970-01-01
        相关资源
        最近更新 更多