@Yano
2019-09-20T10:50:30.000000Z
字数 15339
阅读 5772
LeetCode
coding 笔记、点滴记录,以后的文章也会同步到公众号(Coding Insight)中,希望大家关注^_^
https://github.com/LjyYano/Thinking_in_Java_MindMapping
https://leetcode.com/problems/regular-expression-matching/
Implement regular expression matching with support for '.' and '*'.
The matching should cover the entire input string (not partial).
The function prototype should be:
bool isMatch(const char *s, const char *p)
Some examples:
isMatch("aa","a") → false
isMatch("aa","aa") → true
isMatch("aaa","aa") → false
isMatch("aa", "a*") → true
isMatch("aa", ".") → true
isMatch("ab", ".") → true
isMatch("aab", "c*a*b") → true
class Solution {
public boolean isMatch(String s, String p) {
char[] string = s.toCharArray();
char[] pattern = p.toCharArray();
boolean[][] dp = new boolean[string.length + 1][pattern.length + 1];
dp[0][0] = true;
for(int i = 2; i <= pattern.length; i++) {
if(pattern[i - 1] == '*') dp[0][i] = dp[0][i - 2];
}
for (int i = 1; i <= string.length; i++) {
for (int j = 1; j <= pattern.length; j++) {
if (pattern[j - 1] == string[i - 1] || pattern[j - 1] == '.') {
dp[i][j] = dp[i - 1][j - 1];
} else if(pattern[j - 1] == '*') {
if(!dp[i][j - 1] && j >= 2) {
dp[i][j] = dp[i][j - 2] || ((string[i - 1] == pattern[j - 2] || pattern[j - 2] == '.') && dp[i - 1][j]);
} else if(dp[i][j - 1]) {
dp[i][j] = dp[i - 1][j] && (string[i - 1] == pattern[j - 2] || pattern[j - 2] == '.');
}
}
}
//System.out.println(Arrays.deepToString(dp));
}
return dp[string.length][pattern.length];
}
}
https://leetcode.com/problems/wildcard-matching/
Implement wildcard pattern matching with support for '?' and '*'.
The matching should cover the entire input string (not partial).
The function prototype should be:
bool isMatch(const char *s, const char *p)
Some examples:
isMatch("aa","a") → false
isMatch("aa","aa") → true
isMatch("aaa","aa") → false
isMatch("aa", "") → true
isMatch("aa", "a") → true
isMatch("ab", "?*") → true
isMatch("aab", "c*a*b") → false
class Solution {
public boolean isMatch(String s, String p) {
char[] string = s.toCharArray();
char[] pattern = p.toCharArray();
boolean[][] dp = new boolean[string.length + 1][pattern.length + 1];
dp[0][0] = true;
for(int i = 1; i <= pattern.length; i++) {
if(pattern[i - 1] == '*') dp[0][i] = dp[0][i - 1];
}
for (int i = 1; i <= string.length; i++) {
for (int j = 1; j <= pattern.length; j++) {
if (pattern[j - 1] == '*') {
dp[i][j] = dp[i - 1][j] || dp[i][j - 1];
} else if (string[i - 1] == pattern[j - 1] || pattern[j - 1] == '?') {
dp[i][j] = dp[i - 1][j - 1];
}
}
}
return dp[string.length][pattern.length];
}
}
https://leetcode.com/problems/longest-valid-parentheses/
Given a string containing just the characters '(' and ')', find the length of the longest valid (well-formed) parentheses substring.
For "(()", the longest valid parentheses substring is "()", which has length = 2.
Another example is ")()())", where the longest valid parentheses substring is "()()", which has length = 4.
class Solution {
public int longestValidParentheses(String s) {
int ans = 0, start = 0;
Stack<Integer> stack = new Stack<>();
for (int i = 0; i < s.length(); i++) {
if (s.charAt(i) == '(') {
stack.push(i);
} else {
if (stack.isEmpty()) {
start = i + 1;
} else {
stack.pop();
ans = stack.isEmpty() ? Math.max(ans, i - start + 1) : Math.max(ans, i - stack.peek());
}
}
}
return ans;
}
}
https://leetcode.com/problems/climbing-stairs/
You are climbing a stair case. It takes n steps to reach to the top.
Each time you can either climb 1 or 2 steps. In how many distinct ways can you climb to the top?
Note: Given n will be a positive integer.
Example 1:
Input: 2
Output: 2
Explanation: There are two ways to climb to the top.
1. 1 step + 1 step
2. 2 steps
Example 2:
Input: 3
Output: 3
Explanation: There are three ways to climb to the top.
1. 1 step + 1 step + 1 step
2. 1 step + 2 steps
3. 2 steps + 1 step
class Solution {
public int climbStairs(int n) {
if(n <= 2) return n;
// dp 可以优化为两个数
int[] dp = new int[n + 1];
dp[1] = 1;
dp[2] = 2;
for(int i = 3; i <= n; i++) {
dp[i] = dp[i - 1] + dp[i - 2];
}
return dp[n];
}
}
https://leetcode.com/problems/edit-distance/
Given two words word1 and word2, find the minimum number of steps required to convert word1 to word2. (each operation is counted as 1 step.)
You have the following 3 operations permitted on a word:
a) Insert a character
b) Delete a character
c) Replace a character
class Solution {
public int minDistance(String w1, String w2) {
int[][] dp = new int[w1.length() + 1][w2.length() + 1];
for(int i = 0; i <= w1.length(); i++) {
dp[i][0] = i;
}
for(int i = 0; i <= w2.length(); i++) {
dp[0][i] = i;
}
for(int i = 1; i <= w1.length(); i++) {
for(int j = 1; j <= w2.length(); j++) {
if(w1.charAt(i - 1) == w2.charAt(j - 1)) {
dp[i][j] = dp[i - 1][j - 1];
} else {
// 分别对应删、增、改
dp[i][j] = Math.min(Math.min(dp[i - 1][j], dp[i][j - 1]), dp[i - 1][j - 1]) + 1;
}
}
}
return dp[w1.length()][w2.length()];
}
}
https://leetcode.com/problems/decode-ways/
A message containing letters from A-Z is being encoded to numbers using the following mapping:
'A' -> 1
'B' -> 2
...
'Z' -> 26
Given an encoded message containing digits, determine the total number of ways to decode it.
For example,
Given encoded message "12",
it could be decoded as "AB" (1 2) or "L" (12).
The number of ways decoding "12" is 2.
class Solution {
public int numDecodings(String s) {
if(s == null || s.length() == 0) return 0;
int len = s.length();
int[] dp = new int[len + 1];
dp[0] = 1;
if(s.charAt(0) != '0') dp[1] = 1;
for(int i = 2; i < len + 1; i ++){
if(s.charAt(i - 1) != '0')
dp[i] += dp[i - 1];
int val = Integer.valueOf(s.substring(i - 2, i));
if(val >= 10 && val <= 26)
dp[i] += dp[i - 2];
}
return dp[len];
}
}
https://leetcode.com/problems/distinct-subsequences/
Given a string S and a string T, count the number of distinct subsequences of S which equals T.
A subsequence of a string is a new string which is formed from the original string by deleting some (can be none) of the characters without disturbing the relative positions of the remaining characters. (ie, "ACE" is a subsequence of "ABCDE" while "AEC" is not).
Here is an example:
S = "rabbbit", T = "rabbit"
Return 3.
class Solution {
public int numDistinct(String S, String T) {
// array creation
int[][] dp = new int[T.length()+1][S.length()+1];
// filling the first row: with 1s
for(int j=0; j<=S.length(); j++) {
dp[0][j] = 1;
}
// the first column is 0 by default in every other rows but the first, which we need.
/**
S = [acdabefbc]
mem[1] = [0111222222]
mem[2] = [0000022244]
**/
for(int i=0; i<T.length(); i++) {
for(int j=0; j<S.length(); j++) {
if(T.charAt(i) == S.charAt(j)) {
dp[i+1][j+1] = dp[i][j] + dp[i+1][j];
} else {
dp[i+1][j+1] = dp[i+1][j];
}
}
}
return dp[T.length()][S.length()];
}
}
https://leetcode.com/problems/word-break/
Given a non-empty string s and a dictionary wordDict containing a list of non-empty words, determine if s can be segmented into a space-separated sequence of one or more dictionary words. You may assume the dictionary does not contain duplicate words.
For example, given
s = "leetcode",
dict = ["leet", "code"].
Return true because "leetcode" can be segmented as "leet code".
UPDATE (2017/1/4):
The wordDict parameter had been changed to a list of strings (instead of a set of strings). Please reload the code definition to get the latest changes.
class Solution {
public boolean wordBreak(String s, List<String> wordDict) {
if (s == null || s.length() == 0) {
return false;
}
Set<String> set = new HashSet<>(wordDict);
boolean[] dp = new boolean[s.length() + 1];
dp[0] = true;
for (int i = 0; i < s.length(); i++) {
for (int j = 0; j <= i; j++) {
if (dp[j] && set.contains(s.substring(j, i + 1))) {
dp[i + 1] = true;
break;
}
}
// System.out.println(Arrays.toString(dp));
}
return dp[s.length()];
}
}
https://leetcode.com/problems/word-break-ii/
Given a non-empty string s and a dictionary wordDict containing a list of non-empty words, add spaces in s to construct a sentence where each word is a valid dictionary word. You may assume the dictionary does not contain duplicate words.
Return all such possible sentences.
For example, given
s = "catsanddog",
dict = ["cat", "cats", "and", "sand", "dog"].
A solution is ["cats and dog", "cat sand dog"].
UPDATE (2017/1/4):
The wordDict parameter had been changed to a list of strings (instead of a set of strings). Please reload the code definition to get the latest changes.
class Solution {
public List<String> wordBreak(String s, List<String> wordDict) {
List<String> ans = new ArrayList<>();
if (s == null || s.length() == 0) {
return ans;
}
Set<String> set = new HashSet<>(wordDict);
boolean[] dp = new boolean[s.length() + 1];
dp[0] = true;
for (int i = 0; i < s.length(); i++) {
for (int j = 0; j <= i; j++) {
if (dp[j] && set.contains(s.substring(j, i + 1))) {
dp[i + 1] = true;
break;
}
}
}
if(dp[s.length()]) {
robot(s, 0, set, ans, "", dp);
}
return ans;
}
private void robot(String s, int start, Set<String> set, List<String> ans, String out, boolean[] dp) {
if (start == s.length()) {
ans.add(out.substring(0, out.length() - 1));
return;
}
for (int i = start; i < s.length(); i++) {
String word = s.substring(start, i + 1);
if (dp[i + 1] && set.contains(word)) {
robot(s, i + 1, set, ans, out + word + " ", dp);
}
}
}
}
https://leetcode.com/problems/ugly-number-ii/
Write a program to find the n-th ugly number.
Ugly numbers are positive numbers whose prime factors only include 2, 3, 5. For example, 1, 2, 3, 4, 5, 6, 8, 9, 10, 12 is the sequence of the first 10 ugly numbers.
Note that 1 is typically treated as an ugly number, and n does not exceed 1690.
Credits:Special thanks to @jianchao.li.fighter for adding this problem and creating all test cases.
class Solution {
public int nthUglyNumber(int n) {
List<Integer> ans = new ArrayList<>();
ans.add(1);
int i0 = 0, i1 = 0, i2 = 0;
while (ans.size() < n) {
int m0 = 2 * ans.get(i0),m1 = 3 * ans.get(i1),m2 = 5 * ans.get(i2);
int min = Math.min(m0, Math.min(m1, m2));
if (min == m0) i0++;
if (min == m1) i1++;
if (min == m2) i2++;
ans.add(min);
}
return ans.get(n - 1);
}
}
https://leetcode.com/problems/perfect-squares/
Given a positive integer n, find the least number of perfect square numbers (for example, 1, 4, 9, 16, ...) which sum to n.
For example, given n = 12, return 3 because 12 = 4 + 4 + 4; given n = 13, return 2 because 13 = 4 + 9.
Credits:Special thanks to @jianchao.li.fighter for adding this problem and creating all test cases.
class Solution {
public int numSquares(int n) {
List<Integer> ans = new ArrayList<>();
ans.add(0);
while (ans.size() <= n) {
int m = ans.size(), val = Integer.MAX_VALUE;
for (int i = 1; i * i <= m; i++) {
val = Math.min(val, ans.get(m - i * i) + 1);
}
ans.add(val);
}
return ans.get(n);
}
}
https://leetcode.com/problems/longest-increasing-subsequence/
Given an unsorted array of integers, find the length of longest increasing subsequence.
For example,
Given [10, 9, 2, 5, 3, 7, 101, 18],
The longest increasing subsequence is [2, 3, 7, 101], therefore the length is 4. Note that there may be more than one LIS combination, it is only necessary for you to return the length.
Your algorithm should run in O(n2) complexity.
Follow up: Could you improve it to O(n log n) time complexity?
Credits:Special thanks to @pbrother for adding this problem and creating all test cases.
class Solution {
public int lengthOfLIS(int[] nums) {
int n = nums.length;
int[] dp = new int[n];
int ans = 0;
for (int i = 0; i < n; i++) {
dp[i] = 1;
for (int j = 0; j < i; j++) {
dp[i] = Math.max(dp[i], nums[i] > nums[j] ? dp[j] + 1 : 0);
}
ans = Math.max(ans, dp[i]);
}
return ans;
}
}
https://leetcode.com/problems/range-sum-query-immutable/
Given an integer array nums, find the sum of the elements between indices i and j (i ≤ j), inclusive.
Example:
Given nums = [-2, 0, 3, -5, 2, -1]
sumRange(0, 2) -> 1
sumRange(2, 5) -> -1
sumRange(0, 5) -> -3
Note:
class NumArray {
private int[] nums;
private int[] sums;
public NumArray(int[] tmp) {
this.nums = Arrays.copyOf(tmp, tmp.length);
sums = new int[nums.length + 1];
for (int i = 0; i < nums.length; i++) {
sums[i + 1] += nums[i] + sums[i];
}
}
public int sumRange(int i, int j) {
if (i < 0 || j > nums.length || i > j) {
return 0;
}
return sums[j + 1] - sums[i];
}
}
/**
* Your NumArray object will be instantiated and called as such:
* NumArray obj = new NumArray(nums);
* int param_1 = obj.sumRange(i,j);
*/
https://leetcode.com/problems/range-sum-query-2d-immutable/
Given a 2D matrix matrix, find the sum of the elements inside the rectangle defined by its upper left corner (row1, col1) and lower right corner (row2, col2).
The above rectangle (with the red border) is defined by (row1, col1) = (2, 1) and (row2, col2) = (4, 3), which contains sum = 8.
Example:
Given matrix = [
[3, 0, 1, 4, 2],
[5, 6, 3, 2, 1],
[1, 2, 0, 1, 5],
[4, 1, 0, 1, 7],
[1, 0, 3, 0, 5]
]
sumRegion(2, 1, 4, 3) -> 8
sumRegion(1, 1, 2, 2) -> 11
sumRegion(1, 2, 2, 4) -> 12
Note:
public class NumMatrix {
public long[][] sumMatrix;
public NumMatrix(int[][] matrix) {
if (matrix == null || matrix.length == 0) {
return;
}
sumMatrix = new long[matrix.length + 1][matrix[0].length + 1];
for (int i = 0; i < matrix.length; i++) {
for (int j = 0; j < matrix[0].length; j++) {
sumMatrix[i][j + 1] = sumMatrix[i][j] + matrix[i][j];
}
}
}
public int sumRegion(int row1, int col1, int row2, int col2) {
if (sumMatrix == null || row1 < 0 || row2 < 0 || col1 < 0
|| col2 < 0 || row1 >= sumMatrix.length - 1
|| row2 >= sumMatrix.length - 1
|| col1 >= sumMatrix[0].length - 1
|| col2 >= sumMatrix[0].length - 1 || row1 > row2
|| col1 > col2) {
return Integer.MIN_VALUE;
}
long rt = 0;
for (int i = row1; i <= row2; i++) {
rt += sumMatrix[i][col2 + 1] - sumMatrix[i][col1];
}
return (int) rt;
}
}
// Your NumMatrix object will be instantiated and called as such:
// NumMatrix numMatrix = new NumMatrix(matrix);
// numMatrix.sumRegion(0, 1, 2, 3);
// numMatrix.sumRegion(1, 2, 3, 4);
https://leetcode.com/problems/burst-balloons/
Given n balloons, indexed from 0 to n-1. Each balloon is painted with a
number on it represented by array nums.
You are asked to burst all the balloons. If the you burst
balloon i you will get nums[left] * nums[i] * nums[right] coins. Here left
and right are adjacent indices of i. After the burst, the left and right
then becomes adjacent.
Find the maximum coins you can collect by bursting the balloons wisely.
Note:
(1) You may imagine nums[-1] = nums[n] = 1. They are not real therefore you can not burst them.
(2) 0 ≤ n ≤ 500, 0 ≤ nums[i] ≤ 100
Example:
Given [3, 1, 5, 8]
Return 167
nums = [3,1,5,8] --> [3,5,8] --> [3,8] --> [8] --> []
coins = 3*1*5 + 3*5*8 + 1*3*8 + 1*8*1 = 167
Credits:Special thanks to @dietpepsi for adding this problem and creating all test cases.
public class Solution {
public int robot(int idx, int[] nums, boolean[] v) {
if(idx >= nums.length) {
return 0;
}
int ans = 0;
for(int i = 0; i < nums.length; i++) {
if(!v[i]) {
v[i] = true;
int left = i - 1 >= 0 ? nums[i - 1] : 1;
int right = i + 1 < nums.length ? nums[i + 1] : 1;
ans = nums[i] * left * right + robot(idx + 1, nums, v);
v[i] = false;
}
}
return ans;
}
public int maxCoins(int[] nums) {
boolean[] v = new boolean[nums.length];
return robot(0, nums, v);
}
}
https://leetcode.com/problems/coin-change/
You are given coins of different denominations and a total amount of money amount. Write a function to compute the fewest number of coins that you need to make up that amount. If that amount of money cannot be made up by any combination of the coins, return -1.
Example 1:
coins = [1, 2, 5], amount = 11
return 3 (11 = 5 + 5 + 1)
Example 2:
coins = [2], amount = 3
return -1.
Note:
You may assume that you have an infinite number of each kind of coin.
Credits:Special thanks to @jianchao.li.fighter for adding this problem and creating all test cases.
class Solution {
public int coinChange(int[] coins, int amount) {
int n = coins.length;
int[] dp = new int[amount + 1];
Arrays.fill(dp, amount + 1);
dp[0] = 0;
for (int i = 1; i <= amount; i++) {
for (int j = 0; j < n; j++) {
if (i >= coins[j]) {
dp[i] = Math.min(dp[i], dp[i - coins[j]] + 1);
}
}
}
return dp[amount] > amount ? -1 : dp[amount];
}
}
https://leetcode.com/problems/combination-sum-iv/
Given an integer array with all positive numbers and no duplicates, find the number of possible combinations that add up to a positive integer target.
Example:
nums = [1, 2, 3]
target = 4
The possible combination ways are:
(1, 1, 1, 1)
(1, 1, 2)
(1, 2, 1)
(1, 3)
(2, 1, 1)
(2, 2)
(3, 1)
Note that different sequences are counted as different combinations.
Therefore the output is 7.
Follow up:
What if negative numbers are allowed in the given array?
How does it change the problem?
What limitation we need to add to the question to allow negative numbers?
Credits:Special thanks to @pbrother for adding this problem and creating all test cases.
class Solution {
public int combinationSum4(int[] nums, int target) {
int[] dp = new int[target + 1];
dp[0] = 1;
for (int i = 1; i <= target; i++) {
for (int v : nums) {
if (i >= v) {
dp[i] += dp[i - v];
}
}
}
return dp[target];
}
}