[BJOI2012]最多的方案(记忆化搜索)

Posted zh-comld

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了[BJOI2012]最多的方案(记忆化搜索)相关的知识,希望对你有一定的参考价值。

第二关和很出名的斐波那契数列有关,地球上的OIer都知道:F1=1, F2=2, Fi = Fi-1 + Fi-2,每一项都可以称为斐波那契数。现在给一个正整数N,它可以写成一些斐波那契数的和的形式。如果我们要求不同的方案中不能有相同的斐波那契数,那么对一个N最多可以写出多少种方案呢?

题意是说数列中不能出现相同的数。

显然要记忆化搜索。

直接搜会T,我们枚举下一个数填什么是要从大到小枚举,可以使效率有指数级的提升。

这是枚举上界,枚举下界可以用前缀和+二分来优化枚举复杂度。

加了这两个优化后代码跑的飞快。

Code

#include<iostream>
#include<cstdio>
#include<algorithm>
#include<map>
#define mm make_pair
using namespace std;
typedef long long ll;
ll dp[100],sum[100];
map<pair<ll,int>,ll>mem;
ll n;
ll dfs(ll x,int xian){
    if(!x)return 1;
    if(mem[mm(x,xian)])return mem[mm(x,xian)];
    ll ans=0;
    int p=lower_bound(sum+1,sum+87+1,x)-sum;
    for(int i=p;i<=xian;++i)if(dp[i]<=x)ans+=dfs(x-dp[i],i-1);else break;
    return mem[mm(x,xian)]=ans;
}
int main(){
    scanf("%lld",&n);dp[0]=dp[1]=1;
    for(int i=2;i<=87;++i)dp[i]=dp[i-1]+dp[i-2];
    for(int i=1;i<=87;++i)sum[i]=sum[i-1]+dp[i];
    printf("%lld",dfs(n,87));
    return 0;
} 

 

以上是关于[BJOI2012]最多的方案(记忆化搜索)的主要内容,如果未能解决你的问题,请参考以下文章

P4133 [BJOI2012]最多的方案(记搜)

挖地雷(记忆化搜索)

!HDU 1078 FatMouse and Cheese-dp-(记忆化搜索)

P1141 01迷宫 (记忆化搜索)

hdu 1078 FatMouse and Cheese(简单记忆化搜索)

HDU 1078 FatMouse and Cheese 简单记忆化搜索