题目大意:给出一个森林,每次询问给出u,v,问从u所在连通块中随机选出一个点与v所在连通块中随机选出一个点相连,连出的树的直径期望(不是树输出-1)。(n,q<=10^5)

解法:预处理出各连通块的直径和各点到连通块内一点的最远距离d[x](树形dp+换根),询问若在同一块内输出-1,否则若随机选出两点x,y,直径为max(d[x]+d[y]+1,x所在块直径,y所在块直径),我们把同一连通块内的d排序,枚举小的连通块中的d,到大的连通块中二分d[x]+d[y]+1<=max(x所在块直径,y所在块直径)(或者记下大的块中每种权值出现次数,从max直径开始d[x]递增d[y]递减,复杂度少一个log并仍然只与小的块的大小相关),记忆下相同询问,这样我们枚举小的次数最大只有$O(n^{1.5})$(每次我们复杂度只跟小的有关,那么最劣情况只有每次两个块大小相等,假设所有联通块大小均为k,我们的复杂度是$O(k*min(q,(\frac{n}{k})^2))$,容易证明复杂度最大是$O(n^{1.5})$)。

#include<cstdio>
#include<algorithm>
#include<vector>
#include<map>
using namespace std;
inline int read()
{
int x;char c;
while((c=getchar())<''||c>'');
for(x=c-'';(c=getchar())>=''&&c<='';)x=(x<<)+(x<<)+c-'';
return x;
}
#define MN 100000
#define p(x,y) make_pair(x,y)
struct edge{int nx,t;}e[MN*+];
int f[MN+],h[MN+],en,d[MN+],d2[MN+],mx[MN+];
vector<double> v[MN+];
map<pair<int,int>,double> mp;
int gf(int k){return f[k]?f[k]=gf(f[k]):k;}
inline void ins(int x,int y)
{
e[++en]=(edge){h[x],y};h[x]=en;
e[++en]=(edge){h[y],x};h[y]=en;
}
void upd(int x,int y)
{
if(y>d[x])d2[x]=d[x],d[x]=y;
else if(y>d2[x])d2[x]=y;
}
void dp(int x,int fa)
{
for(int i=h[x];i;i=e[i].nx)if(e[i].t!=fa)
dp(e[i].t,x),upd(x,d[e[i].t]+);
mx[gf(x)]=max(mx[gf(x)],d[x]+d2[x]);
}
void dfs(int x,int fa)
{
v[gf(x)].push_back(d[x]);
for(int i=h[x];i;i=e[i].nx)if(e[i].t!=fa)
upd(e[i].t,d[e[i].t]+==d[x]?d2[x]+:d[x]+),dfs(e[i].t,x);
}
int main()
{
int n,m,q,x,y,i,j,l,r,mid,res;double ans;
n=read();m=read();q=read();
while(m--)ins(x=read(),y=read()),f[gf(x)]=gf(y);
for(i=;i<=n;++i)if(gf(i)==i)
{
dp(i,i);dfs(i,i);v[i].push_back();
sort(v[i].begin(),v[i].end());
for(j=;j<v[i].size();++j)v[i][j]+=v[i][j-];
}
while(q--)
{
if((x=gf(read()))==(y=gf(read()))){puts("-1");continue;}
if(x>y)swap(x,y);
if(mp[p(x,y)]){printf("%.10lf\n",mp[p(x,y)]);continue;}
if(v[x].size()>v[y].size())swap(x,y);
for(i=,ans=;i<v[x].size();++i)
{
for(l=,r=v[y].size()-,res=;l<=r;)
{
mid=l+r>>;
if(v[x][i]-v[x][i-]+v[y][mid]-v[y][mid-]+<=max(mx[x],mx[y]))
res=mid,l=mid+;
else r=mid-;
}
ans+=(double)res*max(mx[x],mx[y])+
(v[y].size()--res)*(v[x][i]-v[x][i-]+)+v[y][v[y].size()-]-v[y][res];
}
if(x>y)swap(x,y);
printf("%.10lf\n",mp[p(x,y)]=ans/(v[x].size()-)/(v[y].size()-));
}
}

Codeforces Round #411 (Div. 1) D. Expected diameter of a tree的更多相关文章

  1. Codeforces Round #411 (Div. 2)

    来自FallDream的博客,未经允许,请勿转载,谢谢. 由于人傻又菜 所以这次又滚去div2了  一堆结论题真的可怕 看见E题不是很有思路   然后就去大力搞F题  T了最后一个点 真的绝望   但 ...

  2. Codeforces Round #411 (Div. 2) A-F

    比赛时候切了A-E,fst了A Standings第一页只有三个人挂了A题,而我就是其中之一,真™开心啊蛤蛤蛤 A. Fake NP time limit per test 1 second memo ...

  3. Codeforces Round 411 Div.2 题解

    A Fake NP standard input/output s, MB Submit Add to favourites x3673 B -palindrome standard input/ou ...

  4. Codeforces Round #411 (Div. 2)(A,B,C,D 四水题)

    A. Fake NP time limit per test:1 second memory limit per test:256 megabytes input:standard input out ...

  5. Codeforces Round #411 div 2 D. Minimum number of steps

    D. Minimum number of steps time limit per test 1 second memory limit per test 256 megabytes input st ...

  6. Codeforces Round #411 (Div. 2) 【ABCDE】

    A. Fake NP 题意:给你l,r,让你输出[l,r]里面除1以外的,出现因子数量最多的那个数. 题解:如果l==r输出l,否则都输出2 #include<bits/stdc++.h> ...

  7. Codeforces Round #411 (Div. 2) C. Find Amir

    C. Find Amir time limit per test   1 second memory limit per test   256 megabytes   A few years ago ...

  8. 【DFS】【贪心】Codeforces Round #411 (Div. 1) C. Ice cream coloring

    对那个树进行dfs,在动态维护那个当前的冰激凌集合的时候,显然某种冰激凌仅会进出集合各一次(因为在树上形成连通块). 于是显然可以对当前的冰激凌集合贪心染色.暴力去维护即可.具体实现看代码.map不必 ...

  9. 【推导】Codeforces Round #411 (Div. 1) B. Minimum number of steps

    最后肯定是bbbb...aaaa...这样. 你每进行一系列替换操作,相当于把一个a移动到右侧. 会增加一些b的数量……然后你统计一下就行.式子很简单. 喵喵喵,我分段统计的,用了等比数列……感觉智障 ...

随机推荐

  1. alpha-咸鱼冲刺day6

    一,合照 emmmmm.自然还是没有的. 二,项目燃尽图 三,项目进展 !!!QAQ可以做到跟数据库交互了!!!!先来撒花花!(然后继续甲板) (然后就没有进展了.翻车+1s) 四,问题困难 数据库交 ...

  2. 项目Alpha冲刺Day1

    一.会议照片 二.项目进展 1.今日安排 讨论完成项目的详细设计,并完成数据库的设计,学习powerDesigner的使用 2.问题困难 powerDesigner导出sql语句因为问题无法导入,特别 ...

  3. mysql基础篇 - 其他基本操作

    基础篇 - 其他基本操作         其他基本操作 一.实验简介 本节实验中我们将学习并实践数据库的其他基本操作:索引.视图,导入和导出,备份和恢复等. 这些概念对于数据库管理员而言都非常重要,请 ...

  4. hibernate.QueryException: ClassNotFoundException: org.hibernate.hql.ast.HqlToken

    环境:weblogic10.3.5,hibernate3,GGTS(groovy/grails tools suite):出现这问题是因为该项目是从weblogic8.1.6下移植到weblogic1 ...

  5. jstree的简单用法

    一般我们用jstree主要实现树的形成,并且夹杂的邮件增删重命名刷新的功能 下面是我在项目中的运用,采用的是异步加载 $('#sensor_ul').data('jstree', false).emp ...

  6. 策略模式(Stratety)

    namespace StrategyPattern //策略模式 { /// <summary> /// 定义所以支持的算法的公共接口 /// </summary> abstr ...

  7. windows 7 netsh wlan命令连接wifi

    显示本机保存的profiles,配置文件是以wifi的ssid命名的. netsh wlan show profiles 用netsh wlan connect name=00_1111 连接其中一个 ...

  8. Andrew Ng机器学习第一章——单变量线性回归

    监督学习算法工作流程 h代表假设函数,h是一个引导x得到y的函数 如何表示h函数是监督学习的关键问题 线性回归:h函数是一个线性函数 代价函数 在线性回归问题中,常常需要解决最小化问题.代价函数常用平 ...

  9. AngularJS1.X学习笔记12-Ajax

    说到Ajax,你一定是思绪万千,想到XMLHttpRequest,$.ajax(),跨域,异步之类的.本文将探讨一下AngularJS的Ajax. 一.一个简单的例子 <!DOCTYPE htm ...

  10. HttpClient 上传多个文件

    using (System.Net.Http.HttpClient client = new System.Net.Http.HttpClient()) { client.BaseAddress = ...