LeetCode 精华75题

LeetCode 75
1768. 交替合并字符串 - 力扣(LeetCode)
题目描述
两个字符串,每个字符串都拆成字符,交替拼接,长的字符串(剩余的字符) 拼接其后。
思路
● 根据短字符的长度进行遍历,最后拼接上长字符的剩余字符
class Solution {
public String mergeAlternately(String word1, String word2) {
int minSize = Math.min(word1.length(),word2.length());

    StringBuilder res = new StringBuilder();
    
    for(int i = 0 ;i < minSize ;i ++){
        res.append(word1.charAt(i));
        res.append(word2.charAt(i));
    }
    // 处理剩余字符
    if(word1.length() > minSize){
        res.append(word1.substring(minSize));
    }
    // 处理剩余字符
    if(word2.length() > minSize){
         res.append(word2.substring(minSize));
    }
    return res.toString();
    
}

}
1071. 字符串的最大公因子 - 力扣(LeetCode)
题目描述
给出两个字符串,找出最长的公因子(比如 str1 = “ABCABC”, str2 = “ABC”, 则最长公因子是“ABC”)

思路
● 最长公因子,最长也就是短串
优化思路
● str1 + str2 = str2 + str1 的时候才会有解,因为有解的情况下,str1可以被除尽,str2也可以被除尽,当然str1+str2 可以被除尽。
● 如果存在这样的字符串x,那么x的长度必须是两个字符串长度的公约数。

class Solution {
public String gcdOfStrings(String str1, String str2) {
// 检查两个字符串连接后是否相等 // 假设str1是N个x,str2是M个x,那么str1+str2肯定是等于str2+str1的。
if (!(str1 + str2).equals(str2 + str1)) {
return “”;
}
// 计算两个字符串长度的最大公约数
int gcdLength = gcd(str1.length(), str2.length());

    // 返回最大公约数长度的前缀子串
    return str1.substring(0, gcdLength);
}

// 计算最大公约数的辅助方法
private int gcd(int a, int b) {
    while (b != 0) {
        int temp = b;
        b = a % b;
        a = temp;
    }
    return a;
}

}

  1. 拥有最多糖果的孩子 - 力扣(LeetCode)
    题目描述
    一群孩子每个有都拥有n个糖果,现在你有m个糖果,你选择将糖果给某个孩子,判断该孩子是不是拥有最多的糖果数

思路
● 先找出最多拥有的糖果数
● 遍历如果将自己的糖果全部给该孩子,是否小于最多的糖果数。
class Solution {
public List kidsWithCandies(int[] candies, int extraCandies) {
int temp = 0;
for(int i = 0; i < candies.length; i ++){
if(candies[i] >= temp){
temp = candies[i];
}
}

    List res = new ArrayList<Boolean>(candies.length);
     for(int i = 0; i < candies.length; i ++){
        if(candies[i] + extraCandies >= temp){
            res.add(true);
        }else{
            res.add(false);
        }
    }

    return res;
}

}

  1. 种花问题 - 力扣(LeetCode)
    题目描述
    一个数组,已经存在一部分元素(不能相邻存在元素),要求插入n个元素,但是不能相邻,判断该n个元素是否能插入。

思路
● 根据数组大小判断最多可以存多少元素
○ 比如说偶数组长度n,最多只能存 n/2.
○ 奇数,可以存n/2 或者 n/ 2 + 1.(根据第一个元素)------有问题不能这莫分析

class Solution {
public boolean canPlaceFlowers(int[] flowerbed, int n) {
int cnt = 0; // 计数
for(int i = 0; i< flowerbed.length;i ++){
if(flowerbed[i] == 1)
{
cnt ++;
}
}
// 最多可以存
//偶数
int max= 0;
if(flowerbed.length % 2 == 0){
max = flowerbed.length / 2;
}
// 奇数
else{
if(flowerbed[0] == 1){
max = flowerbed.length/ 2 +1;
}
else{
max = flowerbed.length / 2;
}
}
if(cnt + n > max){
return false;
}
return true;
}
}
错误

分析:思路有问题,我上述的设想奇数长度的大小,最多可以判断只有在开头种植了才能+1,忽略了结尾,比如上述。
经过实验,对于过一个数组,间隔插入的最大值: 偶数长度 n/2, 奇数长度 (n+1)/2.
但是对于已经存在数据的数组,就不能按上述计算,因为,插入的位置同时可以在偶数位置并且在奇数位置上。

优化思路
● 查找数组的连续0
● 连续0在中间 最多可以插入 (n -1)/ 2
● 连续0 在两边, 最多可以插入 n / 2
如下图
连续0在数组的中间的情况:

连续0在数组两边的情况:

优化思路
● 贪心,能种就种

class Solution {
public boolean canPlaceFlowers(int[] flowerbed, int n) {
int count = 0;
int i = 0;
while (i < flowerbed.length) {
if (flowerbed[i] == 0) {
boolean canPlace = (i == 0 || flowerbed[i - 1] == 0) && // 左边没有花或是数组开头
(i == flowerbed.length - 1 || flowerbed[i + 1] == 0); // 右边没有花或是数组结尾

            if (canPlace) {
                flowerbed[i] = 1; // 种花
                count++;
                if (count >= n) {
                    return true;
                }
                i += 2; // 跳过下一个位置,因为不能相邻
                   // 检查是否能种花,左右都没中种花就可以种花,考虑边界情况,对于边界开头,只需要确保其后没有种花,对于结尾的边界,只需要前面没有种花

            //开头边界
            boolean first = (i==0) && (flowerbed[i+1] == 0);

            //结尾边界
            boolean end =(i == flowerbed.length -1) && (flowerbed[i-1] == 0);

            // 中间
            boolean center = (flowerbed[i-1] ==0 && flowerbed[i + 1] == 0);         } else {
                i++;
            }
        } else {
            i += 2; // 已经有花,跳过下一个位置
        }
    }
    return count >= n;
}

}
边界问题处理了好久,看一下我最初的处理边界
// 检查是否能种花,左右都没中种花就可以种花,考虑边界情况,对于边界开头,只需要确保其后没有种花,对于结尾的边界,只需要前面没有种花

//开头边界
// boolean first = (i==0) && (flowerbed[i+1] == 0);
if(i == 0){
if(flowerbed[i+1] == 0){
first = true;
}
}

            //结尾边界
            // boolean end =(i == flowerbed.length -1) && (flowerbed[i-2] == 0);
            else if(i == flowerbed.length -1){
                if(flowerbed[i-1] == 0){
                    end = true;
                }
            }else{
                 // 中间
             center = (flowerbed[i-1] ==0 && flowerbed[i + 1] == 0);
            }

还是下标溢出,原因是,假如数组长度就1,在判断开头的时候就i+1,下标溢出

  1. 反转字符串中的元音字母 - 力扣(LeetCode)
    题目描述
    一个字符串,将其中的元音字母,反转(交换位置)输出结果字符串。

思路
● 双指针,找到元音字母相互交换位置。
class Solution {
public String reverseVowels(String s) {

    // 元音数组
    Set<Character> vowels = new HashSet<>(
        Arrays.asList('a', 'e', 'i', 'o', 'u', 'A', 'E', 'I', 'O', 'U')
    );

    // sting 不可变---》转成数组
    char [] check = s.toCharArray();

    int l = 0, r = check.size - 1;
  
    while(l < r){ 
        // 从左向右找元音
        while(l < r && !vowels.contains(check[l])){
            l++;
        }
         // 从左向右找元音
        while(l < r && !vowels.contains(check[r])){
            r --;
        }

        // 交换元素

        if( l < r ){
            char t = check[l];
            check[l] = check[r];
            check[r] = t;
            l++;
            r--;
        }


    }
    return new String(check);

}

}

  1. 反转字符串中的单词 - 力扣(LeetCode)
    题目描述
    反转单词 一个字符串,包含多个单词,反转单词的顺序。

题目思路
● 识别单词,存入list,倒叙拼接。
● 主要是处理空格,我们可以使用split()根据空格进行分割,但是假如多个空格分割,值是有空格的?那我存的时候判断一下。

实现有误
class Solution {
public String reverseWords(String s) {
String[] split =s.split(" ");
StringBuilder result = new StringBuilder();
for(int i = split.length - 1 ;i >=0 ;i --){
if(split[i] != " "){
// 最后一个不加空格
if(i == 0){
result.append(split[i]);
}
// 其他的后边要加一个空格
else{
result.append(split[i] + " ");
}
}
}
return result.toString();
}
}
原因如下: 在处理最后一个单词的时候,判断要不要加空格,是根据分割后的下标来判断的,这种逻辑是错误的
tips:比较字符串 不要直接使用 == != 而是使用equals。

问题?什么时候应该不加空格,最后一个有效单词,怎么判断?下标是判断不了。换种思路,在添加元素的前面添加空格
class Solution {
public String reverseWords(String s) {
String[] split =s.split(" ");
StringBuilder result = new StringBuilder();
boolean isfirst = true;
for(int i = split.length - 1 ;i >=0 ;i --){
if(!split[i].isEmpty()){

            // 不是第一个元素,在前面就加空格
            if(!isfirst){
                result.append(" ");
            }
            result.append(split[i]);
            isfirst =false;
        }
    }
    return result.toString();
}

}
更优雅的做法,使用trim()去除首尾空格,同时分割的时候根据一个或者多个空格分割
class Solution {
public String reverseWords(String s) {
// 去除首尾空格
s = s.trim();
// 按一个或多个空格分割
String[] words = s.split(“\s+”);
StringBuilder sb = new StringBuilder();

    // 从后向前添加单词
    for (int i = words.length - 1; i >= 0; i--) {
        sb.append(words[i]);
        if (i > 0) {
            sb.append(" ");
        }
    }
    
    return sb.toString();
}

}
更简洁的做法: 直接reverse
class Solution {
public String reverseWords(String s) {
// 去除首尾空格,并按一个或多个空格分割
String[] words = s.trim().split(“\s+”);
Collections.reverse(Arrays.asList(words));
return String.join(" ", words);
}
}
238. 除自身以外数组的乘积 - 力扣(LeetCode)
题目描述
一个整型数组, 获得出自己之外的数组相乘的值数组,数组中的每个值相乘(但是不能乘自己)

思路
● 全部相乘除以自己, 注意nums[i] =0 要注意 不要除。—实验发现这种思路错误因为有0的存在导致总和是0,所以不能这样算。
优化思路
● 前缀积 + 后缀积
● 前缀鸡[i] = 前缀鸡[i-1] * 数组[i-1]
● 后缀鸡[i] = 后缀鸡[i+1] * 数组[i+1]

class Solution {
public int[] productExceptSelf(int[] nums) {
// 前缀数组
int[] prefix = new int[nums.length];

    //前缀乘积
    prefix[0] = 1; // 第一个元素的前缀乘积是1
    for(int i = 1 ; i < nums.length; i ++){
        prefix[i] = prefix[i-1] * nums[i-1]; // 前缀鸡[i] = 前缀鸡[i-1] * 数组[i-1]
    }
    // 后缀鸡
    int[] suffix = new int[nums.length];
    suffix[nums.length -1] = 1;
    for(int i = nums.length -2 ; i >=0 ; i --){
          suffix[i] = suffix[i+1] * nums[i+1]; // 后缀鸡[i] = 后缀鸡[i+1] * 数组[i+1]
    }

    // 结果
    for(int i = 0; i < nums.length ; i ++){
        nums[i] = prefix[i] * suffix[i];
    }
    return nums;
}

}
内存优化:上述new 了两个数组,可以就弄一个前缀数组,后缀数组不需要,直接使用一个变量存一下,直接计算结果
class Solution {
public int[] productExceptSelf(int[] nums) {
int n = nums.length;
int[] result = new int[n];

    // 计算前缀乘积并直接存入结果数组
    result[0] = 1;
    for (int i = 1; i < n; i++) {
        result[i] = result[i-1] * nums[i-1];
    }
    
    // 使用一个变量来跟踪后缀乘积,避免使用额外数组
    int suffix = 1;
    for (int i = n-1; i >= 0; i--) {
        result[i] *= suffix;
        suffix *= nums[i];
    }
    
    return result;
}

}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值