【BZOJ】1987: Zju2672 Fibonacci Subsequence

题意

给出一个序列\(A\),求一个最长的满足fib性质的子序列,输出其长度及其元素(如果多种方案,输出位置最靠前的)。(\(n \le 3000\)

题解

容易想到dp,即\(d(i, j)\)表示\(i\)作为fib序列的倒数第二项且\(j\)作为fib序列的最后一项的最大长度。
那么很容易通过\(a[i]-a[j]\)得到转移。
然后发现倒推很好求出位置序列字典序最小的方案。
方案的话..有了长度有了前两项,你还不会求?

坑点:
bzoj上题意不对,应该是输出位置最靠前的方案,即位置序列的字典序最小。
开hash和开map就mle系列...还是老老实实二分...

#include <bits/stdc++.h>
#include <hash_map>
using namespace std;
const int N=3005;
int a[N], n, d[N][N], ans, b[N], pos[N], cnt;
int main() {
    scanf("%d", &n);
    for(int i=1; i<=n; ++i) scanf("%d", &a[i]), b[i]=a[i];
    sort(b+1, b+1+n);
    cnt=unique(b+1, b+1+n)-b-1;
    for(int j=n; j>=2; --j) {
        for(int i=1; i<j; ++i) {
            int k=0, ps=lower_bound(b+1, b+1+cnt, a[i]+a[j])-b;
            if(b[ps]==a[i]+a[j]) k=pos[ps];
            if(k) d[i][j]=d[j][k]+1;
            else d[i][j]=2;
            ans=max(ans, d[i][j]);
        }
        pos[lower_bound(b+1, b+1+cnt, a[j])-b]=j;
    }
    if(ans<3) { puts("0"); return 0; }
    printf("%d\n", ans);
    for(int i=1; i<n; ++i) for(int j=i+1; j<=n; ++j) if(d[i][j]==ans) {
        printf("%d", a[i]);
        int x=a[i], y=a[j], t;
        for(int i=2; i<=ans; ++i) {
            printf(" %d", y);
            t=y;
            y=x+y;
            x=t;
        }
        return 0;
    }
    return 0;
}

转载于:https://www.cnblogs.com/iwtwiioi/p/4985698.html

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值