scu-3296: Windy's S

http://cstest.scu.edu.cn/soj/problem.action?id=3296

最小表示法

// File Name: 3296.cpp
// Author: bo_jwolf
// Created Time: 2013年08月14日 星期三 17时30分27秒

#include<vector>
#include<list>
#include<map>
#include<set>
#include<deque>
#include<stack>
#include<bitset>
#include<algorithm>
#include<functional>
#include<numeric>
#include<utility>
#include<sstream>
#include<iostream>
#include<iomanip>
#include<cstdio>
#include<cmath>
#include<cstdlib>
#include<cstring>
#include<ctime>

using namespace std;
int MinimumRepresentation(char *s, int len)
{
    int i = 0, j = 1, count = 0, t;
    while (i < len && j < len && count < len)
    {
       int  x = i + count;
       int  y = j + count;
 
        if (x >= len) x -= len;          //用减法代替求余 
        if (y >= len) y -= len;          //用减法代替求余
 
        if (s[x] == s[y])
            count++;
        else
        {
            if (s[x] > s[y])
            {
                i = i + count + 1;
                j = i + 1;         /*将 j 拖至 i + 1 的地方*/
            }
            else
                j = j + count + 1;
 
            if (i == j) j++;
            count = 0;
        }
    }
 
    return i;                    //直接return i即可 
}
const int MAXN=110000;
char a[MAXN];

int main() 
{
    int n ; 
    scanf( "%d" , &n ) ;
    while( n-- )
    {
        scanf( "%s" , a ) ;
        int len = strlen( a ) ;
        int ans = MinimumRepresentation( a , len ) ;
    //    cout << ans << endl << len << endl ;
        for( int i = 0 ; i < len ; ++i )
            printf( "%c" , a[ ( ans + i ) % len ] ) ;
        cout << endl ;
    }
    return 0;
}


#include<cstdio>
#include<iostream>
#include<algorithm>
#include<sstream>
#include<cstdlib>
#include<cstring>
#include<string>
#include<climits>
#include<cmath>
#include<queue>
#include<vector>
#include<stack>
#include<set>
#include<map>
#define INF 0x3f3f3f3f
#define eps 1e-8
using namespace std;
int MinimumRepresentation(char *a, int n)
{
     int ans ; 
     for(int i = 0 , j = 1 , k = 0; k < n && j+k<2*n; ) 
     {  
        if(a[(i+k)%n]>a[(j+k)%n]) 
        {  
            i=max(i+k+1,j);  
            j=i+1;  
            k=0;  
        } 
        else if(a[(i+k)%n]<a[(j+k)%n]) 
        {  
             j=j+k+1;  
             k=0;  
        }
        else 
        {  
             k++;  
        }  
            ans=i;  
    }  
    return ans;                    //直接return i即可 
}
const int MAXN=110000;
char a[MAXN];

int main() 
{
    int n ; 
    scanf( "%d" , &n ) ;
    while( n-- )
    {
        scanf( "%s" , a ) ;
        int len = strlen( a ) ;
        int ans = MinimumRepresentation( a , len ) ;
    //    cout << ans << endl << len << endl ;
        for( int i = 0 ; i < len ; ++i )
            printf( "%c" , a[ ( ans + i ) % len ] ) ;
        cout << endl ;
    }
    return 0;
}


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值