leetcode刷题之路(持续更新)
前言
用于记录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%的用户
写完之后看了一下题解,除了我这种方法以外还有横向扫描、二分查找等方法,其中二分查找的复杂度最低,准备明天尝试用二分写一下
上一篇: 1429: 箭无虚发
下一篇: 随记_求N位数范围内的所有自幂数