Infinite Tree

Infinite Tree

题意:

一颗无限结点的数,任意大于1的点k与点

题解:

参考博客
看了好一阵子才明白。。。emm。
我们先按照题意画出一部分树
在这里插入图片描述
我们先不考虑复杂度,这题应该怎么做?
题目给了每个点的权值w[i],问一个点到所有的节点路径长度*点权之和最小是多少,很明显是树形dp
dp[i]表示以i为根的子树到i的w和
sum[i]表示乘上距离之后的答案
dep[i]表示深度,now为当前节点,son为子节点
则有:

dp[now]= w[now]
sum[now]=0
dp[now]+=dp[son]
sum[now]+=sum[son]+(dep[son]-dep[now])*sum[son]

但是以now为根并不一定是最佳答案,所以我们还要不断的换根,求出最佳答案
如果将now为根转移到son为根,我们不用重新算一遍,只需要在之前的基础上

sum[now]-=sum[son]+(dep[son]-dep[now])*dp[son]
dp[now]-=dp[son]
sum[son]+=sum[now]+(dep[son]-dep[now])*dp[now]
dp[son]+=dp[now]

这样就OK了
但是!
题目是要求u到i!的距离,i!的增长速度很快,树的增长也是很快,如果全建出来肯定TLE了,所以我们没办法将整棵树建立,只能选择性建立,这要用到虚树
我们将阶乘点当做关键点,保留关键点及其lca,然后建立数就行
现在又有一个新问题,lca怎么计算?
我们首先考虑a!和(a+1)!的dfn谁大?
因为后者的因子一定包含前者的因子,所以除以mindiv后,(a+1)!的深度肯定更大,所以这些阶乘数的dfn是随a值从小到大的,这样我们只需要考虑相邻两个点的lca即可
我们列一个表格记录阶乘数分解后有多少个质数
在这里插入图片描述
我们根据上面的表格来列出相邻的LCA
2!和3 ! : 1,深度为1
3!和4 ! : 6,深度为3
4!和5 !: 1,深度为1
5!和6 !: 15,深度为3
6!和7 !: 1,深度为1
我们可以得出,对于a!和(a+1)!,LCA就是从大到小公共的质因子的乘积,遇到不同的就停止,深度是相同的个数+1
例如5!和6!:
5!分解后:5 3 2 2 2
6!:5 3 3 2 2 2 2
从大到小,一样的是5和3 ,(从第三位开始不一样,停止)
lca就是15,深度就是3
在这里插入图片描述
可以得到dep[lca((i+1)!,i!)] = sum(maxdiv(i+1),n)
sum(maxidv(i+1),n)为原本i!中大于等于maxdiv(i+1)的因子个数
这样我们就可以快速算出LCA
但是还是不够快,如果对于每个数都扫一次的话,还是很慢。所以,需要一个快速地查找求和,修改的算法,那就是用到了线段树或树状数组

代码:

#include<bits/stdc++.h>
#define ll long long
#define inf 1ll<<60
using namespace std;
const int MAXN=1e5+10;
int num[MAXN],w[MAXN<<1],d[MAXN<<1],stk[MAXN];
int ldfn[MAXN],rdfn[MAXN],dep[MAXN],lcad[MAXN],m;
int mndv[MAXN],pcnt=0;
ll dp1[MAXN<<1],dp2[MAXN<<1],ans;
vector<int> vir[MAXN<<1];
ll tr[MAXN<<2];//注意开long long
void Build(int root,int l,int r)
{tr[root]=0;if(l==r) return;int mid=l+r>>1;Build(root<<1,l,mid);Build(root<<1|1,mid+1,r);
}
void Change(int root,int l,int r,int x)
{if(l==r){tr[root]++;return;}int mid=l+r>>1;if(x<=mid) Change(root<<1,l,mid,x);else Change(root<<1|1,mid+1,r,x);tr[root]=tr[root<<1]+tr[root<<1|1];
}//x是插入的质数,要将其计数+1
int Query(int root,int x,int l,int r)
{              if(l>=x) return tr[root];int mid=l+r>>1;if(x<=mid) return Query(root<<1,x,l,mid)+Query(root<<1|1,x,mid+1,r);else if(x>mid) return Query(root<<1|1,x,mid+1,r);
}//查找当前质数的个数
void build()
{//^不等于,相当于!=dep[1]=1;for(int i=2;i<=m;i++){dep[i]=dep[i-1];int j=i;while(j^mndv[j]) j/=mndv[j];lcad[i]=Query(1,j,1,m)+1;for(j=i;j^1;dep[i]++,j/=mndv[j]) Change(1,1,m,mndv[j]);}int top=0,tot=m;stk[++top]=1;for(int i=2;i<=m;i++){if(top==1||lcad[i]==dep[stk[top]]){stk[++top]=i;continue;}while(top>1&&lcad[i]<=dep[stk[top-1]]){vir[stk[top-1]].push_back(stk[top]);top--;}//建虚树的基本操作,不会的建议去学习一下if(lcad[i]^dep[stk[top]]){dep[++tot]=lcad[i];w[tot]=0;vir[tot].push_back(stk[top]);stk[top]=tot;}stk[++top]=i;}while(top>1){vir[stk[top-1]].push_back(stk[top]);top--;}
}//原理同上,供参考
void dfs1(int x,int fa)
{dp1[x]=w[x];dp2[x]=0;for(int i=0;i<vir[x].size();i++){int son=vir[x][i];if(son==fa) continue;dfs1(son,x);dp1[x]+=dp1[son];dp2[x]+=dp2[son]+(dep[son]-dep[x])*dp1[son];}
}
void dfs2(int x,int fa)
{ans=min(ans,dp2[x]);for(int i=0;i<vir[x].size();i++){int son=vir[x][i];if(son==fa) continue;ll x1=dp1[x],x2=dp2[x],son1=dp1[son],son2=dp2[son];dp2[x]-=dp2[son]+(dep[son]-dep[x])*dp1[son];dp1[x]-=dp1[son];dp2[son]+=dp2[x]+(dep[son]-dep[x])*dp1[x];dp1[son]+=dp1[x];dfs2(son,x);dp1[x]=x1,dp2[x]=x2,dp1[son]=son1,dp2[son]=son2;}
}//树形dp+换根,非重点,且是模板一套的问题,上面已经分析
int main()
{mndv[1]=1;for(int i=2;i<MAXN;i++)if(!mndv[i])for(int j=i;j<MAXN;j+=i)if(!mndv[j]) mndv[j]=i;//预处理出每个数的mindiv,之后分解时可以用while(~scanf("%d",&m)){for(int i=1;i<=m;i++)scanf("%d",&w[i]);for(int i=0;i<=m*2;i++){vir[i].clear();dp1[i]=dp2[i]=0;}Build(1,1,m);build();dfs1(1,0);ans=dp2[1];dfs2(1,0);printf("%lld\n",ans);}
}
#include<bits/stdc++.h>
#define lowbit(x) x&-x
using namespace std;
typedef long long ll;
const int MAX = 2e5 + 10;
//建立虚树点数tot < 2n, 空间开两倍int n, w[MAX];
ll ans;//树状数组
int c[MAX];
void upd(int p, int k) { for (; p <= n; p += lowbit(p)) c[p] += k; }
int query(int p) { int res = 0; for (; p; p -= lowbit(p)) res += c[p]; return res; }int mindiv[MAX];
void sieve(int siz) {//筛mindivfor (int i = 2; i <= siz; i++)if (!mindiv[i])for (int j = i; j <= siz; j += i)if (!mindiv[j])mindiv[j] = i;
}int lcadep[MAX], dep[MAX];
int st[MAX], top, tot;//stack, top, tot:虚树点数
vector<int> g[MAX];//虚树
void add_edge(int u, int v) { g[u].push_back(v), g[v].push_back(u); }void buildVirtualTree() {tot = n;st[top = 1] = 1;for (int i = 2; i <= n; i++) {dep[i] = dep[i - 1] + 1; int j = i;for (; j != mindiv[j]; j /= mindiv[j]) dep[i]++;lcadep[i] = query(n) - query(j - 1);for (j = i; j != 1; j /= mindiv[j]) upd(mindiv[j], 1);}//建树for (int i = 2; i <= n; i++) {while (top > 1 && dep[st[top - 1]] >= lcadep[i])add_edge(st[top - 1], st[top]), top--;if (dep[st[top]] != lcadep[i]) {dep[++tot] = lcadep[i];add_edge(tot, st[top]);st[top] = tot;}st[++top] = i;}while (top > 1){add_edge(st[top - 1], st[top]);top--;}
}void dfs(int u, int fa) {ans += 1ll * w[u] * dep[u];//ans最开始是rt = 1时的答案for (auto &v: g[u])if (v != fa) {dfs(v, u);w[u] += w[v];}
}void dfs2(int u, int fa) {//如果rt移动之后答案变小就一直移动下去,直到答案不在变小for (auto &v: g[u])if (v != fa) {//rt从u转移到v的代价//+(w[1] - w[v]) - w[v]if (w[1] - 2 * w[v] < 0) {ans += 1ll * (w[1] - 2 * w[v]) * (dep[v] - dep[u]);//一步的代价*距离dfs2(v, u);}}
}void init() {ans = top = 0;for (int i = 1; i <= tot; i++) {g[i].clear();c[i] = w[i] = lcadep[i] = dep[i] = 0;}
}int main() {sieve(1e5);while (~scanf("%d", &n)) {init();for (int i = 1; i <= n; i++) scanf("%d", &w[i]);buildVirtualTree();int rt = 1;dfs(rt, 0);dfs2(rt, 0);printf("%lld\n", ans);}return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/318115.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

IdentityServer4-从数据库获取User登录并对Claims授权验证(五)

本节将在第四节基础上介绍如何实现IdentityServer4从数据库获取User进行验证&#xff0c;并对Claim进行权限设置。一、新建Web API资源服务&#xff0c;命名为ResourceAPI&#xff08;1&#xff09;新建API项目&#xff0c;用来进行user的身份验证服务。&#xff08;2&#xff…

周末狂欢赛1(玩游戏/Game,函数,JOIOI王国)

狂欢1T1&#xff1a;玩游戏 / Game题目题解代码实现T2&#xff1a;函数题目题解代码实现T3&#xff1a;JOIOI王国题目题解代码实现T1&#xff1a;玩游戏 / Game 题目 ljcc 和他的学妹在玩游戏&#xff0c;这个游戏共有 n 轮&#xff0c;在第 i 轮获胜会获得 i 分&#xff0c;…

用ABP只要加人即可马上加快项目进展(二) - 分工篇 - BDD实战篇 - .NET Core里跑Specflow...

这是<如何用ABP框架快速完成项目 >系列中的一篇文章。BDD很赞&#xff01;比TDD先进很多&#xff0c;能够大大提高编码效率。上一篇文章说了如何在.NET Core里安装Specflow. 然而文章成果只到了hello world级别。要想真的和实际业务结合&#xff0c;比如要能够IOC new cl…

【做题记录】CodeForces 做题记录

链接放的是洛谷上的链接&#xff0c;难度就是 CF 上的评分。 <details><summary>$\texttt{solution}$</summary></details> CF10D LCIS 难度&#xff1a;\(\tt{2800}\) 求两个串的最长公共上升子序列。\(n\le 500\) $\texttt{solution}$ 严重虚高题&am…

周末狂欢赛2(冒泡排序,概率充电器,不勤劳的图书管理员)

狂欢2T1&#xff1a;冒泡排序题目题解CODET2&#xff1a;概率充电器题目题解CODET3&#xff1a;不勤劳的图书管理员题目题解CODE我不这么认为。。。。 T1&#xff1a;冒泡排序 题目 下面是一段实现冒泡排序算法的 C代码&#xff1a; for(int i1; i<n; i)for(int j1; j&l…

P5659-[CSP-S2019]树上的数【贪心】

正题 题目链接:https://www.luogu.com.cn/problem/P5659 题目大意 给出nnn个点的一棵树&#xff0c;每个节点上有一个数字&#xff0c;你每次可以选择一条边删除然后交换连接的两个点的数字&#xff0c;在删完所有数字后设pip_ipi​表示数字iii所在节点编号&#xff0c;要求使…

YBTOJ洛谷P3195:玩具装箱(斜率优化dp)

传送门 文章目录前言解析代码前言 斜率优化dp&#xff0c;就是利用斜率优化的dp &#xff08;逃&#xff09; 解析 第一道斜优的题 分析题目 设sumisum_isumi​为1-i的c的前缀和 容易写出dp转移式&#xff1a; dpimin(dpj(sumi−sumji−j−1−L)2)dp_imin(dp_j(sum_i-sum_ji-…

01.微服务系列介绍

微服务系列实践 .NET CORE在开始之前呢&#xff0c;还是得废话一下&#xff0c;毕竟还是需要介绍一下这个系列我们要实现什么样的一套服务架构&#xff0c;也让大家能初步的有一个了解&#xff0c;后续实践起来也有一个完整的概念&#xff0c;相对也会容易的多。互联网架构演变…

Walker

Walker 题意&#xff1a; 一个区间[0,n]&#xff0c;区间上有两个点&#xff0c;坐标分别是pos1&#xff0c;pos2&#xff0c;速度分别是v1&#xff0c;v2&#xff0c;这两个点是在移动&#xff0c;可以随时改变移动方向&#xff0c;问当区间的每一块均被一个点或两个点移动覆…

【网络流】最大流问题(EK算法带模板,Dinic算法带模板及弧优化,ISAP算法带模板及弧优化)上下界网络流

本blog重点是代码网络流的相关概念流网络(flow network)流(flow)网络的流残留网络(residual network)增广路径(augmenting path)Edmonds-Karp算法思想bfs模板调用EK&更新残留网络流模板luogu的AC代码(EK版)Dinic算法思路时间复杂度证明bfs模板模板1模板2dfs模板不带弧优化模…

Rainbond 5.0正式发布, 支持对接管理已有Kubernetes集群

今天很高兴的向大家宣布Rainbond v5.0正式发布&#xff0c;Rainbond是开源的企业应用云操作系统&#xff0c;支撑企业应用的开发、架构、交付和运维的全流程&#xff0c;通过无侵入架构&#xff0c;无缝衔接各类企业应用&#xff0c;底层资源可以对接和管理IaaS、虚拟机和物理服…

Fibonacci

Fibonacci 题意&#xff1a; f[i]表示第i位的斐波那契数列 给定n&#xff0c;求 题解&#xff1a; 这种题一开始没什么思路&#xff0c;那么枚举就行 g(x,y) 1 是当x * y为偶数时 x * y为偶数说明&#xff1a; x是偶数&#xff0c;y也是偶数 x是奇数&#xff0c;y是偶数 而…

基于.NET Standard的分布式自增ID算法--美团点评LeafSegment

概述前一篇文章讲述了最流行的分布式ID生成算法snowflake&#xff0c;本篇文章根据美团点评分布式ID生成系统文章&#xff0c;介绍另一种相对更容易理解和编写的分布式ID生成方式。实现原理Leaf这个名字是来自德国哲学家、数学家莱布尼茨的一句话&#xff1a;There are no two …

[费用流专题]Going Home,Minimum Cost,工作安排

文章目录T1&#xff1a;Going Home题目题解CODET2&#xff1a;Minimum Cost题目题解CODET3&#xff1a;工作安排题解CODET1&#xff1a;Going Home 题目 On a grid map there are n little men and n houses. In each unit time, every little man can move one unit step, e…

Sky Garden

Sky Garden 题意&#xff1a; 画n个圆和m条直线&#xff0c;圆的中心点为(0,0)&#xff0c;圆的半径分别从1到n&#xff0c;而直线都必经过(0,0)点&#xff0c;并且所有直线会把每个圆平均分成2m个面积相等的区域&#xff0c;直线会和圆形成交点&#xff0c;求所有交点两两经…

IdentityServer4-前后端分离的授权验证(六)

上两节介绍完Hybrid模式在MVC下的使用&#xff0c;包括验证从数据获取的User和Claim对MVC的身份授权。本节将介绍Implicit模式在JavaScript应用程序中的使用&#xff0c;使用Node.jsExpress构建JavaScript客户端&#xff0c;实现前后端分离。本节授权服务和资源服务器基于第四和…

人类智慧贪心

题意看起来很清新&#xff0c;代码实现也基本在入门难度&#xff0c;但是为什么我不会&#xff01; 另&#xff1a;反悔贪心 <details><summary>$\texttt{solution}$</summary></details> P2672 [NOIP2015 普及组] 推销员 $\texttt{solution}$ 发现答案…

周末狂欢赛3(跳格子,英雄联盟,排序问题)

文章目录T1&#xff1a;跳格子题目题解CODET2&#xff1a;英雄联盟题目题解CODET3&#xff1a;排序问题题目题解CODET1&#xff1a;跳格子 题目 n 个格子排成一列&#xff0c;一开始&#xff0c;你在第一个格子&#xff0c;目标为跳到第 n 个格子。在每个格子 i 里面你可以做…

想让AI在企业落地?微软最新Azure AI不容错过!

Microsoft Connect(); 2018 如期举行&#xff0c;大会上发布的众多顶尖技术&#xff0c;瞬间引爆了全球&#xff01;AI的高速发展&#xff0c;正在掀起新一波的创新浪潮。对于很多企业来说&#xff0c;AI创造的巨大价值&#xff0c;是不容错过的风口&#xff0c;大会上&#xf…

[费用流]数字配对,新生舞会

文章目录T1&#xff1a;数字配对题目题解CODET2&#xff1a;新生舞会题目题解CODE&#xff08;最大费用最大流版&#xff09;CODE&#xff08;最小费用最大流版&#xff09;T1&#xff1a;数字配对 题目 有 n 种数字&#xff0c;第 i 种数字是 ai、有 bi 个&#xff0c;权值是…