gisempire100
捉鬼专家
捉鬼专家
  • 注册日期2004-08-13
  • 发帖数552
  • QQ
  • 铜币2462枚
  • 威望0点
  • 贡献值0点
  • 银元0个
阅读:1245回复:0

计算点、线、面等元素之间的交点、交线、封闭区域面积和闭合集(待续)

楼主#
更多 发布于:2008-03-06 20:31
<P >地理信息系统软件开发中经常需要求取点、线、面之间的交点、交线、封闭区域面积和闭合集等结果,采用以矢量点乘和叉乘为基础的求取算法符合实际工作中已给出点位置和法向量等条件的情况,效率较高。</P>
<P >首先给出基本公式的推导。</P>
<P >矢量的结合率和交换率:</P>
<P >U+(V+W) = (U+V) + W;</P>
<P >U+V=V+U;</P>
<P >而设线段的起点和终点为P,Q;则中间任意一点R可以表示为:</P>
<P >R = P + r(Q - P); (0<r<1)</P>
<P >由P0,P1和P2三点构成的共面参数方程可表示为如下一种形式:</P>
<P >P(s,t) = (1-s-t)P0+sP1+tP2,其中s,t为参数</P>
<P >矢量的长度定义为:</P>
<P >设 V = (a,b); 则 |V| = sqrt(a*a + b*b); (即各轴数据平方和的开方)</P>
<P >而向量的规范化是 U = V/|V|,使得向量的长度为单位1。</P>
<P>矢量的点乘定义:</P>
<P >V&#150;W = v1w1+v2w2,V= (v1,v2) , W = (w1,w2); </P>
<P >此定义可从二维空间类推到三维和其他高次空间。</P>
<P >我们知道,二维平面上面一点P(x,y)在坐标轴转动某角度α后,新坐标为P(xnew,ynew)</P>
<P>设P点距离原点为r,此点与原X轴夹角为δ,则</P>
<P>x = rcosδ, y = rsinδ;</P>
<P>可求得新坐标为:xnew = rcos(δ-α)=rcosδcosα + rsinδsinα = xcosα + ysinα;</P>
<P >ynew =rsin(δ-α)=rsinδcosα –rcosδsinα = ycosα –xsinα;</P>
<P>下面推导广泛运用的一个公式:</P>
<P >V&#150;W = |V||W|cos&#981;,&#981;为两向量的交角;</P>
<P >若V = (v1,0),既此点落在X轴上,则V&#150;W = v1w1 + 0w2 = v1w1;</P>
<P>因为V点落在轴上,则 |V| = v1,|W| cos&#981; = w1;所以 V&#150;W = |V||W|cos&#981;;</P>
<P >若为一般情况,即V= (v1,v2),则我们可旋转原始坐标轴角度δ后使得V点落在新X轴上,</P>
<P>由于: (v1w1 + v2w2)/|V| = w1cosδ+ w2sinδ = W点在新坐标系下的X轴数值 = |W| cos&#981; </P>
<P>因此 v1w1 + v2w2 = |V||W|cos&#981;;</P>
<P >在平时的运用中,计算cos&#981; = V&#150;W / |V||W| ,可对两向量的交角大小进行判断,若V&#150;W为0,则说明两向量互相垂直;若V&#150;W大于0,则|&#981;|<90度,说明交角为锐角。</P>
<P >设向量V(a,b),对其旋转90度后,得到U= (acos90 + bsin90, ycos90-xsin90) = (b,-a);可由此方法得到向量的垂直向量。</P>
<P >由V&#150;W = |V||W|cos&#981;公式的推导中可以类推|V||W|sin&#981;等于什么呢?推导如下:</P>
<P >|W|sin&#981; = W点在新坐标系下的Y轴数值 = w2cosδ-w1sinδ= w2v1/|V|-w1v2/|V|= </P>
<P >(v1w2-v2w1)/|V|,即</P>
<P >v1w2-v2w1 =  |V||W|sin&#981;,&#981;为两向量的交角;</P>
<P>       在平时的运用中,可采用此公式计算三角形和四边形的面积。也能判断两向量的位置关系,如果计算v1w2-v2w1为0,则说明两向量交角为0度或180度,即共线。如果计算结果大于0,则说明两向量交角0度到180度之间。</P>
<P>求取向量之间位置关系和三角形面积的参考代码:</P>
<P  align=left>//    Input: three points P0, P1, and P2</P>
<P  align=left>//    Return: >0 for P2 left of the line through P0 and P1</P>
<P  align=left>//            =0 for P2 on the line</P>
<P  align=left>//            <0 for P2 right of the line</P>
<P  align=left>double CDEMAlgorithm::isLeft( Point P0, Point P1, Point P2 )</P>
<P  align=left>{</P>
<P  align=left>    return ( (P1.x - P0.x) * (P2.y - P0.y)</P>
<P  align=left>        - (P2.x - P0.x) * (P1.y - P0.y) );       //两向量的叉乘</P>
<P  align=left>}</P>
<P  align=left>double CDEMAlgorithm::orientation2D_Triangle( Point V0, Point V1, Point V2 )</P>
<P  align=left>{</P>
<P  align=left>    return isLeft(V0, V1, V2);                        //判断位置关系</P>
<P  align=left>}</P>
<P  align=left>double CDEMAlgorithm::area2D_Triangle( Point V0, Point V1, Point V2 )</P>
<P  align=left>{</P>
<P  align=left>    return isLeft(V0, V1, V2) / 2.0;                  //求三角形的面积</P>
<P>}<BR></P>
<P><FONT face=Verdana>double CDEMAlgorithm::dist3D_Line_to_Line( Line L1, Line L2)<BR>{<BR> Vector   u = L1.P1 - L1.P0;<BR> Vector   v = L2.P1 - L2.P0;<BR> Vector   w = L1.P0 - L2.P0;<BR> double    a = Dot(u,u);        // always >= 0<BR> double    b = Dot(u,v);<BR> double    c = Dot(v,v);        // always >= 0<BR> double    d = Dot(u,w);<BR> double    e = Dot(v,w);<BR> double    D = a*c - b*b;       // always >= 0<BR> double    sc, tc;</FONT></P>
<P><FONT face=Verdana> // compute the line parameters of the two closest points<BR> if (D < EPS) {         // 几乎平行<BR>  sc = 0.0;<BR>  tc = (b>c ? d/b : e/c);   // <BR> }<BR> else {<BR>  sc = (b*e - c*d) / D;<BR>  tc = (a*e - b*d) / D;<BR> }</FONT></P>
<P><FONT face=Verdana>   //sc,tc分别指示了在两条 直线上的 比例参数<BR>  Vector   dP = w + (sc * u) - (tc * v);  // = L1(sc) - L2(tc)</FONT></P>
<P><FONT face=Verdana> return sqrt(Dot(dP,dP));   // return the closest distance</FONT></P>
<P><FONT face=Verdana>}<BR><BR>附图:<BR><IMG src="http://www.cnblogs.com/images/cnblogs_com/wuhanhoutao/125287/r_AreaCompute.jpg" border=0></FONT></P>
喜欢0 评分0
A friend is never known till a man has need. ...CL
游客

返回顶部