FZU 2275 - Game ( KMP字符串匹配 )

题意

Alice和Bob玩游戏, 他们各自给出两个数字A, B ( 0<=A,B<=10^100000 )
他们可以对这两个数字进行两种操作: 除10 或 反转数字
例如X=123456 , 除10变为12345(而不是 12345.6). 特别注意 0/0=0 (一个坑, 如果Bob给数字是0, 则Alice一定可以一直除10达到0, 也就是说B = 0, Alice一定赢 ).
Alice先手, 问谁会赢

思路

对数字的两种操作实际上就是删A这个字符串的两端, 如果串A中能查找到B或者B的反转串, 那么Alice一定赢, 反之Bob一定赢
这个题对应的是严格的 KMP 字符串匹配算法
一开始写的string类里的find()函数超时了, 后来用的 < cstring>下的strstr() 函数也超时了

KMP 时间复杂度O(n)
strstr() 时间复杂度O(n^2)
.find() 时间复杂度O(m*n)
关于字符串匹配的效率 strstr 对比 KMP

KMP :
KMP算法最浅显理解——一看就明白
从头到尾彻底理解KMP

AC代码

#include <iostream>
#include <cstdio>
#include <cstring>
#define mst(a) memset(a, 0, sizeof(a))
using namespace std;
const int maxn = 100000 + 10;
char s1[maxn], s2[maxn], ss[maxn];
int nxt[maxn];

bool kmp(int n, char* a, int m, char* b){
    int i, j;
    for( nxt[0] = j = -1, i = 1; i < n; nxt[i++] = j ){
        while(~j && a[j+1] != a[i]) j = nxt[j];
        if(a[j+1] == a[i]) j++;
    }
    for( j = -1, i = 0; i < m; i++ ){
        while(~j && a[j+1] != b[i]) j = nxt[j];
        if(a[j+1] == b[i]) j++;
        if(j == n-1){
            return true;
            j = nxt[j];
        }
    }
    return false;
}

int main()
{
    int T;
    scanf("%d",&T);
    while(T--){
        scanf("%s%s",s1, s2);
        int len1 = strlen(s1), len2 = strlen(s2);
        if( strcmp(s2,"0") == 0 ){
            printf("Alice\n");
            continue;
        }
        if( strcmp(s1, s2) == 0 ){
            printf("Alice\n");
            continue;
        }
        if( len1 < len2 ){
            printf("Bob\n");
            continue;
        }
        if( kmp(len2, s2, len1, s1) )
            printf("Alice\n");
        else{
            int i, j;
            for( i = len2-1, j = 0; i >= 0; i--, j++ )
                ss[j] = s2[i];
            ss[j] = '\0';
            //printf(" %s\n",ss);
            if( kmp(len2, ss, len1, s1) )
                printf("Alice\n");
            else
                printf("Bob\n");
        }
    }
    return 0;
}
posted @ 2018-04-09 22:28  JinxiSui  阅读(159)  评论(0编辑  收藏  举报