题目

分析

发现,当原图是一棵树的时候,那么新建一条边后,就会变成环套树,

而环内的所有点对都是安全点对,如果环中有k个点,答案就是\(k(k-1)\)

联想到,当把原图做一遍tarjan缩点,每个环缩成一个点,点权为环中的点数,然后就变成了一棵树,那么新建一条边后,就会变成环套树,

经过计算,增加的点对数就是点权和的平方减去点权的平方和

至于如何求出点权和的平方以及点权的平方和,对于每个询问(x,y)

答案就是x到y的路径上的点权和的平方以及点权的平方和,用lca来做,

如果手贱,可以打树链剖分

#include <cmath>
#include <iostream>
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <algorithm>
#include <queue>
const long long maxlongint=2147483647;
const long long mo=1000000007;
const long long N=200005;
using namespace std;
long long g[N][25],n,m,q,ans,last[N*4],next[N*4],to[N*4],tot,deep[N],d[N*2],dfn[N],low[N],dd;
long long f[N][25],f1[N][25],last1[N*4],next1[N*4],to1[N*4],belong[N],be,sum[N],k;
bool bz[N*4];
int bj(int x,int y)
{
next[++tot]=last[x];
last[x]=tot;
to[tot]=y;
}
int bj1(int x,int y)
{
next1[++tot]=last1[x];
last1[x]=tot;
to1[tot]=y;
}
int tarjan(int x)
{
dfn[x]=low[x]=++dd;
d[++tot]=x;
for(int i=last[x];i;i=next[i])
if(bz[i])
{
bz[i]=bz[i^1]=false;
int j=to[i];
if(!dfn[j])
{
tarjan(j);
low[x]=min(low[x],low[j]);
}
else
low[x]=min(low[x],low[j]);
}
if(dfn[x]==low[x])
{
be++;
while(low[d[tot]]>=dfn[x])
{
belong[d[tot--]]=be;
sum[be]++;
}
}
}
int dg(int x)
{
bz[x]=false;
for(int i=last1[x];i;i=next1[i])
{
if(bz[to1[i]])
{
deep[to1[i]]=deep[x]+1;
g[to1[i]][0]=x;
f[to1[i]][0]=sum[to1[i]];
f1[to1[i]][0]=sum[to1[i]]*sum[to1[i]];
dg(to1[i]);
}
}
}
int llca(int x,int y)
{
if(deep[x]>deep[y])
{
x=x^y;
y=x^y;
x=x^y;
}
for(int i=log2(n);i>=0;i--)
{
if(deep[g[y][i]]>=deep[x])
{
k+=f[y][i];
ans-=f1[y][i];
y=g[y][i];
}
}
for(int i=log2(n);i>=0;i--)
{
if(g[y][i]!=g[x][i])
{
k+=f[y][i]+f[x][i];
ans-=f1[y][i]+f1[x][i];
y=g[y][i];
x=g[x][i];
}
}
if(x!=y)
{
k+=f[x][0]+f[y][0];
ans-=f1[y][0]+f1[x][0];
return g[x][0];
}
else
return x; }
int main()
{
scanf("%lld%lld%lld",&n,&m,&q);
tot=1;
for(int i=1;i<=m;i++)
{
int x,y;
scanf("%d%d",&x,&y);
bj(x,y);
bj(y,x);
}
memset(bz,true,sizeof(bz));
tot=0;
tarjan(1);
tot=0;
for(int x=1;x<=n;x++)
{
for(int i=last[x];i;i=next[i])
{
int j=to[i];
if(belong[x]!=belong[j])
{
bj1(belong[x],belong[j]);
bj1(belong[j],belong[x]);
}
}
}
f[1][0]=sum[1];
f1[1][0]=sum[1]*sum[1];
deep[1]=1;
memset(bz,true,sizeof(bz));
dg(1);
for(int i=1;i<=log2(n);i++)
for(int j=1;j<=n;j++)
{
g[j][i]=g[g[j][i-1]][i-1];
f[j][i]=f[j][i-1]+f[g[j][i-1]][i-1];
f1[j][i]=f1[j][i-1]+f1[g[j][i-1]][i-1];
}
ans=0;
for(int i=1;i<=q;i++)
{
int x1,y1;
scanf("%d%d",&x1,&y1);
k=0;
int lca=llca(belong[x1],belong[y1]);
k+=sum[lca];
ans-=sum[lca]*sum[lca];
ans+=k*k;
}
printf("%lld",ans);
}

【NOIP2016提高A组模拟9.15】Map的更多相关文章

  1. 【JZOJ4784】【NOIP2016提高A组模拟9.15】Map

    题目描述 输入 输出 样例输入 4 4 2 1 2 2 3 3 2 3 4 1 2 1 4 样例输出 14 数据范围 样例解释 upd:保证原图连通. "不相交路径"的定义为不存在 ...

  2. NOIP2016提高A组模拟10.15总结

    第一题,就是将原有的式子一步步简化,不过有点麻烦,搞了很久. 第二题,枚举上下边界,维护一个单调队列,二分. 比赛上没有想到,只打了个暴力,坑了80分. 第三题,贪心,最后的十多分钟才想到,没有打出来 ...

  3. 【NOIP2016提高A组模拟10.15】打膈膜

    题目 分析 贪心, 先将怪物按生命值从小到大排序(显然按这个顺序打是最优的) 枚举可以发对少次群体攻击, 首先将所有的群体攻击发出去, 然后一个一个怪物打,当当前怪物生命值大于2,如果还有魔法值就放重 ...

  4. 【NOIP2016提高A组模拟10.15】最大化

    题目 分析 枚举两个纵坐标i.j,接着表示枚举区域的上下边界, 设对于每个横坐标区域的前缀和和为\(s_l\),枚举k, 显然当\(s_k>s_l\)时,以(i,k)为左上角,(j,k)为右下角 ...

  5. 【NOIP2016提高A组模拟10.15】算循环

    题目 分析 一步步删掉循环, 首先,原式是\[\sum_{i=1}^n\sum_{j=1}^m\sum_{k=i}^n\sum_{l=j}^m\sum_{p=i}^k\sum_{q=j}^l1\] 删 ...

  6. 【NOIP2016提高A组模拟9.15】Osu

    题目 分析 考虑二分答案, 二分小数显然是不可取的,那么我们将所有可能的答案求出来,记录在一个数组上,排个序(C++调用函数很容易超时,手打快排,时间复杂度约为\(O(>8*10^7)\),但相 ...

  7. 【NOIP2016提高A组模拟9.15】Math

    题目 分析 因为\((-1)^2=1\), 所以我们只用看\(\sum_{j=1}^md(i·j)\)的值模2的值就可以了. 易证,一个数x,只有当x是完全平方数时,d(x)才为奇数,否则为偶数. 那 ...

  8. 【NOIP2016提高A组模拟8.15】Garden

    题目 分析 其实原题就是[cqoi2012][bzoj2669]局部极小值. 有一个n行m列的整数矩阵,其中1到nm之间的每个整数恰好出现一次.如果一个格子比所有相邻格子(相邻是指有公共边或公共顶点) ...

  9. 【NOIP2016提高A组模拟8.15】Throw

    题目 分析 首先对于一个状态(a,b,c),假定a<=b<=c: 现在考虑一下这个状态,的转移方案: \[1,中间向两边跳(a,b,c)-->(a*2-b,a,c).(a,b,c)- ...

随机推荐

  1. centos v7.0解决乱码

    [root@localhost ~]# ll 鎬荤敤閲4-rw-------. 1 root root 1045 8鏈 24 21:17 anaconda-ks.cfg [root@localhost ...

  2. 【神经网络与深度学习】leveldb的实现原理

    郑重声明:本篇博客是自己学习 Leveldb 实现原理时参考了郎格科技系列博客整理的,原文地址:http://www.samecity.com/blog/Index.asp?SortID=12,只是为 ...

  3. excel常用公式--时间序列类

    year,month,day:返回对应于某个日期的年月日.Year作为1900 - 9999之间的整数返回. weekday:返回对应于某个日期的一周中的第几天. WEEKNUM:返回特定日期的周数. ...

  4. [转帖]预警 | Linux 爆“SACK Panic”远程DoS漏洞,大量主机受影响

    预警 | Linux 爆“SACK Panic”远程DoS漏洞,大量主机受影响   https://cloud.tencent.com/developer/article/1447879 所有的 版本 ...

  5. 【挣扎失败】2019CSP-S 游记

    妈耶…… 今年是作为高中生参赛的第一年……然而…… 心痛 洛谷评分,一橙一蓝两紫两黑 我个菜鸡瑟瑟发抖 在考完后三天的信息课码的,刚写没几个字就要下课了 抽空把这个写完

  6. List<HashMap<String,String>> list, 根据hashmap中的某个键的值排序

    来源https://blog.51cto.com/zhaodan/1725249 //可以使用Collections.sort(List list, Comparator c)来实现 这里举例hash ...

  7. P3964 [TJOI2013]松鼠聚会

    传送门 首先题意就是求一个点到所有其他点的切比雪夫距离和最小 考虑枚举所有点作为答案,那么我们需要快速计算切比雪夫距离和,发现不太好算 根据一些奇怪的套路,我们把坐标系变化,把 $(x,y)$ 变成 ...

  8. RabbitMQ入门教程(十三):虚拟主机vhost与权限管理

    原文:RabbitMQ入门教程(十三):虚拟主机vhost与权限管理 版权声明:本文为博主原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接和本声明. 本文链接:https://b ...

  9. 096、运行第一个Service (Swarm03)

    参考https://www.cnblogs.com/CloudMan6/p/7874609.html   上一节我们部署好了 Swarm 集群,下面部署一个运行httpd镜像的service进行演示 ...

  10. oracle链接报错shared memory realm does not exist

    问题描述: 前两天能够正常链接,今天来了突然不能链接,原因不详. 处理方法: 连接linux进行如下操作: 1.sqlplus /nolog 2.conn / as sysdba 3.startup ...