J - Super Mario HDU - 4417

这个题目我开始直接暴力,然后就超时了,不知道该怎么做,直接看了题解,这个习惯其实不太好。

不过网上的思路真的很厉害,看完之后有点伤心,感觉自己应该可以写的,但是没有写出来。

因为我们要查找一个区间小于等于c的数有多少个,这种区间处理可以用线段树很好的解决,

但是怎么解决呢,我习惯性的直接建树,然后再想怎么做,其实这样不对,

线段树对于我们来说是一个解决问题的工具,不要本末倒置了。

我们想查找一个区间小于等于c的数的个数,用线段树查询很简单,但是怎么建树才可以

让我们每次查询都得到想要的结果呢,这个肯定是不能直接把这个墙的高度直接建树的,应该要先处理一下。

我们可以把查询的高度按升序排序,这样线段树之前更新过的值就不会影响后面的值了。

但是线段树怎么更新呢,这个也可以按升序对墙排个序,这样就解决问题了。

这个思路其实没有特别难想。。。

代码很好敲,但是要注意不要漏掉了,也许都已经更新完了,但是这个答案没有在for循环里面找到。

这个要注意!!!

然后这个题目还可以用主席树写

#include <cstdio>
#include <cstring>
#include <cstdlib>
#include <queue>
#include <vector>
#include <algorithm>
#include <iostream>
#define inf 0x3f3f3f3f
#define inf64 0x3f3f3f3f3f3f3f3f
using namespace std;
typedef long long ll;
const int maxn = 2e5+;
ll sum[maxn*], ans[maxn];
int n, m;
struct node
{
ll num, id;
}a[maxn];
struct edge
{
ll l, r, h, id;
edge(ll l=,ll r=,ll h=,ll id=):l(l),r(r),h(h),id(id){}
}ex[maxn]; bool cmp(node a,node b)
{
return a.num < b.num;
} bool cmp1(edge a,edge b)
{
return a.h < b.h;
} void update(int id,int l,int r,int pos)
{
if(l==r)
{
sum[id] = ;
return;
}
int mid = (l + r) >> ;
if (pos <= mid) update(id << , l, mid, pos);
else update(id << | , mid + , r, pos);
sum[id] = sum[id << ] + sum[id << | ];
//printf("sum[%d]=%d\n", id, sum[id]);
} ll query(int id,int l,int r,int x,int y)
{
if (x <= l && y >= r) return sum[id];
int mid = (l + r) >> ;
int ans = ;
if (x <= mid) ans += query(id << , l, mid, x, y);
if (y > mid) ans += query(id << | , mid + , r, x, y);
return ans;
} int main()
{
int t;
scanf("%d", &t);
for(int cas=;cas<=t;cas++)
{
memset(ans, , sizeof(ans));
memset(sum, , sizeof(sum));
scanf("%d%d", &n, &m);
for (int i = ; i <= n; i++) scanf("%lld", &a[i].num), a[i].id = i;
for (int i = ; i <= m; i++) {
ll l, r, h;
scanf("%lld%lld%lld", &l, &r, &h);
l++, r++;
ex[i] = edge(l, r, h, i);
}
sort(a + , a + + n, cmp);
sort(ex + , ex + + m, cmp1);
printf("Case %d:\n", cas);
int k = ;
for(int i=;i<=n;i++)
{
while(a[i].num>ex[k].h&&k<=m)
{
ans[ex[k].id] = query(, , n, ex[k].l, ex[k].r);
k++;
}
update(, , n, a[i].id);
}
for (int i = k; i <= m; i++) ans[ex[i].id] = query(, , n, ex[i].l, ex[i].r);//这个注意不要丢掉了!!!
for (int i = ; i <= m; i++) printf("%lld\n", ans[i]);
}
return ;
}

线段树

#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <algorithm>
#include <queue>
#include <vector>
#define inf 0x3f3f3f3f
#define inf64 0x3f3f3f3f3f3f3f3f
using namespace std;
const int maxn = 1e5 + ;
int n, m, root[maxn], a[maxn], b[maxn], cnt;
int sum[maxn << ], lc[maxn << ], rc[maxn << ]; void build(int &rt,int l,int r)
{
rt = ++cnt;
sum[rt] = ;
if (l == r) return;
int mid = (l + r) >> ;
build(lc[rt], l, mid);
build(rc[rt], mid + , r);
// printf("rt=%d l=%d r=%d\n",rt,l,r);
} int update(int rt,int l,int r,int pos)
{
// printf("ww rt=%d l=%d r=%d pos=%d\n", rt, l, r, pos);
int id = ++cnt;
sum[id] = sum[rt] + ;
// printf("rt=%d sum[%d]=%d\n", rt, id, sum[id]);
lc[id] = lc[rt], rc[id] = rc[rt];
if (l == r) return id;
int mid = (l + r) >> ;
// printf("mid=%d rt=%d l=%d r=%d pos=%d\n", mid,rt,l,r,pos);
if (pos <= mid) lc[id] = update(lc[id], l, mid, pos);
else rc[id] = update(rc[id], mid + , r, pos);
// printf("rt=%d l=%d r=%d pos=%d\n", rt, l, r, pos);
return id;
} int query(int l,int r,int u,int v,int h)
{
int mid = (l + r) >> ;
int x = sum[lc[v]] - sum[lc[u]];
if (l == r) return sum[v] - sum[u];
int ans = ;
if (h <= mid) ans = query(l, mid, lc[u], lc[v], h);
else ans = x + query(mid + , r, rc[u], rc[v], h);
return ans;
} int main()
{
int t;
scanf("%d", &t);
for(int cas=;cas<=t;cas++)
{
cnt = ;
scanf("%d%d", &n, &m);
for (int i = ; i <= n; i++) scanf("%d", &a[i]), b[i] = a[i];
sort(b + , b + + n);
int len = unique(b + , b + + n) - b - ;
build(root[], , len);
// printf("\n\n");
for (int i = ; i <= n; i++) {
a[i] = lower_bound(b + , b + + len, a[i]) - b;
//printf("a[%d]=%d\n", i, a[i]);
root[i] = update(root[i - ], , len, a[i]);
// printf("sum[%d]=%d\n", root[i], sum[root[i]]);
// printf("\n");
}
printf("Case %d:\n", cas);
while(m--)
{
int l, r, h;
scanf("%d%d%d", &l, &r, &h);
l++, r++;
h = upper_bound(b + , b + + len, h) - b - ;
if (h == ) printf("0\n");
else printf("%d\n", query(, len, root[l - ], root[r], h));
}
}
return ;
}

主席树

J - Super Mario HDU - 4417 线段树 离线处理 区间排序的更多相关文章

  1. Super Mario HDU 4417 主席树区间查询

    Super Mario HDU 4417 主席树区间查询 题意 给你n个数(编号从0开始),然后查询区间内小于k的数的个数. 解题思路 这个可以使用主席树来处理,因为这个很类似查询区间内的第k小的问题 ...

  2. E - No Pain No Game 线段树 离线处理 区间排序

    E - No Pain No Game  HDU - 4630 这个题目很好,以后可以再写写.这个题目就是线段树的离线写法,推荐一个博客:https://blog.csdn.net/u01003321 ...

  3. Super Mario HDU - 4417 (主席树询问区间比k小的个数)

    Mario is world-famous plumber. His “burly” figure and amazing jumping ability reminded in our memory ...

  4. hdu 1166线段树 单点更新 区间求和

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

  5. AC日记——Super Mario hdu 4417

    Super Mario Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total ...

  6. HDU 3308 线段树单点更新+区间查找最长连续子序列

    LCIS                                                              Time Limit: 6000/2000 MS (Java/Oth ...

  7. Super Mario HDU - 4417 (主席树)

    Mario is world-famous plumber. His “burly” figure and amazing jumping ability reminded in our memory ...

  8. hdu 4288 线段树+离线+离散化

    http://acm.hdu.edu.cn/showproblem.php?pid=4288 開始的时候,果断TLE,做的方法是,线段树上只维护%5==3的坐标,比方1 2 3 4 5 6 7  假设 ...

  9. I Hate It HDU - 1754 线段树 单点修改+区间最值

    #include<iostream> #include<cstring> using namespace std; ; int m,n,p; struct node{ int ...

随机推荐

  1. 钩子函数 Function类

    Function 为 com.google.common.base包下接口类: public interface Function<F, T> { @Nullable T apply(@N ...

  2. Java的多线程编程模型5--从AtomicInteger开始

    Java的多线程编程模型5--从AtomicInteger开始 2011-06-23 20:50 11393人阅读 评论(9) 收藏 举报 java多线程编程jniinteger测试 AtomicIn ...

  3. python填写金数据

    ''' 将 main 中的{url}改为真 url 将 setd_data中 {姓名} {纬度} {经度} {地址} 改为确切数据 数据自行参考post 结果判断基于响应中是否包含“谢谢参与”字样 ' ...

  4. [算法总结]DFS(深度优先搜索)

    目录 一.关于DFS 1. 什么是DFS 2. DFS的搜索方式 二.DFS的具体实现 三.剪枝 1. 顺序性剪枝 2. 重复性剪枝 3. 可行性剪枝 4. 最优性剪枝 5. 记忆化剪枝 四.练习 一 ...

  5. L25词嵌入进阶GloVe模型

    词嵌入进阶 在"Word2Vec的实现"一节中,我们在小规模数据集上训练了一个 Word2Vec 词嵌入模型,并通过词向量的余弦相似度搜索近义词.虽然 Word2Vec 已经能够成 ...

  6. PHP函数:array_chunk

    array_chunk()  -  将一个数组分割成多个. 说明: array_chunk ( array $array , int $size [, bool $preserve_keys = fa ...

  7. [转]PHP利用PCRE回溯次数限制绕过某些安全限制

    这次Code-Breaking Puzzles中我出了一道看似很简单的题目pcrewaf,将其代码简化如下: <?php function is_php($data){ return preg_ ...

  8. kubernetes的Service是什么?

    service到底是什么? k8s的service定义了一个服务的访问入口地址,前端的应用通过这个入口地址访问其背后的一组由pod副本组成的集群实例.来自外部的访问请求被负载均衡到后端的各个容器应用上 ...

  9. python爬虫面试题集锦及答案

    1.爬取数据后使用哪个数据库存储数据的,为什么? - 2.你用过的爬虫框架或者模块有哪些?优缺点? - 3.写爬虫是用多进程好?还是多线程好? - 4.常见的反爬虫和应对方法? - 5.需要登录的网页 ...

  10. 蒲公英 · JELLY技术周刊 Vol.03

    蒲公英 · JELLY技术周刊 Vol.03 「蒲公英」期刊全新升级--JELLY技术周刊!深度挖掘业界热点动态,来自团队大咖的专业点评,带你深入了解团队研究的技术方向. 登高远眺 天高地迥,觉宇宙之 ...