[loj6271]生成树求和
将每一位拆开考虑,即不妨假设$0\le c<3$
考虑矩阵树定理,即统计所有生成树边权乘积的和,但我们这里要将边权相加,很明显将其作为幂次(如果作为$cx+1$无法对3取模)
更具体的,也就是将每一个位置从1变为$x^{c}$,系数对$10^{9}+7$取模,相乘时幂次对3取模
另外,高斯消元时需要对该多项式求逆,考虑$(ax^{2}+bx+c)^{-1}$即找到$dx^{2}+ex+f$满足
$$
(ax^{2}+bx+c)(dx^{2}+ex+f)\equiv 1(mod\ x^{3}-1)
$$
(幂次对3取模等价于原多项式对$x^{3}-1$取模,关于这个正确性不证明也没关系,可以将这个对$x^{3}-1$看作一个符号)
展开后,也就得到了如下的三元一次方程组
$$
\begin{cases}ad+bf+ce=0\\ ae+bd+cf=1\\af+be+cd=0\end{cases}
$$
通过一些计算,可以解得即
$$
\begin{cases}d=\frac{b^{2}-ac}{a^{3}+b^{3}+c^{3}-3abc}\\e=\frac{a^{2}-bc}{{a^{3}+b^{3}+c^{3}-3abc}}\\f=\frac{c^{2}-ab}{{a^{3}+b^{3}+c^{3}-3abc}}\end{cases}
$$
根据$a^{3}+b^{3}+c^{3}-3abc=\frac{(a+b+c)((a-b)^{2}+(b-c)^{2}+(c-a)^{2})}{2}$,若其为0其实与原来的0多项式是一样的,因此要判定要求$a\ne b$或$b\ne c$且$a+b+c\ne 0$($a+b+c\not\equiv 0(mod\ 10^{9}+7)$
更特别的,在如果最后一个式子不满足上述条件,则还要乘上去
由此计算即可,时间复杂度为$o(n^{3}\log_{3}c)$


1 #include<bits/stdc++.h>
2 using namespace std;
3 #define N 105
4 #define mod 1000000007
5 struct Edge{
6 int x,y,z;
7 }e[N*N];
8 int n,m,ans;
9 int pow(int n,int m){
10 int s=n,ans=1;
11 while (m){
12 if (m&1)ans=1LL*ans*s%mod;
13 s=1LL*s*s%mod;
14 m>>=1;
15 }
16 return ans;
17 }
18 struct poly{
19 int a[3];
20 poly(){
21 a[0]=a[1]=a[2]=0;
22 }
23 void init(){
24 a[0]=a[1]=a[2]=0;
25 }
26 bool zero(){
27 return (a[0]==a[1])&&(a[1]==a[2])||((0LL+a[0]+a[1]+a[2])%mod==0);
28 }
29 poly operator + (const poly &k)const{
30 poly o;
31 for(int i=0;i<3;i++)o.a[i]=(a[i]+k.a[i])%mod;
32 return o;
33 }
34 poly operator - ()const{
35 poly o;
36 for(int i=0;i<3;i++)o.a[i]=(mod-a[i])%mod;
37 return o;
38 }
39 poly operator * (const poly &k)const{
40 poly o;
41 for(int i=0;i<3;i++)
42 for(int j=0;j<3;j++)o.a[(i+j)%3]=(o.a[(i+j)%3]+1LL*a[i]*k.a[j])%mod;
43 return o;
44 }
45 poly inv(){
46 int s=mod-3LL*a[0]*a[1]%mod*a[2]%mod;
47 for(int i=0;i<3;i++)s=(s+1LL*a[i]*a[i]%mod*a[i])%mod;
48 s=pow(s,mod-2);
49 poly o;
50 o.a[0]=(1LL*a[0]*a[0]-1LL*a[1]*a[2]%mod+mod)%mod*s%mod;
51 o.a[1]=(1LL*a[2]*a[2]-1LL*a[0]*a[1]%mod+mod)%mod*s%mod;
52 o.a[2]=(1LL*a[1]*a[1]-1LL*a[0]*a[2]%mod+mod)%mod*s%mod;
53 return o;
54 }
55 }a[N][N];
56 poly guess(){
57 poly ans;
58 ans.a[0]=1;
59 for(int i=1;i<n;i++){
60 int k=-1;
61 for(int j=i;j<n;j++)
62 if (!a[j][i].zero()){
63 k=j;
64 break;
65 }
66 if (k<0)return ans*a[i][i];
67 if (k!=i){
68 ans=(-ans);
69 for(int j=i;j<n;j++)swap(a[i][j],a[k][j]);
70 }
71 ans=ans*a[i][i];
72 for(int j=i+1;j<n;j++){
73 poly o=a[j][i]*a[i][i].inv();
74 for(int k=i;k<n;k++)a[j][k]=a[j][k]+(-o*a[i][k]);
75 }
76 }
77 return ans;
78 }
79 int main(){
80 freopen("sum.in","r",stdin);
81 freopen("sum.out","w",stdout);
82 scanf("%d%d",&n,&m);
83 for(int i=1;i<=m;i++)scanf("%d%d%d",&e[i].x,&e[i].y,&e[i].z);
84 int s=1;
85 while (1){
86 bool flag=0;
87 for(int i=1;i<n;i++)
88 for(int j=1;j<n;j++)a[i][j].init();
89 for(int i=1;i<=m;i++){
90 poly o;
91 o.a[e[i].z%3]=1;
92 if (e[i].x<n)a[e[i].x][e[i].x]=a[e[i].x][e[i].x]+o;
93 if (e[i].y<n)a[e[i].y][e[i].y]=a[e[i].y][e[i].y]+o;
94 if ((e[i].x<n)&&(e[i].y<n)){
95 a[e[i].x][e[i].y]=a[e[i].x][e[i].y]+(-o);
96 a[e[i].y][e[i].x]=a[e[i].y][e[i].x]+(-o);
97 }
98 if (e[i].z)flag=1;
99 e[i].z/=3;
100 }
101 if (!flag)break;
102 poly o=guess();
103 ans=(ans+1LL*s*o.a[1]+2LL*s*o.a[2])%mod;
104 s=3LL*s%mod;
105 }
106 printf("%d",ans);
107 return 0;
108 }
[loj6271]生成树求和的更多相关文章
- loj6271 「长乐集训 2017 Day10」生成树求和 加强版(矩阵树定理,循环卷积)
loj6271 「长乐集训 2017 Day10」生成树求和 加强版(矩阵树定理,循环卷积) loj 题解时间 首先想到先分开三进制下每一位,然后每一位分别求结果为0,1,2的树的个数. 然后考虑矩阵 ...
- 4.9 省选模拟赛 生成树求和 变元矩阵树定理 生成函数 iDFT 插值法
有同学在loj上找到了加强版 所以这道题是可以交的.LINK:生成树求和 加强版 对于30分 爆搜 可实际上我爆搜只过了25分 有同学使用按秩合并并茶几的及时剪枝通过了30分. const int M ...
- loj6271「长乐集训 2017 Day10」生成树求和 加强版
又是一个矩阵树套多项式的好题. 这里我们可以对每一位单独做矩阵树,但是矩阵树求的是边权积的和,而这里我们是要求加法,于是我们i将加法转化为多项式的乘法,其实这里相当于一个生成函数?之后如果我们暴力做的 ...
- LOJ#6271. 「长乐集训 2017 Day10」生成树求和 加强版
传送门 由于是边权三进制不进位的相加,那么可以考虑每一位的贡献 对于每一位,生成树的边权相当于是做模 \(3\) 意义下的加法 考虑最后每一种边权的生成树个数,这个可以直接用生成函数,在矩阵树求解的时 ...
- 康复计划#5 Matrix-Tree定理(生成树计数)的另类证明和简单拓展
本篇口胡写给我自己这样的什么都乱证一通的口胡选手 以及那些刚学Matrix-Tree,大致理解了常见的证明但还想看看有什么简单拓展的人- 大概讲一下我自己对Matrix-Tree定理的一些理解.常见版 ...
- uoj#335. 【清华集训2017】生成树计数(prufer序列+生成函数+多项式)
传送门 好神仙的题目--又一次有了做一题学一堆的美好体验 据说本题有第二类斯特林数+分治\(FFT\)的做法,然而咱实在看不懂写的是啥,题解贴这里,有兴趣的可以自己去瞅瞅,看懂了记得回来跟咱讲讲 前置 ...
- POJ 2728 Desert King (最优比例生成树)
POJ2728 无向图中对每条边i 有两个权值wi 和vi 求一个生成树使得 (w1+w2+...wn-1)/(v1+v2+...+vn-1)最小. 采用二分答案mid的思想. 将边的权值改为 wi- ...
- [CSP-S模拟测试]:小P的生成树(数学+Kruskal)
题目描述 小$P$是个勤于思考的好孩子,自从学习了最大生成树后,他就一直在想:能否将边权范围从实数推广到复数呢?可是马上小$P$就发现了问题,复数之间的大小关系并没有定义.于是对于任意两个复数$z_1 ...
- Java程序:从命令行接收多个数字,求和并输出结果
一.设计思想:由于命令行接收的是字符串类型,因此应先将字符串类型转化为整型或其他字符型,然后利用for循环求和并输出结果 二.程序流程图: 三.源程序代码: //王荣荣 2016/9/23 ...
随机推荐
- null与undefined到底有啥区别?
话不多说,直接先上结论: null 和 undefined 基本相同,只有细微差别 null 是表示缺少的标识,指示变量未指向任何对象,转为数值为 0 undefined 表示 "缺少值&q ...
- .Net Core with 微服务 - 使用 AgileDT 快速实现基于可靠消息的分布式事务
前面对于分布式事务也讲了好几篇了(可靠消息最终一致性 分布式事务 - TCC 分布式事务 - 2PC.3PC),但是还没有实战过.那么本篇我们就来演示下如何在 .NET 环境下实现一个基于可靠消息的分 ...
- [AGC023D] Go Home 题解
题目传送门 Solution 首先排除掉特殊情况:若 \(S\) 在两侧,肯定会顺序/逆序直接走完,答案就是边界减去出发点. 考虑到若 \(P_1\geq P_n\),那么显然 \(1\) 不到家 \ ...
- FastAPI 学习之路(六)查询参数,字符串的校验
系列文章: FastAPI 学习之路(一)fastapi--高性能web开发框架 FastAPI 学习之路(二) FastAPI 学习之路(三) FastAPI 学习之路(四) FastAPI 学习之 ...
- Java(31)泛型和可变参数
作者:季沐测试笔记 原文地址:https://www.cnblogs.com/testero/p/15228443.html 博客主页:https://www.cnblogs.com/testero ...
- logging的基本使用
logging模块打印log的时候主要有一下几个,级别顺序:CRITICAL>ERROR>WARNING>INFO>DEBUG: 1.日志输出到file: import log ...
- 爬虫逆向基础,理解 JavaScript 模块化编程 webpack
关注微信公众号:K哥爬虫,QQ交流群:808574309,持续分享爬虫进阶.JS/安卓逆向等技术干货! 简介 在分析一些站点的 JavaScript 代码时,比较简单的代码,函数通常都是一个一个的,例 ...
- Node.js CMS——基于 NestJS/NuxtJS 的完整开源项目
这是一款轻量级的基于 Node.js 的开源 CMS,采用前后端分离开发模式,集成了 API.后台管理.WEB 展示三个完整项目.开箱即是一套完整的企业网站,适合企业.个人直接使用或二次开发. API ...
- Alpha阶段发布声明
发布声明 Alpha 1.Alpha版本功能说明 功能列表和详情图 模块 功能 展示 首页 查看首页博文,搜索博文,可供未登录用户使用 动态 查看推荐动态给未登录用户使用,登录用户可以查看关注动态.我 ...
- ES查询区分大小写
ES查询区分大小写 ES查询在默认的情况下是不区分大小写的,在5.0版本之后将string类型拆分成两种新的数据类型,text用于全文搜索(模糊搜索),keyword用于关键字搜索(精确搜索). 注意 ...