Fish eating fruit

\[Time Limit: 1000 ms \quad Memory Limit: 262144 kB
\]

题意

大体的题意就是给出一棵树,求每一对点之间的距离,然后把该距离存在距离 \(\mod 3\) 的位置,输出总和。

思路

令两个 \(dp\) 数组和两个辅助 \(dp\) 的数组。

\(dp1[i][j]\) 表示从 \(i\) 为起点往下到各个点距离 \(\mod 3\) 后为 \(j\) 的距离总和。

\(cnt1[i][j]\) 表示以 \(i\) 为起点往下到各个点距离 \(\mod 3\) 后为 \(j\) 的节点个数。

\(dp2[i][j]\) 表示从 \(i\) 起点往上一步后到各个点距离 \(\mod 3\) 后为 \(j\) 的距离总和。

\(cnt2[i][j]\) 表示以 \(i\) 为起点往上一步后到各个点距离 \(\mod 3\) 后为 \(j\) 的节点个数。


对于两个 \(dp\) 分别跑一遍 \(dfs\)


对于 \(dp1\) 比较好处理,直接往下 \(dfs\)

以 \(u\) 开始的答案等于从 \(v\) 开始的答案加上这一条边 \(w\) 的贡献,可以得到

\[ dp1[u][(j+w)\%3] = \sum (dp1[v][j] + cnt1[v][j]*w)\\
cnt1[u][(j+w)\%3] = \sum cnt1[v][j]
\]


对于 \(dp2\) 会比较麻烦,需要用 \(fa\) 节点向上的贡献在加上 \(fa\) 节点往下的贡献在减去 \(fa\) 节点往 \(u\) 走的贡献。这些节点就是 \(u\) 往上走一步后可以走到的所有节点。这样算出真实的节点数和距离总和,然后 \(u\) 才能开始转移。

设 \(faw\) 为从 \(u\) 到 \(fa\) 的路径长度

计算真实的节点数:

\[ c[j] = cnt2[fa][j]+cnt1[fa][j]\\
c[(j+faw)\%3] -= cnt1[u][j]
\]

计算真实的距离总和:

\[ d[j] = dp2[fa][j]+dp1[fa][j] \\
d[(j+faw)\%3] -= dp1[u][0]+cnt1[u][j]*faw
\]

则最后的 \(dp2\) 就可以利用 \(d\) 和 \(c\) 得到了

\[ dp2[u][(j+faw)\%3] = c[j]*faw+d[j] \\
cnt2[u][(j+faw)\%3] = c[j]
\]

/***************************************************************
> File Name : a.cpp
> Author : Jiaaaaaaaqi
> Created Time : Mon 16 Sep 2019 08:55:33 PM CST
***************************************************************/ #include <map>
#include <set>
#include <list>
#include <ctime>
#include <cmath>
#include <stack>
#include <queue>
#include <cfloat>
#include <string>
#include <vector>
#include <cstdio>
#include <bitset>
#include <cstdlib>
#include <cstring>
#include <iostream>
#include <algorithm>
#include <unordered_map>
#define lowbit(x) x & (-x)
#define mes(a, b) memset(a, b, sizeof a)
#define fi first
#define se second
#define pb push_back
#define pii pair<int, int> typedef unsigned long long int ull;
typedef long long int ll;
const int maxn = 1e5 + 10;
const int maxm = 1e5 + 10;
const ll mod = 1e9 + 7;
const ll INF = 1e18 + 100;
const int inf = 0x3f3f3f3f;
const double pi = acos(-1.0);
const double eps = 1e-8;
using namespace std; int n, m;
int cas, tol, T; vector< pii > vv[maxn];
ll cnt1[maxn][3], cnt2[maxn][3];
ll dp1[maxn][3], dp2[maxn][3]; void dfs1(int u, int fa) {
cnt1[u][0] = 1;
for(auto i : vv[u]) {
int v = i.fi, w = i.se;
if(v == fa) continue;
dfs1(v, u);
dp1[u][(0+w)%3] += (cnt1[v][0]*w%mod+dp1[v][0])%mod;
dp1[u][(1+w)%3] += (cnt1[v][1]*w%mod+dp1[v][1])%mod;
dp1[u][(2+w)%3] += (cnt1[v][2]*w%mod+dp1[v][2])%mod;
for(int j=0; j<3; j++) dp1[u][j] %= mod;
cnt1[u][(0+w)%3] += cnt1[v][0];
cnt1[u][(1+w)%3] += cnt1[v][1];
cnt1[u][(2+w)%3] += cnt1[v][2];
}
} void dfs2(int u, int fa) {
if(u!=1) {
int faw;
for(auto i : vv[u]) {
if(i.fi == fa) {
faw = i.se;
break;
}
}
int c[3] = { 0 };
c[0] = cnt2[fa][0]+cnt1[fa][0];
c[1] = cnt2[fa][1]+cnt1[fa][1];
c[2] = cnt2[fa][2]+cnt1[fa][2];
c[(0+faw)%3] -= cnt1[u][0];
c[(1+faw)%3] -= cnt1[u][1];
c[(2+faw)%3] -= cnt1[u][2];
ll d[3] = { 0 };
d[0] = (dp2[fa][0]+dp1[fa][0])%mod;
d[1] = (dp2[fa][1]+dp1[fa][1])%mod;
d[2] = (dp2[fa][2]+dp1[fa][2])%mod;
d[(0+faw)%3] = ((d[(0+faw)%3] - (cnt1[u][0]*faw%mod+dp1[u][0])%mod+mod)%mod+mod)%mod;
d[(1+faw)%3] = ((d[(1+faw)%3] - (cnt1[u][1]*faw%mod+dp1[u][1])%mod+mod)%mod+mod)%mod;
d[(2+faw)%3] = ((d[(2+faw)%3] - (cnt1[u][2]*faw%mod+dp1[u][2])%mod+mod)%mod+mod)%mod; dp2[u][(0+faw)%3] = (c[0]*faw%mod+d[0])%mod;
dp2[u][(1+faw)%3] = (c[1]*faw%mod+d[1])%mod;
dp2[u][(2+faw)%3] = (c[2]*faw%mod+d[2])%mod;
cnt2[u][(0+faw)%3] += c[0];
cnt2[u][(1+faw)%3] += c[1];
cnt2[u][(2+faw)%3] += c[2];
}
for(auto i : vv[u]) {
int v = i.fi, w = i.se;
if(v == fa) continue;
dfs2(v, u);
}
} int main() {
// freopen("in", "r", stdin);
while(~scanf("%d", &n)) {
for(int i=1; i<=n; i++) {
vv[i].clear();
}
mes(dp1, 0), mes(dp2, 0);
mes(cnt1, 0), mes(cnt2, 0);
for(int i=1, u, v, w; i<n; i++) {
scanf("%d%d%d", &u, &v, &w);
u++, v++;
vv[u].pb(make_pair(v, w));
vv[v].pb(make_pair(u, w));
}
dfs1(1, 0);
dfs2(1, 0);
// for(int i=1; i<=n; i++) {
// for(int j=0; j<3; j++) {
// printf("dp1[%d][%d] = %lld, cnt1[%d][%d] = %lld\n", i, j, dp1[i][j], i, j, cnt1[i][j]);
// }
// }
// cout << "-----------------" << endl;
// for(int i=1; i<=n; i++) {
// for(int j=0; j<3; j++) {
// printf("dp2[%d][%d] = %lld, cnt2[%d][%d] = %lld\n", i, j, dp2[i][j], i, j, cnt2[i][j]);
// }
// }
ll ans0, ans1, ans2;
ans0 = ans1 = ans2 = 0;
for(int i=1; i<=n; i++) {
ans0 = (ans0+dp1[i][0]+dp2[i][0])%mod;
ans1 = (ans1+dp1[i][1]+dp2[i][1])%mod;
ans2 = (ans2+dp1[i][2]+dp2[i][2])%mod;
}
printf("%lld %lld %lld\n", ans0, ans1, ans2);
}
return 0;
}

Fish eating fruit 沈阳网络赛(树形dp)的更多相关文章

  1. HDU 6201 2017沈阳网络赛 树形DP或者SPFA最长路

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=6201 题意:给出一棵树,每个点有一个权值,代表商品的售价,树上每一条边上也有一个权值,代表从这条边经过 ...

  2. 2019沈阳网赛树形dp

    https://nanti.jisuanke.com/t/41403 2019沈阳网络赛D题 树形dp.一棵树,求任意两个点的距离之和.u-v和v-u算两次.两点之间的距离分为三类,模3等于0,1,2 ...

  3. hdu 4274 2012长春赛区网络赛 树形dp ***

    设定每个节点的上限和下限,之后向上更新,判断是否出现矛盾 #include<cstdio> #include<iostream> #include<algorithm&g ...

  4. 2018 ICPC 沈阳网络赛

    2018 ICPC 沈阳网络赛 Call of Accepted 题目描述:求一个算式的最大值与最小值. solution 按普通算式计算方法做,只不过要同时记住最大值和最小值而已. Convex H ...

  5. 2019 沈阳网络赛 D Fish eating fruit ( 树形DP)

    题目传送门 题意:求一颗树中所有点对(a,b)的路径长度,路径长度按照模3之后的值进行分类,最后分别求每一类的和 分析:树形DP \(dp[i][j]\) 表示以 i 为根的子树中,所有子节点到 i ...

  6. 2019 沈阳网络赛 Fish eating fruit

    这题看了三个月,终于过了,第一次看的时候没学树形DP,想用点分治但是不会 后来学了二次扫描,就有点想法了.... 这东西也真就玄学了吧... #include<iostream> #inc ...

  7. The Preliminary Contest for ICPC Asia Shenyang 2019 D. Fish eating fruit(树形dp)

    题意:求一棵树上所有路径和模3分别为0 1 2 的权值的和 思路:树形dp 增加一个记录儿子节点满足条件的个数的数组 不要放在一起dp不然答案跟新会有问题 #include <bits/stdc ...

  8. 5.21 省选模拟赛 luogu P4297 [NOI2006]网络收费 树形dp

    LINK:网络收费 还是自己没脑子. 早上思考的时候 发现树形dp不可做 然后放弃治疗了. 没有合理的转换问题的模型是我整个人最大的败笔. 暴力也值得一提 爆搜之后可以写成FFT的形式的计算贡献的方法 ...

  9. 沈阳网络赛 F - 上下界网络流

    "Oh, There is a bipartite graph.""Make it Fantastic." X wants to check whether a ...

随机推荐

  1. pgsql_pg的数据类型

    PostgreSQL 提供了丰富的数据类型.用户可以使用 CREATE TYPE 命令在数据库中创建新的数据类型.PostgreSQL 的数据类型被分为四种,分别是基本数据类型.复合数据类型.域和伪类 ...

  2. ng的动画过渡

    动画过渡两种方法 1.使用angular+animation实现 在app-module.ts中引入 BrowserAnimationsModule 1.import { BrowserAnimati ...

  3. go-gin-api 路由中间件 - 日志记录

    概述 首先同步下项目概况: 上篇文章分享了,规划项目目录和参数验证,其中参数验证使用的是 validator.v8 版本,现已更新到 validator.v9 版本,最新代码查看 github 即可. ...

  4. springboot+RabbitMQ 问题 RabbitListener 动态队列名称:Attribute value must be constant

    因为多机环境fanout广播模式,每台机器需要使用自己的队列接受消息 所以尝试使用以下的方案 private static final String QUEUE_NAME="foo.&quo ...

  5. 图解微信小程序---scroll_view实现首页排行榜布局

    图解微信小程序---scroll_view实现首页排行榜布局 什么是scroll-view? 滚动视图可滚动视图区域.使用竖向滚动时,需要给scroll-view一个固定高度,通过 WXSS 设置 h ...

  6. Asp.Net页面刷新防止跳转到其他浏览器或新的选项卡

    前端页面js代码: <head> <script> window.name = "PremaritalCheckup_ManSocietyAgreeForm" ...

  7. winform按钮美化(非图片)

    在开发过程中,突然发现vs自带的按钮属性中通过修改Button控件的BackColor的颜色和字体颜色(ForeColor属性)及大小,如下图 就能达到简单美化按钮的效果,下面是显示效果 有兴趣的同学 ...

  8. Python关于多继承

    大部分面向对象的编程语言(除了C++)都只支持单继承,而不支持多继承,为什么呢?因为多继承不仅增加编程复杂度,而且容易导致莫名其妙的错误. Python虽然语法上支持多继承,但是却不推荐使用多继承,而 ...

  9. 架构师小跟班:如何高效又安全的清理Linux服务器上的缓存?

    操作服务器上的生产环境,一定要慎之又慎,安全第一,优化第二! 一些基本原理 说到清理内存,那么不得不提到/proc这一个虚拟文件系统,这里面的数据和文件都是内存中的实时数据,很多参数的获取都可以从下面 ...

  10. python高级编程——网络编程(三)

    TCP和并发服务器 与UDP不同的是,他是一个面向连接的,可靠的数据传输协议 TCP通信比较复杂 先写一个TCP服务器,一般步骤如下: 1.首先是要创建一个socket套接字:socket() 2.服 ...