程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> C語言 >> C++ >> C++入門知識 >> POJ 1692 Crossed Matchings(dp).

POJ 1692 Crossed Matchings(dp).

編輯:C++入門知識

POJ 1692 Crossed Matchings(dp).


~~~~

題意:給出兩行數字序列,求上下匹配的最多組數是多少.
匹配規則:
1.匹配對的數字必須相同。
2.每個匹配必須有且只能有一個匹配與之相交叉,且相交叉的兩組匹配數字必須不同。
3.一個數最多只能匹配一次。

最長公共子序列的改編,令f[i][j]表示第一個序列的前i個數字和第二個序列的前j個數字的匹配的最優值。

狀態轉移:f[i][j]=max(f[p-1][q-1]+2,max(f[i-1][j],f[i][j-1]))

&&(a[p]==b[j],b[q]==a[i]);

*以f[p-1][q-1]更新f[i][j]就能保證一個數最多只能匹配一次。

題目鏈接:http://poj.org/problem?id=1692

~~~~

#include
#include
#include
#include
#include
#define N 1000
using namespace std;

int f[N][N];
int a[N],b[N];
int main()
{
    int t;
    scanf("%d",&t);
    while(t--)
    {
        int n,m;
        scanf("%d%d",&n,&m);
        int i,j,p,q;
        for(i=1;i<=n;i++)
            scanf("%d",&a[i]);
        for(j=1;j<=m;j++)
            scanf("%d",&b[j]);
        memset(f,0,sizeof(f));
        for(i=1;i<=n;i++)
        {
            for(j=1;j<=m;j++)
            {
                for(q=j-1;q>0;q--)
                    if(b[q]==a[i]) break;
                for(p=i-1;p>0;p--)
                    if(a[p]==b[j]) break;
                if(p>0 && q>0 && a[i]!=b[j])   //~~~
                    f[i][j]=max(f[p-1][q-1]+2,max(f[i][j-1],f[i-1][j]));
                else
                    f[i][j]=max(f[i][j-1],f[i-1][j]);
            }
        }
        printf("%d\n",f[n][m]);
    }
    return 0;
}


  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved