字符串

字符串

1、反转字符串

题目链接:力扣
描述
编写一个函数,其作用是将输入的字符串反转过来。输入字符串以字符数组 s 的形式给出。
不要给另外的数组分配额外的空间,你必须原地修改输入数组、使用 O(1) 的额外空间解决这一问题。
示例

输入:s = ["h","e","l","l","o"]
输出:["o","l","l","e","h"]

思路

使用两个指针分别指向数组的第一个和最后一个元素,两个指针同时向中间移动,依次交换数值

实现代码
方法一:

使用一个额外空间temp 来交换数值

class Solution {
    public void reverseString(char[] s) {
        int left = 0;
        int right = s.length-1;
        while(left < right ){
            char temp = s[left];
            s[left] = s[right];
            s[right] = temp;
            left++;
            right--;
        }
    }
}

方法二:

位运算 ^ 原地交换两个数 (a , b)
参与运算的两个值,如果两个相应bit位相同,则结果为0,否则为1。
即: 0^0 = 0, 1^0 = 1, 0^1 = 1, 1^1 = 0
任意一个变量a与其自身进行异或运算,结果为0,即a^a=0。
任意一个变量a与0进行异或运算,结果不变,即 a^0=a。
异或运算具有可结合性 异或运算具有可交换性

step1: a = a ^ b (完成后 a 变量的结果,代入step2)
step2: b = a ^ b (b = a ^ b = a ^ b ^ b = a ^ 0 = a)
经过前两步,a = a ^ b , b = a;
step3: a = a ^ b (a = a ^ b = a ^ b ^ a = a ^ a ^ b = 0 ^ b = b)

class Solution {
    public void reverseString(char[] s) {
        int left = 0;
        int right = s.length-1;
        while(left < right ){
            s[left]  ^= s[right];
            s[right] ^= s[left];
            s[left]  ^= s[right];
            left++;
            right--;
        }
    }
}

2、反转字符串 II

题目链接: 力扣
描述
给定一个字符串 s 和一个整数 k,从字符串开头算起,每计数至 2k 个字符,就反转这 2k 字符中的前 k 个字符。
如果剩余字符少于 k 个,则将剩余字符全部反转。
如果剩余字符小于 2k 但大于或等于 k 个,则反转前 k 个字符,其余字符保持原样。
示例

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

思路

使用 l 和 r 两个指针分别圈出每次需要翻转的范围,每次翻转完更新 l 和 r,同时注意范围 [l,r] 内不足 k 个的情况(将 r 与真实边界 n - 1取个 min)。

实现代码

class Solution {
    public String reverseStr(String s, int k) {
        char[] ch = s.toCharArray();
        for (int i = 0; i < ch.length-1; i += 2*k){
            int l = i;
            int r = Math.min(l+k-1, ch.length-1);
            while(l < r){
                ch[l] ^= ch[r];
                ch[r] ^= ch[l];
                ch[l] ^= ch[r];
                l++;
                r--;
            }
        }
        return String.valueOf(ch);
    }
}

3、字符串变形

题目链接:牛客
描述
对于一个长度为 n 字符串,我们需要对它做一些变形。
首先这个字符串中包含着一些空格,就像"Hello World"一样,然后我们要做的是把这个字符串中由空格隔开的单词反序,同时反转每个字符的大小写。
比如"Hello World"变形后就变成了"wORLD hELLO"。
输入描述:
给定一个字符串s以及它的长度n(1 ≤ n ≤ 10^6)
返回值描述:
请返回变形后的字符串。题目保证给定的字符串均由大小写字母和空格构成。

示例

输入:"This is a sample",16
返回值:"SAMPLE A IS tHIS"

思路

step 1:遍历字符串,反转每个字符的大小写。
step 2:第一次反转整个字符串,这样基本的单词逆序就有了,但是每个单词的字符也是逆的。
step 3:再次遍历字符串,以每个空间为界,将每个单词反转回正常。

实现代码

public class Solution {
    /**
     * 代码中的类名、方法名、参数名已经指定,请勿修改,直接返回方法规定的值即可
     * @param s string字符串 
     * @param n int整型 
     * @return string字符串
     */
    public String trans (String s, int n) {
        // write code here
        char[] ch = s.toCharArray();
        for(int i = 0; i < n; i++){
            if('A'<=ch[i]&&ch[i]<='Z') ch[i] = (char)(ch[i] - 'A' + 'a');
            else if('a'<=ch[i]&&ch[i]<='z') ch[i] = (char)(ch[i] - 'a' + 'A');
        }
        // 先整体翻转
        reverseArr(ch, 0, n-1);
        // 此时每个单词也逆序了,需要根据空格将每个单元重新反转回去
        for(int i=0; i<n; i++){
            if(ch[i] == ' ') continue;
            int j=i;
            while(j<n && ch[j]!=' ') j++;
            reverseArr(ch, i, j-1);
            i=j;
        }
        return String.valueOf(ch);
    }
    // 字符串数组翻转函数
    public void reverseArr(char[] arr, int i, int j){
        int l = i;
        int r = j;
        while(l < r){
            char temp = arr[l];
            arr[l] = arr[r];
            arr[r] = temp;
            l++;
            r--;
        }
    }
}

4、验证IP地址

题目链接: 牛客
描述
编写一个函数来验证输入的字符串是否是有效的 IPv4 或 IPv6 地址
IPv4 地址由十进制数和点来表示,每个地址包含4个十进制数,其范围为 0 - 255, 用(".")分割。比如,172.16.254.1;
同时,IPv4 地址内的数不会以 0 开头。比如,地址 172.16.254.01 是不合法的。
IPv6 地址由8组16进制的数字来表示,每组表示 16 比特。这些组数字通过 (":")分割。比如, 2001:0db8:85a3:0000:0000:8a2e:0370:7334 是一个有效的地址。而且,我们可以加入一些以 0 开头的数字,字母可以使用大写,也可以是小写。所以, 2001:db8:85a3:0:0:8A2E:0370:7334 也是一个有效的 IPv6 address地址 (即,忽略 0 开头,忽略大小写)。
然而,我们不能因为某个组的值为 0,而使用一个空的组,以至于出现 (:😃 的情况。 比如, 2001:0db8:85a3::8A2E:0370:7334 是无效的 IPv6 地址。
同时,在 IPv6 地址中,多余的 0 也是不被允许的。比如, 02001:0db8:85a3:0000:0000:8a2e:0370:7334 是无效的。
说明: 你可以认为给定的字符串里没有空格或者其他特殊字符。

示例1

输入:"172.16.254.1"
返回值:"IPv4"
说明:这是一个有效的 IPv4 地址, 所以返回 "IPv4"

示例2

输入:"2001:0db8:85a3:0:0:8A2E:0370:7334"
返回值:"IPv6"
说明:这是一个有效的 IPv6 地址, 所以返回 "IPv6"

备注

ip地址的类型,可能为
IPv4, IPv6, Neither

思路

分别验证IPv4和IPv6
Integer.valueOf(s,16)将16进制的 s 转换成10进制,如果抛异常则说明 s 不符合16进制

代码实现

public class Solution {
    /**
     * 代码中的类名、方法名、参数名已经指定,请勿修改,直接返回方法规定的值即可
     * 验证IP地址
     * @param IP string字符串 一个IP地址字符串
     * @return string字符串
     */
    public static String solve (String IP) {
        return isIPv4(IP) ? "IPv4" : isIPv6(IP) ? "IPv6" : "Neither";
    }

    public static boolean isIPv4(String IP) {
        String[] arr = IP.split("\\.", -1);
        if (arr.length != 4) {
            return false;
        }
        for (String s : arr) {
            //判断字符串是否以0开头
            if (s.startsWith("0")) {
                return false;
            }
            try {
                int num = Integer.valueOf(s);
                if (!(0 <= num && num <= 255)) {
                    return false;
                }
            } catch (NumberFormatException e) {
                return false;
            }

        }
        return true;
    }
    public static boolean isIPv6(String IP) {
        String[] arr = IP.split(":", -1);
        if (arr.length != 8) {
            return false;
        }
        for (String s : arr) {
            //每个分割不能缺省,不能超过4位
            if (s.length() > 4 || s.length() == 0) {
                return false;
            }
            try {
                int val = Integer.valueOf(s, 16);
            } catch (NumberFormatException e) {
                return false;
            }
        }
        return true;
    }
}
posted @ 2024-10-29 13:44  dwhere  阅读(16)  评论(0编辑  收藏  举报