Successor HDU - 4366 分块
代码+注释:
1 /*
2 题意:
3 一共有n个人,其中0号是总裁(金字塔顶尖)。后面输入其他n-1个人的信息啊a、b、c,分别代表第i个人的上级是a,他的
4 忠诚度为b,他的能力为c。后面有m次询问。他问你能不能找到一个能力比他高的,且忠诚度最高的人。(注意能力只需要
5 大于此人就可以,不需要最高)
6
7
8 题解:
9 这道题目用的是分块的方法
10
11 首先需要跑一遍dfs来确定每一个人所处于金字塔的位置。从总裁开始向他的下属跑dfs(注意是dfs不是bfs)
12 因为这样我们最后会得到一个pre数组和一个last数组。这两个数组的作用就是来看某个人x他在金字塔中的位置
13 以及他的下属最大的位置(这个位置的值是cnt)。那么pre[x]到last[x]之间的所有人都是x的下属
14
15 然后分块也是按照位置的编号进行分块的,这个样子维护也好维护。分块后会得到一个maxx数组,这个数组的作用就是
16 maxx[x][y]:在第x个块里,能力大于y且忠诚度最高的值,这个p[x][y]存放的是这个值所对应的人
17
18 id数组里面放的是就是,id[x]表示:位置为x的人的在输入中的编号是多少
19
20 具体看代码
21 */
22 #include<stdio.h>
23 #include<string.h>
24 #include<iostream>
25 #include<algorithm>
26 #include<math.h>
27 #include<vector>
28 using namespace std;
29 const int maxn=50005;
30 const int maxm=sqrt(50000)+5;
31 int n,m,block,num;
32 int belong[maxn],l[maxn],r[maxn],last[maxn],cnt=0,pre[maxn];
33 int id[maxn],v[maxn],a[maxn],maxx[maxm][maxm],p[maxm][maxm];
34 struct node
35 {
36 int ability,loyalty,id;
37 node(int l=0,int a=0,int id=0):loyalty(l),ability(a),id(id) {};
38 bool operator <(const node& r)const
39 {
40 return ability<r.ability;
41 }
42 } c[maxn];
43 vector<node>b[maxn];
44 vector<int>g[maxn];
45 void build()
46 {
47 block = sqrt(n);
48 num = n / block;
49 if(n % block) num++;
50 for(int i = 1; i <= num; i++)
51 {
52 l[i] = (i-1)*block+1;
53 r[i] = i*block;
54 }
55 r[num] = n;
56 for(int i = 1; i <= n; i++)
57 {
58 belong[i] = (i-1)/block+1;
59 }
60 for(int i = 1; i <= num; i++)
61 {
62 for(int j = l[i]; j <= r[i]; j++)
63 {
64 int idd = id[j];
65 c[j] = node(v[idd], a[idd], idd);
66 b[i].push_back(node(v[idd], a[idd], idd));
67 }
68 sort(b[i].begin(), b[i].end());
69 int len = b[i].size();
70 maxx[i][len] = -1;
71 for(int j = len-1; j >= 0; j--)
72 {
73 if(maxx[i][j+1] < b[i][j].loyalty)
74 {
75 maxx[i][j] = b[i][j].loyalty;
76 p[i][j] = b[i][j].id;
77 }
78 else
79 {
80 maxx[i][j] = maxx[i][j+1];
81 p[i][j] = p[i][j+1];
82 }
83 }
84 }
85 }
86 int query(int fa,int x,int y)
87 {
88 int ans=-1,id=-1;
89 if(belong[x]==belong[y])
90 {
91 for(int i=x; i<=y; ++i)
92 {
93 if(a[fa]<c[i].ability && ans<c[i].loyalty)
94 {
95 ans=c[i].loyalty;
96 id=c[i].id;
97 }
98 }
99 }
100 else
101 {
102 for(int i=x; i<=r[belong[x]]; ++i)
103 {
104 if(a[fa]<c[i].ability && ans<c[i].loyalty)
105 {
106 ans=c[i].loyalty;
107 id=c[i].id;
108 }
109 }
110
111 for(int i=belong[x]+1; i<belong[y]; ++i)
112 {
113 int pos = upper_bound(b[i].begin(), b[i].end(), node(0, a[fa], 0)) - b[i].begin();
114 int cur = maxx[i][pos];
115 if(ans < cur)
116 {
117 ans = cur;
118 id = p[i][pos];
119 }
120 }
121
122 for(int i=l[belong[y]]; i<=y; ++i)
123 {
124 if(a[fa]<c[i].ability && ans<c[i].loyalty)
125 {
126 ans=c[i].loyalty;
127 id=c[i].id;
128 }
129 }
130 }
131 return id;
132 }
133 void init()
134 {
135 for(int i=0; i<=n; ++i)
136 {
137 b[i].clear();
138 g[i].clear();
139 }
140 cnt=0;
141 }
142 void dfs(int root,int fa)
143 {
144 pre[root]=++cnt;
145 id[cnt]=root;
146 int len=g[root].size();
147 for(int i=0; i<len; ++i)
148 {
149 int v=g[root][i];
150 if(v!=fa) dfs(v,root);
151 }
152 last[root]=cnt;
153 }
154 int main()
155 {
156 int t;
157 scanf("%d",&t);
158 while(t--)
159 {
160 scanf("%d%d",&n,&m);
161 init();
162 for(int i=1; i<=n-1; ++i)
163 {
164 int x;
165 scanf("%d%d%d",&x,&v[i],&a[i]);
166 g[x].push_back(i);
167 }
168 dfs(0,0);
169 build();
170 while(m--)
171 {
172 int x;
173 scanf("%d",&x);
174 printf("%d\n",query(x,pre[x],last[x]));
175 }
176 }
177 return 0;
178 }
Successor HDU - 4366 分块的更多相关文章
- Successor HDU - 4366 (预处理,线段树,dfs序)
Successor HDU - 4366 Sean owns a company and he is the BOSS.The other Staff has one Superior.every s ...
- Successor hdu 4366 线段树
题意: 现在n个人,其中编号0的是老板,之后n-1个员工,每个员工只有一个上司,有一个忠诚值和能力值.每次要解雇一个人的时候,从他的下属中选取能力值大于他的且忠诚值最高的一个,若不存在则输出-1.共m ...
- HDU - 4366 Successor DFS区间+线段树
Successor:http://acm.hdu.edu.cn/showproblem.php?pid=4366 参考:https://blog.csdn.net/colin_27/article/d ...
- HDU 5213 分块 容斥
给出n个数,给出m个询问,询问 区间[l,r] [u,v],在两个区间内分别取一个数,两个的和为k的对数数量. $k<=2*N$,$n <= 30000$ 发现可以容斥简化一个询问.一个询 ...
- HDU 5145 分块 莫队
给定n个数,q个询问[l,r]区间,每次询问该区间的全排列多少种. 数值都是30000规模 首先考虑计算全排列,由于有同种元素存在,相当于每次在len=r-l+1长度的空格随意放入某种元素即$\bin ...
- HDU 2920 分块底数优化 暴力
其实和昨天写的那道水题是一样的,注意爆LL $1<=n,k<=1e9$,$\sum\limits_{i=1}^{n}(k \mod i) = nk - \sum\limits_{i=1}^ ...
- hdu 4366 Successor - CDQ分治 - 线段树 - 树分块
Sean owns a company and he is the BOSS.The other Staff has one Superior.every staff has a loyalty an ...
- HDU 4366 Successor 分块做法
http://acm.hdu.edu.cn/showproblem.php?pid=4366 今日重新做了这题的分块,果然是隔太久了,都忘记了.. 首先,用DFS序变成一维的问题 关键是它有两个权值, ...
- HDU 4366 Successor(dfs序 + 分块)题解
题意:每个人都有一个上司,每个人都有能力值和忠诚值,0是老板,现在给出m个询问,每次询问给出一个x,要求你找到x的所有直系和非直系下属中能力比他高的最忠诚的人是谁 思路:因为树上查询很麻烦,所以我们直 ...
随机推荐
- Restful API是什么、为什么、怎么使用
Restful API 文章目录 Restful API 1.REST是什么以及它的 6 个限制 REST是什么? REST的6个限制 2. Restful是什么 Restful是什么 RESTful ...
- kubernets之服务的实现方式
一 服务如何通过kubernetes集群的组件来实现其功能 1.1 节点上的所有的服务相关的功能实现都是通过节点上面的kube-proxy来实现的,服务提供了一个或者多个服务IP以及端口对客户端开 ...
- 攻防世界 - Misc(一)
base64÷4: 1.下载附件,是一个.txt文件,打开是一串字符, 666C61677B453333423746443841334238343143413936393945444442413234 ...
- oracle新增ID主键列,如何补全旧数据的ID值
1.创建SEQUENCE CREATE SEQUENCE MONKEY.TEST_ADD_IDCOL_ID CACHE 100; 2.新增表栏位 ALTER TABLE MONKEY.TEST_ADD ...
- Scrapy——將爬取圖片下載到本地
1. Spider程序: 1 import scrapy, json 2 from UnsplashImageSpider.items import ImageItem 3 4 class Unspl ...
- 图解ARP协议
前置知识:MAC地址 在说到ARP协议之前,需要了解MAC地址,在OSI七层模型中,第三层是网络层,要解决的是下一跳机制确定的设备位置,具体来说就是网卡地址,MAC地址用于在网络中唯一标示一个网卡,一 ...
- Service Mesh架构的持续演进 单体模块化 SOA 微服务 Service Mesh
架构不止-严选Service Mesh架构的持续演进 网易严选 王育松 严选技术团队 2019-11-25 前言同严选的业务一样,在下层承载它的IT系统架构一样要生存.呼吸.增长和发展,否则过时的.僵 ...
- 数据水印 watermark
外发数据创建水印 产品通过对外发数据进行添加数据标记.自动生成水印.数据源追溯等功能,避免了内部人员外发数据泄露无法对事件追溯,提高了数据传递的安全性和可追溯能力. 数据水印系统_数据安全管理工具_[ ...
- Python 2.x 和 Python 3.x
Python 2.x 默认不支持中文,具体原因,等到介绍 字符编码 时给大家讲解 Python 2.x 的解释器名称是 python Python 3.x 的解释器名称是 python3 目前市场上有 ...
- Java多线程--两种实现方式
进程概述: 在这之前,有必要了解一下什么是进程? 在一个操作系统中,每个独立的执行的程序都可称为一个进程,也就是"正在运行的程序".如图所示: 线程概述: 如上所述,每个运行的程序 ...