一共分两部分:
1.方法概述
2.C#实现点到线段起点距离
第一部分
http://www.gispower.org/article/arcgis/ao/2008/122/081222025470KDK5965183CCA19ABJ5.html
我们常常需要计算某个线状几何要素(可以是直线Iline,IpolyLine,也可以是曲线Icurve,Iring)上任意两点间的距离。对于直线而言我们只需要用两点间直线距离公式就可以得到,但是对于曲线来说在Ae中没有提供直接获取的方法和属性。本人在实验后找到了可以求得曲线上任意一两点距离的方法,和大家分享,希望共同提高。
方法一:
在AE中有个方法QueryPointAndDistance(Icurve为例),通过这个方法我们可以获取空间上一点到曲线上最近的点的位置和据起点的距离。这样我们就可以利用这个方法将空间上的点转移到我们要求得的两点间距离的两个点上,也就是将曲线上求距离的两个点看作是空间上任意两个点。通过QueryPointAndDistance方法求得在曲线上与这两个点最近的两个点与两个据起点的距离。
*注意*:实际上我们获取的最近的两个点实际上就是要求得距离的两个点本身。
假设曲线上两点位PtA,PtB 那么通过计算得到的两个距离就是DisPtA和DisPtB。
用DisPtA - DisPtB 就是我们要获取的曲线上两点间距离。
Dim outpt1 As IPoint
Set outpt1 = New Point
Dim disalong1 As Double
Dim dis1 As Double
pcurve.QueryPointAndDistance esriExtendTangentAtFrom, pt1, False, outpt1, disalong1, dis1, False
Dim outpt2 As IPoint
Set outpt2 = New Point
Dim disalong2 As Double
Dim dis2 As Double
pcurve.QueryPointAndDistance esriExtendTangentAtFrom, pt2, False, outpt2, disalong2, dis2, False
''曲线两点间距离
resultDis=disalong2-disalong2
方法二:
还有一种方法是利用 GetSubcurve方法来将曲线上两点间的曲线裁剪出一个SubCurve,然后利用ICurve接口的Length方法就可以直接获取。但本方法实施起来有一定难度。
方法三:
如果我们要查询距离的两点 是曲线上的 节点,那么我们就可以通过IEnumPointAndDistance接口下的 DistanceAlongCurve方法直接获取一个点到参照点的曲线距离(参照点一般为曲线的起点 frompoint),然后将获取到的两个距离相减也同样可以获取到两点间的曲线距离。
注意:本方法一定是节点 才可以使用。
第二部分
http://hi.baidu.com/llinkin_park/blog/item/1200cb3cc5a15f3070cf6cb4.html
在做垃圾桶服务能力检测功能时,需要求两个垃圾桶之间在道路上的实际距离,由于道路使用的是线要素,而考虑到采集数据的精度问题,我假设在地图上道路两边都有可能出现垃圾桶。所以求这个距离的确是个难题。后来想了很多种方法,都一一否定了。 后来在论坛上看见一个帖子,他是使用的vb写的,被我改成c#了,其实也挺简单的,主要使用了IPolyline自带的QueryPointAndDistance方法,它能返回一个点至线的起始点的距离,然后通过这两个距离相减,就得到两点之间的距离了。
ICurve.QueryPointAndDistance Method :Finds the point on the curve closest to inPoint, then copies that point to outPoint; optionally calculates related items。
/// <summary>
/// 得到两个垃圾桶在道路上的距离
/// </summary>
/// <param name="pnt1">垃圾桶1</param>
/// <param name="pnt2">垃圾桶2</param>
/// <param name="roadFeature">所在道路</param>
/// <returns></returns>
public double getDistanceOfTwoPoint(IPoint pnt1, IPoint pnt2, IFeature roadFeature)
{
double distanceOfpnt1 = distanceFromStartPoint(pnt1, roadFeature);
double distanceOfpnt2 = distanceFromStartPoint(pnt2, roadFeature);
return distanceOfpnt2 - distanceOfpnt1;
}
/// <summary>
/// 垃圾桶到道路起点的距离
/// </summary>
/// <param name="pPoint"></param>
/// <param name="pFeature"></param>
/// <returns></returns>
public double distanceFromStartPoint(IPoint pPoint, IFeature pFeature)
{
IPolyline pPolyline = pFeature.Shape as IPolyline;
IPoint outPoint = new PointClass();
double distanceAlongCurve = 0;//该点在曲线上最近的点距曲线起点的距离
double distanceFromCurve = 0;//该点到曲线的直线距离
bool bRightSide = false;
bool asRatio = false; //asRatio:byval方式,bool类型,表示上面两个参数给定的长度是以绝对距离的方式给出还是以占曲线总长度的比例的方式给出
pPolyline.QueryPointAndDistance(esriSegmentExtension.esriNoExtension, pPoint, asRatio, outPoint, ref distanceAlongCurve, ref distanceFromCurve,ref bRightSide);
return distanceAlongCurve;
}