CF1335F Robots on a Grid
比较简单的倍增
但还是看了题解才会
题意
给出一个 \(n\times m\) 的网格,每个格子有颜色,\(0\) 黑 \(1\) 白,每个格子还有一个方向,表示这个格子上的机器人会向那个方向走,并保证不会走出格子
摆放机器人,它们同时开始运动,在任意时刻不能有两个机器人在同一个格子里
先最大化机器人个数,如果多种方案机器人个数相等,再最大化摆在黑格子里的机器人数量
首先,这个路线肯定是循环的,如果不循环,就会走到无限多个格子,不合理
然后,对于任意一个格子,从那里开始走到完成一个完整的循环,步数肯定小于等于 \(nm\),显然它不能走了比 \(nm\) 还多的格子仍然不开始循环
而对于两个不同的循环,它们一定没有交点(就是一个相同的格子),如果有,肯定就不会行成两个循环了
如果有两个机器人分别走了至少 \(nm\) 步,那么它们肯定已经各子循环了一次或以上了
所以,如果它们不在同一个循环,显然不会相遇
如果在一个循环,且没有在 \(nm\) 步内相遇,说明这时它们已经“同步”了,就是会一直保持这一个距离不断的走,永不相遇
当然,如果在 \(nm\) 步之前就已经相遇,那么它们会一直一起走,不会有影响
至此,我们判断两个格子上的机器人是否会相遇的方法,就是看它们走了 \(nm\) 步以后,是不是在同一个格子
所以,假设我们在所有格子都摆上机器人,让他们走,一旦有几个相遇了,就说明我们要去掉这些机器人只剩下其中的一个
这时,给这 \(nm\) 个点编号 \(1\ldots nm\),并用\(white_i,black_i\)表示的分别是有没有从白/黑格出发的机器人,\(nm\) 格后会走到这里
然后从 \(1\) 到 \(nm\) 统计答案,如果某个格子可以由黑色格子中的机器人走来,就保留黑色格子的那个,否则任意保留
就可以计算出答案了
对于走 \(nm\) 步那个操作,就要用倍增实现
#include<cstdio>
#include<algorithm>
#include<iostream>
#include<cmath>
#include<map>
#include<iomanip>
#include<cstring>
#define reg register
#define EN std::puts("")
#define LL long long
inline int read(){
register int x=0;register int y=1;
register char c=std::getchar();
while(c<'0'||c>'9'){if(c=='-') y=0;c=std::getchar();}
while(c>='0'&&c<='9'){x=x*10+(c^48);c=std::getchar();}
return y?x:-x;
}
int black[1000006],white[1000005];
int nex[24][1000006];
int color[1000006];
char s[1000006];
int main(){int T=read();while(T--){
int n=read(),m=read();
for(reg int i=1;i<=n;i++){
std::scanf("%s",s+1);
for(reg int j=1;j<=m;j++) color[(i-1)*m+j]=s[j]=='0'?0:1;
}
for(reg int i=1;i<=n;i++){
std::scanf("%s",s+1);
for(reg int j=1;j<=m;j++){
int now=(i-1)*m+j;
if(s[j]=='U') nex[0][now]=now-m;
else if(s[j]=='D') nex[0][now]=now+m;
else if(s[j]=='L') nex[0][now]=now-1;
else nex[0][now]=now+1;
}
}
n*=m;
for(reg int i=1;i<=20;i++)
for(reg int j=1;j<=n;j++) nex[i][j]=nex[i-1][nex[i-1][j]];
for(reg int j=1;j<=n;j++){
int to=j;
for(reg int i=20;~i;i--){//倒着循环,倍增传统套路,不过想想也能知道,要先走大的步数
if((1<<i)&n) to=nex[i][to];
}
color[j]?white[to]=1:black[to]=1;
}
int ans=0,black_num=0;
for(reg int i=1;i<=n;i++)
if(black[i]) black_num++,ans++,black[i]=white[i]=0;
else if(white[i]) ans++,black[i]=white[i]=0;
std::printf("%d %d\n",ans,black_num);
}
return 0;
}
原文地址:https://www.cnblogs.com/suxxsfe/p/12699630.html
- Numpy教程第1部分 - 阵列简介(常用基础操作总结)
- Session和Cookies的基本原理
- 浅析Numpy.genfromtxt及File I/O讲解
- 损失函数详解
- 排查Java的内存问题
- 使用两种方法让 ASP.NET Core 实现遵循 HATEOAS 结构的 RESTful API
- 设计模式六大原则(5):迪米特法则
- Selenium2+python自动化61-Chrome浏览器(chromedriver)
- 区块链可以减少社会不平等吗?
- 【干货】不止准确率:为分类任务选择正确的机器学习度量指标(附代码实现)
- python爬虫beautifulsoup4系列1
- 区块链入门教程
- python爬虫beautifulsoup4系列2
- python爬虫beautifulsoup4系列3
- JavaScript 教程
- JavaScript 编辑工具
- JavaScript 与HTML
- JavaScript 与Java
- JavaScript 数据结构
- JavaScript 基本数据类型
- JavaScript 特殊数据类型
- JavaScript 运算符
- JavaScript typeof 运算符
- JavaScript 表达式
- JavaScript 类型转换
- JavaScript 基本语法
- JavaScript 注释
- Javascript 基本处理流程
- Javascript 选择结构
- Javascript if 语句
- Javascript if 语句的嵌套
- Javascript switch 语句
- Javascript 循环结构
- Javascript 循环结构实例
- Javascript 跳转语句
- Javascript 控制语句总结
- Javascript 函数介绍
- Javascript 函数的定义
- Javascript 函数调用
- Javascript 几种特殊的函数
- JavaScript 内置函数简介
- Javascript eval() 函数
- Javascript isFinite() 函数
- Javascript isNaN() 函数
- parseInt() 与 parseFloat()
- escape() 与 unescape()
- Javascript 字符串介绍
- Javascript length属性
- javascript 字符串函数
- Javascript 日期对象简介
- Javascript 日期对象用途
- Date 对象属性和方法
- Javascript 数组是什么
- Javascript 创建数组
- Javascript 数组赋值与取值
- Javascript 数组属性和方法
- Python保留字总结
- Python进阶 | 五分钟带你弄懂迭代器与生成器,夯实代码能力
- [Go]GO语言实战项目-gin框架上传图片文件
- [Go] Golang练习项目-GO语言实现选择排序
- 设计模式~策略模式
- Java单元测试框架(一)——JUnit4
- Java单元测试框架(二)——JUnit5
- 绘制双坐标轴图
- 用箭头和文字来标记重要的点
- 32.Python字符串方法split
- 程序员过关斩将--解决分布式session问题
- 常见的C编程段错误及对策
- Python 内置函数之——zip()
- js 将数据保存到本地
- MySQL遇见SELECT list is not in GROUP BY clause and contains nonaggre的问题