欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页

leetcode刷题之路(持续更新)

程序员文章站 2024-03-16 16:09:52
...

前言

用于记录leetcode刷题过程,目前计划是每天至少一题

1. 两数之和(简单)

题目:
给定一个整数数组 nums 和一个目标值 target,请你在该数组中找出和为目标值的那 两个 整数,并返回他们的数组下标。

你可以假设每种输入只会对应一个答案。但是,你不能重复利用这个数组中同样的元素。

示例:

给定 nums = [2, 7, 11, 15], target = 9

因为 nums[0] + nums[1] = 2 + 7 = 9
所以返回 [0, 1]

这是在leetcode上开始的第一题,很简单的一题,但好久没写算法题加上JAVA基础忘得差不多了,写起来还不太熟练

首先很容易想出的就是两个for循环暴力解法

class Solution {
    public int[] twoSum(int[] nums, int target) {
        for(int i=0;i<nums.length;i++)
        {
            for(int j=i+1;j<nums.length;j++)
            {
                if(nums[i]+nums[j]==target)
                {
                    return new int[]{i,j};
                }
            }
        }
        throw new IllegalArgumentException("No two sum solution");
    }
}

执行用时27ms
时间复杂度O(n^2)

然后是根据题解学习到的借助哈希表的解法

import java.util.HashMap;

class Solution {
    public int[] twoSum(int[] nums, int target) {
        HashMap<Integer, Integer> map = new HashMap<>();
        for(int i=0;i<nums.length;i++)
        {
            map.put(nums[i],i);
        }
        for(int i=0;i<nums.length;i++)
        {
            int a=target-nums[i];
            if(map.containsKey(a)&&map.get(a)!=i)
            {
                return new int[]{i,map.get(a)};
            }
        }
        throw new IllegalArgumentException("No two sum solution");
    }
}

执行时间6ms
时间复杂度O(n)

7. 整数反转(简单)

题目:
给出一个 32 位的有符号整数,你需要将这个整数中每位上的数字进行反转。
示例 1:

输入: 123
输出: 321

示例 2:

输入: -123
输出: -321

示例 3:

输入: 120
输出: 21

注意:
假设我们的环境只能存储得下 32 位的有符号整数,则其数值范围为 [−231, 231 − 1]。请根据这个假设,如果反转后整数溢出那么就返回 0。

这是在力扣上做的第二题了,题目很简单,主要是要考虑反转后的数有没有溢出,先贴下自己的代码

import java.util.Scanner;

class Solution {
    public int reverse(int x) {
        long y = 0;
        int xx, num = 0;
        if (x < 0) xx = -x;
        else xx = x;
        while (xx > 0) {
            xx /= 10;
            num = num + 1;
        }
        for (int i = 0; i < num; i++) {
            y *= 10;
            y += x % 10;
            x /= 10;
        }
        if (y >= -Math.pow(2, 31) && y <= Math.pow(2, 31) - 1) return (int) y;
        else return 0;
    }
}

执行用时 :2 ms, 在所有 java 提交中击败了57.43%的用户
内存消耗 :33.8 MB, 在所有 java 提交中击败了79.13%的用户

写完之后发现代码还是有挺多问题的,
第一是使用了Long类型用来存放反转后的数,实际上不太符合题意,
第二是完全没有必要计算数的位数,可以直接用一个while(x!=0)代替,
第三是int的最大值和最小值可以用Integer.MAX_VALUE和Integer.MIN_VALUE代替,没必要用幂函数算。
最后再贴一下官方题解

class Solution {
    public int reverse(int x) {
        int rev = 0;
        while (x != 0) {
            int pop = x % 10;
            x /= 10;
            if (rev > Integer.MAX_VALUE/10 || (rev == Integer.MAX_VALUE / 10 && pop > 7)) return 0;
            if (rev < Integer.MIN_VALUE/10 || (rev == Integer.MIN_VALUE / 10 && pop < -8)) return 0;
            rev = rev * 10 + pop;
        }
        return rev;
    }
}

9. 回文数(简单)

题目:
判断一个整数是否是回文数。回文数是指正序(从左向右)和倒序(从右向左)读都是一样的整数。

示例 1:

输入: 121
输出: true

示例 2:

输入: -121
输出: false
解释: 从左向右读, 为 -121 。 从右向左读, 为 121- 。因此它不是一个回文数。

示例 3:

输入: 10
输出: false
解释: 从右向左读, 为 01 。因此它不是一个回文数。

这题也很简单,和第七题差不多,可以想到主要有两种方法,第一种是把数字当做字符串,然后进行判断,第二种是把它反转过后的数和它本身进行对比,相等则返回true
下面贴一下我的代码

class Solution {
    public boolean isPalindrome(int x) {
        if (x < 0) return false;
        int y = 0, xx = x;
        while (xx != 0) {
            y *= 10;
            y += xx % 10;
            xx /= 10;
        }
        if (y == x) return true;
        else return false;
    }
}

执行用时 :9 ms, 在所有 java 提交中击败了98.60%的用户
内存消耗 :35.7 MB, 在所有 java 提交中击败了97.46%的用户

13. 罗马数字转整数(简单)

题目:
罗马数字包含以下七种字符: I, V, X, L,C,D 和 M。

字符          数值
I             1
V             5
X             10
L             50
C             100
D             500
M             1000

例如, 罗马数字 2 写做 II ,即为两个并列的 1。12 写做 XII ,即为 X + II 。 27 写做 XXVII, 即为 XX + V + II 。

通常情况下,罗马数字中小的数字在大的数字的右边。但也存在特例,例如 4 不写做 IIII,而是 IV。数字 1 在数字 5 的左边,所表示的数等于大数 5 减小数 1 得到的数值 4 。同样地,数字 9 表示为 IX。这个特殊的规则只适用于以下六种情况:

I 可以放在 V (5) 和 X (10) 的左边,来表示 4 和 9。
X 可以放在 L (50) 和 C (100) 的左边,来表示 40 和 90。
C 可以放在 D (500) 和 M (1000) 的左边,来表示 400 和 900。
给定一个罗马数字,将其转换成整数。输入确保在 1 到 3999 的范围内。

示例 1:

输入: "III"
输出: 3

示例 2:

输入: "IV"
输出: 4

示例 3:

输入: "IX"
输出: 9

示例 4:

输入: "LVIII"
输出: 58
解释: L = 50, V= 5, III = 3.

示例 5:

输入: "MCMXCIV"
输出: 1994
解释: M = 1000, CM = 900, XC = 90, IV = 4.

这是一道比较简单的模拟题,我的思路是遍历字符串中的每一个字符,对于一些特殊规则中的字符,看一看它的下一位是什么,是否符合特殊规则,下面贴一下我的代码,不过写的还是不怎么简洁,有点冗杂

class Solution
{
    public int romanToInt(String s)
    {
        int sum = 0;
        for (int i = 0; i < s.length(); i++)
        {
            if (s.charAt(i) == 'I')
            {
                if (i == s.length() - 1) sum += 1;
                else if (s.charAt(i + 1) == 'V')
                {
                    sum += 4;
                    i += 1;
                }
                else if (s.charAt(i + 1) == 'X')
                {
                    sum += 9;
                    i += 1;
                }
                else sum += 1;
            }
            else if (s.charAt(i) == 'V')
            {
                sum += 5;
            }
            else if (s.charAt(i) == 'X')
            {
                if (i == s.length() - 1) sum += 10;
                else if (s.charAt(i + 1) == 'L')
                {
                    sum += 40;
                    i += 1;
                }
                else if (s.charAt(i + 1) == 'C')
                {
                    sum += 90;
                    i += 1;
                }
                else sum += 10;
            }
            else if (s.charAt(i) == 'L')
            {
                sum += 50;
            }
            else if (s.charAt(i) == 'C')
            {
                if (i == s.length() - 1) sum += 100;
                else if (s.charAt(i + 1) == 'D')
                {
                    sum += 400;
                    i += 1;
                }
                else if (s.charAt(i + 1) == 'M')
                {
                    sum += 900;
                    i += 1;
                }
                else sum += 100;
            }
            else if (s.charAt(i) == 'D')
                sum += 500;
            else sum += 1000;
        }
        return sum;
    }
}

执行用时 :6 ms, 在所有 java 提交中击败了78.53%的用户
内存消耗 :36.1 MB, 在所有 java 提交中击败了98.27%的用户

14. 最长公共前缀(简单)

题目:
编写一个函数来查找字符串数组中的最长公共前缀。

如果不存在公共前缀,返回空字符串 “”。

示例 1:

输入: ["flower","flow","flight"]
输出: "fl"

示例 2:

输入: ["dog","racecar","car"]
输出: ""

解释: 输入不存在公共前缀。
说明: 所有输入只包含小写字母 a-z 。

这是一道字符串处理题,我的思路是纵向扫描每个字符串,也就是从0位置开始不停往下判断每个字符串该位置的字符是否相等,如果有不相等的情况发生就返回,下面贴一下代码

class Solution
{
    public String longestCommonPrefix(String[] strs)
    {
        char c = '0';
        String s = "";
        if (strs.length == 0) return "";
        for (int i = 0; ; i++)
        {
            for (int j = 0; j < strs.length; j++)
            {
                if (i >= strs[j].length()) return s;
                if (j == 0) c = strs[0].charAt(i);
                if (strs[j].charAt(i) != c)
                {
                    return s;
                }
            }
            s += c;
        }
    }
}

执行用时 :1 ms, 在所有 java 提交中击败了87.22%的用户
内存消耗 :37.3 MB, 在所有 java 提交中击败了79.14%的用户

写完之后看了一下题解,除了我这种方法以外还有横向扫描、二分查找等方法,其中二分查找的复杂度最低,准备明天尝试用二分写一下