CQBZOJ 【重庆市NOIP模拟赛】避难向导
题目描述
“特大新闻,特大新闻!全国爆发了一种极其可怕的病毒,已经开始在各个城市 中传播开来!全国陷入了巨大的危机!大量居民陷入恐慌,想要逃到其它城市以 避难!经调查显示,该病毒来自于C 市的A 学校的一次非法的……” “哎。”你关上电视,叹了口气。作为A 学校的校长,你一天前为了保命,独自 逃离了A 学校,抛弃了全校师生,包括那个曾经帮你计算并拆除道路的工程师。 你良心受到了巨大的谴责,因此决定做出一些补救,回答一些逃难的人提出的询 问。 已知该国一共有n 个城市,并且1 号城市是首都。(n-1)条双向的公路连接这些 城市,通过这些公路,任意两个城市之间存在且仅存在一条路径。每条公路有一 个长度。如果一个城市只与一条公路相连,则称它为边境城市。 该国政府有一个奇怪的规定:每个城市有一个封闭系数di,定义di 为离这个城 市最远的边境城市到这个城市的距离。市民们认为,一个城市的安全系数Si 和 它的封闭系数有很重要的联系。a,b,c 是该国的幸运数字,所以大家公认一个 城市的安全系数Si = (di + a) * b mod c。 市民们一共会提出m 次询问。每个询问包含三个信息,xi,yi 和qi。xi 是询问 者所在的城市编号。你要为这个询问者在xi 到yi 的必经之路上找出一个离xi 最近的避难城市,并且要求这个避难城市的安全系数大于等于qi。如果存在这 样的城市(包含xi 和yi),则输出城市编号,否则输出一行包括一个数-1。
输入
第一行五个数:依次是n, m, a, b, c。 接下来n-1 行描述公路的信息。每行三个数,前两个数代表这条公路连接的两个 城市的编号,第三个数表示这条公路的长度。 再接下来m 行,每行描述一个询问,包含三个数xi, yi 和qi。
输出
对于每个询问,输出一行包含一个整数,存在符合要求的城市则输出城市编号, 不存在则输出-1。
样例输入
7 6 5 6 20
1 2 4
2 4 2
2 5 3
1 3 5
3 6 6
6 7 7
7 5 15
3 4 5
5 4 2
4 5 2
6 6 10
3 5 19
样例输出
6
3
2
4
6
-1
这个题目本身就是两种题型的合并
1、树的最长路径
2、树形倍增数组寻找最值
m^2p 刚学倍增就做这种有(wei)趣(suo)的题真的好吗
先解决最长链的问题,我用的是DP做法,时间复杂度为O(n):
设g[root][0]与g[root][1]为以root为根节点的子树由树根到叶节点的最长链和次长链,注意,最长链与次长链无重合边
一次DFS即可初始化
再设f[root][0]与f[root][1]表示以root为起点的最长链和次长链,也无重合边,一次DFS也可做到
最麻烦的就是倍增查询了:
先看看m的范围 300000
300000 是一个转折点,因为O(nlognlogn)在300000下会TLE
这就让我们只能用O(nlogn)来解决问题
状态应该不难定义:
设f[i][j]表示i节点的第2j个祖先的节点编号
设g[i][j]表示i节点到第2j个祖先之间的点权最大值
接下来便是O(logn)的查询,O(longnlogn)很好想,但现实就是这么骨感QwQ
设c=LCA(a,b),将一个路径拆分成a−>c−>b两部分
将总任务拆分成两个子任务
将一条x−y的直链看成A和B两部分
A中包含2的整数次幂个节点,且A的长度大于x−y总长度的一半
通过这样的观点来考察上述两个子任务:
一、离x最近的:
知道k时,可以O(1)判断答案是否在A中。
① 如果在A中,枚举i从k−1到0即可得到答案。
② 如果A中不存在合法解,则令x=f[x][k],递归进行以上过程。
对于①,整个递归过程中只有得出答案前会被执行一次。
对于②,每次递归后k的值减少,而k上界为logn,下界为0,所以递归深度为O(logn)
综上,该操作复杂度为O(logn)
二、离x最远的:
① 优先令x=f[x][k],递归在B段中求解。如递归返回后已经得到答案,则直接返回答案。
② 如经过①没有获得答案,则O(1)检查A段中是否有答案,如果没有,则x−y整段路径中无解。如果有,则枚举i从k−1到0即可定位答案。
类似子任务一,该部分的复杂度为O(logn)。
码了三个小时,终于AC了!!!!!!
代码如下:
#include<cstdio>
#include<cmath>
#include<cstring>
#include<algorithm>
#define LOG 20
#define maxn 100000
using namespace std;
inline int getint()
{
int num=0,flag=1;char c;
while((c=getchar())<'0'||c>'9')if(c=='-')flag=-1;
while(c>='0'&&c<='9')num=num*10+c-48,c=getchar();
return num*flag;
}
int s[maxn+5];
int fir[2*maxn+5],nxt[2*maxn+5],to[2*maxn+5],dis[2*maxn+5],cnt;
int n,m,a,b,c;
inline void newnote(int u,int v,int w){to[++cnt]=v,dis[cnt]=w,nxt[cnt]=fir[u],fir[u]=cnt;}
struct node1{
int f[maxn+5][2],g[maxn+5][2];
bool vis[maxn+5];
inline void work(int root)
{
int i;
vis[root]=1;
for(i=fir[root];i;i=nxt[i]) if(!vis[to[i]])
{
work(to[i]);
if(g[to[i]][0]+dis[i]>g[root][0])
g[root][1]=g[root][0],g[root][0]=g[to[i]][0]+dis[i];
else if(g[to[i]][0]+dis[i]>g[root][1])
g[root][1]=g[to[i]][0]+dis[i];
}
}
inline void dp(int x,int fa,int c)
{
vis[x]=1;
if(x==1)
f[x][0]=g[x][0],f[x][1]=g[x][1];
else if(f[fa][0]==g[x][0]+c)
{
if(g[x][0]>=f[fa][1]+c)
f[x][0]=g[x][0],f[x][1]=max(f[fa][1]+c,g[x][1]);
else
f[x][0]=f[fa][1]+c,f[x][1]=g[x][0];
}
else
f[x][0]=f[fa][0]+c,f[x][1]=g[x][0];
for(int i=fir[x];i;i=nxt[i])
if(!vis[to[i]])dp(to[i],x,dis[i]);
}
}farthest;
struct node2{
int f[maxn+5][LOG+1],g[maxn+5][LOG+1],dep[maxn+5];
bool vis[maxn+5];
inline void dfs(int x)
{
for(int i=fir[x];i;i=nxt[i]) if(!vis[to[i]])
{
vis[to[i]]=1;dep[to[i]]=dep[x]+1;f[to[i]][0]=x;
g[to[i]][0]=max(s[to[i]],s[x]);dfs(to[i]);
}
}
inline void dp()
{
for(int j=1;j<=LOG;j++)for(int i=1;i<=n;i++)
{
f[i][j]=f[f[i][j-1]][j-1];
g[i][j]=max(g[f[i][j-1]][j-1],g[i][j-1]);
}
}
inline int getg(int u,int k)
{
int ans=-(1<<30);
for(int i=LOG;i>=0;i--)
if(k&(1<<i))ans=max(ans,g[u][i]),u=f[u][i];
return ans;
}
inline int getk(int u,int k)
{
for(int i=LOG;i>=0;i--)
if(k&(1<<i))u=f[u][i];
return u;
}
inline int getd(int u,int d){return getk(u,dep[u]-d);}
inline int LCA(int u,int v)
{
int x=getd(u,min(dep[v],dep[u]));
int y=getd(v,min(dep[u],dep[v]));
if(x==y)return x;
for(int i=LOG;i>=0;i--)
if(f[x][i]!=f[y][i])x=f[x][i],y=f[y][i];
return f[x][0];
}
inline int getg1(int x,int y,int w)
{
if(x==y)return s[x]>=w?x:-1;
int t=log2(dep[x]-dep[y]);
if(g[x][t]>=w)
{
for(int i=t-1;i>=0;i--)
if(g[x][i]<w)x=f[x][i];
return s[x]>=w?x:f[x][0];
}
else return getg1(f[x][t],y,w);
}
inline int getg2(int x,int y,int w)
{
if(x==y)return s[x]>=w?x:-1;
int t=log2(dep[x]-dep[y]);
int tmp=getg2(f[x][t],y,w);
if(tmp!=-1) return tmp;
if(g[x][t]>=w)
{
for(int i=t-1;i>=0;i--)
if(g[f[x][i]][i]>=w)x=f[x][i];
return s[f[x][0]]>=w?f[x][0]:x;
}
return -1;
}
}getans;
int main()
{
int i,u,v,w;
n=getint(),m=getint(),a=getint(),b=getint(),c=getint();
for(i=1;i<n;i++)
{
u=getint(),v=getint(),w=getint();
newnote(u,v,w),newnote(v,u,w);
}
farthest.work(1);
memset(farthest.vis,0,sizeof farthest.vis);
farthest.dp(1,0,0);
for(i=1;i<=n;i++)
s[i]=((long long)((farthest.f[i][0]+a)%c)*(b%c))%c;
getans.dep[1]=getans.vis[1]=1,getans.dfs(1);
getans.dp();
for(i=1;i<=m;i++)
{
u=getint(),v=getint(),w=getint();
if(u==v){printf("%d\n",s[u]<w?-1:u);continue;}
int lca=getans.LCA(u,v);
int g1=getans.getg1(u,lca,w),g2=getans.getg2(v,lca,w);
if(~g1)printf("%d\n",g1);
else printf("%d\n",g2);
}
}
CQBZOJ 【重庆市NOIP模拟赛】避难向导的更多相关文章
- NOIP模拟赛20161022
NOIP模拟赛2016-10-22 题目名 东风谷早苗 西行寺幽幽子 琪露诺 上白泽慧音 源文件 robot.cpp/c/pas spring.cpp/c/pas iceroad.cpp/c/pas ...
- contesthunter暑假NOIP模拟赛第一场题解
contesthunter暑假NOIP模拟赛#1题解: 第一题:杯具大派送 水题.枚举A,B的公约数即可. #include <algorithm> #include <cmath& ...
- NOIP模拟赛 by hzwer
2015年10月04日NOIP模拟赛 by hzwer (这是小奇=> 小奇挖矿2(mining) [题目背景] 小奇飞船的钻头开启了无限耐久+精准采集模式!这次它要将原矿运到泛光之源的矿 ...
- 大家AK杯 灰天飞雁NOIP模拟赛题解/数据/标程
数据 http://files.cnblogs.com/htfy/data.zip 简要题解 桌球碰撞 纯模拟,注意一开始就在袋口和v=0的情况.v和坐标可以是小数.为保险起见最好用extended/ ...
- 队爷的讲学计划 CH Round #59 - OrzCC杯NOIP模拟赛day1
题目:http://ch.ezoj.tk/contest/CH%20Round%20%2359%20-%20OrzCC杯NOIP模拟赛day1/队爷的讲学计划 题解:刚开始理解题意理解了好半天,然后发 ...
- 队爷的Au Plan CH Round #59 - OrzCC杯NOIP模拟赛day1
题目:http://ch.ezoj.tk/contest/CH%20Round%20%2359%20-%20OrzCC杯NOIP模拟赛day1/队爷的Au%20Plan 题解:看了题之后觉得肯定是DP ...
- 队爷的新书 CH Round #59 - OrzCC杯NOIP模拟赛day1
题目:http://ch.ezoj.tk/contest/CH%20Round%20%2359%20-%20OrzCC杯NOIP模拟赛day1/队爷的新书 题解:看到这题就想到了 poetize 的封 ...
- CH Round #58 - OrzCC杯noip模拟赛day2
A:颜色问题 题目:http://ch.ezoj.tk/contest/CH%20Round%20%2358%20-%20OrzCC杯noip模拟赛day2/颜色问题 题解:算一下每个仆人到它的目的地 ...
- CH Round #52 - Thinking Bear #1 (NOIP模拟赛)
A.拆地毯 题目:http://www.contesthunter.org/contest/CH%20Round%20%2352%20-%20Thinking%20Bear%20%231%20(NOI ...
随机推荐
- Jquery Validate表单验证,自定义校验正整数
// 添加自定义校验规则,校验正整数 jQuery.validator.addMethod("positiveinteger", function(value, element) ...
- PHP常用函数拾遗
PHP中常用过滤函数addslashes().mysql_real_escape_string().mysql_escape_string() 如addslashes().mysql_real_es ...
- 使用 HttpClient 进行表单提交时,遇到的问题
问题 在开发微信支付的小微商户进件接口时,需要通过表单来上传身份证图片等数据.在微信支付接口文档也说明了,需要使用 multipart/form-data 的方式发送请求..NET 提供了 Multi ...
- 为什么IIS应用程序池回收时间默认被设置为1740分钟?
作者:斯科特 福赛斯/Scott Forsyth日期:2013/04/06地址:http://weblogs.asp.net/owscott/why-is-the-iis-default-app-po ...
- Hive性能优化(全面)
1.介绍 首先,我们来看看Hadoop的计算框架特性,在此特性下会衍生哪些问题? 数据量大不是问题,数据倾斜是个问题. jobs数比较多的作业运行效率相对比较低,比如即使有几百行的表,如果多次关联多次 ...
- 从零开始のcocos2dx生活(六)EventDispatcher
EventDispatcher可能是所有的里面比较不容易理解也不容易看的 我说自己的理解可能会误导到你们-[索了你们看不下去>< 我写了几乎所有的代码的注释,有的是废话跳过就好 主要的代码 ...
- 《美国纽约摄影学院摄影教材》PDF教材
下载地址: 美国纽约摄影学院摄影教材(上册).pdf 美国纽约摄影学院摄影教材(下册).pdf 欢迎你到纽约摄影学院来,我们急切地等待着开课,你们也在 跃跃欲试了.那就让我们马上开始吧! 你已 ...
- 「Luogu P3931」SAC E#1 - 一道难题 Tree 解题报告
圆原题面 我环顾四周,发现大佬们的写法都好高端! 比较差劲的我,只能交上一份DFS的题解 思路: DFS(当然了,其他算法也行) 要想切断叶子节点到根节点的连接 就是在叶子节点和根节点之间砍掉一条边 ...
- git命令的基本使用流程
git命令的基本使用流程 前提:已经基本熟悉git的常用命令 1.提交项目 首先,在对项目进行完善后,如果想要将项目更新到github或是阿里code上去,先要使用 git add -A 这一命令的作 ...
- json查询结果绑定
M_Hisorder.doQuery = function (){ $("#dataList").empty(); var data = ""; var url ...