Codeforces 464E The Classic Problem (最短路 + 主席树 + hash)
这个题加深了我对主席树的理解,是个好题。每次更新某个点的距离时,是以之前对这个点的插入操作形成的线段树为基础,在O(logn)的时间中造出了一颗新的线段树,相比直接创建n颗线段树更省时间。比较的时候二分比较,为了加快比较给每个点设置一个hash值。
代码:
#include <bits/stdc++.h>
using namespace std;
const unsigned long long P = 13331;
const int mod = 1000000007;
const int maxn = 100010; int head[maxn], Next[maxn * 2], edge[maxn * 2], ver[maxn * 2];
int tot, tote;
int b[maxn * 2], pre[maxn];
int mx, dis[maxn];
bool vis[maxn];
int Stack[maxn], Top; void add(int x, int y, int z) {
ver[++tote] = y;
edge[tote] = z;
Next[tote] = head[x];
head[x] = tote;
}
struct SegementTree {
int ls, rs;
int sum;
unsigned long long hash;
}tr[maxn * 200]; void pushup(int x) {
tr[x].sum = tr[tr[x].ls].sum + tr[tr[x].rs].sum;
unsigned long long tmp1 = tr[tr[x].ls].hash, tmp2 = tr[tr[x].rs].hash;
tr[x].hash = (((tmp1 * P + tmp2) ^ tmp1) * P ^ tmp2) * P;
} void insert(int &now, int l, int r, int pos) {
int p = now;
now = ++tot;
tr[now] = tr[p];
if(l == r) {
tr[now].sum = 1;
tr[now].hash = l + P;
return;
}
int mid = (l + r) >> 1;
if(pos <= mid) insert(tr[now].ls, l, mid, pos);
else insert(tr[now].rs, mid + 1, r, pos);
pushup(now);
} void del(int &now, int l, int r, int ql, int qr) {
if(l >= ql && r <= qr) {
now = 0;
return;
}
int p = now;now = ++tot;
tr[now] = tr[p];
int mid = (l + r) >> 1;
if(ql <= mid) del(tr[now].ls, l, mid, ql, qr);
if(qr > mid) del(tr[now].rs, mid + 1, r, ql, qr);
pushup(now);
} int query(int now, int l, int r, int pos) {
if(l == r) {
if(tr[now].sum == 0) return l;
return -1;
}
int mid = (l + r) >> 1;
if(pos > mid) return query(tr[now].rs, mid + 1, r, pos);
int ans = query(tr[now].ls, l, mid, pos);
if(ans != -1) return ans;
return query(tr[now].rs, mid + 1, r, pos);
} int add(int now, int pos) {
int p = query(now, 0, mx, pos);
if(p > pos) del(now, 0, mx, pos, p - 1);
insert(now, 0, mx, p);
return now;
} int get_sum(int now, int l, int r) {
int mid = (l + r) >> 1;
if(now == 0) return 0;
if(l == r) return b[l];
return (get_sum(tr[now].ls, l, mid) + get_sum(tr[now].rs, mid + 1, r)) % mod;
} bool cmp(int x, int y, int l, int r) {
if(l == r) return tr[x].sum >= tr[y].sum;
int mid = (l + r) >> 1;
if(tr[tr[x].rs].hash != tr[tr[y].rs].hash)
return cmp(tr[x].rs, tr[y].rs, mid + 1, r);
else return cmp(tr[x].ls, tr[y].ls, l, mid);
} struct node {
int x, y;
bool operator < (const node& rhs) const {
return cmp(x, rhs.x, 0, mx);
}
}; priority_queue<node> q; void dijkstra(int s) {
dis[s] = ++tot, q.push((node){dis[s], s});
while(!q.empty()) {
node tmp = q.top();
q.pop();
if(vis[tmp.y]) continue;
vis[tmp.y] = 1;
int x = tmp.y;
for (int i = head[x]; i; i = Next[i]) {
int y = ver[i], z = add(dis[x], edge[i]);
if(!dis[y] || !cmp(z, dis[y], 0, mx)) {
pre[y] = x;
dis[y] = z;
q.push((node){z, y});
}
}
}
} void print(int x, int deep) {
if(x == 0) {
printf("%d\n", deep);
return;
}
print(pre[x], deep + 1);
printf("%d ", x);
}
int main() {
int n, m, x, y, z, s, t;
scanf("%d%d", &n, &m);
for (int i = 1; i <= m; i++) {
scanf("%d%d%d", &x, &y, &z);
add(x, y, z);
add(y, x, z);
mx = max(mx, z);
}
b[0] = 1;
mx += 20;
for (int i = 1; i <= mx; i++) {
b[i] = (b[i - 1] << 1) % mod;
}
scanf("%d%d", &s, &t);
dijkstra(s);
if(dis[t] == 0) printf("-1\n");
else {
printf("%d\n", get_sum(dis[t], 0, mx));
while(t) {
Stack[++Top] = t;
t = pre[t];
}
printf("%d\n", Top);
while(Top) {
printf("%d ", Stack[Top]);
Top--;
}
}
}
Codeforces 464E The Classic Problem (最短路 + 主席树 + hash)的更多相关文章
- [Codeforces 464E] The Classic Problem(可持久化线段树)
[Codeforces 464E] The Classic Problem(可持久化线段树) 题面 给出一个带权无向图,每条边的边权是\(2^{x_i}(x_i<10^5)\),求s到t的最短路 ...
- Codeforces 464E The Classic Problem(主席树+最短路+哈希,神仙题)
题目链接 题意:给出一张 \(n\) 个点 \(m\) 条边的无向图,第 \(i\) 条边连接 \(u_i,v_i\),边权为 \(2^{w_i}\),求 \(s\) 到 \(t\) 的最短路. \( ...
- CodeForces 464E The Classic Problem | 呆克斯歘 主席树维护高精度
题意描述 有一个\(n\)点\(m\)边的无向图,第\(i\)条边的边权是\(2^{a_i}\).求点\(s\)到点\(t\)的最短路长度(对\(10^9 + 7\)取模). 题解 思路很简单--用主 ...
- Codeforces 464E. The Classic Problem
题目大意 给定一张$n$个点, $m$条边的无向图,求$S$ 到$T$的最短路,其中边权都是$2^k$的形式$n,m,k<=10^5$,结果对$10^9+7$取模 题解 大佬好厉害 跑一边dij ...
- Codeforces 464E #265 (Div. 1) E. The Classic Problem 主席树+Hash
E. The Classic Problem http://codeforces.com/problemset/problem/464/E 题意:给你一张无向带权图,求S-T的最短路,并输出路径.边权 ...
- CF 464E The Classic Problem
补一补之前听课时候的题. 考虑使用dij算法求最短路,因为边权存不下,所以考虑用主席树维护二进制位,因为每一次都只会在一个位置进行修改,所以可以暴力进位,这样均摊复杂度是对的. <算法导论> ...
- codeforces gym #101161E - ACM Tax(lca+主席树)
题目链接: http://codeforces.com/gym/101161/attachments 题意: 给出节点数为$n$的树 有$q$次询问,输出$a$节点到$b$节点路程中,经过的边的中位数 ...
- 【bzoj3218】a+b Problem 最小割+主席树
数据范围:$n≤5000$,$a,l,r≤10^9$,$b,w,p≤2\times 10^5$. 我们考虑一种暴力的最小割做法: 首先令$sum=\sum\limits_{i=1}^{n} b_i+w ...
- HDU 4729 An Easy Problem for Elfness 主席树
题意: 给出一棵树,每条边有一个容量. 有若干次询问:\(S \, T \, K \, A \, B\),求路径\(S \to T\)的最大流量. 有两种方法可以增大流量: 花费\(A\)可以新修一条 ...
随机推荐
- JavaScript如何处理解析JSON数据详解
JSON (JavaScript Object Notation)一种简单的数据格式,比xml更轻巧. JSON 是 JavaScript 原生格式,这意味着在 JavaScript 中处理 JSON ...
- 利用HTML5开发Android笔记(中篇)
资源来自于www.mhtml5.com 杨丰盛老师成都场的PPT分享 一个很简明的demo 可以作为入门基础 学习的过程中做了点笔记 整理如下 虽然内容比较简单 但是数量还是比较多的 所以分了3篇 ( ...
- MySQL安装后默认自带数据库的作用
大家在学习MySQL时,安装后都会发现里边已经自带了几个默认的数据库,我装的MySQL5.5里边自带六个数据库 网上查了一些资料对这几个数据库的功能做一下学习. 1.information_schem ...
- group by 和 聚合函数
1.在oracle中 select * from Table group by id 会报错. 会报不是group by 表达式.为什么一定不能是 * ,而必须是分组的列或者某个列的聚合函数. 在my ...
- ENTRYPOINT 与 CMD
在Dockerfile中 ENTRYPOINT 只有最后一条生效,如果写了10条,前边九条都不生效 ENTRYPOINT 的定义为运行一个Docker容器像运行一个程序一样,就是一个执行的命令 两种写 ...
- Python 2.7_爬取妹子图网站单页测试图片_20170114
1.url= http://www.mzitu.com/74100/x,2为1到23的值 2.用到模块 os 创建文件目录; re模块正则匹配目录名 图片下载地址; time模块 限制下载时间;req ...
- karma
一个简单的工具,允许你在多个浏览器中执行JavaScript代码. Karma的主要目的是使您的测试驱动开发变得简单.快速和有趣. 我什么时候该用Karma? 您希望在真正的浏览器中测试代码. 您希望 ...
- 首次db查询时延迟明显高于后面几次。
1.如果排查到时db相关的问题的话,一般都是连接池的配置问题. 在配置好连接池时一定要注意对连接也进行初始化配置,否则可能出现连接池初始化了,但是连接并没有初始化,这样在第一次查询的时候可能会出现较大 ...
- 分区工具parted的详解及常用分区使用方法
一. parted的用途及说明 概括使用说明: parted用于对磁盘(或RAID磁盘)进行分区及管理,与fdisk分区工具相比,支持2TB以上的磁盘分区,并且允许调整分区的大小. ...
- Azure CLI脚本查看未挂载的ManagedDisk
本文介绍如何用Azure CLI的脚本查看未挂载的Managed Disk,以及Managed Disk挂载到哪些资源. 具体的脚本如下: #!/bin/bash rm -rf noownerdisk ...