【JZOJ4784】【NOIP2016提高A组模拟9.15】Map
题目描述
输入
输出
样例输入
4 4 2
1 2
2 3
3 2
3 4
1 2
1 4
样例输出
14
数据范围
样例解释
upd:保证原图连通。
“不相交路径”的定义为不存在相同的边。可以存在相同的点。重边视为不同的边。
对于样例:
原图有2个安全点对为(2,3),(3,2)
询问1答案为4,新增的安全点对为(1,2),(1,3),(2,1)(3,1)
询问2答案为10,新增的安全点对为(1,2),(1,3),(1,4),(2,1)(2,4),(3,1),(3,4),(4,1),(4,2),(4,3)
因此输出14
解法
利用边双连通分量缩点,再利用lca统计答案。
代码
#include<iostream>
#include<stdio.h>
#include<math.h>
#include<string.h>
#include<algorithm>
#define ll long long
#define ln(x,y) (ll)(log(x)/log(y))
#define sqr(x) ((x)*(x))
using namespace std;
const char* fin="aP3.in";
const char* fout="aP3.out";
const ll inf=0x7fffffff;
const ll maxn=400007,maxm=maxn*4,maxk=22;
ll n,m,t,i,j,k,l,tot,limit,tmp,cnt;
ll fi[maxn],ne[maxm],la[maxm],de[maxn];
ll fa[maxn][maxk],sum[maxn][maxk],qsum[maxn][maxk];
ll dfn[maxn],low[maxn],num,stack[maxn],blg[maxn];
ll az[maxn];
ll ans;
bool bz[maxn],cz[maxn],dz[maxn];
ll read(){
ll x=0;
char ch=getchar();
while (ch<'0' || ch>'9') ch=getchar();
while (ch>='0' && ch<='9') x=x*10+ch-'0',ch=getchar();
return x;
}
void add_line(ll a,ll b){
tot++;
ne[tot]=fi[a];
la[tot]=b;
fi[a]=tot;
}
void dfs(ll v,ll from){
ll i,j,k;
cz[v]=true;
fa[v][0]=from;
qsum[v][0]=sqr(sum[v][0]);
de[v]=de[from]+1;
for (i=1,j=ln(de[v],2);i<=j;i++){
k=fa[v][i-1];
fa[v][i]=fa[k][i-1];
sum[v][i]=sum[v][i-1]+sum[k][i-1];
qsum[v][i]=qsum[v][i-1]+qsum[k][i-1];
}
for (k=fi[v];k;k=ne[k]){
if (!cz[az[blg[la[k]]]]){
dfs(az[blg[la[k]]],v);
}
}
}
void up(ll &a,ll i,ll &j,ll &k){
j+=sum[a][i];
k+=qsum[a][i];
a=fa[a][i];
}
ll lca(ll a,ll b){
ll i,j=0,k=0;
if (de[a]<de[b]) swap(a,b);
for (i=ln(de[a]-de[b],2);i>=0;i--) if (de[fa[a][i]]>de[b]) up(a,i,j,k);
if (de[a]!=de[b]) up(a,0,j,k);
for (i=ln(de[a],2);i>=0;i--) if (fa[a][i]!=fa[b][i]) up(a,i,j,k),up(b,i,j,k);
if (a!=b) up(a,0,j,k),up(b,0,j,k);
j+=sum[a][0];
k+=qsum[a][0];
return sqr(j)-k;
}
void tarjan(ll v,ll from){
ll i,j,k;
dfn[v]=low[v]=++num;
bz[j=stack[++stack[0]]=v]=true;
for (k=fi[v];k;k=ne[k])
if (!dz[(k+1)/2]){
dz[(k+1)/2]=true;
if (!dfn[la[k]]){
tarjan(la[k],v);
low[v]=min(low[la[k]],low[v]);
}else if (bz[la[k]]) low[v]=min(low[v],dfn[la[k]]);
}
if (low[v]==dfn[v]){
while (stack[stack[0]]!=j){
blg[stack[stack[0]]]=v;
bz[stack[stack[0]--]]=false;
}
blg[stack[stack[0]]]=v;
bz[stack[stack[0]--]]=false;
}
}
int main(){
cnt=n=read();m=read();t=read();
for (i=1;i<=m;i++){
j=read();k=read();
add_line(j,k);
add_line(k,j);
}
tarjan(1,0);
for (i=1;i<=n;i++){
if (!az[blg[i]]) az[blg[i]]=++cnt;
sum[az[blg[i]]][0]++;
for (k=fi[i];k;k=ne[k]) add_line(az[blg[i]],la[k]);
}
dfs(n+1,0);
for (i=1;i<=t;i++){
j=read();
k=read();
ans+=lca(az[blg[j]],az[blg[k]]);
}
printf("%lld\n",ans);
return 0;
}
启发
边双连通分量在某种程度上等同于强连通分量。
【JZOJ4784】【NOIP2016提高A组模拟9.15】Map的更多相关文章
- NOIP2016提高A组模拟10.15总结
第一题,就是将原有的式子一步步简化,不过有点麻烦,搞了很久. 第二题,枚举上下边界,维护一个单调队列,二分. 比赛上没有想到,只打了个暴力,坑了80分. 第三题,贪心,最后的十多分钟才想到,没有打出来 ...
- 【NOIP2016提高A组模拟10.15】打膈膜
题目 分析 贪心, 先将怪物按生命值从小到大排序(显然按这个顺序打是最优的) 枚举可以发对少次群体攻击, 首先将所有的群体攻击发出去, 然后一个一个怪物打,当当前怪物生命值大于2,如果还有魔法值就放重 ...
- 【NOIP2016提高A组模拟10.15】最大化
题目 分析 枚举两个纵坐标i.j,接着表示枚举区域的上下边界, 设对于每个横坐标区域的前缀和和为\(s_l\),枚举k, 显然当\(s_k>s_l\)时,以(i,k)为左上角,(j,k)为右下角 ...
- 【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\] 删 ...
- 【NOIP2016提高A组模拟9.15】Map
题目 分析 发现,当原图是一棵树的时候,那么新建一条边后,就会变成环套树, 而环内的所有点对都是安全点对,如果环中有k个点,答案就是\(k(k-1)\) 联想到,当把原图做一遍tarjan缩点,每个环 ...
- 【NOIP2016提高A组模拟9.15】Osu
题目 分析 考虑二分答案, 二分小数显然是不可取的,那么我们将所有可能的答案求出来,记录在一个数组上,排个序(C++调用函数很容易超时,手打快排,时间复杂度约为\(O(>8*10^7)\),但相 ...
- 【NOIP2016提高A组模拟9.15】Math
题目 分析 因为\((-1)^2=1\), 所以我们只用看\(\sum_{j=1}^md(i·j)\)的值模2的值就可以了. 易证,一个数x,只有当x是完全平方数时,d(x)才为奇数,否则为偶数. 那 ...
- 【NOIP2016提高A组模拟8.15】Garden
题目 分析 其实原题就是[cqoi2012][bzoj2669]局部极小值. 有一个n行m列的整数矩阵,其中1到nm之间的每个整数恰好出现一次.如果一个格子比所有相邻格子(相邻是指有公共边或公共顶点) ...
- 【NOIP2016提高A组模拟8.15】Throw
题目 分析 首先对于一个状态(a,b,c),假定a<=b<=c: 现在考虑一下这个状态,的转移方案: \[1,中间向两边跳(a,b,c)-->(a*2-b,a,c).(a,b,c)- ...
随机推荐
- extern关键字及C\C++相互调用
extern关键字主要修饰变量或函数,表示该函数可以跨文件访问,或者表明该变量在其他文件定义,在此处引用. 1.extern修饰变量 (1)如果某变量int m在a.c中定义声明,则其他b.c文件访问 ...
- 内容溢出文字用"..."代替 以及超出文本内容换行
text-overflow:ellipsis; white-space:nowrap; overflow:hidden; 超出.....代替 overflow: hidden; word-break: ...
- Jeecg-Boot 开发环境准备(二):开发工具安装
目录索引: 后端开发工具 前端开发工具 Nodejs镜像 WebStorm入门配置 JeecgBoot采用前后端分离的架构,官方推荐开发工具 前端开发: Webstrom 或者 IDEA 后端开发: ...
- PHP数组循环遍历的四种方式
1.使用for循环遍历数组 conut($arr);用于统计数组元素的个数. for循环只能用于遍历,纯索引数组!!!! 如果存在关联数组,count统计时会统计两种数组的总 ...
- Jquery选择器总结一
jquery 是javaScript框架,封装了js. 好处:使用方便,少代码多功能. 实现同一个功能的代码量少. 屏蔽浏览器差异. 简化ajax开发. 选择器 基本选择器 1. id选择器 $(&q ...
- mavenjar 一些拉取不下来问题
http://search.maven.org/这里找相近版本替换试试.拉取不下来是因为官方版本不足或者网络问题.
- Win10操作系统安装—U盘作为启动盘—系统安装到固态硬盘中
利用U盘作为启动盘安装win10操作系统 1.U盘制作为启动盘,制作工具,我选择的是大白菜(个人觉得还是很好用的) 大白菜http://www.bigbaicai.com/rjjc/syjc/3269 ...
- PHP--Smarty的template模式
function change_year() { var ss = $('#select_year').children('option:selected').val(); $.ajax({ type ...
- android的AIDL
一.AIDL的意义: AIDL全称是Android Interface Definition Language,是android接口定义语言.AIDL就是为了避免我们一遍遍的写一些 ...
- [转]js的垃圾回收机制
javascript具有自动垃圾收集机制,执行环境会负责管理代码执行过程中使用的内存.在编写javascript程序时,开发人员不用再关心内存使用问题,所需内存的分配以及无用内存的回收完全实现了自动管 ...