首页 > 其他分享 >CF888G题解

CF888G题解

时间:2023-10-27 10:11:18浏览次数:38  
标签:CF888G int 题解 tr tp LCA 节点 nw

  • 分析

    看到异或不难想到 01Trie。
    不难想到,当两个数的值相等的时候,我们可以当这两个点是一个点,因为连边的费用为 \(0\)。
    那么对于一个序列 \(n\),若存在 \(m\) 种不同的权值,那么在 Trie 树上子节点数为 \(2\) 的节点就有 \(m-1\) 个(因为如果一个数新加进来与所有数都不同,那么一定会从已有的树上分出一条新链,使得一个节点的度数加 \(1\),因为这个节点必是度数为 \(1\) 的非叶节点,所以会新产生一个度数为 \(2\) 的节点),然后我们发现我们实际要处理的边数也是 \(m-1\) 个,而且权值也是这 \(m\) 种。
    我们肯定优先选取不连通而且异或值最小的两个数,所以很好想的是在 Trie 树上做 dfs,然后找到深度最大的点取 LCA。
    但是这个做法由于要保证两个点不连通,所以不好直接贪心找两个点求LCA使答案最小。
    发现有 \(m-1\) 个点的子树数为 \(2\),而一个点是 LCA 的条件就是子树数为 \(2\),而且我们只需要 \(m-1\) 个 LCA,同时Trie的左右两颗子树肯定不连通(因为 Trie 树也是一棵树),那么我们发现是 LCA 的点已经确定下来了,就是那些子树数为 \(2\)的点,接下来就是找连边花费最小的两个子节点。
    因为是LCA,所以需要连边的两个节点一定是分局两个子树的,所以需要在左右子树各取一个点,保证这两个点的异或和最小。这个还是可以用枚举左左子树内的点在右子树中查询来解决。为了方便枚举,我们要保证子树内的点是连续的,所以我们要先排序。
    那么最后的答案就是 \(\sum_{i\in\{x|x有两个子节点\}} \min_{x\in\{x|tr_{i,0}\},y\in\{y|tr_{i,1}\}} x \oplus y\)。、

  • 代码

#include <iostream>
#include <algorithm>
#define int long long 
using namespace std;
constexpr int MAXN(200007);
constexpr int INF(0x3f3f3f3f);
inline void read(int &temp) { cin >> temp; }
int tr[MAXN << 5][2], l[MAXN << 5], r[MAXN << 5], tot;
int a[MAXN], n;
struct Trie{
	inline void insert(int x) {
		int nw(0), tp(0);
		int t = a[x];
		for (int i(30); ~i; --i) {
			tp = (t >> i) & 1; 
			if (!tr[nw][tp])  tr[nw][tp] = ++tot;
			if (!l[nw])  l[nw] = x;
			r[nw] = x;
			nw = tr[nw][tp];
		}
	}
	inline int query(int x, int nw, int k) {
		if (k == -1)  return 0;
		int tp = (x >> k) & 1; 
		if (!tr[nw][tp])  return (1 << k) + query(x, tr[nw][tp ^ 1], k - 1);
		return query(x, tr[nw][tp], k - 1);
	}
}Kafu;
int dfs(int nw, int k) {
	if (k == -1)  return 0;
	if (!tr[nw][0])  return dfs(tr[nw][1], k - 1);
	if (!tr[nw][1])  return dfs(tr[nw][0], k - 1);
	int ans(INF);
	for (int i(l[tr[nw][0]]); i <= r[tr[nw][0]]; ++i)  ans = min(ans, Kafu.query(a[i], tr[nw][1], k - 1) + (1 << k));
	return dfs(tr[nw][0], k - 1) + dfs(tr[nw][1], k - 1) + ans;
}
signed main() {
	ios::sync_with_stdio(false), cin.tie(nullptr), cout.tie(nullptr);
	read(n);
	for (int i(1); i <= n; ++i)  read(a[i]);
	sort(a + 1, a + n + 1);
	for (int i(1); i <= n; ++i)  Kafu.insert(i);
	cout << dfs(0, 30) << endl;
	return 0;
}

标签:CF888G,int,题解,tr,tp,LCA,节点,nw
From: https://www.cnblogs.com/Kazdale/p/17791134.html

相关文章

  • P8865 [NOIP2022] 种花 题解
    前言去年多测不清空导致即便CCF放过了我的\(O(n^2m)\)的代码但依然挂成了\(0pts\)。当时看清空数组后能过CCF数据就没再管。时隔\(1\)年,重做这道题写了\(O(nm)\)的正解,终于完成了当年的心愿。\(O(n^2m)\)思路想到计算方案的话可以维护两个数组\(c1_{i,j}\)表......
  • SP4082 MBLAST - BLAST 题解
    几万年前做的dp题了,有亿点点水题意简述求一个字符串添加多少个空格距离最小解法求距离最小,可以考虑动规,其实这题的写法和最长公共子序列的写法类似。我们设\(f(i,j)\)表示\(a[1]\sima[i]\)和\(b[1]\simb[j]\)的距离不加空格的时候为\(f(i,j)=f(i-1,j-1)+\le......
  • [AGC061A] Long Shuffle 题解
    题意给定一个满足\(A_i=i\)的排列\(A\),求对其进行一次\(\mathrm{shuffle}(1,N)\)操作后其第\(K\)项的值。其中\(\mathrm{shuffle}(L,R)\)的定义如下:若\(R=L+1\),那么交换\(A_L\)和\(A_R\)的值否则,依次执行\(\mathrm{shuffle}(L,R-1)\),\(\mathrm{shuffle}(......
  • P4678 [BalticOI 2005] Bus Trip 题解
    P4678[BalticOI2005]BusTrip题解(RE:题解再改造!!)贴码#include<bits/stdc++.h>#defineMAXN500010usingnamespacestd;//ifstreamis("trip.in",ios::in);//ofstreamos("trip.out",ios::out);//#definecinis//#definecoutosintn,m,p,t,tote......
  • [HDU 3483] A Very Simple Problem 题解
    题目描述快速求出下面式子的值:\[\left(\sum\limits_{k=1}^{N}k^{x}x^{k}\right)\bmodM\]其中\(1≤N,M≤2\times10^9\),并且\(1≤x≤50\)。题解(solution)对于该类题目,\(N\)很大,而\(x\)很小,考虑矩阵快速幂优化。对于每一个\(N\)的答案,我们用\(f(N)\)来......
  • P5537 【XR-3】系统设计 题解-哈希+线段树二分
    20231026P5537【XR-3】系统设计题解-哈希+线段树二分这个东西怎么会和哈希有关?!直接寄。Statement这个系统首先需要输入一棵\(n\)个点的有根树和一个长度为\(m\)的序列\(a\),接下来需要实现\(q\)个操作。操作分两种:1xlr表示设定起点为有根树的节点\(x\),接下来......
  • CF888E题解
    分析看到\(n\leq35\)的数据范围就想到了meet-in-middle。先爆搜出对于\(1\sim\frac{n}{2}\)和\(\frac{n}{2}\simn\)两个下标范围内在模意义下所有的和。然后用一个常见trick,就是枚举第二个部分的和,然后匹配第一个部分的和的最优解。显然匹配有两种情况,一种是......
  • chrome新版本http自动跳转https问题解决
    虽然是个好功能,但是部分内网业务还没做好https兼容,有的时候手工访问,还是变成https 解决办法:chrome://flags/找到:HTTPSUpgrades,修改disabled,重启解决,当然这个需要需要用户去调整,真正还需要服务去兼容https  ......
  • 在使用 Unity 2022 打包安卓项目时,遇到 gradle 无法访问或下载超级慢最终超时出错的问
    一般表现是打包最后一步会等待超长时间,最后报错,错误信息:PickedupJAVA_TOOL_OPTIONS:-Dfile.encoding=UTF-8FAILURE:Buildfailedwithanexception.*Whatwentwrong:Aproblemoccurredconfiguringrootproject'Gradle'.>Couldnotresolveallartifactsfor......
  • CCC 2023 题解 和 思考过程
    Trianglane水题,只要分情况判断中间和两侧有边叠牢的情况,每次减2#include<iostream>#include<cstdlib>#include<cstdio>#include<cmath>#include<algorithm>#include<cstring>usingnamespacestd;typedeflonglongll;constintN=200010......