洛谷P3943 星空
题目背景
命运偷走如果只留下结果, 时间偷走初衷只留下了苦衷。
你来过,然后你走后,只留下星空。
题目描述
逃不掉的那一天还是来了,小 F 看着夜空发呆。
天上空荡荡的,没有一颗星星——大概是因为天上吹不散的乌云吧。
心里吹不散的乌云,就让它在那里吧,反正也没有机会去改变什么了。
小 C 拿来了一长串星型小灯泡,假装是星星,递给小 F,想让小 F 开心一点。
不过,有 着强迫症的小 F 发现,这串一共 n 个灯泡的灯泡串上有 k 个灯泡没有被点亮。
小 F 决定 和小 C 一起把这个灯泡串全部点亮。
不过,也许是因为过于笨拙,小 F 只能将其中连续一段的灯泡状态给翻转——点亮暗灯 泡,熄灭亮灯泡。
经过摸索,小 F 发现他一共能够翻转 m 种长度的灯泡段中灯泡的状态。
小 C 和小 F 最终花了很长很长很长很长很长很长的时间把所有灯泡给全部点亮了。
他 们想知道他们是不是蠢了,因此他们找到了你,让你帮忙算算:在最优的情况下,至少需要 几次操作才能把整个灯泡串给点亮?
输入输出格式
输入格式:
从标准输入中读入数据。
输入第 1 行三个正整数 n,k,m。
输入第 2 行 $k$ 个正整数,第 i 个数表示第 i 个被没点亮的灯泡的位置 $a_i$。
输入第 3 行 $m$ 个正整数,第 i 个数表示第 i 种操作的长度 $b_i$。
保证所有 $b_i$ 互不相同;保证对于 $1 \le i < k$,有 $a_i< a_{i+1}$;保证输入数据有解。
输出格式:
输出标准输入中。
输出一行一个非负整数,表示最少操作次数。
输入输出样例
5 2 2
1 5
3 4
2
说明
【样例 1 解释】
【数据范围与约定】
子任务会给出部分测试数据的特点。如果你在解决题目中遇到了困难,可以尝试只解 决一部分测试数据。
每个测试点的数据规模及特点如下表
特殊性质:保证答案小于 4
题解Here!
一道状压$DP$好题。
首先发现$n$特别的大,但是$k$特别小,显然是要在$k$上面做文章。
注意到$k\leq8$,感觉有点状压$DP$的样子啊。
我们先对原序列进行异或差分。
异或差分是啥?$emmmm\cdots\cdots$
假如有一个$01$序列$10010101$,对其进行异或差分得到了$110111111$。
注意末尾多了一位。
有什么用呢?
我们可以发现,当我们要对$[l,r]$进行区间异或时,我们只需要对$l-1,r$这两个点异或就好。
于是现在我们要求的就变成了:
将异或序列每次取反两个间隔一定的点,求序列中所有元素变成$0$的最小次数。
而我们一次在两个$0$上取反没有任何意义。
所以每一次修改要么是把两个$1$变成$0$,要么是一个$0$变$1$,一个$1$变$0$。
如果是后者,我们可以把它看成前面的$1$往后跳了一段。
但是最后还是要和另外一个$1$同时消掉。
于是就变成了点与点之间两两配对求最小代价的问题。
点与点之间匹配的最小代价可以跑$SPFA$搞出来。
当然这个匹配不是二分图匹配。。。
这个匹配时一般图带权匹配。
但是这玩意好像要用带花树啊。。。
蒟蒻表示根本不会。。。
($PS$:一般图带权匹配可以去$UOJ$找板子。。。我这个菜鸡就算了。。。)
那怎么办???
等一下!有效点数好像不超过$16$?
我们可以状压$DP$啊!
复杂度表面上是$O(2^kk^2)$。
但是我们发现我们只要按顺序找到一对没有匹配过的点直接转移就可以了。
所以复杂度实际上是$O(2^kk)$。
仍然可以遍历所有状态。
于是这个题被$\text{差分}+SPFA+\text{状压}DP$完美搞定。
一开始把$m,k$的读入顺序搞反了还$WA$了一发。。。
我的代码里$k$换成了$q$,注意一下。
附代码:
#include<iostream>
#include<algorithm>
#include<cstdio>
#include<queue>
#include<cstring>
#define MAXN 40010
#define MAXM 70
#define MAXK 20
using namespace std;
int n,m,q;
int A[MAXN],B[MAXM],path[MAXN],pos[MAXK],dis[MAXK][MAXK],dp[1<<MAXK];
inline int read(){
int date=0,w=1;char c=0;
while(c<'0'||c>'9'){if(c=='-')w=-1;c=getchar();}
while(c>='0'&&c<='9'){date=date*10+c-'0';c=getchar();}
return date*w;
}
void spfa(int x){
int u,s=pos[x];
queue<int> que;
for(int i=1;i<=n;i++)path[i]=0;
path[s]=1;
que.push(s);
while(!que.empty()){
u=que.front();
que.pop();
for(int i=1;i<=m;i++){
if(u+B[i]<=n&&!path[u+B[i]]){
path[u+B[i]]=path[u]+1;
que.push(u+B[i]);
}
if(u-B[i]>=1&&!path[u-B[i]]){
path[u-B[i]]=path[u]+1;
que.push(u-B[i]);
}
}
}
for(int i=1;i<=q;i++)dis[x][i]=path[pos[i]]-1;
}
void work(){
int S=(1<<q)-1;
dp[0]=0;
for(int i=0;i<S;i++)
for(int j=1;j<=q;j++){
if((1<<(j-1))&i)continue;
for(int k=j+1;k<=q;k++){
if(((1<<(k-1))&i)||dis[j][k]==-1)continue;
int x=(i|(1<<(j-1))|(1<<(k-1)));
dp[x]=min(dp[x],dp[i]+dis[j][k]);
}
break;
}
printf("%d\n",dp[S]);
}
void init(){
int x;
memset(dp,0x3f,sizeof(dp));
n=read()+1;q=read();m=read();
for(int i=1;i<=q;i++){
x=read();
A[x]^=1;A[x+1]^=1;
}
q=0;
for(int i=1;i<=n;i++)if(A[i])pos[++q]=i;
for(int i=1;i<=m;i++)B[i]=read();
for(int i=1;i<=q;i++)spfa(i);
}
int main(){
init();
work();
return 0;
}
洛谷P3943 星空的更多相关文章
- 洛谷 P3943 星空
题目背景 命运偷走如果只留下结果, 时间偷走初衷只留下了苦衷. 你来过,然后你走后,只留下星空. 题目描述 逃不掉的那一天还是来了,小 F 看着夜空发呆. 天上空荡荡的,没有一颗星星——大概是因为天上 ...
- [洛谷P3943]:星空(DP+最短路)
题目传送门 题目背景 命运偷走如果只留下结果, 时间偷走初衷只留下了苦衷.你来过,然后你走后,只留下星空. 题目描述 逃不掉的那一天还是来了,小$F$看着夜空发呆.天上空荡荡的,没有一颗星星——大概是 ...
- 洛谷P3943星空
啦啦啦啦——又是五月天的歌,题目传送门 这道题比之前两道真的不是同一级别的,这里我这个蒟蒻也讲不清,不如看下这位大佬的吧,他的写的已经非常清楚了:Z-Y-Y-S,这里我就只放下我的代码,也是按照这位大 ...
- 洛谷P3943 星空——题解
一道很好的锻炼思维难度的题,如果您能在考场上直接想出来的话,提高组450分以上就没问题了吧.(别像作者一样看了好几篇题解才勉强会) 先提取出题目大意:给定一个长度n<=40000的01串,其中1 ...
- CodeForces 79D 【Password】,洛谷P3943 【星空】
其实我做的是洛谷的P3943,但是听说fstqwq窃题...... 题目描述: 小 C 拿来了一长串星型小灯泡,假装是星星,递给小 F,想让小 F 开心一点.不过,有 着强迫症的小 F 发现,这串一共 ...
- 洛谷P3941入阵曲
题目传送门 这道题也是今年湖南集训队Day8的第一题,昨天洛谷的公开赛上又考了一遍,来发个记录(其实是因为五月天,另外两道题分别是将军令和星空,出这次题目的人肯定同为五迷(✪㉨✪)) 话不多说.先理解 ...
- 洛谷1640 bzoj1854游戏 匈牙利就是又短又快
bzoj炸了,靠离线版题目做了两道(过过样例什么的还是轻松的)但是交不了,正巧洛谷有个"大牛分站",就转回洛谷做题了 水题先行,一道傻逼匈牙利 其实本来的思路是搜索然后发现写出来类 ...
- 洛谷P1352 codevs1380 没有上司的舞会——S.B.S.
没有上司的舞会 时间限制: 1 s 空间限制: 128000 KB 题目等级 : 钻石 Diamond 题目描述 Description Ural大学有N个职员,编号为1~N.他们有 ...
- 洛谷P1108 低价购买[DP | LIS方案数]
题目描述 “低价购买”这条建议是在奶牛股票市场取得成功的一半规则.要想被认为是伟大的投资者,你必须遵循以下的问题建议:“低价购买:再低价购买”.每次你购买一支股票,你必须用低于你上次购买它的价格购买它 ...
随机推荐
- 【共享单车】—— React后台管理系统开发手记:AntD Form基础组件
前言:以下内容基于React全家桶+AntD实战课程的学习实践过程记录.最终成果github地址:https://github.com/66Web/react-antd-manager,欢迎star. ...
- 自己定义控件-LinearListView
一.描写叙述 用LinearLayout 实现的一个ListView ,重写了ListView中的经常使用函数,所以使用起来和ListView 没有区别. 比方:setAdapter.addHeade ...
- 自定义ViewPager的兼容性问题及解决办法
通过它我们可以给图片增加组合动画效果,也可以写成一个图片查看器. 比如我们首次安装应用的时候,很多就会用到ViewPager给我们做一个应用简介.今天要写的也是这个--怎么用ViewPager实现动画 ...
- TestNG+ReportNG+IDEA+Git+Jenkins+surefire持续集成数据驱动dubbo接口测试
一.pom.xml增加testng相关配置 <!--添加插件 关联testNg.xml--><plugin> <groupId>org.apache.maven.p ...
- Location配置与ReWrite语法
1 Location语法规则 1.1 Location规则 语法规则: location [=|~|~*|^~] /uri/ {… } 首先匹配 =,其次匹配^~,其次是按文件中顺序的正则匹配,最后是 ...
- Unity3D的脚本-script入门
来自:http://blog.163.com/shininglore@126/blog/static/961841802013412101454833/ Unity3D的基本操作很容易就能掌握了,接下 ...
- 【转】Intellij IDEA常用配置详解
1. IDEA内存优化 先看看你机器本身的配置而配置. \IntelliJ IDEA 8\bin\idea.exe.vmoptions -------------------------------- ...
- R语言初识
# 创建数据集&基本数据管理1.向量 创建函数 c() a <- c(1,2,3,4) a[c(i,j)] :[]给定元素所处位置的数值,即向量a中第i和第j个元素,a[2]第二个元素即 ...
- vim 标签页管理
一.打开关闭标签页 1. :tabnew 新建标签页 2. :tabc 关闭当前标签页 3. :tabo 关闭其他标签页保留当前标签页 4. :tabe file 在新标签页中打开 ...
- thread.join() 方法存在的必要性是什么?
好久远的问题,为什么关注这个问题的人这么少? 或许是用到这个功能的情形比较少吧. 1.等待处理结果 为什么要用join()方法在很多情况下,主线程生成并起动了子线程,如果子线程里要进行大量的耗时的运算 ...