计算几何 _ 基础知识
前置知识点
(1)
(2) 余弦定理
浮点数的比较
const double eps = 1e-8; int sign(double x) // 符号函数 { if (fabs(x) < eps) return 0; if (x < 0) return -1; return 1; } int cmp(double x, double y) // 比较函数 { if (fabs(x - y) < eps) return 0; if (x < y) return -1; return 1; }
向量
向量的加减法和数乘运算
加减法: x和y分别相加(减)即可。
几何意义:
向量相加等价于两向量构成的平行四边形的一条对角线。
向量相减等价于两向量构成的平行四边形的另一条对角线。
数乘法:x和y分别乘上 数值 即可。
内积(点积)
A·B = |A||B|cos(C)
(1) 几何意义:
向量A在向量B上的投影与B的长度的乘积。
(2) 代码实现
double dot(Point a, Point b) { return a.x * b.x + a.y * b.y; }
3.3 外积(叉积)
AxB = |A||B|sin(C)
几何意义:向量A与B张成的平行四边形的有向面积(面积可能为负数)。B在A的逆时针方向为正。
代码实现
double cross(Point a, Point b) { return a.x * b.y - b.x * a.y; }
3.4 常用函数
3.4.1 取模
double get_length(Point a) { return sqrt(dot(a, a)); }
3.4.2 计算向量夹角
double get_angle(Point a, Point b) { return acos(dot(a, b) / get_length(a) / get_length(b)); }
3.4.3 计算两个向量构成的平行四边形有向面积
double area(Point a, Point b, Point c) { return cross(b - a, c - a); }
3.4.5 向量A顺时针旋转C的角度:
相当与乘上这个矩阵
Point rotate(Point a, double angle) { return Point(a.x * cos(angle) + a.y * sin(angle), -a.x * sin(angle) + a.y * cos(angle)); }
4. 点与线
4.1 直线定理
(1) 一般式 ax + by + c = 0
(2) 点向式
(3) 斜截式 y = kx + b
4.2 常用操作
判断点在直线上 = 0
A是点O和点向式的原点的向量,即
判断点在直线左右
两直线交点
// cross(v, w) == 0则两直线平行或者重合
Point get_line_intersection(Point p, Vector v, Point q, vector w) { vector u = p - q; double t = cross(w, u) / cross(v, w); return p + v * t; }
(3) 点到直线的距离
三角形面积除以底边长度。
double distance_to_line(Point p, Point a, Point b) { vector v1 = b - a, v2 = p - a; return fabs(cross(v1, v2) / get_length(v1)); }
(4) 点到线段的距离
double distance_to_segment(Point p, Point a, Point b) { if (a == b) return get_length(p - a); Vector v1 = b - a, v2 = p - a, v3 = p - b; if (sign(dot(v1, v2)) < 0) return get_length(v2); if (sign(dot(v1, v3)) > 0) return get_length(v3); return distance_to_line(p, a, b); }
(5) 点在直线上的投影
Point get_line_projection(Point p, Point a, Point b) { Vector v = b - a; return a + v * (dot(v, p - a) / dot(v, v)); }
(6) 点是否在线段上
bool on_segment(Point p, Point a, Point b) { return sign(cross(p - a, p - b)) == 0 && sign(dot(p - a, p - b)) <= 0; }
(7) 判断两线段是否相交
判断线段A的两端点,是不是在线段B的两侧。
bool segment_intersection(Point a1, Point a2, Point b1, Point b2) { double c1 = cross(a2 - a1, b1 - a1), c2 = cross(a2 - a1, b2 - a1); double c3 = cross(b2 - b1, a2 - b1), c4 = cross(b2 - b1, a1 - b1); return sign(c1) * sign(c2) <= 0 && sign(c3) * sign(c4) <= 0; }
5. 多边形
5.1 三角形
5.1.1 面积
(1) 叉积
(2) 海伦公式(知道三边边长时用)
p = (a + b + c) / 2; S = sqrt(p(p - a) * (p - b) * (p - c));
5.1.2 三角形四心
(1) 外心,外接圆圆心
三边中垂线交点。到三角形三个顶点的距离相等
(2) 内心,内切圆圆心
角平分线交点,到三边距离相等
(3) 垂心
三条垂线交点
(4) 重心
三条中线交点(到三角形三顶点距离的平方和最小的点,三角形内到三边距离之积最大的点)
5.2 普通多边形
通常按逆时针存储所有点
5.2.1 定义
(1) 多边形
由在同一平面且不再同一直线上的多条线段首尾顺次连接且不相交所组成的图形叫多边形
(2) 简单多边形
简单多边形是除相邻边外其它边不相交的多边形
(3) 凸多边形
过多边形的任意一边做一条直线,如果其他各个顶点都在这条直线的同侧,则把这个多边形叫做凸多边形
任意凸多边形外角和均为360°
任意凸多边形内角和为(n−2)180°
5.2.2 常用函数
(1) 求多边形面积
(不一定是凸多边形)
我们可以从第一个顶点除法把凸多边形分成n − 2个三角形,然后把面积加起来。
double polygon_area(Point p[], int n) { double s = 0; for (int i = 1; i + 1 < n; i ++ ) s += cross(p[i] - p[0], p[i + 1] - p[i]); return s / 2; }
(2) 判断点是否在多边形内
(不一定是凸多边形)
a. 射线法,从该点任意做一条和所有边都不平行的射线。交点个数为偶数,则在多边形外,为奇数,则在多边形内。
b. 转角法
(3) 判断点是否在凸多边形内
只需判断点是否在所有边的左边(逆时针存储多边形)。
5.3 皮克定理
皮克定理是指一个计算点阵中顶点在格点上的多边形面积公式该公式可以表示为:
其中a表示多边形内部的点数,
b表示多边形边界上的点数,
S表示多边形的面积。
- 圆
(1) 圆与直线交点
(2) 两圆交点
(3) 点到圆的切线
(4) 两圆公切线
(5) 两圆相交面积
作者:yxc
链接:https://www.acwing.com/activity/content/code/content/635453/
来源:AcWing
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。
本文作者:kingwzun
本文链接:https://www.cnblogs.com/kingwz/p/16675978.html
版权声明:本作品采用知识共享署名-非商业性使用-禁止演绎 2.5 中国大陆许可协议进行许可。
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】凌霞软件回馈社区,博客园 & 1Panel & Halo 联合会员上线
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】博客园社区专享云产品让利特惠,阿里云新客6.5折上折
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
2021-09-10 21/9/9补题 [ 欧拉筛] [数组去重] | [进制转换]
2021-09-10 Java入门
2021-09-10 JAVA:类和对象基础