1 条题解
-
0
自动搬运
来自洛谷,原作者为

warzone
梦违科学世纪搬运于
2025-08-24 22:36:19,当前版本为作者最后更新于2022-03-11 15:30:49,作者可能在搬运后再次修改,您可在原文处查看最新版自动搬运只会搬运当前题目点赞数最高的题解,您可前往洛谷题解查看更多
以下是正文
题目大意
你在一个 维空间中。
给出空间中 个点的坐标 及它们与你的距离 ,
请你求出你所在的坐标 。
若有多个解,输出任意一个。
,坐标范围 ,保证数据随机且有解。
题解
给出一个 naive 的数形结合的做法。
根据题意,不难列出方程
$$\begin{cases} (\vec{x}-\vec{y}_1)^2=r_1^2\\ (\vec{x}-\vec{y}_2)^2=r_2^2\\ \cdots \\ (\vec{x}-\vec{y}_n)^2=r_n^2 \end{cases} $$这个方程含有平方,所以没法直接高斯消元。
但注意到
$$(\vec{x}-\vec{y})^2=\vec{x}^2-2\vec{y}\cdot\vec{x}+\vec{y}^2 $$其中 是相同的。
考虑将原点平移到 ,设 $\vec{x}'=\vec{x}-\vec{y}_1,\forall k,\vec{y}_k'=\vec{y}_k-\vec{y}_1$,则
$$\begin{cases} \vec{x}'^2=r_1^2&(1)\\ (\vec{x}'-\vec{y}_2')^2=r_2^2\\ (\vec{x}'-\vec{y}_3')^2=r_3^2\\ \cdots\\ (\vec{x}'-\vec{y}_n')^2=r_n^2 \end{cases} $$下面 个式子同时减去 ,得到
$$\begin{cases} -2\vec{y}'_2\cdot\vec{x}+\vec{y}_2'^2=r_2^2-r_1^2\\ -2\vec{y}'_3\cdot\vec{x}+\vec{y}_3'^2=r_3^2-r_1^2\\ \cdots\\ -2\vec{y}'_n\cdot\vec{x}+\vec{y}_n^2=r_n^2-r_1^2 \end{cases} $$这等价于线性方程组
$$A=\begin{bmatrix} (\vec{y}_2')^T\\ \hline(\vec{y}_3')^{T^{^{^{}}}}\\ \hline\vdots \\ \hline(\vec{y}_n')^{T^{^{^{}}}} \end{bmatrix},\vec{a}=-\dfrac{1}{2}\begin{bmatrix} r_2^2-r_1^2-\vec{y}_2^2\\ r_3^2-r_1^2-\vec{y}_3^2\\ \vdots\\ r_n^2-r_1^2-\vec{y}_n^2 \end{bmatrix}$$高斯消元求解该线性方程组,若其有唯一解,则解就是答案。
如果解不是唯一的怎么办?这时“形”便派上用场。
让我们转换一下题意:给定 维空间中 个超球,
球心为 ,半径为 ,求它们的交点。

先把这个问题搁置一边,来看看这个高中文化课常用的 trick:
给定二维空间两个圆的方程:
$$\begin{cases} (x-a_1)^2+(y-b_1)^2=r_1^2&(1)'\\ (x-a_2)^2+(y-b_2)^2=r_2^2&(2)' \end{cases} $$它们的交线如图所示:

现在我们要求交线的方程,一般人的想法是将 联立,求得交点后再求交线。
但最快的方法是将 相减,直接求得交线的方程
$$(x-a_1)^2-(x-a_2)^2+(y-b_1)^2-(y-b_2)^2=r_1^2-r_2^2 $$接下来我们简短证明下这么做的正确性:
- 中 系数均为 ,因此 一定是个二元一次不定方程,
也就一定是一条直线。 - ,因此圆 与该直线的交点一定在圆 上。
同理,圆 与该直线的交点一定在圆 上。
因此该直线与两个圆中一个的交点等价于两个圆的交点。
综上, 就是两个圆的交线方程。
显然这个结论及其证明可以推广到高维空间,若已知三维空间两个球的方程:
$$\begin{cases} (x-a_1)^2+(y-b_1)^2+(z-c_1)^2=r_1^2&(1)''\\ (x-a_2)^2+(y-b_2)^2+(z-c_2)^2=r_2^2&(2)''\\ \end{cases} $$它们的交面如图所示:

把 和 相减,可直接得到交面的方程
$$(x-a_1)^2-(x-a_2)^2+(y-b_1)^2-(y-b_2)^2+(z-c_1)^2+(z-c_2)^2=r_1^2-r_2^2 $$且交面与其中一个球的交点等价于两球的交点。
同理,若已知 维空间的两个超球:
$$\begin{cases} (\vec{x}-\vec{y}_1)^2=r_1^2\\ (\vec{x}-\vec{y}_2)^2=r_2^2 \end{cases} $$将两个超球的方程相减,可得到 维超交面的方程
$$(\vec{x}-\vec{y}_1)^2-(\vec{x}-\vec{y}_2)^2=r_1^2-r_2^2 $$且超交面与其中一个超球的交点等价于两个超球的交点。
此时我们已经能解释之前的高斯消元在做什么了:
- 将所有方程减去 等价于求超球 与其他所有超球的超交面方程。
- 高斯消元等价于求这些超交面的交。
- 这些超交面的交与超球 的交点都是合法的解,即所有超球的交点。
分析完这些后,解不唯一的情况变成了一个这样的问题:
设高斯消元后自由元的个数为 ,
则我们需要求解一个 维超球和一个 维 “平面” (准确的说是线性流形)的交点。
同样考虑二维空间的情况:求解一个圆和一条直线的交点。

一个显然的做法是过圆心向直线作垂线,然后勾股定理算出垂足到交点的距离:

拓展到三维空间时,情况是类似的:
- 对于三维空间一条直线和球相交,跟刚才的情况是一样的。

- 对于三维空间一个平面和球相交,可以确定截面一定是个圆。
同样作垂线得到圆心后勾股定理求解。

想一想怎么归纳到高维空间,提示一下:一维空间的超球是两个点。
答案揭晓: 维空间一个超球与一个 维线性流形相交,截面一定是个 维超球。
过原超球球心向线性流形作垂线即可得到截面的球心,再勾股定理即可得到球心的半径。我们由此得到求解原问题的一个大致的思路:
- 过超球心向线性流形作垂线,求出垂足的坐标 。
- 勾股定理算出截面的半径 。
- 在线性流形上随便找一个长度为 的向量 , 就是一个合法的解。
回到之前的线性方程组。高斯消元后,其一定会变为如下形式:
$$\left[ \begin{array}{ccccc|} 1&&&&&v_{1,1}&v_{2,1}&\cdots&v_{k,1}\\ &1&&&&v_{1,2}&v_{2,2}&\cdots&v_{k,2}\\ &&\ddots&&&\vdots&\vdots&\vdots&\vdots\\ &&&1&&v_{1,d-k-1}&v_{2,d-k-1}&\cdots&v_{k,d-k-1}\\ &&&&1&v_{1,d-k}&v_{2,d-k}&\cdots&v_{k,d-k}\\ \hline 0&0&0&0&0&0&0&0&0\\ \vdots&\vdots&\vdots&\vdots&\vdots&\vdots&\vdots&\vdots&\vdots \end{array}\right] \begin{bmatrix} x_1\\x_2\\\vdots\\x_{d-k}\\\hline x_{d-k+1}\\ x_{d-k+2}\\ \vdots \\x_d \end{bmatrix}=\begin{bmatrix} a_1'\\ a_2'\\ \vdots\\ a_{d-k}'\\\hline 0\\0\\ \vdots \\0 \end{bmatrix} $$简记为如下形式:
$$\left[\begin{array}{c|c|c|c|c} I&\vec{v}_1&\vec{v}_2&\cdots&\vec{v}_k\\ \hline O&\vec{0}&\vec{0}&\vec{0}&\vec{0} \end{array}\right] \begin{bmatrix}\vec{x}_0\\ \hline\vec{x}_1\end{bmatrix} =\begin{bmatrix}\vec{a}'\\ \hline\vec{0}\end{bmatrix} $$我们需要确定线性方程组的解集,从而确定整个线性流形。
- 若 ,显然 。
因此 $\begin{bmatrix}\vec{a}'\\ \hline\vec{0}\end{bmatrix}$ 是该线性方程组的一组特解。 - 若 ,
则 $\vec{x}_0+\lambda\vec{v}_1=\vec{a}',\vec{x}_0=\vec{a}-\lambda\vec{v}_1$。
因此 每增加 , 就要减去 。
综上,设
$$\vec{u}_1=\begin{bmatrix}-\vec{v}_1\\\hline 1\\0\\0\\0\\ \vdots\end{bmatrix} ,\vec{u}_2=\begin{bmatrix}-\vec{v}_2\\\hline 0\\1\\0\\0\\ \vdots\end{bmatrix}, \vec{u}_3=\begin{bmatrix}-\vec{v}_3\\\hline 0\\0\\1\\0\\ \vdots\end{bmatrix},\cdots $$$$\vec{b}=\begin{bmatrix} \vec{a}'\\ \hline \vec{0} \end{bmatrix},V=\{[\vec{u}_1|\vec{u}_2|\cdots|\vec{u}_k]\vec{x}_1|\vec{x}_1\in\mathbb{R}_k\} $$则线性方程组的解集为
即线性空间 内的所有向量加上特解 ,与线性流形的定义一致。
接下来考虑作垂线,由于球心为 , 的基底为 ,
$$\vec{u}_1\cdot \vec{p}=\vec{u}_2\cdot\vec{p}=\cdots=\vec{u}_k\cdot \vec{p}=0 \qquad{(2)}$$
我们的目标是找出 ,使得 与 中所有的基向量垂直,即把 拆开:
$$\vec{p}=\vec{b}+[\vec{u}_1|\vec{u}_2|\cdots|\vec{u}_k]\vec{x}_1\qquad{(3)} $$然后将 表示成矩阵形式:
$$\begin{bmatrix} \vec{u}_1^T\\ \hline\vec{u}_2^{T^{^{^{}}}}\\ \hline\cdots\\ \hline\vec{u}_k^{T^{^{^{}}}} \end{bmatrix}(\vec{b}+[\vec{u}_1|\vec{u}_2|\cdots|\vec{u}_k]\vec{x}_1)=\vec{0} $$注意到 $\begin{bmatrix}\vec{u}_1^T\\\hline\vec{u}_2^{T^{^{^{}}}}\\\hline\cdots\\\hline\vec{u}_k^{T^{^{^{}}}} \end{bmatrix}[\vec{u}_1|\vec{u}_2|\cdots|\vec{u}_k]$ 为 阶方阵 ,因此高斯消元解线性方程组
$$\begin{bmatrix}\vec{u}_1^T\\\hline\vec{u}_2^{T^{^{^{}}}}\\\hline\cdots\\\hline\vec{u}_k^{T^{^{^{}}}} \end{bmatrix}[\vec{u}_1|\vec{u}_2|\cdots|\vec{u}_k]\vec{x}_1=-\begin{bmatrix}\vec{u}_1^T\\\hline\vec{u}_2^{T^{^{^{}}}}\\\hline\cdots\\\hline\vec{u}_k^{T^{^{^{}}}} \end{bmatrix}\vec{b} $$即可得到 ,代入 即可求出 。
因为过线性流形(直线/平面)上一点有且只有一条直线与该流形(直线/平面)垂直,
这个线性方程组一定有唯一解,也就一定能解出 。
解出 后,勾股定理得到
然后 就是一组合法的解了。
最后不要忘记,我们一开始将原点平移到了 ,要平移回去才行哦!
Code
/* this code is made by warzone 2022.3.11 15:30 */ #include<stdio.h> #include<string.h> #include<math.h> #include<algorithm> typedef double db; struct READ{//快读 char c,w; inline READ(){c=getchar();} template<typename type> inline READ& operator >>(type& num){ for(w=1;'0'>c||c>'9';c=getchar()) w=c=='-'? -1:1; for(num=0;'0'<=c&&c<='9';c=getchar()) num=num*10+(c-'0'); if(c=='.') for(db i=1;c=getchar(),'0'<=c&&c<='9';) i*=0.1L,num+=i*(c-'0'); return num*=w,*this; } }cin; int d,n; int pos[512],pos_[512]; db y[512],r; db A[512][512],a[512]; db B[512][512],b[512]; int top,top_; template<typename type1,typename type2,typename type3> inline void gauss(type1 &A,type2 &a,type3 &pos,//高斯消元 int &top,const int d,const int n){ top=d; for(int id=0;id<top&&id<n;++id){ while(A[id][id]==0){ for(int i=id+1;i<n;++i) if(A[i][id]){ for(int j=id;j<d;++j) std::swap(A[id][j],A[i][j]); std::swap(a[id],a[i]);break; } if(A[id][id]==0){//把消元失败的维度放到最后面 if(id>=--top) return; std::swap(pos[top],pos[id]); for(int i=0;i<n;++i) std::swap(A[i][id],A[i][top]); } } if(A[id][id]!=1){ db get=A[id][id]; for(int i=id+1;i<d;++i) A[id][i]/=get; A[id][id]=1,a[id]/=get; } for(int i=0;i<n;++i) if(i!=id&&A[i][id]){ db get=A[i][id]; for(int j=id+1;j<d;++j) A[i][j]-=A[id][j]*get; A[i][id]=0,a[i]-=a[id]*get; } } } inline void print(){//平移回来并输出答案 for(int i=0;i<d;++i) y[pos[i]]+=a[i]; for(int i=0;i<d;++i) printf("%0.6lf ",y[i]); } int main(){ cin>>d>>n; for(int i=0;i<d;++i) cin>>y[i],pos[i]=i;//输入 (1) 式 cin>>r,--n; for(int id=0;id<n;++id){ for(int i=0;i<d;++i){//减去 (1) 式 cin>>A[id][i],A[id][i]-=y[i]; a[id]+=A[id][i]*A[id][i]; } db get; cin>>get,a[id]=(a[id]+r*r-get*get)/2; } gauss(A,a,pos,top,d,n),top=std::min(top,n);//第一次高斯消元 if(top==d) return print(),0;//唯一解 for(int i=top;i<d;++i) A[i][i]=-1; const int k=d-top; for(int id=0;id<k;++id){//计算 \vec{p} 的线性方程组 pos_[id]=id; for(int i=0;i<k;++i) for(int j=0;j<d;++j) B[id][i]+=A[j][id+top]*A[j][i+top]; for(int j=0;j<d;++j) b[id]-=A[j][id+top]*a[j]; } gauss(B,b,pos_,top_,k,k);//第二次高斯消元 for(int id=0;id<k;++id)//将 \vec{x}_1 代回 \vec{p} for(int i=0;i<d;++i) a[i]+=b[id]*A[i][id+top]; db r_=r*r,r__=0; for(int i=0;i<d;++i){//勾股定理计算 r' r__+=A[i][top]*A[i][top]; r_-=a[i]*a[i]; } r_=sqrtl(r_/r__); for(int i=0;i<d;++i) a[i]+=A[i][top]*r_; return print(),0; } - 中 系数均为 ,因此 一定是个二元一次不定方程,
- 1
信息
- ID
- 7487
- 时间
- 6000ms
- 内存
- 512MiB
- 难度
- 7
- 标签
- 递交数
- 0
- 已通过
- 0
- 上传者