题意:
求n个数中长度为m的上升子序列的个数
解题思路:dp[i][j]表示第i个数长度为j的上升序列的个数。dp[i][j] = sum{dp[k][j-1] | a[k] < a[i]},这里的时间复杂度有O(n³),会超时,所以这里要有优化。其实可以将a[i]离散化,对于每一个j,构造一个树状数组,这样求sum{dp[k][j-1]}就可以用树状数组的求和了,时间复杂度可以降为O(n²logn)。
#include<iostream>
#include<cstdio>
#include<cstring>
#include<set>
#include<map>
using namespace std;
const int maxn = 1005;
const int mod = 1e9+7;
int n,m,dp[maxn][maxn];
int a[maxn],tree[maxn][maxn];
set<int> Set;
map<int,int> Map;
int lowbit(int x)
{
return x & -x;
}
void add(int i,int j,int c)
{
while(j <= n)
{
tree[i][j] = (tree[i][j] + c) % mod;
j += lowbit(j);
}
}
int sum(int i,int j)
{
int ans = 0;
while(j > 0)
{
ans = (ans + tree[i][j]) % mod;
j -= lowbit(j);
}
return ans;
}
int main()
{
int t,cas = 1;
scanf("%d",&t);
while(t--)
{
scanf("%d%d",&n,&m);
Set.clear();
Map.clear();
for(int i = 1; i <= n; i++)
{
scanf("%d",&a[i]);
Set.insert(a[i]);
}
int cnt = 0;
for(set<int>::iterator it = Set.begin(); it != Set.end(); it++)
Map[*it] = ++cnt;
memset(dp,0,sizeof(dp));
memset(tree,0,sizeof(tree));
for(int i = 1; i <= n; i++)
for(int j = 1; j <= m; j++)
{
if(j == 1) dp[i][j] = 1;
else dp[i][j] = (dp[i][j] + sum(j-1,Map[a[i]]-1)) % mod;
add(j,Map[a[i]],dp[i][j]);
}
int ans = 0;
for(int i = 1; i <= n; i++)
ans = (ans + dp[i][m]) % mod;
printf("Case #%d: %d\n",cas++,ans);
}
return 0;
}