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\)可以新修一条 ...
随机推荐
- 配置SSH密码登录
在客户端生成公钥: ssh-keygen –t rsa 生成的公钥默认位置在~/.ssh/目录 把公钥上传到服务器端: scp id_rsa.pub root@ip地址:文件保存路径 cat id_r ...
- ItelliJ IDEA下载及获取注册码详解
Idea安装很简单: 官网地址下载:https://www.jetbrains.com/idea/ 注册码获取:http://idea.lanyus.com/ 进入此网址,点击“获得注册码”复制使 ...
- HTML中可以连接资源的标签集合
1.<a>标签,href属性指示链接的目标,可以是HTML也可以是内部css样式.<a href="http://www.w3school.com.cn"> ...
- 16_游戏编程模式ServiceLocator 服务定位
####简单说,就是某个系统作为一个服务,对全局系统可见. Service Locator (服务定位) ``` //简单粗暴的代码, 使用声音系统 // Use a static class? Au ...
- luogu P4848 崂山白花蛇草水
https://www.luogu.org/problemnew/show/P4848 我的数据结构大概已经废了. 外层权值线段树内层kdtree,外层线段树上二分答案. 码数据结构一时爽,码完deb ...
- C++对C语言的拓展(5)—— 函数重载和函数指针结合
1.函数指针的介绍 函数指针指向某种特定类型,函数的类型由其参数及返回类型共同决定,与函数名无关.举例如下: int add(int nLeft,int nRight);//函数定义 该函数类型为in ...
- npm install -d
nodejs Error: Cannot find module 'xxx'错误 解决方案: 确定package.json里有添加相应的依赖配置 使用npm install -d 可以自动配置pack ...
- Python函数-cmp()
cmp(x, y) 作用: 比较两个对象x和y,如果x < y ,返回负数:x == y, 返回0:x > y,返回正数. 注:在python2所有版本中都可用,但在pyt ...
- Linux基础命令-echo
echo命令 功能:显示字符 (末尾自带换行功能) 语法:echo [-neE][字符串] 说明:echo会将输入的字符串送往标准输出.输出的字符串间以空白字符隔开, 并在最后加上换行号 -n 不在字 ...
- Angular5学习笔记 - 项目目录结构(二)
一.项目总体目录 README.md:项目的说明和一些常用指令说明,建议看看. e2e:看不懂暂时空着??? node_modules/:存放npm下载的组件(npm install 后自动产生,不需 ...