SA例题

题面

对于串 \(S\) 的两个子串 \(A\) 和 \(B\) ,满足 \(k = |A| = |B|\),\(\exists c \forall i\, a_i + c=b_i\),且 \(A\) 和 \(B\) 在原串中没有重叠,求最大的满足条件的 \(k\)

先考虑 \(\forall i\, a_i=b_i\) 的情况,二分 \(k\) ,如果在\(height\)上存在一段满足 \(\forall L \le i \le R \,height_i\ge k\),且存在 \(SA[p_1]-SA[p_2] \ge k\),说明 \(k\) 满足条件

接下来将上述做法用在原数组的差分数组上即可,注意判定时条件要改为\(SA[p_1]-SA[p_2] > k\),且要在第一个字符前添加特殊字符,否则判定时可能出问题

#include<cstdio>
using namespace std;

const int MAXN = 5001;
int str[MAXN], sa[MAXN], rank[MAXN], ht[MAXN], buc[MAXN];

void Swap(int*& a, int*& b){
    int *tmp = a; a = b; b = tmp;
}

void getSA(int s[], int len){
    int *rk = rank, *tp = ht;
    int crd = 180, p;
    for (int i = 1; i <= len; ++i)
        rk[i] = s[i], tp[i] = i;
    for (int i = 0; i <= crd; ++i)
        buc[i] = 0;
    for (int i = 1; i <= len; ++i)
        ++buc[rk[i]];
    for (int i = 1; i <= crd; ++i)
        buc[i] += buc[i - 1];
    for (int i = len; i >= 1; --i)
        sa[buc[rk[tp[i]]]--] = tp[i];
    for (int w = 1; p != len; w <<= 1, crd = p){
        p = 0;
        for (int i = len - w + 1; i <= len; ++i)
            tp[++p] = i;
        for (int i = 1; i <= len; ++i)
            if (sa[i] > w)
                tp[++p] = sa[i] - w;
        for (int i = 0; i <= crd; ++i)
            buc[i] = 0;
        for (int i = 1; i <= len; ++i)
            ++buc[rk[i]];
        for (int i = 1; i <= crd; ++i)
            buc[i] += buc[i - 1];
        for (int i = len; i >= 1; --i)
            sa[buc[rk[tp[i]]]--] = tp[i];
        Swap(rk, tp);
        rk[sa[1]] = p = 1;
        for (int i = 2; i <= len; ++i)
            rk[sa[i]] = tp[sa[i]] == tp[sa[i - 1]] && tp[sa[i] + w] == tp[sa[i - 1] + w] ? p : ++p;
    }
    for (int i = 1; i <= len; ++i)
        rank[sa[i]] = i;
}

void getHeight(int s[], int len){
    ht[1] = 0;
    for (int i = 1, j = 0, pos; i <= len; ++i, --j){
        if (rank[i] == 1)
            continue;
        if (j < 0)
            j = 0;
        pos = sa[rank[i] - 1];
        while (s[i + j] == s[pos + j] && i + j <= len && pos + j <= len)
            ++j;
        ht[rank[i]] = j;
    }
}

bool judge(int k, int len){
    int lp, rp;
    for (int i = 1, j; i < len;){
        if (ht[i + 1] >= k){
            lp = rp = sa[i];
            j = i + 1;
            while (ht[j] >= k && j <= len){
                if (sa[j] < lp)
                    lp = sa[j];
                if (sa[j] > rp)
                    rp = sa[j];
                ++j;
            }
            if (rp - lp > k)
                return 1;
            i = j;
        }
        else
            ++i;
    }
    return 0;
}

int main(){
    int slen;
    scanf("%d", &slen);
    for (int i = 1, pre = 89, tmp; i <= slen; ++i){
        scanf("%d", &tmp);
        str[i] = tmp - pre + 89; pre = tmp;
    }
    getSA(str, slen);
    getHeight(str, slen);
    int L = 1, R = slen >> 1, mid;
    while (L + 1 < R){
        mid = L + R >> 1;
        if (judge(mid, slen))
            L = mid;
        else
            R = mid;
    }
    int ans;
    if (judge(R, slen))
        ans = R;
    else
        ans = L;
    if (ans < 4)
        printf("0");
    else
        printf("%d", ans + 1);
    return 0;
}

乐曲主题Musical Themes的更多相关文章

  1. 洛谷P2743 乐曲主题Musical Themes [USACO5.1] SA

    正解:SA 解题报告: 传送门 这题三个条件嘛,那就一个个考虑下都解决了就把这题解决了嘛QwQ 那就直接分别针对三个条件写下各个击破就欧克辣? 1)长度大于等于5:求出答案之后和5比大小 2)不能有公 ...

  2. Luogu P2743 [USACO5.1]乐曲主题Musical Themes

    链接 \(Click\) \(Here\) 人生第一道后缀数组的题目.首先要对输入的串进行差分处理,差分后长度为(\(n - 1\))的相同子段就是原串中长度为\(n\)的相同(可变调)子段.求出来\ ...

  3. [USACO5.1] 乐曲主题Musical Themes

    题目链接:戳我 Emmm......hash怎么做啊不会啊 这里是SA后缀数组版本的 就是先两两做差分,作为要处理后缀的数组.普通地求出来h数组之后,我们二分这个答案,然后判定是否合法就行了.是否合法 ...

  4. P2743(poj1743) Musical Themes[差分+后缀数组]

    P2743 乐曲主题Musical Themes(poj1743) 然后呢这题思路其实还是蛮简单的,只是细节特别多比较恶心,忘记了差分带来的若干疏漏.因为转调的话要保证找到相同主题,只要保证一段内相对 ...

  5. [转]jQuery EasyUI 扩展-- 主题(Themes)

    主题(Themes)允许您改变站点的外观和感观.使用主题可以节省设计的时间,让您腾出更多的时间进行开发.您也可以创建一个已有主题的子主题. 主题生成器(Theme Builder) jQuery UI ...

  6. USACO Section 5.1 Musical Themes(枚举)

    直接枚举O(n^3)会TLE,只要稍微加点优化,在不可能得到更优解时及时退出.其实就是道水题,虽说我提交了6次才过= =..我还太弱了 -------------------------------- ...

  7. USACO 5.1 Musical Themes(哈希+二分)

    Musical ThemesBrian Dean A musical melody is represented as a sequence of N (1 <= N <= 5000) n ...

  8. Android应用开发中的风格和主题(style,themes)

    http://www.cnblogs.com/playing/archive/2011/04/01/2002469.html 越来越多互联网企业都在Android平台上部署其客户端,为了提升用户体验, ...

  9. [USACO 5.1.3]乐曲主题

    Description 我们用N(1 <= N <=5000)个音符的序列来表示一首乐曲,每个音符都是1..88范围内的整数,每个数表示钢琴上的一个键.很不幸这种表示旋律的方法忽略了音符的 ...

随机推荐

  1. xamarin.IOS App闪退日志----crash

    一.查找日志文件位置:通过xcode/windows/device/你的IPhone/crash 二.拿到日志可以直接查看,但是日志记录太乱看不懂,需要转换处理,查找.DSYM文件,文件位置:/Use ...

  2. VC 读取服务器上的文件(HTTP方式) [转]

    CString GetStringFromUrl(LPCTSTR pszUrl){    CString str ;    HINTERNET hSession = ::InternetOpen( _ ...

  3. Python 自动登录网站(处理Cookie)

    http://digiter.iteye.com/blog/1300884 Python代码   def login():     cj = cookielib.CookieJar()     ope ...

  4. eclipse下的ssh框架整合过程及測试

    最近在搭建Stuts2+hibernate+spring的框架,网上看的教程,大部分都是非常easy的步骤.没有比較具体的步骤以及每一个步骤完毕之后怎样检查是否配置成功.下面是笔者依据自己搭建的过程进 ...

  5. 改变其他iframe的src

    window.parent.$("#ifr").location="????";);来改变

  6. python(34)- 模块与包练习

    创建如下目录结构 keystone/ ├── __init__.py └── auth     ├── __init__.py     └── plugins         └── core.py ...

  7. 【每日Scrum】第八天(4.29) TD学生助手Sprint2

    站立会议 组员 今天 签到 刘铸辉 (组长) 绩效考核 Y 刘静 测试用例书写 测试bug报告 测试详细报告 Y 解凤娇 Y 王洪叶 项目可行性报告 项目开发计划书 需求分析(已完成并发布) Y 胡宝 ...

  8. ubuntu安装交叉编译工具链

    一.sudo su 切换到root用户权限 二.将arm-linux-gcc-4.3.2.tgz从windows拷贝到Linux  /home/ttt/arm-linux-gcc-4.3.2.tgz ...

  9. Leetcode 002-Search Insert Position

    #Given a sorted array and a target value, return the index if the target is found. If not, return th ...

  10. ios6.0,程序为横屏,出现闪退

    本文转载至 http://blog.csdn.net/huanghuanghbc/article/details/10150355   ios6.0,程序为横屏,出现闪退 *** Terminatin ...