Problem 2236 第十四个目标

Accept: 17    Submit: 35 Time Limit: 1000 mSec    Memory Limit : 32768 KB

 Problem Description

目暮警官、妃英里、阿笠博士等人接连遭到不明身份之人的暗算,柯南追踪伤害阿笠博士的凶手,根据几起案件现场留下的线索发现凶手按照扑克牌的顺序行凶。在经过一系列的推理后,柯南发现受害者的名字均包含扑克牌的数值,且扑克牌的大小是严格递增的,此外遇害者与毛利小五郎有关。

为了避免下一个遇害者的出现,柯南将可能遭到暗算的人中的数字按关联程度排列了出来,即顺序不可改变。柯南需要知道共有多少种可能结果,满足受害人名字出现的数字严格递增,但是他柯南要找出关键的证据所在,所以这个任务就交给你了。

(如果你看不懂上面在说什么,这题是求一个数列中严格递增子序列的个数。比如数列(1,3,2)的严格递增子序列有(1)、(3)、(2)、(1,3)、(1,2),共5个。长得一样的但是位置不同的算不同的子序列,比如数列(3,3)的答案是2。)

 Input

多组数据(<=10),处理到EOF。

第一行输入正整数N(N≤100 000),表示共有N个人。

第二行共有N个整数Ai(1≤Ai≤10^9),表示第i个人名字中的数字。

 Output

每组数据输出一个整数,表示所有可能的结果。由于结果可能较大,对1 000 000 007取模后输出。

 Sample Input

3 1 3 2

 Sample Output

5

 Source

福州大学第十三届程序设计竞赛

题解:dp的思路很好想,dp[i]代表i结尾的递增序列的个数;
dp[i] = 1 + sum{dp[j]}(j < i, a[j] < a[i]);
看到sum(dp[j]),a[]j < a[i],我们很容易想到树状数组;由于数字过大,那么我们离散化下就好了;
代码:
#include<iostream>
#include<cmath>
#include<cstring>
#include<cmath>
#include<algorithm>
#include<set>
#include<queue>
#include<vector>
using namespace std;
const int MAXN = ;
int dp[MAXN];
int a[MAXN], b[MAXN];
int tree[MAXN];
typedef long long LL;
const int MOD = 1e9 + ;
int lowbit(int x){return x & (-x);}
void update(int i, int x){
while(i < MAXN){
tree[i] += x;
tree[i] %= MOD;
i += lowbit(i);
}
}
int sum(int i){
int ans = ;
while(i > ){
ans += tree[i];
ans %= MOD;
i -= lowbit(i);
}
return ans;
}
int main()
{
int n;
while (~scanf("%d", &n)) {
for (int i = ; i < n; i++) {
scanf("%d", a + i);
b[i] = a[i];
}
sort(a, a + n);
memset(dp, , sizeof(dp));
memset(tree, , sizeof(tree));
LL ans = ;
int k = unique(a, a + n) - a;
for (int i = ; i < n; i++) {
int p = lower_bound(a, a + k, b[i]) - a + ;
dp[i] = sum(p - ) + ;
update(p, dp[i]);
}
for(int i = ; i < n; i++){
ans += dp[i];
ans %= MOD;
}
printf("%lld\n", ans % MOD);
}
return ;
}

比赛的时候刚开始想着用优先队列优化的,没什么用,还是超时;

代码:

#include<iostream>
#include<cmath>
#include<cstring>
#include<cmath>
#include<algorithm>
#include<set>
#include<queue>
using namespace std;
const int MAXN = ;
int dp[MAXN];
int a[MAXN];
int tree[MAXN];
typedef long long LL;
const int MOD = 1e9 + ;
struct Node{
int v,p;
friend bool operator < (Node a, Node b){
return a.v > b.v;
}
};
int main()
{
int n;
while (~scanf("%d", &n)) {
for (int i = ; i <= n; i++) {
scanf("%d", a + i);
}
memset(dp, , sizeof(dp));
LL ans = ;
priority_queue<Node>Q, Q1;
Node x;
for (int i = ; i <= n; i++) {
dp[i] = ;
x.v = a[i];x.p = i;
Q.push(x);
while(Q.top().v < a[i]){
dp[i] = (dp[i] + dp[Q.top().p])%MOD;
Q1.push(Q.top());
Q.pop();
}
while(!Q1.empty()){
Q.push(Q1.top());
Q1.pop();
}
}
for(int i = ; i <= n; i++){
ans += dp[i];
ans %= MOD;
}
printf("%lld\n", ans % MOD);
}
return ;
}

线段树又写了下,发现还是用二分好,用了结构体各种错。。。

代码:

#include<iostream>
#include<cstdio>
#include<cmath>
#include<cstring>
#include<algorithm>
using namespace std;
#define ll root<<1
#define rr root<<1|1
#define lson ll,l,mid
#define rson rr,mid+1,r
const int MOD = 1e9 + ;
const int MAXN = ;
typedef long long LL;
LL tree[MAXN << ];
LL dp[MAXN];
void pushup(int root){
tree[root] = (tree[ll] + tree[rr]) % MOD;
}
void update(int root, int l, int r, int a, int b){
int mid = (l + r) >> ;
if(a == l && a == r){
tree[root] += b;
return;
}
if(mid >= a)
update(lson, a, b);
else
update(rson, a, b);
pushup(root);
}
LL ans;
void query(int root, int l, int r, int a, int b){
int mid = (l + r) >> ;
if(l >= a && r <= b){
ans += tree[root];
ans %= MOD;
return;
}
LL ans = ;
if(mid >= a)
query(lson, a, b);
if(mid < b)
query(rson, a, b);
return ;
}
int a[MAXN], b[MAXN];
int main(){
int N;
while(~scanf("%d", &N)){
for (int i = ; i < N; i++) {
scanf("%d", a + i);
b[i] = a[i];
}
sort(a, a + N);
memset(tree, , sizeof(tree));
memset(dp, , sizeof(dp));
int k = unique(a, a + N) - a;
for(int i = ; i < N; i++){
int p = lower_bound(a, a + k, b[i]) - a + ;
ans = ;
query(, , N + , , p - );
dp[i] = ans + ;
dp[i] %= MOD;
update(, , N + , p, dp[i]);
}
printf("%lld\n", tree[] % MOD);
}
return ;
}

第十四个目标(dp + 树状数组 + 线段树)的更多相关文章

  1. 洛谷P2414 阿狸的打字机 [NOI2011] AC自动机+树状数组/线段树

    正解:AC自动机+树状数组/线段树 解题报告: 传送门! 这道题,首先想到暴力思路还是不难的,首先看到y有那么多个,菜鸡如我还不怎么会可持久化之类的,那就直接排个序什么的然后按顺序做就好,这样听说有7 ...

  2. 树状数组 && 线段树应用 -- 求逆序数

    参考:算法学习(二)——树状数组求逆序数 .线段树或树状数组求逆序数(附例题) 应用树状数组 || 线段树求逆序数是一种很巧妙的技巧,这个技巧的关键在于如何把原来单纯的求区间和操作转换为 求小于等于a ...

  3. hdu1394(枚举/树状数组/线段树单点更新&区间求和)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1394 题意:给出一个循环数组,求其逆序对最少为多少: 思路:对于逆序对: 交换两个相邻数,逆序数 +1 ...

  4. hdu 5147 Sequence II【树状数组/线段树】

    Sequence IITime Limit: 5000/2500 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) Problem ...

  5. hdu 1166:敌兵布阵(树状数组 / 线段树,入门练习题)

    敌兵布阵 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)Total Submis ...

  6. 51nod 1680区间求和 (dp+树状数组/线段树)

    不妨考虑已知一个区间[l,r]的k=1.k=2....k=r-l+1这些数的答案ans(只是这一个区间,不包含子区间) 那么如果加入一个新的数字a[i](i = r+1) 则新区间[l, i]的答案为 ...

  7. HDU 5293 Train chain Problem - 树链剖分(树状数组) + 线段树+ 树型dp

    传送门 题目大意: 一颗n个点的树,给出m条链,第i条链的权值是\(w_i\),可以选择若干条不相交的链,求最大权值和. 题目分析: 树型dp: dp[u][0]表示不经过u节点,其子树的最优值,dp ...

  8. 数据结构--树状数组&&线段树--基本操作

    随笔目的:方便以后对树状数组(BIT)以及基本线段树的回顾 例题链接:http://acm.hdu.edu.cn/showproblem.php?pid=1166 例题:hdu 1166 敌兵布阵 T ...

  9. BZOJ_1901_&_ZJU_2112_Dynamic_Rankings_(主席树+树状数组/线段树+(Treap/Splay))

    描述 http://www.lydsy.com/JudgeOnline/problem.php?id=1901 给出一个长度为n的数列A,有m次询问,询问分两种:1.修改某一位置的值;2.求区间[l, ...

随机推荐

  1. Mongo服务器集群配置【转】

    http://www.cnblogs.com/wly923/tag/MongoDB/ 当前标签: MongoDB   Mongo服务器集群配置学习三——分片 风行影者 2013-04-14 22:35 ...

  2. 【技术文档】《算法设计与分析导论》R.C.T.Lee等·第7章 动态规划

    由于种种原因(看这一章间隔的时间太长,弄不清动态规划.分治.递归是什么关系),导致这章内容看了三遍才基本看懂动态规划是什么.动态规划适合解决可分阶段的组合优化问题,但它又不同于贪心算法,动态规划所解决 ...

  3. debian 源

    把下面的源覆盖在/etc/apt/sources.list deb http://http.debian.net/debian wheezy main deb-src http://http.debi ...

  4. Java数据结构漫谈-LinkedList

    同样是List的数据结构,LinkedList是使用了前后指针,指明节点的方式来表示链表的,这与之前介绍的ArrayList http://www.cnblogs.com/yakovchang/p/j ...

  5. "ORA-00942: 表或视图不存在 "的原因和解决方法[转]

    采用Oracle数据库,使用Powerdesigner设计,生成Sql文件导入后查询出现“ORA-00942: 表或视图不存在 ”,很是郁闷,这个问题以前出现过,当初解决了,但因好久没有使用,这次竟然 ...

  6. silverlight .net后台 设置visifire控件图表样式 属性说明

    .net后台 代码: 如图 Chart chart = new MyCharts();  //图表            //chart.Watermark = false;  //没好使       ...

  7. JSP 实现 之 调用java方法实现MySQL数据库备份和恢复

    package cn.qm.db; import java.io.BufferedReader; import java.io.DataInputStream; import java.io.IOEx ...

  8. C#设置鼠标在控件上面时,改变光标形状

    //设置鼠标在控件上面时,改变光标形状 private void pictureBox_macroLogo_MouseHover(object sender, System.EventArgs e) ...

  9. paip.输入法编程----一级汉字1000个

    paip.输入法编程----一级汉字1000个.txt 作者Attilax ,  EMAIL:1466519819@qq.com  来源:attilax的专栏 地址:http://blog.csdn. ...

  10. 我的django之旅(二)模板和静态文件

    我的django之旅(二)模板和静态文件 标签(空格分隔): django 1.为什么要使用模板 在上一篇博文中,提到了HttpReponse,但是HttpReponse只能传送字符串,如果要构建一个 ...