题目链接:

http://poj.org/problem?id=3899

题目意思:

求给定区间内,只含4、7的数的个数以及通过反转后在该区间内的个数和。

解题思路:

模拟+数学。

代码解释的很详细,请看代码。

#include<iostream>
#include<cmath>
#include<cstdio>
#include<cstdlib>
#include<string>
#include<cstring>
#include<algorithm>
#include<vector>
#include<map>
#include<set>
#include<stack>
#include<list>
#include<queue>
#define eps 1e-6
#define INF 0x1f1f1f1f
#define PI acos(-1.0)
#define ll __int64
#define lson l,m,(rt<<1)
#define rson m+1,r,(rt<<1)|1
using namespace std; /*
freopen("data.in","r",stdin);
freopen("data.out","w",stdout);
*/
char A[50],B[50];
//g1(a,b,len) 表示1-a中后缀为b的lucky数,其中len是b的长度
//g2(a,b)表示1-a中反转后大于b的lucky数
//A-B 之间的lucky数个数为 g1(B,0,0)-g1(A-1,0,0)
//反转后在A-B 之间的lucky数为 g2(A,A)-g2(A,B)+g2(B,B)-g2(A,B) ll g1(char * a,char * b,int len)
{
int alen=strlen(a);
ll res=0;
bool ism=false; for(int i=0;i<len;i++) //比较a的后len位与b的大小,>=
{
int j=i+alen-len;
if(a[j]>b[i])
break;
else if(a[j]<b[i])
{
ism=true;
break;
}
}
if(len==0) //先算出低于alen位的总的lucky数
{
for(int i=1;i<alen;i++)
res+=(ll)1<<i; //i位的话一共有2^i个lucky数
}//如果len!=0 那么前面的m位中也有1-m,这种情况好像没考虑啊
int m=alen-len; //计算与a位数相同lucky数
int i;
for(i=0;i<m;i++) //一位一位从前往后考虑
{
if(a[i]>'7') //4和7都可以
{
res+=(ll)1<<(m-i);
break;
}
else if(a[i]=='7') //4一定可以,7再往后计算
res+=(ll)1<<(m-i-1);//把是4的情况计算清楚,后面就是7的情况
else if(a[i]>'4'&&a[i]<'7')
{
res+=(ll)1<<(m-i-1); //放4的情况,后面可以任意
break;
}
else if(a[i]<'4')
break;
}
if((i==m)&&!ism) //计算临界情况
res++;
return res;
} ll g2(char * a,char * b) //1-a之间,反转后大于b的lucky数
{ //b的长度肯定要>=a的长度
int alen=strlen(a);
char tmp[50],*last=&tmp[49]; //从后往前
ll res=0; for(int i=0;i<alen;i++)
{
if(b[i]>'7') //高位已经超过7了,不可能超过它了
break ;
else if(b[i]=='7') //边界情况
*(last--)='7';
else if(b[i]>'4'&&b[i]<'7') //只要满足这
{
*last='7'; //只要把后面的这一位置成7,前面的可以任意了
res+=g1(a,last,i+1);
break;
}
else if(b[i]=='4')
{
*last='7';
res+=g1(a,last,i+1); //把这位放7,前面的就任意了
*(last--)='4'; //然后把它放4作为临界情况
}
else
{
*last='7'; //后面放7,前面就任意了
res+=g1(a,last,i+1);
*last='4'; //后面放4,前面就任意了
res+=g1(a,last,i+1);
break;
}
}
//因为是算大于的情况,临界情况就不用考虑了
return res;
} int main()
{
int t; char aa[4]="999",bb[2]="7";
printf("%I64d\n",g1(aa,bb,1)); scanf("%d",&t);
while(t--)
{
scanf("%s%s",A,B);
int lea=strlen(A),leb=strlen(B); if(A[lea-1]!='0')
--A[lea-1]; //是零的话就无所谓了
ll ans=0;
ans+=(g1(B,NULL,0)-g1(A,NULL,0)+g2(A,A)+g2(B,B));
if(lea==leb)
ans-=(2*g2(A,B));
printf("%I64d\n",ans); }
return 0;
}

poj-3899-The Lucky Numbers 模拟+数学的更多相关文章

  1. HDU 5676 ztr loves lucky numbers (模拟)

    ztr loves lucky numbers 题目链接: http://acm.hust.edu.cn/vjudge/contest/121332#problem/I Description ztr ...

  2. codeforces 630C Lucky Numbers

    C. Lucky Numbers time limit per test 0.5 seconds memory limit per test 64 megabytes input standard i ...

  3. hdu 5676 ztr loves lucky numbers(dfs+离线)

    Problem Description ztr loves lucky numbers. Everybody knows that positive integers are lucky if the ...

  4. codeforces 630C - Lucky Numbers 递推思路

    630C - Lucky Numbers 题目大意: 给定数字位数,且这个数字只能由7和8组成,问有多少种组合的可能性 思路: 假设为1位,只有7和8:两位的时候,除了77,78,87,88之外还哇哦 ...

  5. hdu 5676 ztr loves lucky numbers 打表+二分

    ztr loves lucky numbers Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/ ...

  6. ZCMU 2177 Lucky Numbers (easy)

    传送门: http://acm.zcmu.edu.cn/JudgeOnline/problem.php?id=2177 2177: Lucky Numbers (easy) 时间限制: 2 Sec   ...

  7. hdu-5676 ztr loves lucky numbers(乱搞题)

    题目链接: ztr loves lucky numbers  Time Limit: 2000/1000 MS (Java/Others)  Memory Limit: 65536/65536 K ( ...

  8. C - Lucky Numbers (easy)

    Problem description Petya loves lucky numbers. Everybody knows that positive integers are lucky if t ...

  9. Codeforces Round #160 (Div. 2)---A. Roma and Lucky Numbers

    Roma and Lucky Numbers time limit per test 1 second memory limit per test 256 megabytes input standa ...

随机推荐

  1. HDU 2136 Largest prime factor

    题目大意:求出比给出数小的互质的质数个数. 题解:直接用筛法求素数,稍微改编一下,将原先的布尔数组变为数组用来记录信息就可以了. 注意点:大的数组定义要放在程序的开头,不要放在main里面,不然会栈溢 ...

  2. Matlab pivotgolf

    function scoreout = pivotgolf(course,pivotstrat) % PIVOTGOLF Pivot Pickin' Golf. % Your goal is to u ...

  3. asp.neti 加密三种方式

    public string Get_MD5_Method1(string strSource) { System.Security.Cryptography.MD5 md5 = new System. ...

  4. C#高级编程零散知识点

    1.206-实现单链表的添加和插入 207-实现单链表的其他功能和 3.209-Lambda表达式 4.301-栈的介绍和BCL中的栈 4.501-进程和线程的概念[00_12_06][2015122 ...

  5. js 特效 手风琴效果

    $(document).ready(function(){ //定义展开的块 var lastBlock = $('#a1'); //展开的块的宽度 var maxWidth = 406; //折叠的 ...

  6. MongoDB shell常用命令

    Shell操作数据库: 1.  超级用户相关: 1. #进入数据库admin use admin 2. #增加或修改用户密码 db.addUser('name','pwd') 3. #查看用户列表 d ...

  7. select into from 和 insert into select 的用法和区别(转)

    转自:http://www.studyofnet.com/news/182.html select into from 和 insert into select都是用来复制表,两者的主要区别为: se ...

  8. 面向对象程序设计-C++ Finial exam review NOTES【第十六次上课笔记】

    写在前面: 我记得也不全,如果有记录的更全的同学可以留言,我会添加哒 :) 常量 内敛函数 为什么需要内敛函数 内敛函数适用于什么场合 内敛函数本身,最大优点是,避免了真正函数调用的开销 因为普通函数 ...

  9. Sticks(poj 1011)

    题目描述: Description George took sticks of the same length and cut them randomly until all parts became ...

  10. Week15(12月19日):授课综述2

    Part I:提问 =========================== 1.为了编辑应用程序的统一布局,可打开位于Views\Shared子目录中的(    )文件. A.MasterPage.h ...