最小(大)表示法主要是用来得到字符串循环移位后字典序最小(大)的字符串。
例如,给你一个字符串 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中最小的
}