【set&&sstream||floyed判环算法】【UVa 11549】Calculator Conundrum
CALCULATOR CONUNDRUM
Alice got a hold of an old calculator that can display n digits. She was bored enough to
come up with the following time waster.
She enters a number k then repeatedly squares it until the result
overflows. When the result overflows, only the n most significant digits are displayed on the screen and an error flag appears. Alice can clear the error and continue squaring the displayed number.
She got bored by this soon enough, but wondered:
“Given n and k, what is the largest number I can get by wasting time in this manner?”
Program
Input
The first line of the input contains an integer t (1 ≤ t ≤ 200), the number of
test cases. Each test case contains two integers n (1 ≤ n ≤ 9) and k (0 ≤ k < 10n) where n is the number of digits this calculator can display k is the starting number.
Program
Output
For each test case, print the maximum number that Alice can get by repeatedly squaring the starting
number as described.
Sample
Input & Output
INPUT
2
1 6
2 99
OUTPUT
9
99
Calgary Collegiate Programming Contest 2008
题解就不写了
刘汝佳老师写的更深入浅出
题目已经暗示了计算器显示出的数将出现循环(想一想,为什么),所以不妨一个一个地模拟,每次判断新得到的数是否以前出现过。如何判断呢?一种方法是把所有计算出来的数放到一个数组里,然后一一进行比较。不难发现,这样每次判断需要花费非常多的时间,相当慢。能否开一个数组vis,直接读vis[k]判断整数k是否出现过呢?很遗憾,k的范围太大,开不下。在这种情况下,一个简便的方法是利用STL的集合,代码如下。
#include <cstdio>
#include <cstdlib>
#include <cmath>
#include <cstring>
#include <ctime>
#include <algorithm>
#include <set>
#include <iostream>
#include <sstream>
#define uns unsigned
#define int64 long long
#ifdef WIN32
#define fmt64 "%I64d"
#else
#define fmt64 "%lld"
#endif
#define oo 0x13131313
#define REP(i, n) for (int i = 1; i <= (n); ++i)
#define FOR(start,end,n) for(int i=start,i <= (end);++i))
using namespace std; int next(int n,int k)
{
stringstream ss;
ss<<(int64)k*k;
string s=ss.str();
if(s.length()>n) s=s.substr(0,n);
int ans;
stringstream ss2(s);
ss2>>ans;
return ans;
} int main()
{
int T;
cin>>T;
while(T--)
{
int n,k;
cin>>n>>k;
set<int> s;
int ans=k;
while(!s.count(k))
{
s.insert(k);
if(k>ans) ans=k;
k=next(n,k);
}
cout <<ans<<endl;
}
return 0;
}
set:
s.count(i)返还集合中i的次数
s.insert(k)将k插入
sstream
ss.str()可以返还存在流中的字符串
ss(s) ss赋初值
string
ss.substr 切断
还有几个成员函数记忆一下
sub.swap()//交换
sub.clear()
//清除
sub.size(),length()
sub.copy()
sub.c_str()
//将内容以C_string返回
sub.substr(a,b)
//返回某个子字符串(a到b)
上述程序在UVa OJ上的运行时间为4.5秒。有经验的读者应该知道,STL的string很慢,stringstream更慢,所以需要考虑把它们换掉。
#include <cstdio>
#include <cstdlib>
#include <cmath>
#include <cstring>
#include <ctime>
#include <algorithm>
#include <set>
#include <iostream>
#include <sstream>
#define uns unsigned
#define int64 long long
#ifdef WIN32
#define fmt64 "%I64d"
#else
#define fmt64 "%lld"
#endif
#define oo 0x13131313
#define REP(i, n) for (int i = 1; i <= (n); ++i)
#define FOR(start,end,n) for(int i=start,i <= (end);++i))
using namespace std;
int buf[30];
int next(int n,int k)
{
if(!k) return 0;
int tot=0,ans=0;
int64 temp;
temp=(long long)k*k;
while(temp>0)
{
buf[++tot]=temp%10;
temp=temp/10;
}
temp=1;
for(int i=tot-n+1;i<=tot;i++)
{
ans+=buf[i]*temp;
temp=temp*10;
}
return ans;
}
int main()
{
int T;
cin>>T;
while(T--)
{
int n,k;
cin>>n>>k;
set<int> s;
int ans=k;
while(!s.count(k))
{
s.insert(k);
if(k>ans) ans=k;
k=next(n,k);
}
cout <<ans<<endl;
}
return 0;
}
上述程序的运行时间降为1秒。
当然,也可以用哈希表(详见《入门经典》的相关部分),但和set一样,空间开销比较大。有没有空间开销比较小且速度也不错的方法呢?答案是肯定的。
想象一下,假设有两个小孩子在一个“可以无限向前跑”的跑道上赛跑,同时出发,但其中一个小孩的速度是另一个的两倍。如果跑道是直的(如图1-25(a)所示),跑得快的小孩永远在前面;但如果跑道有环(如图1-25(b)所示),则跑得快的小孩将“追上”跑得慢的小孩。
图 1-25
这个算法称为Floyd判圈算法,不仅空间复杂度将降为O(1),运行时间也将缩短到0.5秒。主程序如下。
int main() {
int T;
cin>> T;
while(T--){
int n, k;
cin>> n >> k;
int ans =k;
int k1 =k, k2 = k;
do {
k1 =next(n, k1); //小孩1
k2 =next(n, k2); if(k2 > ans) ans = k2; //小孩2,第一步
k2 =next(n, k2); if(k2 > ans) ans = k2; //小孩2,第二步
}while(k1 != k2); //追上以后才停止
cout<< ans << endl;
}
return 0;
}
实际可能next 操作多了一倍 但是因为没用set所以反而更快了
省时间又省空间
最终的程序:
#include <cstdio>
#include <cstdlib>
#include <cmath>
#include <cstring>
#include <ctime>
#include <algorithm>
#include <set>
#include <iostream>
#include <sstream>
#define uns unsigned
#define int64 long long
#ifdef WIN32
#define fmt64 "%I64d"
#else
#define fmt64 "%lld"
#endif
#define oo 0x13131313
#define REP(i, n) for (int i = 1; i <= (n); ++i)
#define FOR(start,end,n) for(int i=start,i <= (end);++i))
using namespace std;
int buf[30];
int next(int n,int k)
{
if(!k) return 0;
int tot=0,ans=0;
int64 temp;
temp=(long long)k*k;
while(temp>0)
{
buf[++tot]=temp%10;
temp=temp/10;
}
temp=1;
for(int i=tot-n+1;i<=tot;i++)
{
ans+=buf[i]*temp;
temp=temp*10;
}
return ans;
}
int main()
{
int T;
cin>>T;
while(T--)
{
int n,k,k1,k2;
cin>>n>>k;
int ans=k;
k1=k;
k2=k;
do
{
k1=next(n,k1);
k2=next(n,k2);if(k2>ans) ans=k2;
k2=next(n,k2);if(k2>ans) ans=k2;
} while(k1!=k2);
cout <<ans<<endl;
}
return 0;
}
【set&&sstream||floyed判环算法】【UVa 11549】Calculator Conundrum的更多相关文章
- Floyd判圈算法 UVA 11549 - Calculator Conundrum
题意:给定一个数k,每次计算k的平方,然后截取最高的n位,然后不断重复这两个步骤,问这样可以得到的最大的数是多少? Floyd判圈算法:这个算法用在循环问题中,例如这个题目中,在不断重复中,一定有一个 ...
- UVA 11549 CALCULATOR CONUNDRUM(Floyd判圈算法)
CALCULATOR CONUNDRUM Alice got a hold of an old calculator that can display n digits. She was bore ...
- UVA 11549 Calculator Conundrum (Floyd判圈算法)
题意:有个老式计算器,每次只能记住一个数字的前n位.现在输入一个整数k,然后反复平方,一直做下去,能得到的最大数是多少.例如,n=1,k=6,那么一次显示:6,3,9,1... 思路:这个题一定会出现 ...
- 龟兔赛跑算法 floyed判环算法
今天写线段树写到要用到这个算法的题目,简单的学习一下. https://blog.csdn.net/javaisnotgood/article/details/89243876 https://blo ...
- Uva 11549 - Calculator Conundrum 找规律加map
题目链接:http://uva.onlinejudge.org/index.php?option=com_onlinejudge&Itemid=8&page=show_problem& ...
- Floyed判环/龟兔算法
求[(5+2√6)2^x+1 ] mod p 的值,其中 0 ≤ x < 232 , p 是个质数,p ≤ 46337 .(这里介绍的是一种暴力的做法) (5+2√6)2^n+1 = an + ...
- floyd判环算法(龟兔赛跑算法)
floyd判环算法(龟兔赛跑算法) 注意,这个算法是用来判断一条链+一条环的图,环的长度或者环与链的交界处的,所以此floyd非彼floyd(虽然都是一个人想出来的). (图不是我的) 如果只要求环的 ...
- UVa 11549 计算器谜题(Floyd判圈算法)
https://vjudge.net/problem/UVA-11549 题意: 有一个老式计算器,只能显示n位数字,输入一个整数k,然后反复平方,如果溢出的话,计算器会显示结果的最高n位.如果一直这 ...
- hdu4975 A simple Gaussian elimination problem.(正确解法 最大流+删边判环)(Updated 2014-10-16)
这题标程是错的,网上很多题解也是错的. http://acm.hdu.edu.cn/showproblem.php?pid=4975 2014 Multi-University Training Co ...
随机推荐
- 【算法】插入排序 insertion_sort
准备写个<STL 源代码剖析>的读书笔记,开个专栏.名为<STL 的实现>,将源代码整理一遍.非常喜欢侯捷先生写在封底的八个字:天下大事.必作于细.他在书中写到:"我 ...
- Linux命令之nano -
我使用过的Linux命令之nano - 比vi简单易用的文本编辑器 本文链接:http://codingstandards.iteye.com/blog/802593 (转载请注明出处) 用途说明 ...
- SQL语句中的DQL、DML、DCL、DDL、CCL、TPL
结构化查询语言(Structured Query Language)简称SQL,是一种特殊目的的编程语言,是一种数据库查询和程序设计语言,用于存取数据以及查询.更新和管理关系数据库系统:同时也是数据库 ...
- NET中级课--文件,流,序列化1
1.对于机器的角度来看,任何文件都是二进制的0和1. 2. 位:bit,一个1或0就是1位. 字节:byte,每8位一个字节.一个字节的范围就是00000000到1111111,换成10进制就是0 ...
- jquery简单切换插件
<!DOCTYPE html> <html> <head> <meta charset="utf-8"> <meta http ...
- Android adb不是内部或外部命令 (转)
dos窗口运行adb命令出现错误:adb不是内部或外部命令…. 出现问题原因及解决办法: 1.没有配置相关环境变量. 只要将android 的sdk安装路径添加到系统变量Path中即可. (以win7 ...
- PHP学习笔记二十九【接口】
<?php //定义接口 //接口可以定义属性,但必须是常量而且是public //接口的所有方法必须是public interface Iusb{ public function start( ...
- iOS 面试题 2
1. 描述应用程序的启动顺序. 1.程序入口main函数创建UIApplication实例和UIApplication代理实例 2.在UIApplication代理实例中重写启动方法, ...
- gulp压缩js
1.安装nodejs -> 全局安装gulp -> 项目安装gulp以及gulp插件 -> 配置gulpfile.js -> 运行任务 2.查看nodejs的版本号 3.npm ...
- ruby更换为淘宝的源
我们在使用gem更新的时候,经常会为速度抓狂,其实gem默认的源是https://rubygems.org,比较慢众所周至的原因了. 可以将源更换到国内的taobao源 查看当前有的源 gem sou ...