原文链接https://www.cnblogs.com/zhouzhendong/p/NowCoder-2018-Summer-Round4-A.html

题目传送门 - https://www.nowcoder.com/acm/contest/142/A

题意

  给定一个长度为 $n$ ,只包含 $0,1,2$ 的数列。

  每一秒会依次进行如下操作:

  1. 所有的 $1$ 后面生出一个 $0$

  2. 所有的 $2$ 后面生出一个 $1$

  3. 第一个数字消失了

  问经过多少时间之后,数列全部消失。

  多组数据。

  答案对于 $10^9+7$ 取模。 $n\leq 10^5,\sum n\leq 2\times 10^6$

题解

  假设当前时间为 $t$ ,我们可以简单推一波式子分别得到消灭下一个数字及其生出来的数字之后的时间:

  如果下一个数字是   :则

          0 : $t^\prime=t+1$

          1 : $t^\prime=2t+2$

          2 : $t^\prime=6\times 2^t-3$

  那么由于 $t$ 会出现在指数上面,我们不能随意将 $t$ 对 $10^9+7$ 取模。

  我当场写了个 $O(n\log n)$ 的,一个有点低级的错误续了我很久(看来昨天念诗之后还有后遗症啊)

  考虑欧拉定理:

  当 $\gcd(a,b)=1$ 时,$a^{\phi (b)}\equiv 1 \pmod b$ 。

  由于这里 $a=2$ ,所以我们可以进行推广。

  记 $b=x\times 2^y,a=2^{k+y}$ ,则当 $k\geq 0$ 时 :

$$2^{k+y}\equiv 2^{(k \mod {\phi(x)}) + y}\pmod {x\times 2^y}$$

  于是我们可以使 $k$ 取模。

  但是如果 $k<0$ 呢,那么显然这个数字很小,直接算出来。

  注意一下 $y$ 的值也是很小的。

  我们处理一下前面较小的一部分的答案,然后从后往前根据上式递归求解即可。

  时间复杂度 $O(n\log n)$ 。

代码

#include <bits/stdc++.h>
using namespace std;
typedef long long LL;
const int N=100005;
int T,ps[N],x[N],y[N],px[N],pt[N];
char s[N];
int Pow(int x,int y,int mod){
int ans=1;
for (;y;y>>=1,x=1LL*x*x%mod)
if (y&1)
ans=1LL*ans*x%mod;
return ans;
}
int phi(int x){
int ans=x;
for (int i=2;i*i<=x;i++){
if (x%i==0){
ans=ans/i*(i-1);
while (x%i==0)
x/=i;
}
}
if (x>1)
ans=ans/x*(x-1);
return ans;
}
int solve(int j,int k){
if (j==0||pt[j]>0)
return pt[j];
int i=j;
while (i>0&&s[i]!='2')
i--;
int t=i==0?0:((6LL*Pow(2,solve(i-1,k+1),x[k])-3)%x[k]);
for (int p=i+1;p<=j;p++){
if (s[p]=='0')
t=(t+1)%x[k];
if (s[p]=='1')
t=(2*t+2)%x[k];
}
t=((t-y[k])%x[k]+x[k])%x[k];
return t+y[k];
}
int main(){
ps[0]=1e9+7;
for (int i=1;i<=100000;i++)
ps[i]=phi(ps[i-1]);
for (int i=1;i<=100000;i++){
for (x[i]=ps[i-1],y[i]=0;x[i]%2==0;x[i]>>=1,y[i]++);
x[i]=phi(x[i]);
}
x[0]=ps[0],y[0]=0;
scanf("%d",&T);
while (T--){
scanf("%s",s+1);
int n=strlen(s+1),i=0;
for (int x=0;i<n;i++,pt[i]=x){
if (s[i+1]=='0')
x++;
if (s[i+1]=='1')
x=x*2+2;
if (s[i+1]=='2')
x=6*Pow(2,x,1e9+7)-3;
if (x>=21)
break;
}
printf("%d\n",(solve(n,0))%ps[0]);
for (;i>0;i--)
pt[i]=0;
}
return 0;
}

  

2018牛客网暑假ACM多校训练赛(第四场)A Ternary String 数论的更多相关文章

  1. 2018牛客网暑假ACM多校训练赛(第二场)E tree 动态规划

    原文链接https://www.cnblogs.com/zhouzhendong/p/NowCoder-2018-Summer-Round2-E.html 题目传送门 - 2018牛客多校赛第二场 E ...

  2. 2018牛客网暑假ACM多校训练赛(第三场)I Expected Size of Random Convex Hull 计算几何,凸包,其他

    原文链接https://www.cnblogs.com/zhouzhendong/p/NowCoder-2018-Summer-Round3-I.html 题目传送门 - 2018牛客多校赛第三场 I ...

  3. 2018牛客网暑假ACM多校训练赛(第三场)G Coloring Tree 计数,bfs

    原文链接https://www.cnblogs.com/zhouzhendong/p/NowCoder-2018-Summer-Round3-G.html 题目传送门 - 2018牛客多校赛第三场 G ...

  4. 2018牛客网暑假ACM多校训练赛(第三场)D Encrypted String Matching 多项式 FFT

    原文链接https://www.cnblogs.com/zhouzhendong/p/NowCoder-2018-Summer-Round3-D.html 题目传送门 - 2018牛客多校赛第三场 D ...

  5. 2018牛客网暑假ACM多校训练赛(第十场)H Rikka with Ants 类欧几里德算法

    原文链接https://www.cnblogs.com/zhouzhendong/p/NowCoder-2018-Summer-Round10-H.html 题目传送门 - https://www.n ...

  6. 2018牛客网暑假ACM多校训练赛(第十场)F Rikka with Line Graph 最短路 Floyd

    原文链接https://www.cnblogs.com/zhouzhendong/p/NowCoder-2018-Summer-Round10-F.html 题目传送门 - https://www.n ...

  7. 2018牛客网暑假ACM多校训练赛(第十场)D Rikka with Prefix Sum 组合数学

    原文链接https://www.cnblogs.com/zhouzhendong/p/NowCoder-2018-Summer-Round10-D.html 题目传送门 - https://www.n ...

  8. 2018牛客网暑假ACM多校训练赛(第八场)H Playing games 博弈 FWT

    原文链接https://www.cnblogs.com/zhouzhendong/p/NowCoder-2018-Summer-Round8-H.html 题目传送门 - https://www.no ...

  9. 2018牛客网暑假ACM多校训练赛(第七场)I Tree Subset Diameter 动态规划 长链剖分 线段树

    原文链接https://www.cnblogs.com/zhouzhendong/p/NowCoder-2018-Summer-Round7-I.html 题目传送门 -  https://www.n ...

  10. 2018牛客网暑假ACM多校训练赛(第六场)I Team Rocket 线段树

    原文链接https://www.cnblogs.com/zhouzhendong/p/NowCoder-2018-Summer-Round6-I.html 题目传送门 - https://www.no ...

随机推荐

  1. 【原创】大叔经验分享(40)hdfs关闭kerberos

    hadoop.security.authentication: Kerberos -> Simple hadoop.security.authorization: true -> fals ...

  2. grep,find

    grep是强大的文本搜索工具,他可以对文件逐行查看,如果找到匹配的模式,就可以打印出包含次模式的所有行,并且支持正则表达式 find查找文件的grep是来查找字符串的,文件的内容 grep 文件的内容 ...

  3. Linux学习之CentOS(三)--初识linux的文件系统以及用户组等概念

    Linux学习之CentOS(三)--初识linux的文件系统以及用户组等概念 进入到了Linux学习之CentOS第三篇了,这篇文章主要记录下对linux文件系统的初步认识,以及用户组.用户权限.文 ...

  4. Go语言中的byte和rune区别、对比

    Go语言中byte和rune实质上就是uint8和int32类型.byte用来强调数据是raw data,而不是数字:而rune用来表示Unicode的code point.参考规范: uint8 t ...

  5. STM32应用实例十五:STM32的ADC通道间干扰的问题

    最近我们在开发一个项目时,用到了MCU自带的ADC,在调试过程中发现通道之间村在相互干扰的问题.以前其实也用过好几次,但要求都不高所以没有太关注,此次因为物理量的量程较大,所以看到了变化. 首先来说明 ...

  6. Oracle_plsql_开发工具搭建最小化客户端

    一:资源下载获取路径: 二:配置方法 1:前提是安装好plsql开发工具 具体安装步骤略 2:配置 简化版的客户端工具. 具体格式:可以参照下文来修改编写使用. orcl_1521 = (DESCRI ...

  7. Confluence 6 使用 Apache 的 mod_jk

      在 Confluence 6 及其后续版本中,不能使用 mod_jk 来做代理.这是因为 Synchrony 服务导致的这个限制. Synchrony 在协同编辑的时候需要启动,同时还不能接受 A ...

  8. Rational Rose 2007下载、安装和破解

    一.文件下载 (1)DAEMON Tools Lite(虚拟光驱)下载地址 链接:https://pan.baidu.com/s/19L1FT6T1MlyhkfXyobd26A 提取码:drfs (2 ...

  9. centos忘记密码

    1.启动时按上下箭头,然后按e进入进入编辑模式 2.上下箭头切换在选择 linux ...这行在末尾输入 LANG=en_US.UTF-8 init=/bin/sh 然后按 ctrl+x 进行引导 3 ...

  10. java----DOS命令

    dir /?   查看帮助 dir /s   查看当前的目录,以及子目录