You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

90 lines
3.4 KiB

2 years ago
##[$AcWing$ $1016$. 最大上升子序列和](https://www.acwing.com/problem/content/1018/)
### 一、题目描述
一个数的序列 $b_i$,当 $b_1<b_2<<b_S$
对于给定的一个序列($a_1,a_2,…,a_N$),我们可以得到一些上升的子序列($a_{i1},a_{i2},…,a_{iK}$),这里$1≤i_1<i_2<<i_KN$
比如,对于序列$(1,7,3,5,9,4,8)$,有它的一些上升子序列,如$(1,7),(3,4,8)$等等。
这些子序列中和最大为$18$,为子序列$(1,3,5,9)$的和。
你的任务,就是对于给定的序列,**求出最大上升子序列和**。
<font color='red' size=4><b>注意:最长的上升子序列的和不一定是最大的</b></font>,比如序列$(100,1,2,3)$的最大上升子序列和为$100$,而最长上升子序列为$(1,2,3)$。
**输入格式**
输入的第一行是序列的长度$N$。
第二行给出序列中的$N$个整数,这些整数的取值范围都在$0$到$10000$(可能重复)。
**输出格式**
输出一个整数,表示最大上升子序列和。
**数据范围**
$1≤N≤1000$
**输入样例**
```cpp {.line-numbers}
7
1 7 3 5 9 4 8
```
**输出样例**
```cpp {.line-numbers}
18
```
### 二、题目分析
计算最长上升子序列长度的时候,我们定义$f[i]$为以第$i$个元素结尾的最长子序列的长度,并且采用双重循环的方法,固定每一个数字$A$时,向前查找可以衔接到哪个数字$B_i$后面,如果$A>B_i$,则尝试接在$B_i$后面,可以获取到一个可能的最长子序列长度$L_i$,所有的$L_i$ $PK$一下$MAX$,就是答案。
本题不再要求计算$LIS$的最长长度,而是计算总和,和上面的思考方式一样,但状态表示有了变化:
#### 状态表示
$f[i]$:以第$i$个元素结尾的最大子序列和
也采用双重循环的方法,固定每一个数字$A$时,向前查找可以衔接到哪个数字$B_i$后面,如果$A>B_i$,则尝试接在$B_i$后面,可以获取到一个可能的最大子序列和$S_i$,所有的$S_i$ $PK$一下$MAX$,就是答案。
综上所述,此题与$LIS$还是有区别的,并不是$LIS$的扩展,而是$LIS$问题的变形,不是在求解$LIS$问题基础上再增加点什么就能解决,而是采用了类似的思路。$LIS$问题有三种解法:
* **朴素版动态规划**  可以继承思想,变形解决
* **贪心+二分** 已经无效,不能继承,思路不同
* **树状数组**  
### 三、$O(n^2)$实现代码
```cpp {.line-numbers}
#include <bits/stdc++.h>
using namespace std;
// 最大上升子序列和
int n;
const int N = 100010;
int a[N];
1 year ago
int f[N];//以第i个元素结尾的最大子序列和
2 years ago
int res;
int main() {
cin >> n;
for (int i = 1; i <= n; i++) cin >> a[i];
for (int i = 1; i <= n; i++) {
f[i] = a[i]; // 最大上升子序列个数这里是1,此处是a[i]
for (int j = 1; j < i; j++)
// 最大上升子序列个数这里是加1,此处是+a[i]
if (a[i] > a[j]) f[i] = max(f[i], f[j] + a[i]);
res = max(res, f[i]);
}
printf("%d ", res);
return 0;
}
```
### 四、$O(nlogn)$实现代码(树状数组版本)
**[题解传送门](https://www.cnblogs.com/littlehb/p/17510484.html)**
1 year ago
### 五、关键字
- 魔改$LIS$
- 最大上升子序列和