剑指offer-42-翻转单词顺序--左旋转字符串

题目一

输入一个一个英文句子,翻转句子中单词的顺序,
但单词内字符的顺序不变。标点符号和普通字母一样处理,
例如:输入"I am a student." ,则输出"student.a am I"。

接下来看一下解题思路:
    首先根据空格将句子分割成单个单词,
* 然后翻转数组,再将单词拼接起来
代码示例:

public String reverseSentence(String str) {
        if (str.length() <= 1 || str.trim().equals("")) {
            return str;
        }
        String[] spit = str.split(" ");
        String res = "";
        for (int i = spit.length - 1; i >= 0; i--) {
            res += (i == 0) ? spit[i] : (spit[i] + " ");
        }

        return res;
    }
总结

    拼接字符串的时候注意空格,上面通过一个三目运算来拼接空格。
    解决了这个问题,我们来进一步的看一下下面这道题。

题目二

对于一个给定的字符序列S,请你把其循环左移K位后的序列输出。
例如,字符序列S=”abcXYZdef”,要求输出循环左移3位后的结果,即“XYZdefabc”。

    可以从第一个问题中受到启发,也是通过翻转字符的方法来解决。

思路一

    左移几位就从第几位将字符串分割成两部分;
* 分别对这两部分字符串翻转,
* 然后再翻转整个字符串,就可以得到左移后的字符串了。

代码示例:

public String leftRotateString2(String str, int n) {
        if (str.length() <= 0 || n <=0) {
            return str;
        }

        char[] chars = str.toCharArray();
        //翻转第一部分的字符串
        reverse(chars, 0, n - 1);
        //翻转第二部分的字符串
        reverse(chars, n, chars.length - 1);
        //翻转整个字符串
        reverse(chars, 0, chars.length - 1);
        String res = "";
        //将每个字符拼接起来
        for (char aChar : chars) {
            res += aChar;
        }

        return res;
    }

    /**
     * 翻转字符
     * @param chars
     * @param low
     * @param high
     */
    private void reverse(char[] chars, int low, int high) {
         while (low < high) {
             char tmp = chars[low];
             chars[low] = chars[high];
             chars[high] = tmp;
             low++;
             high--;
         }
}
总结

    使用这种方法需要对一个字符串做三次翻转,效率不高。其实java里面对String的处理做的很好,可以用String的方法来解决此问题。

思路二

    通过拆分和拼接字符串
* 例如:若是要左移3位,则将该字符串下标为3之前的拆分拼接到该字符的后面
代码示例:

public String leftRotateString(String str, int n) {
        if (str.length() <= 0 || n <=0) {
            return str;
        }
        //可能出现左移位数大于字符串长度的情况
        n = n % str.length();
        //截取0-n的字符串
        String str1 = str.substring(0, n);
        //截取n之后的字符串
        String str2 = str.substring(n);

        return str2 + str1;
    }
总结

     这种方法对字符串做了两次截取操作,还可以用下面这种方法对字符串只做一次截取操作。

思路三

    将两个相等的字符串拼接成一个字符串
* 左移几位就从第几位开始截取原字符串长度的字符串
代码示例:

public String leftRotateString1(String str, int n) {
        if (str.length() <= 0 || n <=0) {
            return str;
        }
        int len = str.length();
        //可能出现左移位数大于字符串长度的情况
        n = n % len;
        //增长字符串以便后面的截取
        str += str;

        return str.substring(n, n + len);
    }
总结

     这种方法虽然比较巧妙,但是对字符串做了多余操作。例如给定字符串"abcdef",只是左移一位,只需要将‘a’ ,移到 “bcdef”后面就好,但是这种方法先拼接一个"abcdefabcdef"字符串,然后截取"bcdefa"部分,如果字符串比较长还是比较耗费空间的。

你可能感兴趣的:(算法,Java,字符串)