UOJ 176 新年的繁荣

编程入门 行业动态 更新时间:2024-10-20 20:44:17

UOJ 176 新年的<a href=https://www.elefans.com/category/jswz/34/1751170.html style=繁荣"/>

UOJ 176 新年的繁荣

挺妙的解法。

发现边权很小,我们可以考虑从大到小枚举边权来进行$kruskal$算法,这样子对于每一个边权$i$,我们只要枚举$0 \leq j < m$,找到一个点使它的点权为$i | 2^j$,尝试连边即可。

另外,如果同一个点权重复出现,一定有办法使这个边权连满,这样子直接累加到答案里就可以了。

时间复杂度$O(m * 2^m)$,再套一个并查集的复杂度。

Code:

#include <cstdio>
#include <cstring>
using namespace std;
typedef long long ll;const int N = 18;int n, m, a[1 << N], ufs[1 << N];
ll ans = 0LL;inline void read(int &X) {X = 0; char ch = 0; int op = 1;for(; ch > '9' || ch < '0'; ch = getchar())if(ch == '-') op = -1;for(; ch >= '0' && ch <= '9'; ch = getchar())X = (X << 3) + (X << 1) + ch - 48;X *= op; 
}inline int find(int x) {return ufs[x] == x ? x : ufs[x] = find(ufs[x]);
}inline bool merge(int x, int y) {int fx = find(x), fy = find(y);if(fx == fy) return 0;ufs[fx] = fy;return 1;
}int main() {
//    freopen("Sample.txt", "r", stdin);
    read(n), read(m);for(int x, i = 1; i <= n; i++) {read(x);if(a[x]) ans += 1LL * x;else a[x] = x;}for(int i = 1; i < (1 << m); i++) ufs[i] = i;for(int i = (1 << m) - 1; i >= 0; i--) {for(int j = 0; j < m && (!a[i]); j++)a[i] = a[i | (1 << j)];for(int j = 0; j < m; j++)if(a[i | (1 << j)] && merge(a[i], a[i | (1 << j)]))ans += 1LL * i;}printf("%lld\n", ans);return 0;
}
View Code

 

转载于:.html

更多推荐

UOJ 176 新年的繁荣

本文发布于:2024-02-06 02:07:29,感谢您对本站的认可!
本文链接:https://www.elefans.com/category/jswz/34/1745807.html
版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系,我们将在24小时内删除。
本文标签:繁荣   新年   UOJ

发布评论

评论列表 (有 0 条评论)
草根站长

>www.elefans.com

编程频道|电子爱好者 - 技术资讯及电子产品介绍!