题解:CF593D Happy Tree Party
题解:CF593D Happy Tree Party
Description
Bogdan has a birthday today and mom gave him a tree consisting of \(n\) vertecies. For every edge of the tree \(i\) , some number \(x_i\) was written on it. In case you forget, a tree is a connected non-directed graph without cycles. After the present was granted, \(m\) guests consecutively come to Bogdan's party. When the \(i\)-th guest comes, he performs exactly one of the two possible operations:
- Chooses some number \(y_i\) , and two vertecies \(a_i\) and \(b_i\). After that, he moves along the edges of the tree from vertex \(a_i\) to vertex \(b_i\) using the shortest path (of course, such a path is unique in the tree). Every time he moves along some edge \(j\) , he replaces his current number \(y_i\) by \(\large y_i =\lfloor \frac{y_i}{x_j} \rfloor\), that is, by the result of integer division \(y_i \ div \ x_j\).
- Chooses some edge \(p_i\) and replaces the value written in it \(x_{p_i}\) by some positive integer \(c_i < x_{p_i}\) .
As Bogdan cares about his guests, he decided to ease the process. Write a program that performs all the operations requested by guests and outputs the resulting value \(y_i\) for each \(i\) of the first type.
数据保证 \(2 \leq n,m \leq 2e5, \ 1 \leq x_i \leq 1e18\)
题意:
给出一棵有边权的树。树上共有 \(n\) 个结点,现在给出 \(m\) 个操作,操作有两类:
给你一个数 \(y\) ,走一条 \(u \to \dots \to v\) 的简单路径,每经过一条权为 \(x\) 的边就令
\[\large y := \lfloor \frac{y}{x} \rfloor
\]询问最终 \(y\) 的值。
更改某条边的权。
Algorithm
又到了我最喜欢的板子题环节!
而且本题在洛谷的难度评级是NOI/NOI+/CTSC ,快来水黑题吧!
首先注意到有
\lfloor \frac {\lfloor \frac {\lfloor \frac {\lfloor \frac{y}{x_1} \rfloor}{x_2} \rfloor} {\vdots} \rfloor} {x_n} \rfloor = \lfloor \frac{y} {\prod _ {i = 1} ^ n x_i} \rfloor
\]
于是问题转化为询问树上简单路径边权积。
这是个并不困难的问题,可以简单地用树链剖分 + 线段树解决。
具体地,我们首先提出一个点作为根,这样每条边的两端结点就有了父子关系。然后将边权下放到子节点上,将根节点的权赋为1。
这样就将点权转化为了边权。不过要注意统计边权积的时候,不应该记录端点的权(此处指每条链的顶部结点)。
然后跑一下树链剖分,建立线段树即可,只需要写单点修改和区间查询,懒惰标记都不用挂。
然而,观察数据范围,\(x_i\) 在 \(1e18\) 的量级,走一条简单路径的乘积最大值可能达到
\({10^{18}}^{200000}\),你就是写高精度也存不下这玩意,何况要高精度除呢。
等等,除法?
给定 \(y\) 的值也在 \(1e18\) 的量级,这意味着 \(\prod_{i=1}^n x_i\) 一旦超过了 \(1e18\) ,结果就必然为0了。
当然了,两个 \(1e18\) 乘起来依然会爆 long long ,我们可以
- 正统地,取对数解决。或有精度误差,但影响不大。
- 还有我
__int128_t和long double解决不了的事吗? //啊这,CF 用__int128_t会CE的
于是,开始写代码吧:
#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
int n, m;
template<typename T>
inline void read(T &x)
{
char c = getchar(); x = 0;
while(c < '0' || '9' < c) c = getchar();
while('0' <= c && c <= '9')
{
x = (x << 1) + (x << 3) + c - 48;
c = getchar();
}
}
struct Mint {
long double val;
Mint():val(1) {}
Mint(ll x):val(x) {}
friend Mint operator * (Mint a, Mint b)
{
long double ret = a.val * b.val;
return (ret > 1e18)? 0: ret;
}
};
Mint &operator *= (Mint &a, Mint b) {
return a = a * b;
}
template<const int N, const int M>
class Tree {
private:
int beg[N], nex[M], tar[M], len;
int dep[N], siz[N], fat[N], hea[N];
int top[N], dfn[N], id[N], cnt;
Mint cst[M], vap[N], seg[N << 2];
public:
Tree():len(1), cnt(0) {}
inline void add_edge(int a, int b, ll c)
{
++len;
nex[len] = beg[a], beg[a] = len;
tar[len] = b, cst[len] = c;
}
void dfs1(int cur, int pa)
{
dep[cur] = dep[pa] + 1;
fat[cur] = pa, siz[cur] = 1;
for(int i = beg[cur]; i; i = nex[i])
{
if(tar[i] == pa) continue;
dfs1(tar[i], cur);
vap[tar[i]] = cst[i];
siz[cur] += siz[tar[i]];
if(hea[cur] == -1 || siz[tar[i]] > siz[hea[cur]])
hea[cur] = tar[i];
}
}
void dfs2(int cur, int pa)
{
top[cur] = pa, ++cnt;
dfn[cur] = cnt, id[cnt] = cur;
if(hea[cur] == -1) return;
dfs2(hea[cur], pa);
for(int i = beg[cur]; i; i = nex[i])
if(tar[i] != hea[cur] && tar[i] != fat[cur])
dfs2(tar[i], tar[i]);
}
#define lson (nod <<1)
#define rson ((nod <<1) | 1)
void build(int nod, int lef, int rig)
{
if(lef == rig) seg[nod] = vap[id[lef]];
else
{
int mid = (lef + rig) >> 1;
build(lson, lef, mid);
build(rson, mid + 1, rig);
seg[nod] = seg[lson] * seg[rson];
}
}
inline void init_treecut() {
memset(hea, -1, sizeof(hea));
dfs1(1, 0);
dfs2(1, 1);
vap[id[1]] = 1;
build(1, 1, n);
}
void update(int nod, int lef, int rig, int goa, Mint val)
{
if(lef == rig) seg[nod] = val;
else
{
int mid = (lef + rig) >> 1;
if(goa <= mid) update(lson, lef, mid, goa, val);
else update(rson, mid + 1, rig, goa, val);
seg[nod] = seg[lson] * seg[rson];
}
}
Mint query(int nod, int lef, int rig, int goal, int goar)
{
if(goar < lef || rig < goal) return Mint(1);
if(goal <= lef && rig <= goar) return seg[nod];
Mint ret = 1;
int mid = (lef + rig) >> 1;
ret *= query(lson, lef, mid, goal, goar);
ret *= query(rson, mid + 1, rig, goal, goar);
return ret;
}
Mint query_path(int u, int v)
{
Mint ret = 1;
while(top[u] != top[v])
{
if(dep[top[u]] < dep[top[v]]) swap(u, v);
ret *= query(1, 1, n, dfn[top[u]], dfn[u]);
u = fat[top[u]];
}
if(dfn[u] > dfn[v]) swap(u, v);
ret *= query(1, 1, n, dfn[u] + 1, dfn[v]);
return ret;
}
void update_point(int x, Mint y)
{
int u = tar[x << 1], v = tar[x << 1 | 1];
if(dep[u] < dep[v]) swap(u, v);
update(1, 1, n, dfn[u], y);
}
};
Tree<262144, 524288> T;
int main()
{
read(n), read(m);
ll x, res;
for(int i = 1, u, v; i != n; ++i)
{
read(u), read(v), read(x);
T.add_edge(u, v, x);
T.add_edge(v, u, x);
}
T.init_treecut();
for(int i = 0, a, b, key; i != m; ++i)
{
read(key);
if(key == 1)
{
read(a), read(b), read(x);
res = (ll)T.query_path(a, b).val;
if(res) printf("%lld\n", x / res);
else puts("0");
}
else
{
read(a), read(x);
T.update_point(a, x);
}
}
return 0;
}
这板子我还记得怎么写,宝刀未老啊。
题解:CF593D Happy Tree Party的更多相关文章
- 【题解】Digit Tree
[题解]Digit Tree CodeForces - 716E 呵呵以为是数据结构题然后是淀粉质还行... 题目就是给你一颗有边权的树,问你有多少路径,把路径上的数字顺次写出来,是\(m\)的倍数. ...
- 【题解】[P4178 Tree]
[题解]P4178 Tree 一道点分治模板好题 不知道是不是我见到的题目太少了,为什么这种题目都是暴力开值域的桶QAQ?? 问点对,考虑点分治吧.直接用值域树状数组开下来,统计的时候直接往树状数组里 ...
- leetcode 题解:Binary Tree Inorder Traversal (二叉树的中序遍历)
题目: Given a binary tree, return the inorder traversal of its nodes' values. For example:Given binary ...
- [LeetCode 题解]: Flatten Binary Tree to Linked List
Given a binary tree, flatten it to a linked list in-place. For example,Given 1 / \ 2 5 / \ \ 3 4 6 T ...
- leetcode题解:Construct Binary Tree from Preorder and Inorder Traversal (根据前序和中序遍历构造二叉树)
题目: Given preorder and inorder traversal of a tree, construct the binary tree. Note:You may assume t ...
- CF593D Happy Tree Party(不用树剖)
题面 题解 我们发现,对于除法有效的xi最小为2,yi最多除log次就会变成0,所以我们可以每次找路径上下一个>=2的xi,暴力除,当发现y=0时就停止 于是我们维护每个点向上走一直走到根最近的 ...
- LeetCode题解之Binary Tree Right Side View
1.题目描述 2.问题分析 使用层序遍历 3.代码 vector<int> v; vector<int> rightSideView(TreeNode* root) { if ...
- LeetCode题解之Binary Tree Pruning
1.题目描述 2.问题分析 使用递归 3.代码 TreeNode* pruneTree(TreeNode* root) { if (root == NULL) return NULL; prun(ro ...
- LeetCode题解Maximum Binary Tree
1.题目描述 2.分析 找出最大元素,然后分割数组调用. 3.代码 TreeNode* constructMaximumBinaryTree(vector<int>& nums) ...
随机推荐
- JSON<前后端的沟通>
1.什么是JSON ==>1什么是json json:是一种轻量级数据交互格式 数据交互:每一种语言的编码都不一样,他们之间互不认识.但是现在的情况是不同的语言开发出的系统也需要进行数据交互,这 ...
- python基础:异常捕捉
一.异常 python在程序运行过程中,可能会出现一些错误和异常,导致程序停止运行.我们可以通过捕捉异常,并对异常进行处理,使得程序可以正常运行 异常有很多类型,可以根据类型挨个捕捉.也可统一捕获: ...
- 2. 构建DNS集群
DNS是什么 DNS(Domain Name System,域名系统),是互联网上存储域名和IP映射关系的一个分布式数据库,他负责把域名转换为IP地址,或IP转换为域名,工作于OSI应用层之上,DNS ...
- 简单地 Makefile 书写
注意事项 每个标签分支前都不能用空格,必须用tab 标签外调用bash命令用 $(shell -),标签内可以正常使用 标签后可以指定其他标签,执行顺序是先执行其他标签,而后在执行自己 比如 all: ...
- Mybatis注解开发案例(入门)
1.创建maven工程,配置pom.xml 文件. 2.创建实体类 3.创建dao接口 4.创建主配置文件SqlMapConfig.xml 5.在SqlMapConfig.xml中导入外部配置文件jd ...
- Kafka 【入门一篇文章就够了】
初识 Kafka 什么是 kafka Kafka 是由 Linkedin 公司开发的,它是一个分布式的,支持多分区.多副本,基于 Zookeeper 的分布式消息流平台,它同时也是一款开源的基于发布订 ...
- Prometheus-Alertmanager告警对接到企业微信
之前写过将Prometheus的监控告警信息通过Alertmanager推送到钉钉群. 最近转移了阵地,需要将Prometheus监控告警信息推送到企业微信群,经过两天的摸索,以及查了网上的一些资料, ...
- ftp客户端自动同步 Windows系统简单操作ftp客户端自动同步
服务器管理工具它是一款功能强大的服务器集成管理器,包含win系统和linux系统的批量连接,vnc客户端,ftp客户端等等实用功能.我们可以使用这款软件的ftp客户端定时上传下载的功能来进实现ftp客 ...
- 【MySQL】面试官:如何添加新数据库到MySQL主从复制环境?
写在前面 今天,一名读者反馈说:自己出去面试,被面试官一顿虐啊!为什么呢?因为这名读者面试的是某大厂的研发工程师,偏技术型的.所以,在面试过程中,面试官比较偏向于问技术型的问题.不过,技术终归还是要服 ...
- python3-day5
模块,用一砣代码实现了某个功能的代码集合. 类似于函数式编程和面向过程编程,函数式编程则完成一个功能,其他代码用来调用即可,提供了代码的重用性和代码间的耦合.而对于一个复杂的功能来,可能需要多个函数才 ...