P16159 [ICPC 2016 NAIPC] Symmetry

ooliver 发布于 2026-04-26 170 次阅读 OI


AI 摘要

**引言:** 想找出最少添加几个点能让图形对称?枚举所有对称中心和对称轴,哈希存储中点、直线;结合公式计算答案,保证整数运算避免小数。对称中心减点,对称轴再减线上点,取最小解。

P16159 [ICPC 2016 NAIPC] Symmetry

思路

先考虑枚举所有可能的对称中心和对称点。

用两个 map 记录每个对称中心或对称轴对应了几组对称点,再用一个 map 记录点的存在情况。

我们 n2n^2 枚举每两个点。
对于对称中心,我们把两个点的中点加一次贡献;
对于对称轴,我们把两个点的连线和两个点的中垂线记录下来,连线不加贡献,中垂线加一次贡献。

统计答案时,我们枚举所有对称中心和对称轴。
对于对称中心,答案应该是点数减去当前对称中心对应对称点组数的两倍,注意如果当前对称中心存在于点集中,答案应再减一。
对于对称轴,我们还需要一个循环找到在当前直线上的点的个数,答案就应该是点数减去当前对称轴对应对称点组数的两倍、再减去点集中在这条直线上的点的个数。
最终取最小答案输出即可。

细节

  1. 为了防止求中点时出现小数,我们在输入时全部乘二。
  2. 对于点的存储,直接哈希;对于直线的存储,我们用结构体存储其一般式 $Ax+By+C=0$ 中的三个常数,注意要存储最简形式。
  3. 已知两点 $(x_1,y_1),(x_2,y_2)$,
    中点:$(\frac{x_1+x_2}{2},\frac{y_1+y_2}{2})$;
    连线:$(y_1−y_2)x+(x_2−x_1)y+(x_1y_2−x_2y_1)=0$;
    中垂线:$2(x_2-x_1)x+2(y_2-y_1)y+(x_1^2-x_2^2+y_1^2-y_2^2)=0$。

代码:

C++
#include<bits/stdc++.h>
using namespace std;

#define int long long
#define fi first
#define sc second

const int N=1e3+5;
int n,ans=INT_MAX;
int x[N],y[N];

struct line{
    int a,b,c;
    bool operator<(const line& l1) const{
        if(a!=l1.a) return a<l1.a;
        if(b!=l1.b) return b<l1.b;
        return c<l1.c;
    }
};
map<int,int> d,tag;
map<line,int> l;

int hsh(int xx,int yy){return xx*100000+yy;}

line func(int a,int b,int c){
    int g=__gcd(__gcd(a,b),c);
    a/=g,b/=g,c/=g;
    if(a<0||(a==0&&b<0)) a=-a,b=-b,c=-c;
    return {a,b,c};
}

void add(int x1,int y1,int x2,int y2){
    int a1=(x2-x1)<<1,b1=(y2-y1)<<1,c1=x1*x1-x2*x2+y1*y1-y2*y2;
    int a2=y1-y2,b2=x2-x1,c2=x1*y2-x2*y1;
    d[hsh(x1+x2>>1,y1+y2>>1)]++;
	l[func(a1,b1,c1)]++,l[func(a2,b2,c2)]+=0;
}

signed main(){
	ios::sync_with_stdio(0);
    cin.tie(0),cout.tie(0);
	cin>>n;
	if(n==1){
        cout<<0;
        return 0;
    }
	for(int i=1;i<=n;i++) cin>>x[i]>>y[i],x[i]<<=1,y[i]<<=1,tag[hsh(x[i],y[i])]=1;
	for(int i=1;i<=n;i++) for(int j=i+1;j<=n;j++) add(x[i],y[i],x[j],y[j]);
	for(auto u:d) ans=min(ans,n-u.sc*2-tag[u.fi]);
    for(auto u:l){
        int cnt=n-u.sc*2;
        line ln=u.fi;
        for(int i=1;i<=n;i++) if(x[i]*ln.a+y[i]*ln.b+ln.c==0) cnt--;
        ans=min(ans,cnt);
    }
	cout<<ans;
	return 0;
}