Noi2018 归途
zz:https://blog.csdn.net/dreaming__ldx/article/details/81106748
以海拔为第一关键字对边进行从大到小的排序,然后修建kruskal重构树,这样就弄出了一颗以海拔为关键字的小根堆。然后对于每一棵子树,如果询问中的水位线是低于子树的根节点的,那么此时这棵子树中的所有叶子结点都是连通的。放到题中就是说这颗子树中任选一个点出发,到子树中的其它点都不需要花费。则它到1的实际开支为这些点中到1的最短路的最小值,这个最小值是在不考虑海平线的前提下,尽量走最短路径的权值。
然后我们假设对于当前询问,我们找到了一个子树的根节点u,满足d[u]>p且d[fa[u]]<=p (d[i]代表i点海拔)且出发点v在子树中,这时从v出发可以直接抵达子树中的任意一个叶子结点。因此我们需要从众多叶子节点(即原图给的点)中选出一个距离1号点花费最小的。时间复杂度O(T*N*LogN)
算法流程如下:
我们首先要求出每个点到1号点的最小花费,这个直接dijstra+最短路预处理。然后是要建出kruskal重构树,再然后维护以每个点作为根节点时子树中距离1号点的最小花费,这个建完树后一个简单的dfs搞定。最后是如何找到点u,这时我们要让一个重要的算法登场:倍增算法。直接加上点权>p的限制在树上倍增即可。
#include<bits/stdc++.h>
#define N 400005
#define M 800005
using namespace std;
inline int read(){
int ans=0;
char ch=getchar();
while(!isdigit(ch))ch=getchar();
while(isdigit(ch))ans=(ans<<3)+(ans<<1)+(ch^48),ch=getchar();
return ans;
}
inline void write(int x)
{
if(x>9)write(x/10);
putchar(x%10+'0');
}
int n,m,T,q,k,s,vis[N],first[N<<1],head[N],cntx=0,d[N],dep[N],f[N][20],fa[N<<1],lastans=0,totx=0;
struct Node、
{
int u,v,l,a;
}e[M],p[N<<1];
struct edge
{
int v,next;
}tr[M<<1];
struct node
{
int v,next,w;
}t[M];
struct heap
{
int u,v;
};
inline bool operator<(heap a,heap b){return a.v>b.v;}
inline void dijstra(int s=1) //找出每个点到s的最短路
{
memset(vis,false,sizeof(vis));
memset(d,0x3f,sizeof(d));
priority_queue<heap>q;
d[s]=0;
q.push((heap){s,d[s]});
while(!q.empty()){
heap x=q.top();
q.pop();
if(vis[x.u])continue;
vis[x.u]=true;
for(int i=head[x.u];i;i=t[i].next){
int v=t[i].v;
if(vis[v])continue;
if(d[v]>d[x.u]+t[i].w){
d[v]=d[x.u]+t[i].w
;
q.push((heap){v,d[v]});
}
}
}
for(int i=1;i<=n;++i)
p[i].l=d[i];//结果放到p[i].l中
}
inline bool cmp(Node a,Node b)
{
return a.a>b.a;
}
inline int find(int x)
{
return x==fa[x]?fa[x]:fa[x]=find(fa[x]);
}
inline void add(int u,int v)//重构树是用来连边的
{
tr[++cntx].v=v;
tr[cntx].next=first[u];
first[u]=cntx;
}
inline void addx(int u,int v,int w)//原图连边用的,带权
{
t[++totx].v=v;
t[totx].next=head[u];
t[totx].w=w;
head[u]=totx;
}
inline void dfs(int u,int pa)
{
dep[u]=dep[pa]+1,f[u][0]=pa;
for(int i=1;i<=19;++i)f[u][i]=f[f[u][i-1]][i-1];
for(int i=first[u];i;i=tr[i].next)
{
int v=tr[i].v;
dfs(v,u);
p[u].l=min(p[u].l,p[v].l);
//u,v是相连的,也就是说如果这边的海拔线高于规定的话,则从u,v之间可以直达
//于是要选两者的最小值
//u要么与两个叶子点相连,要么与一个叶子点一个新增点相连
//最终u为以u为根的子树中,所有点到1的最短路
//也就是说如果能走到u点,则此时p[u].l就是其子树中某个点到1的最短路
}
}
inline int query(int x,int y)
{
for(int i=19;i>=0;--i)
if(dep[x]-(1<<i)>0&&p[f[x][i]].a>y)
//一直向上跳,只要所跳的点海拔高于规定值. 注意是高于,不是高于等于
x=f[x][i];
return p[x].l;
}
inline void kruskal()
{
int tot=0,cnt=n;
for(int i=1;i<=(n<<1);++i)
fa[i]=i;
sort(e+1,e+m+1,cmp);
for(int i=1;i<=m;++i)
{
int u=e[i].u,v=e[i].v;
int fx=find(u),fy=find(v);
if(fx!=fy){
add(++cnt,fx);
add(cnt,fy);
fa[fx]=cnt;
fa[fy]=cnt;
p[cnt].a=e[i].a;
++tot;
}
if(tot==n-1)break;
}
dfs(cnt,0);
while(q--)
{
int x=(k*lastans+read()-1)%n+1,y=(k*lastans+read())%(s+1);
write(lastans=query(x,y));
puts("");
}
}
int main(){
T=read();
while(T--){
lastans=0,n=read(),m=read();
memset(e,0,sizeof(e)),cntx=0,totx=0;
memset(first,0,sizeof(first));
memset(head,0,sizeof(head));
memset(f,0,sizeof(f));
for(int i=1;i<=m;++i)
e[i].u=read(),e[i].v=read(),e[i].l=read(),
e[i].a=read(),addx(e[i].u,e[i].v,e[i].l),addx(e[i].v,e[i].u,e[i].l);
for(int i=n+1;i<=(n<<1);++i) //注意是2*n个点,因为要新增n-1个点
p[i].l=0x3f3f3f3f;
dijstra();
q=read(),k=read(),s=read();
kruskal();
}
return 0;
}
Noi2018 归途的更多相关文章
- 关于NOIP2016与NOI2018
NOIP2016惨淡收场了,距离省一还有相当一大段距离,省队更是差了十条街去了,不过没关系. 既然已经对信息学产生了兴趣,竞赛无疑是最好的锻炼场所. 路是自己选择的,伤痕累累也要走下去. 还有一年,事 ...
- 【NOI2018模拟5】三角剖分Bsh
[NOI2018模拟5]三角剖分Bsh Description 给定一个正 n 边形及其三角剖分,共 2n - 3 条边 (n条多边形的边和n-3 条对角线),每条边的长度为 1. 共 q 次询问,每 ...
- JZOJ 5602.【NOI2018模拟3.26】Cti
JZOJ 5602.[NOI2018模拟3.26]Cti Description 有一个 \(n×m\) 的地图,地图上的每一个位置可以是空地,炮塔或是敌人.你需要操纵炮塔消灭敌人. 对于每个炮塔都有 ...
- 【NOI2018模拟】Yja
[NOI2018模拟]Yja Description 在平面上找\(n\)个点,要求这 \(n\)个点离原点的距离分别为 \(r1,r2,...,rn\) .最大化这\(n\) 个点构成的凸包面积,凸 ...
- BZOJ5418[Noi2018]屠龙勇士——exgcd+扩展CRT+set
题目链接: [Noi2018]屠龙勇士 题目大意:有$n$条龙和初始$m$个武器,每个武器有一个攻击力$t_{i}$,每条龙有一个初始血量$a_{i}$和一个回复值$p_{i}$(即只要血量为负数就一 ...
- BZOJ5417[Noi2018]你的名字——后缀自动机+线段树合并
题目链接: [Noi2018]你的名字 题目大意:给出一个字符串$S$及$q$次询问,每次询问一个字符串$T$有多少本质不同的子串不是$S[l,r]$的子串($S[l,r]$表示$S$串的第$l$个字 ...
- BZOJ5419[Noi2018]情报中心——线段树合并+虚树+树形DP
题目链接: [NOI2018]情报中心 题目大意:给出一棵n个节点的树,边有非负边权,并给出m条链,对于每条链有一个代价,要求选出两条有公共边的链使两条链的并的边权和-两条链的代价和最大. 花了一天的 ...
- 【BZOJ5417】[NOI2018]你的名字(线段树,后缀自动机)
[BZOJ5417][NOI2018]你的名字(线段树,后缀自动机) 题面 BZOJ 洛谷 题解 首先考虑\(l=1,r=|S|\)的做法,对于每次询问的\(T\)串,暴力在\(S\)串的\(SAM\ ...
- 【BZOJ5416】【NOI2018】冒泡排序(动态规划)
[BZOJ5416][NOI2018]冒泡排序(动态规划) 题面 BZOJ 洛谷 UOJ 题解 考场推出了就是两个上升子序列,并且最长下降子序列长度不超过\(2\)...然后大力暴力状压\(dp\)混 ...
随机推荐
- iOS 设备尺寸以及图标尺寸
iPhone 4和iPod Touch 4有一个新的特性:在屏幕尺寸不变的前提下,分辨率提升一倍(320 x 480 => 640 x 960).苹果将这个特性命名为Retina. 用苹果的话讲 ...
- 前端每日实战:12# 视频演示如何用纯 CSS 创作一种文字断开的交互特效
效果预览 按下右侧的"点击预览"按钮在当前页面预览,点击链接全屏预览. https://codepen.io/zhang-ou/pen/LmjNgL 可交互视频教程 此视频是可以交 ...
- 阅读《Effective Java》每条tips的理解和总结(2)(持续更新)
15. 使类和成员的可访问性最小化 一个好用的类的属性必须要隐藏起来,干净的将它与类的api分离开来,类之间只通过api相互使用,降低他们之间的耦合性.为了做到这一点,建议根据情况选择尽可能低的访问级 ...
- Django框架架构总览
Django框架架构总览 理解Django是如何运作的 条目创建于 2013-08-14 1464 views 服务器君一共花费 15.204 ms 进行了 4 次数据库查询,努力地为 ...
- LA 4327 Parade(单调队列优化dp)
题目链接: 题目大意(摘自刘汝佳<<算法竞赛入门经典--训练指南>>):F城是由n+1条横向路和m+1条竖向路组成.你的任务是从最南边的路走到最北边的路,使得走过的路上的高兴值 ...
- proxy配置
关于config.js里面proxy的配置: proxy: { '/api': { target: 'http://192.168.***.**:8500', cha ...
- 【GDOI2016模拟3.9】暴走的图灵机
题目 分析 我们发现当两个字符串合并时,a0.a1表示左右两个字符串中有多少个T,C表示合并处新增的T的个数,那么 a0=a1 a1=a0+a1+C 令s0和s1表示左右手两个字符串,那么每一次操作后 ...
- TinyMCE不可编辑
1. 通过配置在控件初始化时设置 tinyMCE.init({ readonly : 1 }); 2.tinymce.activeEditor.getBody().setAttribute('cont ...
- unigui ios微信界面错位和点击失灵问题
IOS微信下会出现二个严重问题: 1.输入框失去焦点导致控件错位,造成无点正常点击. 此问题是微信自带浏览器,一直遗留问题, 尝试了多种方法始终无解.因此要用来开发公众号的一定要注意. 2.界面下移 ...
- CF D. Labyrinth 01BFS
由于上下走不限制,所以按照贪心,我们应该尽可能走上下方向. 我们可以开一个双端队列,并认为每次提取队首的时候得到的是到达该点的最优策略.(这个一定是唯一的,因为不可能向右走几格,然后再退回去. ) 那 ...