【CF1428D】Bouncing Boomerangs 题解
题意简介
毒瘤大模拟
给你一张n*n的图,在图上摆有一些物体。从每一列的底端往上扔回旋镖,每镖中一个东西,回旋镖就会向右转九十度。现在我们知道从每列i底端往上镖时撞上的物体个数ai,试构造出一个合法的图。若无法构造出这样的图,输出-1。
需要注意的是,题目有几个重要的约束条件:
- \(a_i\leq3\)
- 每一行和每一列上的物品的个数不能超过2
思路分析
以下纯属个人看法。
要构造出一种符合题意的图,我们希望在构造过程中消耗的行和列尽可能少。
换句话说,我们希望能尽可能实现物体的“共用”。因为共用的话就不需要再开一行、一列了。
接下来考虑什么样的情况可以“共用”。我们令 \(i<j\) :
- \(a_j=1\) 时,若 \(a_i>1\) 则j所需要的物品可以与i共用。
- \(a_j=2\) 时,若 \(a_i>2\) 则j所需要的第一个物品可以与i共用。需要注意 \(a_i=2\) 的情况是不行的,因为每行最多只能有两个物体。
- \(a_j=3\) 时,若 \(a_i=3\) 则j所需要的第一个物品可以与i共用。需要注意出现在后面的3是不可能与前面的2共用的,因为若共用,只可能共用2的第二个物体,而这个物体与2的第一个物体在同一行上,若共用的话会撞上它,有 \(a_j>3\) ,不满足题意。
接下来考虑各种情况对空余行和列的数量的影响:
(下面的分析主要取决于影响的统计顺序,不同的统计方式可能会有不同的结论,此处只列出我的做法)
- \(a_j=0\) 时,空余列减1,因为这一列上不能放任何东西
- \(a_j=1\) 时,若与前面的某个i“共用”一个物体,由于已经在i处统计过了,所以不用重复统计影响;若没有共用,则空余行、空余列都要减1。因为我们是往右扫的,\(a_j=1\) 意味着这一行后面没有撞上东西。至于列,由于前面没有共用,且回旋镖只会往右转且最多打到3个物体,所以后面不可能再有某个k与这个j共用了。
- \(a_j=2\) 时,若与前面的某个i共用,\(a_i\) 一定等于3,由于已经在i处统计过了,所以只需空余列减1;若没有共用,则空余行减1、空余列减2。
- \(a_j=3\) 时,若与前面的某个i共用,\(a_i\) 一定等于3,由于已经在i处统计过了,所以只需空余行减1且空余列减1;若没有共用,则空余行减2、空余列减2
其实只要画个图,上面这些结论都可以很快得出。
一轮扫描,先确定每一个回旋镖的共用(拼接)顺序。假若扫描后出现空余列或空余行比0小的情况,就直接输出-1即可。
接下来需要考虑如何根据拼接顺序还原图。
我们知道,每个 \(a_i=2\) 后面必然且只能接一个 \(a_j=1\) ,接2或3的话会使 \(a_i=3\) 。
对于 \(a_i=3\) 的情形,我们需要的行数是最多的,所以我们不妨把这样的物体从最顶行设起。
对于 \(a_i=2 or a_i=1\) 的情形,我们不需要在行的下面再放物品了,不妨从最底行设起,把上面广阔的空间留给需要预留出一行的3。
注意观察图片 最顶行是1 最底行是n
下面给出几组我错过的数据可供测试(如果懒得去cf翻的话) 答案可以自己手推一下合不合理。
数据1:
4
3 3 1 1
数据2:
4
3 3 3 1
其实我主要是错在对行列处理的分类讨论上 还有对3的行分配上。这是需要注意的。
接下来就是实现,具体的一些小细节在代码中有体现,此处不再做赘述。
代码库
#include <cstdio>
const int N=1e5+5;
int n,a[N],pre[N],st2[N],st3[N],s2,s3,line,col,nex[N];
int ans[N<<1][2],cnt; bool vis[N];
inline bool is1(int i){
if(nex[i]) return a[nex[i]]!=3;
return a[i]==1;
}
int main(){
scanf("%d",&n); line=col=n;
for(int i=1;i<=n;i++) scanf("%d",a+i);
for(int i=1;i<=n;i++){
if(a[i]==0){
col--; continue;
}
if(a[i]==1){
if(s2) pre[i]=st2[s2--],nex[pre[i]]=i;
else if(s3) pre[i]=st3[s3--],nex[pre[i]]=i;
else line--,col--;
continue;
}
if(a[i]==2){
if(s3) pre[i]=st3[s3--],nex[pre[i]]=i,col--,st2[++s2]=i;
else line--,col-=2,st2[++s2]=i;
continue;
}
if(a[i]==3){
if(s3) pre[i]=st3[s3--],nex[pre[i]]=i,line--,col--,st3[++s3]=i;
else line-=2,col-=2,st3[++s3]=i;
continue;
}
}
if(line<0||col<0){
printf("-1\n"); return 0;
}
int top=0,low=n+1;
for(int i=1;i<=n;i++){
if(vis[i]||!a[i]) continue;
for(int j=i;j;j=nex[j]){
//printf("Calc %d:%d %d\n",j,top,low);
vis[j]=1;
if(a[j]==1){
if(!pre[j]) ans[++cnt][0]=--low,ans[cnt][1]=j;
//printf("Calc %d:%d %d\n",j,top,low);
continue;
}
if(a[j]==2){
if(!pre[j]){
if(is1(j)) ans[++cnt][0]=--low,ans[cnt][1]=j;
else ans[++cnt][0]=++top,ans[cnt][1]=j;
}
cnt++;
ans[cnt][0]=ans[cnt-1][0]; ans[cnt][1]=nex[j];
//printf("Calc %d:%d %d\n",j,top,low);
continue;
}
if(a[j]==3){
if(!pre[j]){
if(is1(j)) ans[++cnt][0]=low-2,ans[cnt][1]=j;
else ans[++cnt][0]=++top,ans[cnt][1]=j;
}
cnt++;
ans[cnt][0]=ans[cnt-1][0]; ans[cnt][1]=nex[j];
cnt++;
if(!pre[j]&&is1(j)) ans[cnt][0]=--low,ans[cnt][1]=nex[j],low--;
else if(is1(j)) ans[cnt][0]=--low,ans[cnt][1]=nex[j];
else ans[cnt][0]=++top,ans[cnt][1]=nex[j];
//printf("Calc %d:%d %d\n",j,top,low);
continue;
}
}
}
//printf("End with %d %d\n",top,low);
printf("%d\n",cnt);
for(int i=1;i<=cnt;i++) printf("%d %d\n",ans[i][0],ans[i][1]);
//for(int i=1;i<=n;i++) if(a[i]!=1&&a[i]!=0&&nex[i]==0) printf("ERR %d!\n",i);
return 0;
}
END
【CF1428D】Bouncing Boomerangs 题解的更多相关文章
- 算法与数据结构基础 - 哈希表(Hash Table)
Hash Table基础 哈希表(Hash Table)是常用的数据结构,其运用哈希函数(hash function)实现映射,内部使用开放定址.拉链法等方式解决哈希冲突,使得读写时间复杂度平均为O( ...
- LeetCode All in One题解汇总(持续更新中...)
突然很想刷刷题,LeetCode是一个不错的选择,忽略了输入输出,更好的突出了算法,省去了不少时间. dalao们发现了任何错误,或是代码无法通过,或是有更好的解法,或是有任何疑问和建议的话,可以在对 ...
- Codeforces Round#687 Div2 题解
打这场的时候迷迷糊糊的,然后掉分了( A Prison Break: 题面很复杂,但是题意很简单,仅需求出从这个点到四个角的最大的曼哈顿距离即可 #include <bits/stdc++.h& ...
- 2016 华南师大ACM校赛 SCNUCPC 非官方题解
我要举报本次校赛出题人的消极出题!!! 官方题解请戳:http://3.scnuacm2015.sinaapp.com/?p=89(其实就是一堆代码没有题解) A. 树链剖分数据结构板题 题目大意:我 ...
- noip2016十连测题解
以下代码为了阅读方便,省去以下头文件: #include <iostream> #include <stdio.h> #include <math.h> #incl ...
- [LeetCode] Number of Boomerangs 回旋镖的数量
Given n points in the plane that are all pairwise distinct, a "boomerang" is a tuple of po ...
- BZOJ-2561-最小生成树 题解(最小割)
2561: 最小生成树(题解) Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 1628 Solved: 786 传送门:http://www.lyd ...
- Codeforces Round #353 (Div. 2) ABCDE 题解 python
Problems # Name A Infinite Sequence standard input/output 1 s, 256 MB x3509 B Restoring P ...
- 哈尔滨理工大学ACM全国邀请赛(网络同步赛)题解
题目链接 提交连接:http://acm-software.hrbust.edu.cn/problemset.php?page=5 1470-1482 只做出来四道比较水的题目,还需要加强中等题的训练 ...
随机推荐
- python根据密钥得到谷歌两步验证动态验证码
# coding=utf-8 import time import base64 import sys import hashlib import hmac import struct def goo ...
- 北京数途科技有限公司--EAM MAXIMO介绍
一.EAM概述: EAM( Enterprise Asset Management)即企业资产维护管理系统 ,它是面向资产密集型(Asset-intensive)企业的企业信息化 解决方案的总称,其前 ...
- java 单例模式的几种写法
一.懒汉式 public class Singleton{ private static Singleton instance = null; private Singleton(){} public ...
- MFC DLL中如何响应PreTranslateMessage消息
最近项目中使用到MFC,由于工程是DLL的,在使用ToolTip时碰到非模态对话框无法显示的问题.查了一番资料,发现原因是由于:虽然MFC Regular DLL派生了CWinApp类,并有一个the ...
- spring cloud微服务快速教程之(十四)spring cloud feign使用okhttp3--以及feign调用参数丢失的说明
0-前言 spring cloud feign 默认使用httpclient,需要okhttp3的可以进行切换 当然,其实两者性能目前差别不大,差别较大的是很早之前的版本,所以,喜欢哪个自己选择: 1 ...
- hadoop分布式格式化时出现异常java.net.unknownhostexception
当搭建好分布式集群后,准备使用命令格式化时 hdfs namenode format 在日志的最后一行出现 java.net.unknownhostexception的异常,通常是你的主机名没有配置好 ...
- spring:bean的生命周期
1.spring中bean的生命周期 (1)概念 在spring框架中,所有的bean对象都有生命周期,就是指bean的创建.初始化.服务.销毁的一个过程. (2)bean的生命周期 bean的定义 ...
- 【CF1425B】 Blue and Red of Our Faculty! 题解
原题链接 简要翻译: 有一个连通图,A和B同时从点1出发,沿不同的路径前进.原本,图上的每一条边都是灰色的.A将经过的边涂成红色,B将经过的边涂成蓝色的.每个回合每个人只能走灰色的边.当某个回合中不存 ...
- 树(二叉树 & 二叉搜索树 & 哈夫曼树 & 字典树)
树:n(n>=0)个节点的有限集.有且只有一个root,子树的个数没有限制但互不相交.结点拥有的子树个数就是该结点的度(Degree).度为0的是叶结点,除根结点和叶结点,其他的是内部结点.结点 ...
- SFUD+FAL+EasyFlash典型场景需求分析,并记一次实操记录
SFUD+FAL+EasyFlash典型场景需求分析:用整个flash存储数据,上千条数据,读取得时候用easyflash很慢,估计要检索整个flash太慢了. 改进方法:分区检索. 1存数据时,根据 ...