【P5994 [PA2014]Kuglarz】题解
题目链接
题目
魔术师的桌子上有 \(n\) 个杯子排成一行,编号为 \(1,2,…,n\),其中某些杯子底下藏有一个小球,如果你准确地猜出是哪些杯子,你就可以获得奖品。
花费 \(c_{ij}\) 元,魔术师就会告诉你杯子 \(i,i+1,…,j\) 底下藏有球的总数的奇偶性。
采取最优的询问策略,你至少需要花费多少元,才能保证猜出哪些杯子底下藏着球?
思路
前缀和建图+最小生成树。
对于查询区间 \([i,j]\),我们其实知道了区间 \([1, j]\) 和区间 \([1, i]\) 的差为多少。
既然如此,我们可以在 \(j\) 和 \(i-1\) 之间建边。
当所有的点连起来的时候,我们就可以确定所有解。
总结
这是一道很不错的前缀和建图题。
这道题的核心在于对于询问一段区间的奇偶,可以用前缀和表示,通过建图,再求最小生成树,可以构建唯一解。
Code
// Problem: P5994 [PA2014]Kuglarz
// Contest: Luogu
// URL: https://www.luogu.com.cn/problem/P5994
// Memory Limit: 125 MB
// Time Limit: 1000 ms
//
// Powered by CP Editor (https://cpeditor.org)
#include<bits/stdc++.h>
using namespace std;
#define int long long
inline int read(){int x=0,f=1;char ch=getchar();
while(ch<'0'||ch>'9'){if(ch=='-')f=-1;
ch=getchar();}while(ch>='0'&&ch<='9'){x=(x<<1)+
(x<<3)+(ch^48);ch=getchar();}return x*f;}
//#define M
//#define mo
#define N 2010
struct node
{
int x, y, z;
}d[N*N];
int n, m, i, j, k;
int f[N], u, v, ans;
void cun(int x, int y, int z)
{
d[++m].x=x; d[m].y=y;
d[m].z=z;
}
bool cmp(node x, node y)
{
return x.z<y.z;
}
int fa(int x)
{
if(f[x]==x) return x;
return f[x]=fa(f[x]);
}
signed main()
{
// freopen("tiaoshi.in", "r", stdin);
// freopen("tiaoshi.out", "w", stdout);
n=read();
for(i=0; i<=n; ++i) f[i]=i;
for(i=1; i<=n; ++i)
{
for(j=i; j<=n; ++j)
{
u=read();
cun(i-1, j, u);
}
}
sort(d+1, d+m+1, cmp);
for(i=1; i<=m; ++i)
{
u=fa(d[i].x);
v=fa(d[i].y);
// printf("%lld %lld\n", u, v);
if(u!=v)
{
// printf("> %lld %lld %lld\n", d[i].x, d[i].y, d[i].z);
f[u]=v;
ans+=d[i].z;
}
}
printf("%lld", ans);
return 0;
}
本文来自博客园,作者:zhangtingxi,转载请注明原文链接:https://www.cnblogs.com/zhangtingxi/p/15838986.html