CSU1120-病毒-DP/LCS/LIS

G: 病毒

Description

你有一个日志文件,里面记录着各种系统事件的详细信息。自然的,事件的时间戳按照严格递增顺序排列(不会有两个事件在完全相同的时刻发生)。

遗憾的是,你的系统被病毒感染了,日志文件中混入了病毒生成的随机伪事件(但真实事件的相对顺序保持不变)。备份的日志文件也被感染了,但由于病毒采用的随机感染方法,主日志文件和备份日志文件在感染后可能会变得不一样。

给出被感染的主日志和备份日志,求真实事件序列的最长可能长度。

Input

输入第一行为数据组数T (T<=100)。每组数据包含两行,分别描述感染后的主日志和备份日志。

每个日志文件的格式相同,均为一个整数n (1<=n<=1000)(代表感染后的事件总数)和n 个不超过100,000的正整数(表示感染后各事件的时间戳)。

注意,感染后可能会出现时间戳完全相同的事件。

Output

对于每组数据,输出真实事件序列的最长可能长度。

Sample Input

1
9 1 4 2 6 3 8 5 9 1
6 2 7 6 3 5 1

Sample Output

3

Hint

这个题太明显了
先找一次LCS然后找最长上升子序列?不不不那样的话程序写起来贼麻烦
你只需要在找LCS的时候保证上升就好

#include <bits/stdc++.h>
#define N 1010
#define INF 0x3f3f3f3f
#define LL long long
#define mem(a,n) memset(a,n,sizeof(a))
#define fread freopen("in.txt","r",stdin)
#define fwrite freopen("out.txt","w",stdout)
using namespace std;
//int mp[1010][1010],arr1[1010],arr2[1010],dp[1010];愚蠢的做法的痕迹╭(╯^╰)╮
//pair<int,int> path[1010][1010];
//int main()
//{
//  ios::sync_with_stdio(false);
//  int t,n1,n2,x,y;
//  vector<int> ans;
//  cin>>t;
//  path[1][1].first=0,path[1][1].second=0;
//  while(t--){
//      ans.clear();
//      cin>>n1;
//      for(int i=1;i<=n1;++i){
//          cin>>arr1[i];
//      }
//      cin>>n2;
//      for(int i=1;i<=n2;++i){
//          cin>>arr2[i];
//      }
//      for(int i=0;i<=n1||i<=n2;++i){
//          mp[i][0]=mp[0][i]=0;
//      }
//      for(int i=1;i<=n1;++i){
//          for(int j=1;j<=n2;++j){
//              if(arr1[i]==arr2[j]){
//                  mp[i][j]=mp[i-1][j-1]+1;
//                  path[i][j].first=i-1;
//                  path[i][j].second=j-1;
//              }else{
//                  if(mp[i][j-1]>mp[i-1][j]){
//                      mp[i][j]=mp[i][j-1];
//                      path[i][j].first=i;
//                      path[i][j].second=j-1;
//                  }else{
//                      mp[i][j]=mp[i-1][j];
//                      path[i][j].first=i-1;
//                      path[i][j].second=j;
//                  }
//              }
//          }
//      }
//      int tempx,tempy;
//      x=n1,y=n2;
//      while(x!=0&&y!=0){
//          tempx=path[x][y].first,tempy=path[x][y].second;
//          if(mp[x][y]!=mp[tempx][tempy]){
//              ans.push_back(arr1[x]);
//          }
//          x=tempx,y=tempy;
//      }
//      for(int i=0;i<ans.size();++i){
//          cout<<ans[i]<<' ';
//          dp[i]=1;
//      }
//      int m,ma=0;
//      for(int i=1;i<ans.size();++i){
//          m=0;
//          for(int j=0;j<i;++j){
//              if(ans[i]>ans[j]&&dp[j]>m){
//                  m=dp[j];
//              }
//          }
//          dp[i]=m+1;
//          if(dp[i]>ma){
//              ma=dp[i];
//          }
//      }
//      for(int i=0;i<ans.size();++i){
//          if(dp[i]>m){
//              m=dp[i];
//          }
//          cout<<dp[i]<<' ';
//      }
//      cout<<m<<endl;
//  }
//  return 0;
//}
int arr1[N],arr2[N],dp[N];
int main()
{
    ios::sync_with_stdio(false);
    int n,m,t;
    cin>>t;
    while(t--){
        cin>>n;
        for(int i=1;i<=n;++i){
            cin>>arr1[i];
        }
        cin>>m;
        for(int i=1;i<=m;++i){
            cin>>arr2[i];
        }
        mem(dp,0);
        int MAX;
        for(int i=1;i<=n;++i){
            MAX=0;
            for(int j=1;j<=m;++j){
                if(arr1[i]>arr2[j]&&MAX<dp[j]){
                    MAX=dp[j];
                }
                if(arr1[i]==arr2[j]){
                    dp[j]=MAX+1;
                }
            }
        }
        MAX=0;
        for(int i=1;i<=m;++i){
            if(dp[i]>MAX){
                MAX=dp[i];
            }
        }
        cout<<MAX<<endl;
    }
    return 0;
}

/**********************************************************************
    Problem: 1120
    User: CSUzick
    Language: C++
    Result: AC
    Time:228 ms
    Memory:1700 kb
**********************************************************************/
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值