「Codeforces 79D」Password
Description
有一个 01 序列 \(a_1,a_2,\cdots,a_n\),初始时全为 \(0\)。
给定 \(m\) 个长度,分别为 \(l_1\sim l_m\)。
每次可以选择一个长度为某个 \(l_i\) 区间,对其进行翻转操作(\(0\to 1,1\to 0\))。
求最少的操作次数,使得最后有且仅有 \(k\) 个位置为 \(1\)(\(k\) 个位置给定),其余为 \(0\)。
\(1\leq n\leq 10^4,1\leq k\leq 10,1\leq m\leq 100\)。
Solution
原问题等价于:
给定 01 序列 \(a_1,a_2,\cdots,a_n\),有 \(k\) 个位置为 \(1\),其余为 \(0\)。每次可以翻转长度为 \(l_i\) 的区间,求将 \(a\) 清零的最小操作数。
操作为区间修改,考虑差分。
由于是“区间取反”,一般的作差差分无法使用,考虑 异或差分。令 \(b_i=a_i\text{ xor }a_{i+1}\)(设 \(a_0=a_{n+1}=0\))。
那么,将原序列中的区间 \([l,r]\) 翻转,等价于将差分序列中的 \(b_{l-1},b_r\) 取反(其他元素不变)。
Step1
考虑到 \(b\) 序列初始最多只有 \(2k\) 个 \(1\),则问题转化为:
给定 01 序列 \(b_0,b_2,\cdots,b_n\),最多有 \(2k\) 个位置为 \(1\)。每次可以选择一对距离为 \(l_i\) 的位置,将其取反。求将 \(b\) 清零的最小操作次数。
设选择的一对位置为 \((x,y)\)。分类讨论:
若 \(b_x=0,b_y=0\),则操作后 \(b_x=1,b_y=1\),增加 \(2\) 个 \(1\)。(显然会使答案更劣,不会发生)
若 \(b_x=1,b_y=1\),则操作后 \(b_x=0,b_y=0\),相当于 \(2\) 个 \(1\) 碰撞变成 \(0\),减少 \(2\) 个 \(1\)。
若 \(b_x=1,b_y=0\),则操作后 \(b_x=0,b_y=1\),相当于把 \(x\) 上的 \(1\) 移到 \(y\),\(1\) 的数量不变。
若 \(b_x=0,b_y=1\),与 \(b_x=1,b_y=0\) 同理,\(1\) 的数量不变。
Step2
问题等价于:(第 \(i\) 个节点有标记相当于 \(b_i=1\))
给定一个有 \(n+1\) 个节点的图(点的编号为 \(0\sim n\))。当 \(dis(x,y)=l_i\) 时,存在边 \((x,y)\)。初始时最多有 \(2k\) 个节点上有标记,每次可以沿边移动标记。两个标记相遇就会消失。求使所有标记消失的最少移动次数。
设标记点分别为 \(p_0,p_1,\cdots,p_{g-1}\)。
首先,我们可以通过 BFS 计算出所有标记点对之间的距离。
\(2k\leq 20\),考虑 状压 DP(差分序列中为 \(0\) 的位置不用管,只考虑 \(2k\) 个 \(1\),有 \(2^{2k}\) 种状态)。令 \(f_S\) 表示标记点状态为 \(S\) 时使所有标记消失的最少移动次数。
(\(S\) 二进制下的第 \(i\) 位为 \(1\) 表示标记点 \(p_i\) 上的标记未消失。显然转移只需考虑 \(2\) 个 \(1\) 碰撞变成 \(0\) 的情况,其他情况都是没有意义的,所以我们不需要考虑非初始标记点的状态)
转移:设 \(S\) 二进制下为 \(1\) 的其中两个位为 \(i,j\),\(f_{S}=\min\{f_{S-2^i-2^j}+dis(i,j)\}\)。
初始时 \(f_0=0,f_i=\infty\,(i\neq 0)\)。答案即为 \(f_{2^g-1}\)。
#include<bits/stdc++.h>
#define int long long
using namespace std;
const int N=1e4+5,M=22;
int n,m,k,x,a[N],b[N],l[N],g,c[N],d[N],dis[M][M],f[1<<M],ans;
bool v[N];
queue<int>q;
void bfs(int s){ //BFS 计算出所有标记点对之间的距离
for(int i=0;i<=n;i++) d[i]=1e18,v[i]=0;
d[s]=0,v[s]=1,q.push(s);
while(q.size()){
int x=q.front(),y;q.pop();
for(int i=1;i<=m;i++){
if((y=x+l[i])<=n&&!v[y]) d[y]=d[x]+1,v[y]=1,q.push(y);
if((y=x-l[i])>=0&&!v[y]) d[y]=d[x]+1,v[y]=1,q.push(y);
}
}
for(int i=0;i<=n;i++)
if(b[i]) dis[c[s]][c[i]]=d[i];
}
signed main(){
scanf("%lld%lld%lld",&n,&k,&m);
for(int i=1;i<=k;i++)
scanf("%lld",&x),a[x]=1;
for(int i=1;i<=m;i++)
scanf("%lld",&l[i]);
for(int i=0;i<=n;i++)
b[i]=a[i]^a[i+1],c[i]=(b[i]?g++:0); //b 为差分序列。若节点 i 是标号点,也就是 b[i]=1,则节点 i 对应的标记点编号为 c[i](编号从 0 开始)
for(int i=0;i<=n;i++)
if(b[i]) bfs(i); //注意这里是 if(b[i]) 而不是 if(c[i]),因为标记点的编号是从 0 开始的
for(int s=1;s<(1<<g);s++){ //状压 DP
f[s]=1e18;
for(int i=0;i<g;i++){
if(!((s>>i)&1)) continue;
for(int j=i+1;j<g;j++) //枚举 S 二进制下为 1 的两个位为 i,j
if((s>>j)&1) f[s]=min(f[s],f[s-(1<<i)-(1<<j)]+dis[i][j]);
}
}
ans=f[(1<<g)-1],printf("%lld\n",ans==1e18?-1:ans);
return 0;
}
「Codeforces 79D」Password的更多相关文章
- 「CodeForces 581D」Three Logos
BUPT 2017 Summer Training (for 16) #3A 题意 给你三个矩形,需要不重叠不留空地组成一个正方形.不存在输出-1,否则输出边长和这个正方形(A,B,C表示三个不同矩形 ...
- 「CodeForces - 50C 」Happy Farm 5 (几何)
BUPT 2017 summer training (16) #2B 题意 有一些二维直角坐标系上的整数坐标的点,找出严格包含这些点的只能八个方向走出来步数最少的路径,输出最少步数. 题解 这题要求严 ...
- 「CodeForces - 598B」Queries on a String
BUPT 2017 summer training (for 16) #1I 题意 字符串s(1 ≤ |s| ≤ 10 000),有m(1 ≤ m ≤ 300)次操作,每次给l,r,k,代表将r位置插 ...
- 「CodeForces - 717E」Paint it really, really dark gray (dfs)
BUPT 2017 summer training (for 16) #1H 题意 每个节点是黑色or白色,经过一个节点就会改变它的颜色,一开始在1节点.求一条路径使得所有点变成黑色. 题解 dfs时 ...
- 「CodeForces 476A」Dreamoon and Stairs
Dreamoon and Stairs 题意翻译 题面 DM小朋友想要上一个有 \(n\) 级台阶的楼梯.他每一步可以上 \(1\) 或 \(2\) 级台阶.假设他走上这个台阶一共用了 \(x\) 步 ...
- 「CodeForces 546B」Soldier and Badges 解题报告
CF546B Soldier and Badges 题意翻译 给 n 个数,每次操作可以将一个数 +1,要使这 n 个数都不相同, 求最少要加多少? \(1 \le n \le 3000\) 感谢@凉 ...
- CodeForces 79D 【Password】,洛谷P3943 【星空】
其实我做的是洛谷的P3943,但是听说fstqwq窃题...... 题目描述: 小 C 拿来了一长串星型小灯泡,假装是星星,递给小 F,想让小 F 开心一点.不过,有 着强迫症的小 F 发现,这串一共 ...
- 「Codeforces 468C」Hack it!
Description 定义 \(f(x)\) 表示 \(x\) 的各个数位之和.现在要求 \(\sum_{i=l}^rf(i)\bmod a\). 显然 ans=solve(l,r)%a; if(a ...
- 「Codeforces 724F」Uniformly Branched Trees
题目大意 如果两棵树可以通过重标号后变为完全相同,那么它们就是同构的. 将中间节点定义为度数大于 \(1\) 的节点.计算由 \(n\) 个节点,其中所有的中间节点度数都为 \(d\) 的互不同构的树 ...
随机推荐
- Z可读作zed的出处?
Commercial and international telephone and radiotelephone SPELLING ALPHABETS between World War I and ...
- Java偏向锁浅析
偏向锁的定义 顾名思义,偏向锁会偏向第一个访问锁的线程. 如果在接下来的运行过程中,该锁没有被其他线程访问,这持有偏向锁的线程将永远不需要同步 如果在运行过程中,遇到了其他线程抢占锁,则持有偏向锁的线 ...
- day20 系统优化
day20 系统优化 yum源的优化 yum源的优化: 自建yum仓库 使用一个较为稳定的仓库 # 安装华为的Base源 或者使用清华的源也可以 wget -O /etc/yum.repos.d/Ce ...
- Linux的小知识
1. top 命令可以在Linux下查看任务管理器和当前进程使用资源情况. 2. Ctrl+c 即可退出,然后使用 kill+进程号 命令可杀死指定进程 3.在Linux的 /etc/rc.local ...
- 【STM32】使用SDIO进行SD卡读写,包含文件管理FatFs(三)-SD卡的操作流程
其他链接 [STM32]使用SDIO进行SD卡读写,包含文件管理FatFs(一)-初步认识SD卡 [STM32]使用SDIO进行SD卡读写,包含文件管理FatFs(二)-了解SD总线,命令的相关介绍 ...
- Oracle中的job(定时任务)
oracle中的job(定时任务)由dbms_job包下的函数实现.关于job的理论知识可参考https://blog.csdn.net/apextrace/article/details/77675 ...
- final&static
final 1.final修饰类,那么该类不能有子类,那么也就没有子类重写父类的方法,也就没有多态 2.final修饰成员变量,那么成员变量要么显式赋值(用第一种),要么在构造方法中赋值 无论哪一种, ...
- spring注解-属性
一.@Value 基本数值 可以写SpEL: #{} 可以写${}取出配置文件[properties]中的值(在运行环境变量里面的值) @Value("张三") private S ...
- OpenStack之三: 安装MySQL,rabbitmq, memcached
官网地址:https://docs.openstack.org/install-guide/environment-sql-database-rdo.html #:安装mysql [root@mysq ...
- 【VSCode】检测到 #include 错误。请更新 includePath。已为此翻译单元(C:\mingw-w64\i686-8.1.0-posix-dwarf-rt_v6-rev0\mingw32\i686-
win+r 运行cmd 输入"gcc -v -E -x c -"获取mingw路径: 我的: #include "..." search starts here ...