zoukankan      html  css  js  c++  java
  • XDU1131 思维+dp优化

    题意

    分析

    一个直观的做法是,枚举前两项,不断检查前一项,这需要维护一个有序数组并且带下标(可以用一个map<int,vector<int> >,将数字相同的数推倒一个vector中,vector中存的就是相同的数的位置)

    时间复杂度:O(n^2*(logn+logn))

    这个做法不太行啊,两个log,有点卡不过去

    由于有很多重复的枚举,dp优化

    dp[i][j]:以i,j为开始两个的最大长度

    转移:显然需要k且k>j的help,   a[k]=a[i]+a[j] (k>j)即可(二分即可)

    另一种dp[i][j]:以i,j为结束两个的最大长度

    转移:需要k且k<i的help   ,a[k]+a[i]==a[j](k<i)

    但这个状态不太好找二分不好check

    时间复杂度O(n^2*logn)

    #include <cstdio>
    #include <algorithm>
    #include <cstring>
    #include <vector>
    #include <iostream>
    #include <cmath>
    #include <queue>
    #include <ctime>
    using namespace std;
    #define lson l,(l+r)/2,rt<<1
    #define rson (l+r)/2+1,r,rt<<1|1
    #define dbg(x) cout<<#x<<" = "<< (x)<< endl
    #define pb push_back
    #define fi first
    #define se second
    #define ll long long
    #define sz(x) (int)(x).size()
    #define pll pair<long long,long long>
    #define pii pair<int,int>
    #define pq priority_queue
    vector<pll> v;
    int dp[1010][1010];
    ll a[1010];
    int main(){
        int n;
        while(scanf("%d",&n)!=EOF){
            v.clear();
            for(int i=1;i<=n;i++){
                scanf("%lld",&a[i]);
                v.pb({a[i],i});
            }
            if(n==1){
                printf("1
    %lld
    ",a[1]);
                continue;
            }
            sort(v.begin(),v.end());
            pll res;int cnt=0;
            for(int i=n-1;i>=1;i--){
                for(int j=i+1;j<=n;j++){
                    auto it=lower_bound(v.begin(),v.end(),(pll){a[i]+a[j],j+1});
                    if(it!=v.end() && it->fi==a[i]+a[j]){
                        dp[i][j]=dp[j][it->se]+1;
                    }else{
                        dp[i][j]=2;
                    }
                    if(dp[i][j]>cnt || (dp[i][j]==cnt && (pll){a[i],a[j]}<res)){
                        res={a[i],a[j]};
                        cnt=dp[i][j];
                    }
                }
            }
            printf("%d
    ",cnt);
            printf("%lld %lld",res.fi,res.se);
            for(int i=2;i<cnt;i++){
                printf(" %lld",res.fi+res.se);
                res.fi+=res.se;
                swap(res.fi,res.se);
            }
            printf("
    ");
        }
        return 0;
    }
    View Code
  • 相关阅读:
    微信小程序登入实现
    CacheLab实验--深入了解计算机系统实验
    power designer的物理数据模型生成数据字典。
    PowerDesigner15在生成SQL时报错Generation aborted due to errors detected during the verification of the mo
    Mac系统下,如何申请并安装教育版Navicat
    Mac下修改Mysql密码
    数组
    AbstractList源码阅读
    List源码阅读笔记
    AbstractCollection源码阅读笔记
  • 原文地址:https://www.cnblogs.com/Deadline/p/8977350.html
Copyright © 2011-2022 走看看