LOJ2323. 「清华集训 2017」小 Y 和地铁 【搜索】【思维】【好】
思路
首先如果直接算每一个段有三个决策
左/右 上/下 跨不跨过端点
这样的复杂度是\((2^3)^{22}\),显然是无法接受的
然后考虑怎么优化这个东西
首先左右这个决策是没有意义的
因为无论是左还是右,对答案的相对影响是不变的
其次考虑用a和b分别表示上下和跨不跨过端点的决策
然后退一下就可发现有很多种情况是可以归约的
大概成了这样
void dfs(int tmp, int res) {
if (res >= ans) return;
if (tmp > n) {
ans = res;
return;
}
if (cnt[t[tmp]] == 1 || pre[tmp]) {
dfs(tmp + 1, res);
return;
}
int now;
a[tmp] = 0;
b[tmp] = 0;
now = res;
fu(i, 1, tmp - 1)
if (!(a[i] ^ b[i]) && nxt[i] > tmp && nxt[i] < nxt[tmp]) ++now;
dfs(tmp + 1, now);
a[tmp] = 0;
b[tmp] = 1;
now = res;
fu(i, 1, tmp - 1) {
if ((a[i] ^ b[i]) && nxt[i] > nxt[tmp]) ++now;
if (!(a[i] ^ b[i]) && nxt[i] > tmp) ++now;
}
dfs(tmp + 1, now);
a[tmp] = 1;
b[tmp] = 0;
now = res;
fu(i, 1, tmp - 1)
if ((a[i] ^ b[i]) && nxt[i] > tmp && nxt[i] < nxt[tmp]) ++now;
dfs(tmp + 1, now);
a[tmp] = 1;
b[tmp] = 1;
now = res;
fu(i, 1, tmp - 1) {
if (!(a[i] ^ b[i]) && nxt[i] > nxt[tmp]) ++now;
if ((a[i] ^ b[i]) && nxt[i] > tmp) ++now;
}
dfs(tmp + 1, now);
}
这样是可以得到64分的高分的
但是并不够
继续考虑怎么压缩状态
发现前面的每个已经决策的区间对当前区间的影响只有a和b的异或和
所以每次只枚举a和b的异或和
但是a和b的异或和相等的情况有两种
但是这两种对后面决策的影响都是相同的
所以直接贪心取最小的就可以了
复杂度\(n*2^{n/2}\)
AC代码
//Author: dream_maker
#include<bits/stdc++.h>
using namespace std;
//----------------------------------------------
//typename
typedef long long ll;
//convenient for
#define fu(a, b, c) for (int a = b; a <= c; ++a)
#define fd(a, b, c) for (int a = b; a >= c; --a)
#define fv(a, b) for (int a = 0; a < (signed)b.size(); ++a)
//inf of different typename
const int INF_of_int = 1e9;
const ll INF_of_ll = 1e18;
//fast read and write
template <typename T>
void Read(T &x) {
bool w = 1;x = 0;
char c = getchar();
while (!isdigit(c) && c != '-') c = getchar();
if (c == '-') w = 0, c = getchar();
while (isdigit(c)) {
x = (x<<1) + (x<<3) + c -'0';
c = getchar();
}
if (!w) x = -x;
}
template <typename T>
void Write(T x) {
if (x < 0) {
putchar('-');
x = -x;
}
if (x > 9) Write(x / 10);
putchar(x % 10 + '0');
}
//----------------------------------------------
const int N = 50;
int t[N], n, ans;
int pre[N], nxt[N], last[N], cnt[N];
bool a[N], b[N], mark[N];
//a 走上面(0)还是下面(1)
//b 不跨过(0)还是跨过(1)
int T;
/*void dfs(int tmp, int res) {
if (res >= ans) return;
if (tmp > n) {
ans = res;
return;
}
if (cnt[t[tmp]] == 1 || pre[tmp]) {
dfs(tmp + 1, res);
return;
}
int now;
a[tmp] = 0;
b[tmp] = 0;
now = res;
fu(i, 1, tmp - 1)
if (!(a[i] ^ b[i]) && nxt[i] > tmp && nxt[i] < nxt[tmp]) ++now;
dfs(tmp + 1, now);
a[tmp] = 0;
b[tmp] = 1;
now = res;
fu(i, 1, tmp - 1) {
if ((a[i] ^ b[i]) && nxt[i] > nxt[tmp]) ++now;
if (!(a[i] ^ b[i]) && nxt[i] > tmp) ++now;
}
dfs(tmp + 1, now);
a[tmp] = 1;
b[tmp] = 0;
now = res;
fu(i, 1, tmp - 1)
if ((a[i] ^ b[i]) && nxt[i] > tmp && nxt[i] < nxt[tmp]) ++now;
dfs(tmp + 1, now);
a[tmp] = 1;
b[tmp] = 1;
now = res;
fu(i, 1, tmp - 1) {
if (!(a[i] ^ b[i]) && nxt[i] > nxt[tmp]) ++now;
if ((a[i] ^ b[i]) && nxt[i] > tmp) ++now;
}
dfs(tmp + 1, now);
}*/
void dfs(int tmp, int res) {
if (res >= ans) return;
if (tmp > n) {
ans = res;
return;
}
if (cnt[t[tmp]] == 1 || pre[tmp]) {
dfs(tmp + 1, res);
return;
}
int now1, now2;
mark[tmp] = 0;
now1 = now2 = res;
fu(i, 1, tmp - 1) {
if (!mark[i] && nxt[i] > tmp && nxt[i] < nxt[tmp]) ++now1;
if (!mark[i] && nxt[i] > nxt[tmp]) ++now2;
if (mark[i] && nxt[i] > tmp) ++now2;
}
dfs(tmp + 1, min(now1, now2));
mark[tmp] = 1;
now1 = now2 = res;
fu(i, 1, tmp - 1) {
if (mark[i] && nxt[i] > tmp && nxt[i] < nxt[tmp]) ++now1;
if (mark[i] && nxt[i] > nxt[tmp]) ++now2;
if (!mark[i] && nxt[i] > tmp) ++now2;
}
dfs(tmp + 1, min(now1, now2));
}
void solve() {
Read(n);
fu(i, 1, n) pre[i] = nxt[i] = last[i] = cnt[i] = 0;
fu(i, 1, n) {
Read(t[i]);
++cnt[t[i]];
if (last[t[i]]) {
pre[i] = last[t[i]];
nxt[last[t[i]]] = i;
} else {
last[t[i]] = i;
}
}
ans = INF_of_int;
dfs(1, 0);
printf("%d\n", ans);
}
int main() {
freopen("input.txt","r",stdin);
Read(T);
while (T--) solve();
return 0;
}
LOJ2323. 「清华集训 2017」小 Y 和地铁 【搜索】【思维】【好】的更多相关文章
- [LOJ#2323]「清华集训 2017」小Y和地铁
[LOJ#2323]「清华集训 2017」小Y和地铁 试题描述 小Y是一个爱好旅行的OIer.一天,她来到了一个新的城市.由于不熟悉那里的交通系统,她选择了坐地铁. 她发现每条地铁线路可以看成平面上的 ...
- Loj #2324. 「清华集训 2017」小 Y 和二叉树
Loj #2324. 「清华集训 2017」小 Y 和二叉树 小Y是一个心灵手巧的OIer,她有许多二叉树模型. 小Y的二叉树模型中,每个结点都具有一个编号,小Y把她最喜欢的一个二叉树模型挂在了墙上, ...
- loj #2325. 「清华集训 2017」小Y和恐怖的奴隶主
#2325. 「清华集训 2017」小Y和恐怖的奴隶主 内存限制:256 MiB时间限制:2000 ms标准输入输出 题目类型:传统评测方式:文本比较 题目描述 "A fight? Co ...
- [LOJ#2324]「清华集训 2017」小Y和二叉树
[LOJ#2324]「清华集训 2017」小Y和二叉树 试题描述 小Y是一个心灵手巧的OIer,她有许多二叉树模型. 小Y的二叉树模型中,每个结点都具有一个编号,小Y把她最喜欢的一个二叉树模型挂在了墙 ...
- loj2324 「清华集训 2017」小 Y 和二叉树
https://loj.ac/problem/2324 太智障,一开始以为中序遍历的第一个点一定是一个叶子,想了个贪心.然而,手算了一下,第一个点都过不了啊. input 5 2 3 4 1 3 3 ...
- LOJ2324「清华集训 2017」小Y和二叉树
题目链接 瞎jb贪一发就过了.首先度数<=2且编号最小的点一定是中序遍历最靠前的点,我们从这个点开始dfs一遍算出子树中度数<=2且编号最小的点记为\(f(i)\),然后从这个点开始一步一 ...
- 【loj2325】「清华集训 2017」小Y和恐怖的奴隶主 概率dp+倍增+矩阵乘法
题目描述 你有一个m点生命值的奴隶主,奴隶主受伤未死且当前随从数目不超过k则再召唤一个m点生命值的奴隶主. T次询问,每次询问如果如果对面下出一个n点攻击力的克苏恩,你的英雄期望会受到到多少伤害. 输 ...
- LOJ2324. 「清华集训 2017」小 Y 和二叉树【贪心】【DP】【思维】【好】
LINK 思路 首先贪新的思路是处理出以一个节点为根所有儿子的子树中中序遍历起始节点最小是多少 然后这个可以两次dfs来DP处理 然后就试图确定中序遍历的第一个节点 一定是siz<=2的编号最小 ...
- LibreOJ #2325. 「清华集训 2017」小Y和恐怖的奴隶主(矩阵快速幂优化DP)
哇这题剧毒,卡了好久常数才过T_T 设$f(i,s)$为到第$i$轮攻击,怪物状态为$s$时对boss的期望伤害,$sum$为状态$s$所表示的怪物个数,得到朴素的DP方程$f(i,s)=\sum \ ...
随机推荐
- 谈一谈最近关闭的Kindle人论坛
最近Kindle圈子内最大的论坛“Kindle人”关闭了,倒也掀起了一阵小波澜.Kindle人论坛是K友圈子里比较著名的一个“Kindle资源分享论坛”,这么一说其实混了这么久网络,大家都知道这个论坛 ...
- Python 实现99乘法表
首先,我们来回忆一下99乘法表长什么样子吧 进入正题:实现99乘法表 一.For循环 for i in range(1,10): for j in range(1,i+1): print(" ...
- IIS优化整理
IIS 之 在IIS7.IIS7.5中应用程序池最优配置方案 找到Web站点对应的应用程序池,“应用程序池” → 找到对应的“应用程序池” → 右键“高级设置...” 一.一般优化方案 1.基本设置 ...
- web常见问题排查
原帖地址:http://mp.weixin.qq.com/s?__biz=MjM5NzUwNDA5MA==&mid=200596752&idx=1&sn=37ecae802f3 ...
- HttpConnection的使用
项目中需要与第三方系统交互,而交互的方式是XML报文形式,所以会用到HttpConnection与第三方系统连接交互,使用起来并不复杂,但是有几点需要注意的: 1.乱码的问题解决 2.超时的设置,注意 ...
- UVA-11396 Claw Decomposition (二分图判定)
题目大意:给一张无向图,能否把它分成若干个“爪”,即,一个点有三个子节点. 题目分析:每个点的度数3是已知的,只需判断一下是不是二分图即可. 代码如下: # include<iostream&g ...
- 【Python】测算代码运行时间
整理自这里和这里 timeit模块 timeit模块定义了接受两个参数的 Timer 类.两个参数都是字符串. 第一个参数是你要计时的语句或者函数. 传递给 Timer 的第二个参数是为第一个参数语句 ...
- Ansible 小手册系列 九(Playbook)
playbook是由一个或多个"play"组成的列表.play的主要功能在于将事先归并为一组的主机装扮成事先通过ansible中的task定义好的角色.从根本上来讲所谓task无非 ...
- 通过windows的超级终端连接华为交换机
如果你的电脑没有带COM接口 可以找一个COM--->USB的转换器 我前面有介绍过这么安装COM转USB设备的驱动 2 先打开你电脑的设备管理器 看看你的com接口数字编号 这里看到的是COM ...
- 一款连接SqlServer的数据库工具
由于自己使用的电脑系统是xp,而服务器上的数据库是SqlServer2012,于是用SqlServer2005管理端操作2012,总是不成功.在网上也百度谷歌了很久,也没有解决,也发了很多问没有找到解 ...