51nod 1376 最长递增子序列的数量(线段树)
51nod 1376 最长递增子序列的数量
第1行:1个数N,表示数组的长度。(1 <= N <= 50000)
第2 - N + 1行:每行1个数A[i],表示数组的元素(0 <= A[i] <= 10^9)
输出最长递增子序列的数量Mod 1000000007。
5
1
3
2
0
4
2
/*
51nod 1376 最长递增子序列的数量 problem:
给你一个数组,求其中最长递增子序列有多少个 solve:
对于第i个数a[i]而言,它需要知道已经出现的[1,a[i]-1]中最长递增子序列的长度以及数量. 所以可以利用线段树来维护,
然后利用长度和数量来更新a[i] hhh-2016/09/03-16:41:1
*/
#pragma comment(linker,"/STACK:124000000,124000000")
#include <algorithm>
#include <iostream>
#include <cstdlib>
#include <cstdio>
#include <cstring>
#include <vector>
#include <math.h>
#include <queue>
#include <set>
#include <map>
#define lson i<<1
#define rson i<<1|1
#define ll long long
#define clr(a,b) memset(a,b,sizeof(a))
#define scanfi(a) scanf("%d",&a)
#define scanfs(a) scanf("%s",a)
#define scanfl(a) scanf("%I64d",&a)
#define scanfd(a) scanf("%lf",&a)
#define key_val ch[ch[root][1]][0]
#define eps 1e-7
#define inf 0x3f3f3f3f3f3f3f3f
using namespace std;
const ll mod = 1000000007;
const int maxn = 50010;
const double PI = acos(-1.0);
int a[maxn],t[maxn];
struct node
{
int l,r;
ll Max,num;
int mid;
} tree[maxn << 2]; void push_up(int i)
{
if(tree[lson].Max == tree[rson].Max)
{
tree[i].Max = tree[lson].Max;
tree[i].num = tree[lson].num + tree[rson].num;
tree[i].num %= mod;
}
else if(tree[lson].Max > tree[rson].Max)
{
tree[i].Max = tree[lson].Max;
tree[i].num = tree[lson].num;
}
else
{
tree[i].Max = tree[rson].Max;
tree[i].num = tree[rson].num;
}
} void build(int i,int l,int r)
{
tree[i].l = l,tree[i].r = r;
tree[i].Max = tree[i].num = 0;
tree[i].mid = (l+r) >> 1;
if(l == r)
return ;
build(lson,l,tree[i].mid);
build(rson,tree[i].mid + 1,r);
push_up(i);
} void push_down(int i)
{ } void update(int i,int k,ll len,ll many)
{
if(tree[i].l == tree[i].r && tree[i].l == k)
{
if(tree[i].Max < len) tree[i].Max = len,tree[i].num = many;
else if(tree[i].Max == len) tree[i].num += many;
tree[i].num %= mod;
return ;
}
int mid = tree[i].mid;
if(k <= mid)
update(lson,k,len,many);
else update(rson,k,len,many);
push_up(i);
}
ll tans,tnum;
void query(int i,int l,int r)
{
if(l > r)
{
tans = 0,tnum = 1;
return ;
}
if(tree[i].l >= l && tree[i].r <= r)
{
if(tans == -1)
tans = tree[i].Max,tnum = tree[i].num;
else
{
if(tans == tree[i].Max) tnum += tree[i].num;
else if(tans < tree[i].Max)
{
tans = tree[i].Max,tnum = tree[i].num;
}
tnum %= mod;
}
return ;
}
push_down(i);
int mid = tree[i].mid;
if(l <= mid)
query(lson,l,r);
if(r > mid)
query(rson,l,r);
return;
} int cnt = 0;
int main()
{
int n;
while(scanfi(n) != EOF)
{
for(int i = 0; i< n;i++){
scanfi(a[i]);
t[i] = a[i];
}
sort(a,a+n);
cnt = 0;
for(int i = 1;i < n;i++)
{
if(a[i] != a[cnt])
a[++cnt] = a[i];
}
// for(int i = 0;i <= cnt;i++)
// printf("%d\n",a[i]);
build(1,0,cnt);
ll tMax = 0,ans = 0;
int id = lower_bound(a,a+cnt+1,t[0]) -a;
update(1,id,1,1);
tMax = 1,ans = 1;
for(int i = 1;i < n;i++)
{
id = lower_bound(a,a+cnt+1,t[i]) -a;
// cout << id <<" " ;
tans = -1,tnum = 0;
query(1,0,id-1);
if(!tans && !tnum)
tnum = 1;
if(tMax < tans+ 1) tMax = tans+1,ans = tnum;
else if(tMax == tans+1) ans += tnum;
ans %= mod;
update(1,id,tans + 1,tnum);
}
// cout <<endl;
printf("%I64d\n",ans);
}
return 0;
}
51nod 1376 最长递增子序列的数量(线段树)的更多相关文章
- 51Nod 1376 最长递增子序列的数量 —— LIS、线段树
题目链接:http://www.51nod.com/onlineJudge/questionCode.html#!problemId=1376 1376 最长递增子序列的数量 基准时间限制:1 秒 空 ...
- 51NOD 1376 最长递增子序列的数量 [CDQ分治]
1376 最长递增子序列的数量 首先可以用线段树优化$DP$做,转移时取$0...a[i]$的最大$f$值 但我要练习$CDQ$ $LIS$是二维偏序问题,偏序关系是$i<j,\ a_i< ...
- 51nod 1376 最长递增子序列的数量(不是dp哦,线段树 + 思维)
题目链接:https://www.51nod.com/onlineJudge/questionCode.html#!problemId=1376 题解:显然这题暴力的方法很容易想到就是以每个数为结尾最 ...
- 51Nod 1376 最长递增子序列的数量 (DP+BIT)
题意:略. 析:dp[i] 表示以第 i 个数结尾的LIS的长度和数量,状态方程很好转移,先说长度 dp[i] = max { dp[j] + 1 | a[i] > a[j] && ...
- 51nod 1376 最长上升子序列的数量 | DP | vector怒刷存在感!
51nod 1376 最长上升子序列的数量 题解 我们设lis[i]为以位置i结尾的最长上升子序列长度,dp[i]为以位置i结尾的最长上升子序列数量. 显然,dp[i]要从前面的一些位置(设为位置j) ...
- 【51nod】1376 最长递增子序列的数量
数组A包含N个整数(可能包含相同的值).设S为A的子序列且S中的元素是递增的,则S为A的递增子序列.如果S的长度是所有递增子序列中最长的,则称S为A的最长递增子序列(LIS).A的LIS可能有很多个. ...
- 51nod-1134 最长递增子序列,用线段树将N^2的dp降到NlogN
题目链接 给出长度为N的数组,找出这个数组的最长递增子序列.(递增子序列是指,子序列的元素是递增的) 例如:5 1 6 8 2 4 5 10,最长递增子序列是1 2 4 5 10. Input 第1行 ...
- 51nod1376 最长递增子序列的数量
O(n2)显然超时.网上找的题解都是用奇怪的姿势写看不懂TAT.然后自己YY.要求a[i]之前最大的是多少且最大的有多少个.那么线段树维护两个值,一个是当前区间的最大值一个是当前区间最大值的数量那么我 ...
- 51nod 1134 最长递增子序列
题目链接:51nod 1134 最长递增子序列 #include<cstdio> #include<cstring> #include<algorithm> usi ...
随机推荐
- C语言博客作业--数组
一.PTA实验作业 题目1.求整数序列中出现次数最多的数 1.本题PTA提交列表 2.设计思路 定义整形变量n,max,count分别表示整数个数,出现次数最大值,出现次数.定义循环变量i,j. 输入 ...
- bug终结者 团队作业第六、七周
bug终结者 团队作业第六.七周 作业要求:团队作业第六.七周 博客编辑:20162322 朱娅霖 一.修改<需求规格说明书> <需求规格说明书>2.0版(即初稿) <需 ...
- 20162330 实验一 《Java开发环境的熟悉》 实验报告
2016-2017-2 实验报告目录: 1 2 3 4 5 20162330 实验一 <Java开发环境的熟悉> 实验报告 课程名称:<程序设计与数据结构> 学生班级:1623 ...
- NumPy简介
NumPy是什么? NumPy(Numerrical Python 的缩写)是一个开源的Python科学计算库.使用NumPy,就可以很自然的使用数组.NumPy包含很多实用的数学函数,涵盖线性代数运 ...
- SCOI2010 序列操作
2421 序列操作 http://codevs.cn/problem/2421/ 2010年省队选拔赛四川 题目描述 Description lxhgww最近收到了一个01序列,序列里面包含了n个 ...
- JAVA_SE基础——18.方法的递归
方法的递归是指在一个方法的内部调用自身的过程,递归必须要有结束条件,不然就会陷入无限递归的状态,永远无法结束调用,接下来用一个最简单的例子来体现下方法递归,使用递归算法计算自然数之和: public ...
- 【深度学习】深入理解Batch Normalization批标准化
这几天面试经常被问到BN层的原理,虽然回答上来了,但还是感觉答得不是很好,今天仔细研究了一下Batch Normalization的原理,以下为参考网上几篇文章总结得出. Batch Normaliz ...
- 爬虫小探-Python3 urllib.request获取页面数据
使用Python3 urllib.request中的Requests()和urlopen()方法获取页面源码,并用re正则进行正则匹配查找需要的数据. #forex.py#coding:utf-8 ' ...
- C 函数指针与回调函数
函数指针是指向函数的指针变量. 通常我们说的指针变量是指向一个整型.字符型或数组等变量,而函数指针是指向函数. 函数指针可以像一般函数一样,用于调用函数.传递参数. 函数指针变量的声明: #inclu ...
- Mego开发文档 - 处理并发冲突
处理并发冲突 数据库并发是指多个进程或用户同时访问或更改数据库中的相同数据的情况.并发控制是指用于确保存在并发更改时数据一致性的特定机制. Mego实现了乐观并发控制,这意味着它可以让多个进程或用户独 ...