圆的反演 hdu 6097

欢迎关注更多精彩

关注我,学习常用算法与数据结构,一题多解,降维打击。

题目大意

http://acm.hdu.edu.cn/showproblem.php?pid=6097

有一个圆C,它的圆心是O(0,0), 半径是r。

在C内部或边界上有两点P和Q,OP=OQ。

求解圆C上一点D,使得PD+QD最小。图中红色相加最短。

错解

直觉上感觉D是PQ中垂线与圆的交点。

这个直觉是错误的。

一个特殊例子,当PQ是直径时,显然D点是P或Q时取得最值。

利用圆的反演及相似三角形转化圆外点问题

反演定义:有一个圆C以O为圆心半径为r,那么任意点P对于圆C的反演点P'存在于射线OP方向上,且OP' = r*r/OP

寻找相似三角形

图中P'是P关于圆O的反演点。
则存在圆上 1 点 D 使得 △ P ′ D O ∼ △ D P O 相似比为 r O P 则存在圆上1点D使得 \triangle P'DO \sim \triangle DPO 相似比为\frac {r}{OP} 则存在圆上1点D使得△P′DO∼△DPO相似比为OPr

证明也很简单:

根据已知条件,P'是P的反演点可得到

OP' = r*r/OP => OP'/r=r/OP = OD/OP

根据两边之比相等且夹角相等,可以得到上述相似三角形的结论。

问题转化

根据上述结论,Q'D/DQ=P'D/DP=r/OP(已知量)

得到 Q ′ D + P ′ D D Q + D P = r O P \frac {Q'D+P'D}{DQ+DP} = \frac {r}{OP} DQ+DPQ′D+P′D=OPr

那么想要 DQ+DP 最小只要保证Q'D+P'D最小即可,问题就转化成求Q'D+P'D的最小值。

问题求解

对Q'和P'作边线可以发现下述两种情况。

相交

当与圆相交时,最短就是两点的连线,那么D就是圆上与连线的交点。

相离

直觉D点是Q'P'中垂线与圆的交点。

证明如下:

假设Q'P'平等于Y轴,引入辅助线l平行于Q'P'与圆相切。

设向量P'D=(x,-y), Q'D = (x,y), 引入变量y*表示D在直线l上移动。

距离之和可以表示为关于 y ∗ 的函数 f ( y ∗ ) = x 2 + ( − y + y ∗ ) 2 + x 2 + ( y + y ∗ ) 2 距离之和可以表示为关于y*的函数 f(y*) = \sqrt{x^2+(-y+y*)^2}+\sqrt{x^2+(y+y*)^2} 距离之和可以表示为关于y∗的函数f(y∗)=x2+(−y+y∗)2 +x2+(y+y∗)2

对f(y*)求导等于0

f ′ ( y ∗ ) = − y + y ∗ x 2 + ( − y + y ∗ ) 2 + y + y ∗ x 2 + ( y + y ∗ ) 2 f'(y*) = \frac {-y+y*}{\sqrt{x^2+(-y+y*)^2}}+\frac {y+y*}{\sqrt{x^2+(y+y*)^2}} f′(y∗)=x2+(−y+y∗)2 −y+y∗+x2+(y+y∗)2 y+y∗

当y*=0时,f'(y*)=0。

在圆上移动会比直线上更远。

上述结论得证。

如何求P'D+Q'D。

设P'Q'中点为M, 则DM = OM-r。

根据勾股定理可求得P'D。

代码实现

cpp 复制代码
#include<stdio.h>
#include<cmath>

class Point {
public:
	double x, y;

	Point() {}
	Point(double a, double b) :x(a), y(b) {}

	void in() {
		scanf(" %lf %lf", &x, &y);
	}

	double dis() {
		return sqrt(x * x + y * y);
	}

	void operator -=(Point& p) {
		x -= p.x;
		y -= p.y;
	}

	void operator +=(Point& p) {
		x += p.x;
		y += p.y;
	}
	void operator *=(double d) {
		x *= d;
		y *= d;
	}

	void operator /=(double d) {
		this ->operator*= (1 / d);
	}
};



class Circle
{
public:
	Point center;
	double r;
	
	Circle(Point &c, double a):center(c), r(a){}
	Circle();
	void in() {
		center.in();
		scanf("%lf", &r);
	}

};

Circle::Circle()
{
}


void solve() {
	Point P, Q, P1, Q1, M;
	int T;
	Circle c;

	scanf("%d", &T);

	while (T--) {
		scanf("%lf", &c.r);
		c.center = Point(0, 0);
		int a, b;
		scanf("%d %d", &a, &b);
		P.x = a;
		P.y = b;

		scanf("%d %d", &a, &b);
		Q.x = a;
		Q.y = b;
		if (P.dis() < 1e-6) {
			printf("%8f\n", 2*c.r);
			continue;
		}

		// 求反演点
		double k = c.r * c.r / P.dis() / P.dis();
		P1 = P;
		P1 *= k;
		Q1 = Q;
		Q1 *= k;

		// 计算中点
		M = P1;
		M += Q1;
		M /= 2;
		P1 -= Q1;
		double d = P1.dis();
		// 判断相离情况
		if (M.dis() > c.r) {
			double md = M.dis() - c.r;
			Q1 = Point(md, d/2);
			d = Q1.dis() * 2;
		}

		d *= P.dis() / c.r;
		printf("%.8f\n", d);
	}
}

int main() {
	solve();
	return 0;
}

/*
4
4
4 0
0 4
4
0 3
3 0
4
0 2
2 0
4
0 1
1 0

*/

本人码农,希望通过自己的分享,让大家更容易学懂计算机知识。

相关推荐
大桔骑士v3 天前
【算法学习笔记】35:扩展欧几里得算法求解线性同余方程
算法·acm·数论·扩展欧几里得算法·线性同余方程
大桔骑士v3 天前
【算法学习笔记】34:扩展欧几里得算法
算法·acm·数论·最大公约数·扩展欧几里得算法
御风@户外1 个月前
质数生成函数、质数判断备份
算法·acm
Assassin's Creed3 个月前
2024江苏省赛E. Divide
c语言·数据结构·c++·算法·acm·主席树
X_StarX3 个月前
2024JMU第十一届程序设计大赛 部分题解(6题)
数据结构·c++·学习·算法·acm·大学生
AlbertS4 个月前
AWS上迁移WordPress遭遇若干问题记处理办法
acm·aws·wordpress·alb·route53·postfix
mqdlff_python4 个月前
第五届计算机科学与管理科技国际学术会议(ICCSMT 2024)
科技·acm·ei会议·ei·学术会议
WenGyyyL4 个月前
准备蓝桥杯和ACM:C++标准库头文件及其常用功能简介
开发语言·c++·算法·蓝桥杯·acm
怜渠客5 个月前
一次不严谨的C++、C、Pascal、Rust等对比
java·开发语言·c++·rust·acm
Asuka_46_7 个月前
leetcode 第133场双周赛 100333.统计逆序对的数目【计数dp/滚动数组/前缀和优化】
算法·leetcode·前缀和·动态规划·acm·逆序对