\(\mathcal{Description}\)

  Link.

  自己去读题面叭~

\(\mathcal{Solution}\)

  首先,参悟【样例解释 #2】。一种暴力的思路即为钦定集合 \(S\) 内的位置都合法,容斥计数。其中对于每条纸带的每个位置,有三种情况(令 _ 为“保持不变”,注意没有被机器人经过的位置都有这种修改):

  • 同时存在 _*;或者同时存在 01:只能为空,方案数为 \(1\);
  • 否则,存在(_*)且存在(01):只能为空或 01 中的一个,方案数为 \(2\);
  • 否则,方案数为 \(3\)。

  暴力做的复杂度是 \(\mathcal O(2^nn^am^b)\),\(a,b\) 看细节实现,不太可观。

  尝试钦定最后一个在 \(S\) 中的位置为 \(r\),那么所有修改长度大于 \(n-r+1\) 的机器人所在的纸带都必须全空。在此限制下带着容斥系数状压 DP,令 \(f(i,S)\) 表示考虑了纸带的位置 \(i\),前若干个位置的选择情况为 \(S\),注意此处 \(|S|\le2^{\min\{i,n-r\}}=\mathcal O(2^{n/2})\),转移时可以 \(\mathcal O(nm)\) 暴力计算当前位置的贡献,得到了 \(\mathcal O(2^{n/2}nm)\) 的算法。

  顺带,注意到统计当前位置贡献时可以描述为形如“有多少个机器人的修改序列中,下标属于 \(S\) 的位置上存在 _/*/0/1”,可以用 std::bitset 优化。最终复杂度为 \(\mathcal O(\frac{2^{n/2}nm}{\omega})\)。

\(\mathcal{Code}\)

/*~Rainybunny~*/

#include <bitset>
#include <cstdio>
#include <cassert>
#include <cstring> #define rep( i, l, r ) for ( int i = l, rep##i = r; i <= rep##i; ++i )
#define per( i, r, l ) for ( int i = r, per##i = l; i >= per##i; --i ) typedef unsigned long long ULL; const int MAXN = 32, MAXM = 1e3, MOD = 1e9 + 7;
int n, m, pw2[MAXM + 5], pw3[MAXM + 5]; inline int imin( const int a, const int b ) { return a < b ? a : b; }
inline void subeq( int& a, const int b ) { ( a -= b ) < 0 && ( a += MOD ); }
inline int mul( const int a, const int b ) { return int( 1ll * a * b % MOD ); }
inline int add( int a, const int b ) { return ( a += b ) < MOD ? a : a - MOD; }
inline void addeq( int& a, const int b ) { ( a += b ) >= MOD && ( a -= MOD ); } struct Robot {
int len, val[MAXN + 5];
inline int operator [] ( const int k ) const {
return k <= len ? val[k] : 0;
}
inline void read() {
static char str[105]; scanf( "%s", str );
val[len = 1] = 0;
for ( int i = 0; str[i]; ++i ) {
if ( str[i] == 'R' ) val[++len] = 0;
else if ( str[i] == '0' ) val[len] = 1;
else if ( str[i] == '1' ) val[len] = 2;
else val[len] = 3 - val[len];
}
}
} rbt[MAXM + 5]; inline int solve( const int r ) {
static int f[2][1 << 17][2];
static std::bitset<MAXM + 5> sig[MAXN + 5][4], g[1 << 17][4];
int allc = 0; rep ( i, 0, n - 1 ) rep ( j, 0, 3 ) sig[i][j].reset();
rep ( i, 1, m ) if ( rbt[i].len <= n - r + 1 ) {
++allc;
rep ( j, 0, n - 1 )
sig[j][rbt[i][j + 1]].set( i );
}
rep ( i, 0, imin( n - r, r ) ) rep ( j, 0, 3 ) {
g[1 << i][j] = sig[i][j];
} rep ( S, 1, ( 1 << imin( n - r, r ) << 1 ) - 1 ) if ( S & ( S - 1 ) ) {
rep ( i, 0, 3 ) {
g[S][i] = g[S ^ ( S & -S )][i] | g[S & -S][i];
}
} auto calc = [&]( const bool sty, const ULL S )->int {
static std::bitset<MAXM + 5> x, y;
x = ( sty ? g[S][3] : g[S][0] & g[S][3] ) | ( g[S][1] & g[S][2] );
y = ~x & ( sty ? g[S][1] | g[S][2] :
( g[S][0] | g[S][3] ) & ( g[S][1] | g[S][2] ) );
int t = y.count();
return mul( pw2[t], pw3[allc - t - x.count()] );
}; memset( f[0], 0, sizeof f[0] ), f[0][0][0] = 1;
for ( int i = 0, sta = 0; i < r; ++i, sta ^= 1 ) {
memset( f[!sta], 0, sizeof f[!sta] );
rep ( S, 0, ( 1 << imin( i, n - r ) ) - 1 ) {
ULL T = ULL( S ) << 1ull & ( ( 1ull << n >> r ) - 1 );
bool flg = ULL( S ) << 1ull << r >> n & 1ull; rep ( t, 0, 1 ) {
int cur = f[sta][S][t]; f[sta][S][t] = 0;
if ( !cur ) continue;
subeq( f[!sta][r == n ? 0 : T | 1][t || flg || r == n],
mul( cur, calc( t || i + 1 < r, S << 1 | 1 ) ) );
if ( i + 1 < r ) {
addeq( f[!sta][T][t || flg],
mul( cur, calc( 1, S << 1 ) ) );
}
}
}
} int ret = 0;
rep ( t, 0, 1 ) {
for ( int d = 1, sta = r & 1; d <= n - r; ++d, sta ^= 1 ) {
rep ( i, 0, imin( r, n - r ) - 1 ) {
rep ( j, t, 3 ) g[1 << i][j] = sig[i + d][j];
}
rep ( S, 0, ( 1 << imin( r, n - r ) ) - 1 ) {
if ( S & ( S - 1 ) ) {
rep ( j, t, 3 ) {
g[S][j] = g[S ^ ( S & -S )][j] | g[S & -S][j];
}
}
f[!sta][S][t] = mul( f[sta][S][t], calc( t, S ) );
f[sta][S][t] = 0;
}
}
rep ( S, 0, ( 1 << imin( r, n - r ) ) - 1 ) {
addeq( ret, f[n & 1][S][t] );
}
}
return ret;
} int main() {
// freopen( "robot.in", "r", stdin );
// freopen( "robot.out", "w", stdout ); scanf( "%d %d", &n, &m );
pw2[0] = pw3[0] = 1;
rep ( i, 1, m ) {
rbt[i].read();
pw2[i] = mul( 2, pw2[i - 1] ), pw3[i] = mul( 3, pw3[i - 1] );
} int ans = 0;
rep ( i, 1, n ) subeq( ans, solve( i ) );
printf( "%d\n", ans );
return 0;
}

Solution -「NOI 2021」「洛谷 P7740」机器人游戏的更多相关文章

  1. 「区间DP」「洛谷P1043」数字游戏

    「洛谷P1043」数字游戏 日后再写 代码 /*#!/bin/sh dir=$GEDIT_CURRENT_DOCUMENT_DIR name=$GEDIT_CURRENT_DOCUMENT_NAME ...

  2. 洛谷P1118 数字三角形游戏

    洛谷1118 数字三角形游戏 题目描述 有这么一个游戏: 写出一个1-N的排列a[i],然后每次将相邻两个数相加,构成新的序列,再对新序列进行这样的操作,显然每次构成的序列都比上一次的序列长度少1,直 ...

  3. 洛谷P1274-魔术数字游戏

    Problem 洛谷P1274-魔术数字游戏 Accept: 118    Submit: 243Time Limit: 1000 mSec    Memory Limit : 128MB Probl ...

  4. 洛谷 P4012 深海机器人问题【费用流】

    题目链接:https://www.luogu.org/problemnew/show/P4012 洛谷 P4012 深海机器人问题 输入输出样例 输入样例#1: 1 1 2 2 1 2 3 4 5 6 ...

  5. 洛谷P1288 取数游戏II(博弈)

    洛谷P1288 取数游戏II 先手必胜的条件需要满足如下中至少 \(1\) 条: 从初始位置向左走到第一个 \(0\) 的位置,经过边的数目为偶数(包含 \(0\) 这条边). 从初始位置向右走到第一 ...

  6. Solution -「JSOI 2019」「洛谷 P5334」节日庆典

    \(\mathscr{Description}\)   Link.   给定字符串 \(S\),求 \(S\) 的每个前缀的最小表示法起始下标(若有多个,取最小的).   \(|S|\le3\time ...

  7. Solution -「洛谷 P4372」Out of Sorts P

    \(\mathcal{Description}\)   OurOJ & 洛谷 P4372(几乎一致)   设计一个排序算法,设现在对 \(\{a_n\}\) 中 \([l,r]\) 内的元素排 ...

  8. Solution -「POI 2010」「洛谷 P3511」MOS-Bridges

    \(\mathcal{Description}\)   Link.(洛谷上这翻译真的一言难尽呐.   给定一个 \(n\) 个点 \(m\) 条边的无向图,一条边 \((u,v,a,b)\) 表示从 ...

  9. Solution -「APIO 2016」「洛谷 P3643」划艇

    \(\mathcal{Description}\)   Link & 双倍经验.   给定 \(n\) 个区间 \([a_i,b_i)\)(注意原题是闭区间,这里只为方便后文描述),求 \(\ ...

随机推荐

  1. js箭头函数 的 (e) => { } 写法笔记

    1. (e) => {} 是ES 6 新语法,默认是Es 5.1,因此在这里设置一下就不会提示红色下划线了 2.使用: (e) => {}  , 其实就是function (e){} 的缩 ...

  2. 第10组 Alpha冲刺 (6/6)

    1.1基本情况 ·队名:今晚不睡觉 ·组长博客:https://www.cnblogs.com/cpandbb/p/14008187.html ·作业博客:https://edu.cnblogs.co ...

  3. GIS :元宇宙未来发展的有力技术支撑

    摘要:元宇宙是描述未来互联网迭代发展的一个概念,是一个将现实世界和虚拟世界相互融合的一个可感知的持久.共享的3D虚拟空间组成的世界. 本文分享自华为云社区<[云驻共创]元宇宙漫游指南-新一代GI ...

  4. C# TCP传输文件示例代码

    using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.T ...

  5. 【体验】在Adobe After Effects CC 2018中使用脚本创建窗口

    1.主界面 2.脚本编辑器主界面 3.对象浏览器 在脚本编辑器中按F1 4.写一段 ScriptUI var win = new Window('window', 'my win', [100, 10 ...

  6. 【记录一个问题】opencv中 cv::dft()与cv::ocl_dft()计算的结果相差较大

    以一个跟踪算法来测试: 使用cv::dft(), 矩阵未按照2次幂对齐,最终跟踪平均准确率 84.3% 使用cv::dft(),矩阵使用cv::copyMakeBorder对齐,最终跟踪平均准确率 8 ...

  7. Windows蓝牙失效超全攻略

    新电脑蓝牙出现问题,我捣鼓了很久,历经九九八十一难得以修复,说一说我在网上看到的各种方法. 一个功能正常使用,需要经过一个又一个的步骤.任何一个地方出问题,都有可能造成蓝牙失效.以下方法按出现概率从大 ...

  8. FilterConfig接口(Servlet)

    Javax.Servet 包中提供了一个 FilterCofig 接口,它与 ServletConfig 接口相似,用于在过滤器初始化期间向其传递信息.FilterConfig 接口由容器实现,容器将 ...

  9. Redisson 实现分布式锁原理分析

    Redisson 实现分布式锁原理分析   写在前面 在了解分布式锁具体实现方案之前,我们应该先思考一下使用分布式锁必须要考虑的一些问题.​ 互斥性:在任意时刻,只能有一个进程持有锁. 防死锁:即使有 ...

  10. golang中字符串、bytes类型切片、16进制字符串之间的转换

    func main() { // 字符串转bytes类型 name := "马亚南" fmt.Println(name) // 马亚南 bName := []byte(name) ...