最小(大)表示法模板总结

最小(大)表示法主要是用来得到字符串循环移位后字典序最小(大)的字符串。
例如,给你一个字符串 s = “SKYLONG”,然后我们将这个字符串一步步地往左移:
SKYLONG 1
KYLONGS 2
YLONGSK 3
LONGSKY 4
ONGSKYL 5
NGSKYLO 6
GSKYLON 7

可以发现字典序最小的字符串是7,字典序最大的字符串是3,最小(大)表示法就是在O(n)的复杂度内得到这些信息的算法。

由于这个算法模板性较强,就直接记录模板了:

//最小表示法
int get_minstring(char *s)
{
    int len = strlen(s);
    int i = 0, j = 1, k = 0;//最好字符串下标从0开始,这样下面取模方便!!! 
    while(i<len && j<len && k<len)
    {
        int t = s[(i+k)%len]-s[(j+k)%len];
        if(t == 0)
            k++;
        else
        {
            if(t > 0)
                i += k+1;
            else
                j += k+1;
            if(i == j) 
				j++;//相等时比较指针后移
            k = 0;
        }
    }
    return min(i, j);//返回i,j中最小的
}

//最大表示法
int get_maxstring(char *s)
{
    int len = strlen(s);
    int i = 0, j = 1, k = 0;
    while(i<len && j<len && k<len)
    {
        int t = s[(i+k)%len]-s[(j+k)%len];
        if(t == 0)
            k++;
        else
        {
            if(t > 0)//与最小表示法只有这里存在差异!!! 
                j += k+1;
            else
                i += k+1;
            if(i == j) 
				j++;//相等时比较指针后移
            k = 0;
        }
    }
    return min(i, j);//返回i,j中最小的
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值