LeetCode 673. Number of Longest Increasing Subsequence (Java版; Meidum)

welcome to my blog

LeetCode 673. Number of Longest Increasing Subsequence (Java版; Meidum)

题目描述
Given an unsorted array of integers, find the number of longest increasing subsequence.

Example 1:
Input: [1,3,5,4,7]
Output: 2
Explanation: The two longest increasing subsequence are [1, 3, 4, 7] and [1, 3, 5, 7].
Example 2:
Input: [2,2,2,2,2]
Output: 5
Explanation: The length of longest continuous increasing subsequence is 1, and there are 5 subsequences' length is 1, so output 5.
Note: Length of the given array will be not exceed 2000 and the answer is guaranteed to be fit in 32-bit signed int.
第二次做; 这道题和LC300本质是相同的; 核心: 1) 出现递增时考虑更新dp[i]和count[i] 2)两个一维DP
class Solution {
    public int findNumberOfLIS(int[] nums) {
        //input check
        //
        int n = nums.length;
        //dp[i]表示以nums[i]结尾时的LIS长度
        int[] dp = new int[n];
        //count[i]表示以nums[i]结尾的LIS的个数
        int[] count = new int[n];
        for(int i=0; i<n; i++){
            dp[i] = 1;
            count[i] = 1;
        }
        int max=0;
        int res=0;
        for(int i=0; i<n; i++){
            for(int j=0; j<i; j++){
                if(nums[i] > nums[j]){
                    if(1+dp[j] > dp[i]){
                        dp[i] = 1 + dp[j];
                        count[i] = count[j];
                    }
                    else if(1+dp[j]==dp[i]){
                        count[i] += count[j];
                    }
                }
            }
            //update
            if(dp[i] > max){
                max = dp[i];
                res = count[i];
            }
            else if(dp[i]==max){
                res += count[i];
            };
        }
        return res;
    }
}
第一次做; 直接看了优秀题解; 两个一维动态规划, 核心:1) len[i]表示以nums[i]结尾的字符串的最长递增子序列的长度, cnt[i]表示以nums[i]结尾的字符串的最长递增子序列的个数; 2) 在nums[i] > nums[j]的情况下, 比较len[i] 和 len[j]+1; 3) substring往往使用单层循环, subsequence问题往往使用双层循环
/*
动态规划
两个一维动态规划
*/
class Solution {
    public int findNumberOfLIS(int[] nums) {
        if(nums==null || nums.length==0)
            return 0;
        int n = nums.length;
        //len[i]表示以nums[i]结尾的字符串的最长子序列长度
        int[] len = new int[n];
        //cnt[i]表示以nums[i]结尾的字符串的最长子序列的个数
        int[] cnt = new int[n];
        //记录出现过的最大长度
        int max=0;
        //记录最终的结果
        int res = 0;
        //DP主体
        for(int i=0; i<n; i++){
            //自身构成的最长子序列
            len[i] = 1;
            cnt[i] = 1;
            //对于每个i, 只考虑[0,i-1]范围的情况
            for(int j=0; j<i; j++){
                //如果出现递增, 该如何更新数组?
                if(nums[i] > nums[j]){
                    //len[i] 与 len[j] + 1 之间的比较
                    if(len[i] == len[j] + 1 ){
                        cnt[i] += cnt[j];
                    }
                    else if(len[i] < len[j] + 1){
                        len[i] = len[j] + 1;
                        cnt[i] = cnt[j];
                    }
                }
            }
            //
            if(len[i] > max){
                max = len[i];
                res = cnt[i];
            }
            else if(len[i] == max){
                res += cnt[i];
            }
        }
        return res;
    }
}
leetcode优秀题解 分析中还指出了substring问题一般用单层循环,因为当前元素只与相邻的元素有关系; subsequence问题一般用双层循环,因为当前元素可能与任何其他的元素都有关系
for those guys who are not quite familiar with this type of problem, please also check No.549. Binary Tree Longest Consecutive Sequence II to 
have a better understanding..
for each element in the array or on in the tree, they all carry three fields :
1) the maximum increasing / decreasing length ends at the current element,
2) its own value ,
3) the total number of maximum length,
and each time when we visit a element, we will use its 2) to update 1) and 3), the only difference is for array we use iteration while for tree
we use recursion......
Also, for substring problem, we usually use only one for loop because for each index, we only care about the relationship between its two neighbors,
while for subsequence problem, we use two for loops , because for each index, any other indexes can do something...
public int findNumberOfLIS(int[] nums) {
        int n = nums.length, res = 0, max_len = 0;
        int[] len =  new int[n], cnt = new int[n];
        for(int i = 0; i<n; i++){
            len[i] = cnt[i] = 1;
            for(int j = 0; j <i ; j++){
                if(nums[i] > nums[j]){
                    if(len[i] == len[j] + 1)cnt[i] += cnt[j];
                    if(len[i] < len[j] + 1){
                        len[i] = len[j] + 1;
                        cnt[i] = cnt[j];
                    }
                }
            }
            if(max_len == len[i])res += cnt[i];
            if(max_len < len[i]){
                max_len = len[i];
                res = cnt[i];
            }
        }
        return res;
    }
力扣官方题解 使用了线段树
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 可以使用Java中的排序功能来实现。可以使用Arrays.sort()函数,将列表中的元素按照字母顺序排序,或者使用Collections.sort()函数,将列表中的元素按用户指定的排序规则排序。 ### 回答2: 为了实现LeetCode 2561题(Rearranging Fruits)的要求,需要使用Java编程语言。主要思路是遍历给定的水果数组,同时用一个哈希表来记录每个水果出现的次数。然后根据题目要求,重新排列水果使得相同类型的水果尽可能接近,并且按照出现次数的非递增顺序排序。 具体实现步骤如下: 1. 创建一个HashMap来存储每个水果的出现次数。遍历给定的水果数组,如果该水果已经存在于HashMap中,则将其出现次数加1;否则,将该水果添加到HashMap,并将其出现次数初始化为1。 2. 创建一个ArrayList来存储已经排列好的水果。通过HashMap的entrySet方法获取到每种水果和它的出现次数,然后将这些entry按照出现次数的非递增顺序进行排序。 3. 遍历排序好的entry集合,根据每个水果的出现次数,在ArrayList中连续添加相应数量的水果。 4. 返回排列好的水果数组。 以下是Java代码的示例实现: ```java import java.util.*; class Solution { public String[] rearrange(String[] fruits) { HashMap<String, Integer> fruitCountMap = new HashMap<>(); // 统计每个水果的出现次数 for (String fruit : fruits) { if (fruitCountMap.containsKey(fruit)) { fruitCountMap.put(fruit, fruitCountMap.get(fruit) + 1); } else { fruitCountMap.put(fruit, 1); } } ArrayList<Map.Entry<String, Integer>> sortedEntries = new ArrayList<>(fruitCountMap.entrySet()); // 根据出现次数进行非递增排序 Collections.sort(sortedEntries, new Comparator<Map.Entry<String, Integer>>() { public int compare(Map.Entry<String, Integer> entry1, Map.Entry<String, Integer> entry2) { return entry2.getValue().compareTo(entry1.getValue()); } }); ArrayList<String> rearrangedFruits = new ArrayList<>(); // 根据出现次数连续添加水果 for (Map.Entry<String, Integer> entry : sortedEntries) { String fruit = entry.getKey(); int count = entry.getValue(); for (int i = 0; i < count; i++) { rearrangedFruits.add(fruit); } } return rearrangedFruits.toArray(new String[0]); } } ``` 使用以上代码,可以对给定的水果数组进行重新排列,使得相同类型的水果尽可能接近,并且按照出现次数的非递增顺序进行排序。返回的结果就是排列好的水果数组。 ### 回答3: 题目要求将一个字符串中的水果按照特定规则重新排列。我们可以使用Java来实现这个问题。 首先,我们需要定义一个函数来解决这个问题。 ```java public static String rearrangeFruits(String fruits) { // 将字符串转换为字符数组方便处理 char[] fruitArray = fruits.toCharArray(); // 统计每种水果的数量 int[] fruitCount = new int[26]; for (char fruit : fruitArray) { fruitCount[fruit - 'a']++; } // 创建一个新的字符数组来存储重新排列后的结果 char[] rearrangedFruitArray = new char[fruitArray.length]; // 逐个将水果按照规则放入新数组中 int index = 0; for (int i = 0; i < 26; i++) { while (fruitCount[i] > 0) { rearrangedFruitArray[index++] = (char) ('a' + i); fruitCount[i]--; } } // 将字符数组转换为字符串并返回 return new String(rearrangedFruitArray); } ``` 上述代码中,我们首先将字符串转换为字符数组,并使用一个长度为26的数组来统计每一种水果的数量。然后,我们创建一个新的字符数组来存储重新排列后的结果。 接下来,我们利用双重循环将每一种水果按照规则放入新数组中。最后,我们将字符数组转换为字符串并返回。 例如,如果输入字符串为`"acbba"`,则经过重新排列后,输出结果为`"aabbc"`。 这样,我们就用Java实现了题目要求的功能。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值