专栏:算法的魔法世界

个人主页:手握风云

目录

一、例题讲解

1.1. 最长公共前缀

1.2. 最长回文子串

1.3. 二进制求和

1.4. 字符串相乘


一、例题讲解

1.1. 最长公共前缀

        给你一个字符串数组,找出最长公共前缀。

        解法一:两两比较字符串,找出最长公共前缀。利用指针从头开始遍历字符串,如果遇到相同的字符,指针向后移动。当指针越界时,停止。

        完整代码实现:

class Solution {
    public String longestCommonPrefix(String[] strs) {
        int len = strs.length;
        if (strs == null || len == 0) {
            return "";
        }
        if (strs.length == 1) {
            return strs[0];
        }
        // 初始化公共前缀为第一个字符串
        String prefix = strs[0];
        // 遍历数组中的其他字符串,依次比较
        for (int i = 1; i < len; i++) {
            // 找出当前字符串与prefix的公共前缀
            prefix = commonPrefix(prefix, strs[i]);
            // 如果公共前缀为空,提前结束循环
            if (prefix.isEmpty()) {
                break;
            }
        }
        return prefix;
    }
    public String commonPrefix(String s1, String s2) {
        int minLen = Math.min(s1.length(), s2.length());
        int i = 0;
        while (i < minLen && s1.charAt(i) == s2.charAt(i)) {
            i++;
        }
        return s1.substring(0, i);
    }
}

        解法二:统一比较。也是利用指针遍历每一个字符串,如果都相同就像后移动,如果不同或者发生越界,停止移动。

        完整代码实现:

class Solution {
    public String longestCommonPrefix(String[] strs) {
        int len = strs.length;
         if (strs == null || len == 0) {
            return "";
        }
        // 遍历第一个字符串的每个字符
        for (int i = 0; i < strs[0].length(); i++) {
            char c = strs[0].charAt(i);
            // 检查其他字符串的相同位置
            for (int j = 1; j < len; j++) {
                // 如果其他字符串已经到达末尾,或者字符不匹配,返回当前的前缀
                if (i >= strs[j].length() || strs[j].charAt(i) != c) {
                    return strs[0].substring(0, i);
                }
            }
        }
        // 如果第一个字符串是所有字符串的前缀,返回第一个字符串
        return strs[0];
    }
}

1.2. 最长回文子串

        本题采用的是中心扩展。定义两个指针left和right,从回文串的中心分别向左右,当遇到不相同的字符或者两个指针越界时停止。要注意:如果子串长度为奇数,left=right;如果子串长度为偶数,right=left+1。

        完整代码实现:

class Solution {
    public String longestPalindrome(String s) {
        int n = s.length(), start = 0, end = 0;
        for (int i = 0; i < n; i++) {
            // 奇数长度回文串
            int len1 = expandCenter(s, i, i);
            // 偶数长度回文串
            int len2 = expandCenter(s, i, i + 1);
            int len = Math.max(len1, len2);
            if (len > end - start) {
                start = i - (len - 1) / 2;
                end = i + len / 2;
            }
        }
        return s.substring(start, end + 1);
    }
    private int expandCenter(String s, int left, int right) {
        // 从中心向两边扩展,直到不是回文串
        while (left >= 0 && right < s.length() && s.charAt(left) == s.charAt(right)) {
            left--;
            right++;
        }
        // 返回回文串长度
        return right - left - 1;
    }
}

1.3. 二进制求和

        给定两个仅由字符‘0’或‘1’组成的二进制字符串a和b,以二进制字符串的形式返回两数之和。

        本题就是一个模拟手算加法的过程。初始化一个进位carry为 0,结果字符串ret为空。从两个字符串的末尾开始向前遍历,计算当前位上的和,满2向前进1位。遍历结束后,如果还有进位,将进位添加到结果字符串的前面。

        完整代码实现:

class Solution {
    public String addBinary(String a, String b) {
        StringBuilder ret = new StringBuilder();
        int i = a.length() - 1, j = b.length() - 1, carry = 0;
        // 当至少还有一个字符串未遍历完时,继续循环
        while (i >= 0 || j >= 0) {
            int aBit = (i >= 0) ? a.charAt(i--) - '0' : 0;
            int bBit = (j >= 0) ? b.charAt(j--) - '0' : 0;
            // 计算当前位的和
            int sum = aBit + bBit + carry;
            // 将当前位的结果添加到结果中
            ret.append(sum % 2);
            // 更新进位
            carry = sum / 2;
        }
        // 处理最后的进位
        if (carry > 0) {
            ret.append(carry);
        }
        return ret.reverse().toString();
    }
}

1.4. 字符串相乘

        给定两个以字符串形式表示的非负整数num1和num2,计算它们的乘积,并将结果以字符串形式返回,不能直接将输入的字符串转换为整数。

        本题考查的是模拟列竖式相乘:利用其中一个字符串的最后一位开始向前遍历,与另一个字符串每一位进行相乘,满10向前进一位,最后相加得到结果。这里需要注意:高位相乘需要在后面补0,为了方便我们可以将字符串num1、num2,最后的返回结果也得逆序;还需要处理前导"0",比如"123"*"0"="0"(清除前面的"0")。

        上面的代码写起来太长,我们可以进行一些优化。我们还是将其中一个字符串的最后一位开始向前遍历,与另一个字符串每一位进行相乘,乘出来的结果不进行进位,最后再相加进位。

        因为乘出来的结果暂时不进位,所以不知道有多少数位,我们可以用一个数组来存储。

        完整代码实现:

class Solution {
    public String multiply(String num1, String num2) {
        if (num1.equals("0") || num2.equals("0")) {
            return "0";
        }
        int m = num1.length(), n = num2.length();
        int[] res = new int[m + n];
        // 从个位数开始逐位相加
        for (int i = m - 1; i >= 0; i--) {
            for (int j = n - 1; j >= 0; j--) {
                int mul = (num1.charAt(i) - '0') * (num2.charAt(j) - '0');
                int p1 = i + j; // 高位
                int p2 = i + j + 1; // 地位
                int sum = mul + res[p2];
                res[p2] = sum % 10; // 低位直接赋值
                res[p1] += sum / 10; // 高位加进位
            }
        }
        // 结果数组转为字符串
        StringBuffer buffer = new StringBuffer();
        for (int digit : res) {
            if (buffer.length() != 0 || digit !=0) {
                buffer.append(digit);
            }
        }
        return buffer.toString();
    }
}

posted on 2025-09-28 09:59  ycfenxi  阅读(7)  评论(0)    收藏  举报