【ZOJ】3785 What day is that day? ——KMP 暴力打表找规律
转自:http://www.cnblogs.com/kevince/p/3887827.html
首先声明一下,这里的规律指的是循环,即找到最小循环周期。
这么一说大家心里肯定有数了吧,“不就是next数组性质的应用嘛”,没错,正是如此。
在ACM的比赛中有些时候会遇到一些题目,可以或必须通过找出数据的规律来编写代码,这里我们专门来讨论下 如何运用KMP中next数组的性质 来寻找一个长数组中的最小循环周期。
先来看一道题
ZOJ 3785
What day is that day?
Time Limit: 2 Seconds Memory Limit: 65536 KB
It's Saturday today, what day is it after 11 + 22 + 33 + ... + NN days?
Input
There are multiple test cases. The first line of input contains an integer T indicating the number of test cases. For each test case:
There is only one line containing one integer N (1 <= N <= 1000000000).
Output
For each test case, output one string indicating the day of week.
Sample Input
- 2
- 1
- 2
Sample Output
- Sunday
- Thursday
Hint
A week consists of Sunday, Monday, Tuesday, Wednesday, Thursday, Friday and Saturday.
题目的大意是知道今天是周六,让你求 f = 11 + 22 + 33 + ... + NN 这么多天之后是星期几。
也就是求f % 7对于每个输入的N的值。这题在网上一搜题解,都说是打表找规律,当然这题有两种找法,一是对于每个ii % 7 的值都找规律。
这里我们打表可知 前100个值如下所示
1 4 6 4 3 1 0 1 1 4 2 1 6 0 1 2 5 1 5 1 0 1 4 1 4 4 6 0 1 1 3 2 6 1 0 1 2 2 1 2 6 0 1 4 6 4 3 1 0 1 1 4 2 1 6 0 1 2 5 1 5 1 0 1 4 1 4 4 6 0 1 1 3 2 6 1 0 1 2 2 1 2 6 0 1 4 6 4 3 1 0 1 1 4 2 1 6 0 1 2
有一种找规律的方法是当有数字等于第一个数的时候做个标记,再人工判断是否能够构成一个循环。
不可否认的,对于周期较短的一组数字这样找周期并不难,可是如果周期大到数百数千甚至数万时,靠这种方法找周期恐怕是杯水车薪。
当时我就迷茫在了这一长串的数字中不知所措,猛然想起前不久看过的KMP中next数组的性质,当即想到了用KMP求最小重复子串长度的方法,于是脑洞大开……
该性质为:令j=leni-next[i],如果i%j==0且i/j>1,j就是Pi的最小循环节( Pi表示文本串的前i个字符,leni表示该字符串的长度,一般表示为leni = i + 1)
还有一种找规律的方法是直接对 f % 7 的值进行打表找规律,按照上述方法找到的周期为294,下面要做的就很简单了~
- #include <iostream>
- #include<cstdio>
- #include<cstdlib>
- #include<cstring>
- #include<algorithm>
- #include<vector>
- #define N 605
- #define M 200
- #define ll long long
- using namespace std;
- int i,j;
- int T;
- int l;
- int a[N],sum[N];
- char s[N];
- int next[N];
- int fun(int n)
- {
- int re = ;
- for(int i=;i<=n;i++){
- re*=n;
- re%=;
- }
- return re;
- }
- void ini()
- {
- a[]=sum[]=;
- a[]=sum[]=;
- s[]='';
- s[]='';
- for(i=;i<=N-;i++){
- a[i]=fun(i);
- sum[i]+=a[i]+sum[i-];
- sum[i]%=;
- s[i]=sum[i]+'';
- }
- s[i]='\n';
- //printf("%s\n",s);
- // for(i=1;i<=M;i++){
- // printf(" %d %d %d\n",i,a[i],sum[i]);
- // printf(" %d",sum[i]);
- //}
- return;
- }
- void ini2()
- {
- int j = -, i = ;
- next[] = -;
- while(i < N-)
- {
- if(j == - || s[i] == s[j])
- {
- i++;
- j++;
- next[i] = j;
- }
- else
- {
- j = next[j];
- }
- }
- }
- void ini3()
- {
- for(int i = ; i <= N-; ++i)
- {
- int length = i - next[i]; //循环节的长度
- if(i != length && i % length == ) //如果有多个循环
- {
- printf("%d %d\n", i, i / length);
- break;
- }
- }
- }
- int main()
- {
- int x;
- int ans;
- ini();
- //ini2();
- // ini3();
- //freopen("data.txt","r",stdin);
- scanf("%d",&T);
- //while(scanf("%d",&n)!=EOF)
- while(T--)
- {
- scanf("%d",&x);
- ans=sum[x%];
- if(ans==) printf("Saturday\n");
- else if(ans==) printf("Sunday\n");
- else if(ans==) printf("Monday\n");
- else if(ans==) printf("Tuesday\n");
- else if(ans==) printf("Wednesday\n");
- else if(ans==) printf("Thursday\n");
- else if(ans==) printf("Friday\n");
- }
- return ;
- }
后来又练了一次,用了map
4172110 | 2016-03-15 15:26:39 | Accepted | 3785 | C++ | 820 | 280 | njczy2010 |
- #include <iostream>
- #include <cstdio>
- #include <map>
- #include <cstring>
- #include <algorithm>
- #include <cmath>
- #include <string>
- using namespace std;
- #define N 1005
- #define ll long long
- #define mod 7
- int TT;
- int n;
- int s[N];
- int next[N];
- map<int,string> mp;
- void get_next()
- {
- int i,j;
- i = ;
- j = -;
- next[] = -;
- while(i<N)
- {
- if(j == - || s[i] == s[j]){
- i++;j++;next[i] = j;
- }
- else{
- j = next[j];
- }
- }
- }
- int quickpow(int x,int n)
- {
- int re = ;
- while(n)
- {
- if(n&){
- re = (re * x) % mod;
- }
- n /= ;
- x = (x * x) % mod;
- }
- return re;
- }
- void fi()
- {
- int i;
- for(i = ;i < N;i++){
- int length = i - next[i]; //循环节的长度
- if(i != length && i % length == ) //如果有多个循环
- {
- printf("%d %d %d\n", i, length, i / length);
- break;
- }
- }
- }
- void ini()
- {
- int i;
- s[] = ;
- for(i = ;i < N;i++){
- s[i] = (s[i - ] + quickpow(i,i) )%mod;
- }
- mp[] = "Saturday";
- mp[] = "Sunday";
- mp[] = "Monday";
- mp[] = "Tuesday";
- mp[] = "Wednesday";
- mp[] = "Thursday";
- mp[] = "Friday";
- //get_next();
- //fi();
- //
- }
- int main()
- {
- //freopen("in.txt","r",stdin);
- //freopen("out.txt","w",stdout);
- ini();
- scanf("%d",&TT);
- while(TT--){
- //while(scanf("%d",&n)!=EOF){
- scanf("%d",&n);
- cout << mp[ s[ n% ] ] << endl;
- }
- return ;
- }
【ZOJ】3785 What day is that day? ——KMP 暴力打表找规律的更多相关文章
- 【ZOJ】3785 What day is that day? ——浅谈KMP在ACM竞赛中的暴力打表找规律中的应用
转载请声明出处:http://www.cnblogs.com/kevince/p/3887827.html ——By Kevince 首先声明一下,这里的规律指的是循环,即找到最小循环周期. 这 ...
- ZOJ 3622 Magic Number 打表找规律
A - Magic Number Time Limit:2000MS Memory Limit:32768KB 64bit IO Format:%lld & %llu Subm ...
- zoj 3785 What day is that day? (打表找规律)
题目 思路:比赛的时候有想过找循环节,但是,打表打错了. 后来,看着过了挺多人,就急了, 看了一下别人的时间 耗时都挺长的,就以为不是找规律, 没想到真是找规律,不过,这个题的数据可能挺大的. AC代 ...
- zoj 3629 Treasure Hunt IV 打表找规律
H - Treasure Hunt IV Time Limit:2000MS Memory Limit:65536KB 64bit IO Format:%lld & %llu ...
- 【ARC077F】SS kmp+打表找规律
Description 如果某个串可以由两个一样的串前后连接得到,我们就称之为"偶串".比如说"xyzxyz"和"aaaaaa"是偶串, ...
- ZOJ 3785 What day is that day?(今天是星期几?)
Description 题目描述 It's Saturday today, what day is it after 11 + 22 + 33 + ... + NN days? 今天是星期六,11 + ...
- ZOJ 3587 Marlon's String 扩展KMP
链接:http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemCode=3587 题意:给出两个字符串S和T.S,T<=100000.拿出 ...
- zoj 3785 What day is that day?
http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemId=5272 打表找规律. #include <cstdio> #incl ...
- ZOJ 3785 What day is that day?(数论:费马小定理)
What day is that day? Time Limit: 2 Seconds Memory Limit: 65536 KB It's Saturday today, what da ...
随机推荐
- Java IO 之 File 的创建、重命名与遍历
File表示存储设备上的一个文件或目录,使用方式查看API即可,比较简单 package org.zln.io.file; import java.io.File; /** * Created by ...
- jQuery下拉列表二级联动插件
jQuery下拉列表二级联动插件的视图代码: <!doctype html> <html lang="en"> <head> <meta ...
- JDBC连接数据库的过程
以连接MySQL为例: (1)加载MySQL数据库连接的驱动程序.到MySQL官网下载该驱动程序jar包,然后把包复制到WEB-INF/lib目录下,则JDBC会调用Class.forName()方法 ...
- 如何在Javascript中利用封装这个特性
对于熟悉C#和Java的兄弟们,面向对象的三大思想(封装,继承,多态)肯定是了解的,那么如何在Javascript中利用封装这个特性呢? 我们会把现实中的一些事物抽象成一个Class并且把事物的属性( ...
- 怎么把linux的磁盘映射到windows上
步骤如下: 右击如下的computer: 然后选择:Map network drive... 然后在下图按图中所示操作: 最后成功如下图所示:
- bzoj 2525 [Poi2011]Dynamite 二分+树形dp
[Poi2011]Dynamite Time Limit: 30 Sec Memory Limit: 128 MBSubmit: 270 Solved: 138[Submit][Status][D ...
- 修改firefox默认下载路径
菜单栏---编辑---首选项--在常规页就可以看到下载设置了
- Spring 中 AbstractExcelView 支持根据模板生成Excel文件. 通过设置 view 的 URL 属性指定模板的路径
注意:1. 模板需放在 WEB-INF 目录下2. 指定模板路径时不需要添加扩展名, Spring将自动添加 .xls 到URL 属性中.3. 在指定URL前需先设置 view 的 Applicat ...
- checkbox和后面文字无法居中对齐的解决方案
制作前端页面时,表单的页面中都存在表单元素与提示文字无法对齐的问题.下面是针对这一问题的解决方案: 先上结果图看效果,吼吼~ 最上面两个是经过css处理后的效果,已经居中对齐了哦~,最后一个是没有处理 ...
- NYOJ 973 天下第一 (最短路)
题目链接 描述 AC_Grazy一直对江湖羡慕不已,向往着大碗吃肉大碗喝酒的豪情,但是"人在江湖漂,怎能 不挨刀","人在江湖身不由己",如果自己的武功太差,在 ...