左旋转字符串

作者: Phill King

邮箱: phillking1982@163.com

原创文章,转载请注明出处。

字符串的左旋转操作是把字符串前面的若干个字符转移到字符串的尾部。请定义一个函数实现字符串左旋转操作的功能。比如,输入字符串"abcdefg"和数字2,该函数将返回左旋转两位得到的结果"cdefgab"。

示例

输入: s = "abcdefg", k = 2
输出: "cdefgab"

解法一:

最直接的方法就是把字符串的前n位剪切到末尾。

 string reverseLeftWords(string s, int n) {
        return s.substr(n)+ s.substr(0,n);
    }

解法二:

str=AB,BA=($\mathbf{A}^\mathrm{T}$$\mathbf{B}^\mathrm{T}$)^\mathrm{T}

反转A,反转B,再反转str.

时间复杂厚度O(n), 空间复杂度O(1);

 void reverse(string& s, int left, int right){
        while(left<right){
            swap(s[left], s[right]);
            left++;
            right--;
        }
    }
    string reverseLeftWords(string s, int n) {
        if(s.empty()){
            return s;
        }
        int len = s.length();
        n = n%len;
        reverse(s, 0, n-1);
        reverse(s, n, len-1);
        reverse(s, 0, len-1);
        return s;

    }

解法三:

循环拷贝,时间复杂度O(n)空间复杂度O(1)。

考察字符串中每个字符转换前后的位置,可以发现两者是存在映射关系的。

 

输入: s = "abcdefg", k = 2

输出: cdefgba

备注:灰色部分是前面的字符,转移到字符串后方

旋转后:

我们把一整条链按先后顺序排列:a-c-e-g-b-d-f-a, 可以看到这是一个循环链, 其中后一个字符转换到了前一个字符所在的位置,如下图所示:

 

经过观察,我们可以看到(循环链长度)*(左移数)的长度是二者的最小公倍数;

例图里的长度为 7*2 = 14。

如果二者不互质,循环链可能有多条,数目为二者的最大公约数,每条循环链的长度为(字符串长度)/(循环链条数)。

例子:

输入 s= abcdef   k=3

输出:   defabc.

具体代码如下:

 string reverseLeftWords(string s, int n) {
        if(s.empty()){
            return s;
        }
        int len = s.length();
        n = n%len;
        int m = __gcd(len, n);   // number of chain
        int l = len/m;           // length of chain

        for(int i=0; i<m; i++){
            for(int j=0, pre = i; j<l-1; j++,pre+=n){
                swap(s[pre%len], s[(pre+n) %len]);
            }
        }
        return s;

    }

 

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值