P7115-[NOIP2020]移球游戏【构造】
正题
题目链接:https://www.luogu.com.cn/problem/P7115
题目大意
\(n+1\)个柱子,前面\(n\)个上面各有\(m\)个球,球有\(n\)种颜色,每种\(m\)个。
你每次可以把一个柱子最上面的球放到另一个上面,要求在\(820000\)次内使得同种颜色的球都在同一个柱子上。
输出方案
\(2\leq n\leq 50,2\leq m\leq 400\)
解题思路
这题好难啊,用的是洛谷题解上的做法。
首先我们枚举一种颜色\(x\),将这种颜色标记为\(1\)其他都为\(0\)。
然后开始的状态是这样的

然后考虑先构造一个全部都是\(0\)的竖列
我们先记录第一柱的\(1\)的个数\(tmp\),然后把第\(n-1\)柱子的\(tmp\)个丢进第\(n+1\)柱,然后把第一柱分离到后面两个柱子(\(1\)的放到\(n\),\(0\)的放到\(n+1\))

然后把原来的\(0\)放到第一柱,然后分离第二柱,如果是\(0\)放到第一柱否则放到第\(n+1\)柱(如果第一柱已经满了就放进\(n+1\)柱)

然后交换一下柱子序号(用个数组存一下就好了)就变成了

然后再考虑构造全\(1\)柱
我们把同理把第\(1\)柱分裂到第\(n\)和第\(n+1\)柱就变成了

此时第\(n+1\)柱子上面全部是\(1\)而第\(n\)柱上面都是\(0\),然后此时我们再把剩下\(n\)个柱子依次分离就能把所有的\(1\)提到最上面,然后把所有的\(1\)集合就好了。
最后弄出\(n-1\)个全\(0\)柱和一个全\(1\)柱我们就可以把全一柱去掉然后缩小\(n\)的值。
一直重复到\(n=2\)时我们发现我们的方法不再适用,需要特别处理。
我们按照前面的方法把第一柱分离到\(2\)和\(3\)

然后把\(0\)和\(1\)丢到第一个柱子,然后再把\(1\)丢进第\(3\)个柱子

然后分离第二个柱子就好了
然后这样就能过了
code
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<vector>
using namespace std;
const int N=410;
int n,m,a[N][N],cnt[N],p[N];
vector<int> aL,aR;
void mov(int x,int y){
aL.push_back(x);
aR.push_back(y);
a[y][++cnt[y]]=a[x][cnt[x]--];
return;
}
int count(int x,int y){
int ans=0;
for(int i=1;i<=m;i++)
ans+=(a[x][i]==y);
return ans;
}
int main()
{
scanf("%d%d",&n,&m);
for(int i=1;i<=n;i++){
for(int j=1;j<=m;j++)
scanf("%d",&a[i][j]);
cnt[i]=m;p[i]=i;
}
p[n+1]=n+1;
for(int k=n;k>=3;k--){
int tmp=count(p[1],k);
for(int i=1;i<=tmp;i++)mov(p[k],p[k+1]);
for(int i=1;i<=m;i++)
if(a[p[1]][cnt[p[1]]]==k)mov(p[1],p[k]);
else mov(p[1],p[k+1]);
for(int i=1;i<=m-tmp;i++)mov(p[k+1],p[1]);
for(int i=1;i<=m;i++)
if(a[p[2]][cnt[p[2]]]==k)mov(p[2],p[k+1]);
else if(cnt[p[1]]<m)mov(p[2],p[1]);
else mov(p[2],p[k+1]);
swap(p[1],p[k]);swap(p[2],p[k+1]);
for(int i=1;i<k;i++){
int tmp=count(p[i],k);
for(int j=1;j<=tmp;j++)mov(p[k],p[k+1]);
for(int j=1;j<=m;j++)
if(a[p[i]][cnt[p[i]]]==k)mov(p[i],p[k]);
else mov(p[i],p[k+1]);
swap(p[i],p[k+1]);swap(p[k],p[i]);
}
for(int i=1;i<k;i++){
while(a[p[i]][cnt[p[i]]]==k)mov(p[i],p[k+1]);
while(cnt[p[i]]<m)mov(p[k],p[i]);
}
}
int tmp=count(p[1],1);
for(int i=1;i<=tmp;i++)mov(p[2],p[3]);
for(int i=1;i<=m;i++)
if(a[1][cnt[p[1]]]==1)mov(p[1],p[2]);
else mov(p[1],p[3]);
for(int i=1;i<=m-tmp;i++)mov(p[3],p[1]);
for(int i=1;i<=tmp;i++)mov(p[2],p[1]);
while(cnt[p[3]])mov(p[3],p[2]);
for(int i=1;i<=tmp;i++)mov(p[1],p[3]);
for(int i=1;i<=m;i++)
if(a[2][cnt[p[2]]]==1)mov(p[2],p[3]);
else mov(p[2],p[1]);
printf("%d\n",aL.size());
for(int i=0;i<aL.size();i++)
printf("%d %d\n",aL[i],aR[i]);
return 0;
}
P7115-[NOIP2020]移球游戏【构造】的更多相关文章
- NOIP2020 移球游戏
Description 给定 \(n+1\) 个栈,前 \(n\) 个栈内有不定的 \(m\) 个元素,最后一个栈为空,每个栈的最大容量为 \(m\) 每种颜色都有 \(m\) 种,求任意一种方法,使 ...
- nyoj_518_取球游戏_201404161738
取球游戏 时间限制:1000 ms | 内存限制:65535 KB 难度:2 描述 今盒子里有n个小球,A.B两人轮流从盒中取球,每个人都可以看到另一个人取了多少个,也可以看到盒中还剩下多少个 ...
- nyist 518 取球游戏
http://acm.nyist.net/JudgeOnline/problem.php?pid=518 取球游戏 时间限制:1000 ms | 内存限制:65535 KB 难度:2 描述 今 ...
- 躲避球游戏ios源码
躲避球游戏源码,有限源码是一个基于cocos2d的躲避球游戏源码的,并且还引用了大家熟悉google广告的,进行推广,已经还有带game center等,游戏操作很简单,用手指按住物体,然后移动物体避 ...
- 取球游戏|2012年蓝桥杯B组题解析第十题-fishers
(25')取球游戏 今盒子里有n个小球,A.B两人轮流从盒中取球,每个人都可以看到另一个人取了多少个,也可以看到盒中还剩下多少个,并且两人都很聪明,不会做出错误的判断. 我们约定: 每个人从盒子中取出 ...
- 取球游戏_nyoj_518(博弈-蓝桥杯原题).java
取球游戏 时间限制: 1000 ms | 内存限制: 65535 KB 难度: 2 描述 今盒子里有n个小球,A.B两人轮流从盒中取球,每个人都可以看到另一个人取了多少个,也可以看到盒中还剩下 ...
- 放球游戏B
题目描述 校园里在上活动课,Red和Blue两位小朋友在玩一种游戏,他俩在一排N个格子里,自左到右地轮流放小球,每个格子只能放一个小球.第一个人只能放1个球,之后的人最多可以放前一个人的两倍数目的球, ...
- 【题解】放球游戏B
题目描述 校园里在上活动课,Red和Blue两位小朋友在玩一种游戏,他俩在一排N个格子里,自左到右地轮流放小球,每个格子只能放一个小球.第一个人只能放1个球,之后的人最多可以放前一个人的两倍数目的球, ...
- 【题解】放球游戏A
题目描述 校园里在上活动课,Red和Blue两位小朋友在玩一种游戏,他俩在一排N个格子里,自左到右地轮流放小球,每个格子只能放一个小球.每个人一次只能放1至5个球,最后面对没有空格而不能放球的人为输. ...
随机推荐
- jenkins+docker部署java项目
jenkins + maven + jdk + docker + docker register + dockerfile jenkins插件 # 安装插件 SSH # 配置 系统设置-> SS ...
- 分布式文件系统FastDFS搭建实操
转载---------佳先森--- 一.什么是文件系统 分布式文件系统(Distributed File System)是指文件系统管理的物理存储资源不一定直接连接在本地节点上,而是通过计算机网络与节 ...
- vue2.0中模拟数据的配置
在开发过程中,有时候接口跟不上我们的进度,我们要测试,就需要自测. 现在vue已经升级到2.0版本了,早期在vue构建工程文件在build里面有dev-server.js,但是后来构建去除了该文件集成 ...
- web.xml中自定义Listener
Listener可以监听容器中某一执行动作,并根据其要求做出相应的响应. 常用的Web事件的监听接口如下: ServletContextListener:用于监听Web的启动及关闭 ServletCo ...
- 常用cron表达式
0 0 10,14,16 * * ? 每天上午10点,下午2点,4点 0 0/30 9-17 * * ? 朝九晚五工作时间内每半小时 0 0 12 ? * WED 表示每个星期三中午12点 " ...
- 三大操作系统对比使用之·Ubuntu16.04
时间:2018-11-13 整理:byzqy 本篇是一篇个人对 Ubuntu 16.04(桌面版)使用方法.技巧以及应用推荐的文章,以便查询和分享! 打开终端: Ctrl+Alt+T,即可打开&quo ...
- Elasticsearch集群搭建教程及生产环境配置
Elasticsearch 是一个极其强大的搜索和分析引擎,其强大的部分在于能够对其进行扩展以获得更好的性能和稳定性. 本教程将提供有关如何设置 Elasticsearch 集群的一些信息,并将添加一 ...
- 用C++实现的Eratosthenes筛法程序
运行示例 只输出素数总数的运行示例 PS H:\Read\num\x64\Release> .\esieve.exe Eratosthenes sieve: a method to find o ...
- Python3-sqlalchemy-orm 多对多关系建表、插入数据、查询数据
现在来设计一个能描述"图书"与"作者"的关系的表结构,需求是 一本书可以有好几个作者一起出版 一个作者可以写好几本书 此时你会发现,用之前学的外键好像没办法实现 ...
- MeteoInfo-Java解析与绘图教程(四)
MeteoInfo-Java解析与绘图教程(四) 上文我们说到,将地图叠加在色斑图上,但大部分都是卫星绘图,现在开始讲解micaps数据绘图,同样也是更多自定义配置 首先我们解析micaps数据,将之 ...