zoukankan      html  css  js  c++  java
  • 2018 ACM-ICPC 中国大学生程序设计竞赛线上赛 H题 Rock Paper Scissors Lizard Spock.(FFT字符串匹配)

    2018 ACM-ICPC 中国大学生程序设计竞赛线上赛:https://www.jisuanke.com/contest/1227

    题目链接:https://nanti.jisuanke.com/t/26219

    Rock Paper Scissors Lizard Spock

    Description:

    Didi is a curious baby. One day, she finds a curious game, which named Rock Paper Scissors Lizard Spock.

    The game is an upgraded version of the game named Rock, Paper, Scissors. Each player chooses an option . And then those players show their choices that was previously hidden at the same time. If the winner defeats the others, she gets a point.

    The rules are as follows. 

    Scissors cuts Paper

    Paper covers Rock

    Rock crushes Lizard

    Lizard poisons Spock

    Spock smashes Scissors

    Scissors decapitates Lizard

    Lizard eats Paper

    Paper disproves Spock

    Spock vaporizes Rock

    (and as it always has) Rock crushes Scissors.

    (this pic is from baike.baidu.com)

    But Didi is a little silly, she always loses the game. In order to keep her calm, her friend Tangtang writes down the order on a list and show it to her. Didi also writes down her order on another list, like

    .

    (Rock-R Paper-P Scissors-S Lizard-L Spock-K)

    However, Didi may skip some her friends' choices to find the position to get the most winning points of the game, like

     

    Can you help Didi find the max points she can get?

    Input:

    The first line contains the list of choices of Didi's friend, the second line contains the list of choices of Didi.

    (1<=len(s2)<=len(s1)<=1e6)

    Output:

    One line contains an integer indicating the maximum number of wining point.

    忽略每行输出的末尾多余空格

    样例输入1

    RRRRRRRRRLLL
    RRRS

    样例输出1

    3

    样例输入2

    RSSPKKLLRKPS
    RSRS

    样例输出2

    2
    ACM-ICPC Asia Training League   宁夏理工学院

    题解:

    因为之前做过codeforces 528D. Fuzzy Search  ,感觉就不难了,你要是不会这题可以先去做cf528d,有个详细的题解:https://blog.csdn.net/u013368721/article/details/45565729

    【FFT求字符串匹配的问题一般都是将模式串反转,然后将其与主串进行卷积运算】

    枚举五种出拳方式,每种都做fft,最后扫一遍最大值即可求出最佳匹配出的赢的最大次数。(具体fft原理不懂orz,我就是套着原来板子写的...)

    #include<bits/stdc++.h>
    #define CLR(a,b) memset((a),(b),sizeof((a)))
    using namespace std;
    typedef long long ll;
    const int N = 1<<20;
    const double PI = acos(-1.0);
    int n, m;
    struct Complex {
        double x,y;
        Complex(double _x = 0.0,double _y = 0.0){
            x = _x; y = _y;
        }
        Complex operator -(const Complex &b)const{
            return Complex(x-b.x,y-b.y);
        }
        Complex operator +(const Complex &b)const{
            return Complex(x+b.x,y+b.y);
        }
        Complex operator *(const Complex &b)const{
            return Complex(x*b.x-y*b.y,x*b.y+y*b.x);
        }
        Complex operator * (const double &b)const{
            return Complex(x * b,y * b);
        }
        Complex operator / (const double &b)const{
            return Complex(x / b,y / b);
        }
    };
    void change(Complex y[], int len) {
        int i, j, k;
        for(i = 1, j = len/2;i <len-1;i++) {
            if(i < j)swap(y[i],y[j]);
            k = len/2;
            while(j >= k) {
                j -= k;
                k /= 2;
            }
            if(j < k) j += k;
        }
    }
    void fft(Complex y[],int len,int on) {
        change(y,len);
        for(int h = 2; h <= len; h <<= 1) {
            Complex wn(cos(-on*2*PI/h),sin(-on*2*PI/h));
            for(int j = 0;j < len;j+=h) {
                Complex w(1,0);
                for(int k = j;k < j+h/2;k++) {
                    Complex u = y[k];
                    Complex t = w*y[k+h/2];
                    y[k] = u+t;
                    y[k+h/2] = u-t;
                    w = w*wn;
                }
            }
        }
        if(on == -1)
            for(int i = 0;i < len;i++)
                y[i].x /= len;
    }
    Complex a[N], b[N], c[N];
    char s[N], t[N];
    int sum[N];
    int main() {
        int i, j, ans = 0, ma, nn;
        scanf("%s %s", s, t);
        n = strlen(s);
        m = strlen(t);
        reverse(t, t+m);
        ma = max(n, m);  nn = 1;
        while(nn < 2 * ma) nn<<=1;
        CLR(c, 0); CLR(sum, 0);
        //R vs L S
        CLR(a, 0); CLR(b, 0);
        for(i = 0; i < n; ++i) a[i].x = (s[i]=='L'||s[i]=='S');
        for(i = 0; i < m; ++i) b[i].x = (t[i]=='R');
        fft(a, nn, 1); fft(b, nn, 1);
        for(i = 0; i < nn ;++i) c[i] = a[i] * b[i];
        fft(c, nn, -1);
        for(i = m-1; i < n; ++i)
            sum[i] += (int)(c[i].x+0.5);
        //P vs R K
        CLR(a, 0); CLR(b, 0);
        for(i = 0; i < n; ++i) a[i].x = (s[i]=='R'||s[i]=='K');
        for(i = 0; i < m; ++i) b[i].x = (t[i]=='P');
        fft(a, nn, 1); fft(b, nn, 1);
        for(i = 0; i < nn ;++i) c[i] = a[i] * b[i];
        fft(c, nn, -1);
        for(i = m-1; i < n; ++i)
            sum[i] += (int)(c[i].x+0.5);
        //S vs P L
        CLR(a, 0); CLR(b, 0);
        for(i = 0; i < n; ++i) a[i].x = (s[i]=='P'||s[i]=='L');
        for(i = 0; i < m; ++i) b[i].x = (t[i]=='S');
        fft(a, nn, 1); fft(b, nn, 1);
        for(i = 0; i < nn ;++i) c[i] = a[i] * b[i];
        fft(c, nn, -1);
        for(i = m-1; i < n; ++i)
            sum[i] += (int)(c[i].x+0.5);
        //L vs P K
        CLR(a, 0); CLR(b, 0);
        for(i = 0; i < n; ++i) a[i].x = (s[i]=='P'||s[i]=='K');
        for(i = 0; i < m; ++i) b[i].x = (t[i]=='L');
        fft(a, nn, 1); fft(b, nn, 1);
        for(i = 0; i < nn ;++i) c[i] = a[i] * b[i];
        fft(c, nn, -1);
        for(i = m-1; i < n; ++i)
            sum[i] += (int)(c[i].x+0.5);
        //K vs R S
        CLR(a, 0); CLR(b, 0);
        for(i = 0; i < n; ++i) a[i].x = (s[i]=='R'||s[i]=='S');
        for(i = 0; i < m; ++i) b[i].x = (t[i]=='K');
        fft(a, nn, 1); fft(b, nn, 1);
        for(i = 0; i < nn ;++i) c[i] = a[i] * b[i];
        fft(c, nn, -1);
        for(i = m-1; i < n; ++i)
            sum[i] += (int)(c[i].x+0.5);
        for(i = m-1; i < n; ++i) ans = max(ans, sum[i]);
        printf("%d
    ", ans);
        return 0;
    }
  • 相关阅读:
    Android之Margin和Padding属性及支持的长度单位
    java jvm eclipse 性能调优
    spring aop 内部方法调用事务不生效问题解决
    服务器 获取用户 真实ip
    Nginx gzip配置
    全局唯一的支付和订单id生成算法
    spring aop 方法增加日志记录
    linux cp复制文件 直接覆盖
    Twitter分布式自增ID算法snowflake原理解析
    nginx 命令
  • 原文地址:https://www.cnblogs.com/GraceSkyer/p/8921703.html
Copyright © 2011-2022 走看看