bzoj 3528 [Zjoi2014]星系调查【树链剖分+数学】
参考:https://www.cnblogs.com/zhuohan123/p/3698852.html
首先,根据点到直线距离公式
\]
那么XPs的线性假设相斥度为
\]
//以下部分为参考blog截图:


所以只需要维护\( \sum x_i , \sum y_i , \sum x_iy_i , \sum x_i^{2}, \sum y_i^{2} \)即可
注意到题目描述“使用这些跳跃星门,ZeusLeague+的物资就可以在这N个行星系中两两任意互相传输。由于经费问题,跳跃星门的个数不会超过行星系的个数。“也就是说,\( n-1\leq m\leq n \),即,可能是一棵普通树或者基环树。
然后回到树上处理,首先定一个root(假设为1),预处理出所有点到root的\( \sum x_i ,\sum y_i,\sum x_iy_i,\sum x_i^{2},\sum y_i^{2} \)
对于普通树,采用树上差分的方法,即\( \delta_x+\delta_t-\delta_{lca(x,y)}-\delta_{father(lca(x,y))} \)
对于基环树,找到环上的两个点,bfs一遍,然后当成普通树处理即可。
#include<iostream>
#include<cstdio>
#include<cmath>
#include<cstring>
using namespace std;
const int N=100005;
int n,m,Q,tot,tx[N],ty[N],h[N],cnt,id[N],q[N];
int si[N],hs[N],rt[N],fa[N],de[N],fr[N];
bool vis[N];
struct qwe
{
int ne,to;
}e[N<<1];
struct dian
{
int p[10];
void add(int u,int v)
{
p[0]++;
p[1]+=u;
p[2]+=u*u;
p[3]+=v;
p[4]+=v*v;
p[5]+=u*v;
}
double wk()
{
double A=p[2]-1.0*p[1]*p[1]/p[0],B=2*p[5]-2.0*p[1]*p[3]/p[0],C=p[4]-1.0*p[3]*p[3]/p[0];
double a=4.0,b=-4.0*(A+C),c=4*A*C-B*B;
return (-b-sqrt(b*b-4*a*c))/a/2;
}
dian operator + (dian x)
{
for(int i=0;i<6;i++)
x.p[i]+=p[i];
return x;
}
dian operator - (dian x)
{
for(int i=0;i<6;i++)
x.p[i]=p[i]-x.p[i];
return x;
}
}a[N],b[N],u,v;
int read()
{
int r=0,f=1;
char p=getchar();
while(p>'9'||p<'0')
{
if(p=='-')
f=-1;
p=getchar();
}
while(p>='0'&&p<='9')
{
r=r*10+p-48;
p=getchar();
}
return r*f;
}
void add(int u,int v)
{
tot++;
e[tot].ne=h[u];
e[tot].to=v;
h[u]=tot;
}
void dfs1(int u,int z,int fat)
{
hs[u]=0;si[u]=1;fa[u]=fat;de[u]=de[fat]+1;
vis[u]=1;rt[u]=z;
a[u]=a[fa[u]];a[u].add(tx[u],ty[u]);
for(int i=h[u];i;i=e[i].ne)
if(!vis[e[i].to]&&e[i].to!=fat)
{
dfs1(e[i].to,z,u);
si[u]+=si[e[i].to];
if(si[e[i].to]>si[hs[u]])
hs[u]=e[i].to;
}
}
void dfs2(int u,int fir)
{
fr[u]=fir;
if(hs[u])
dfs2(hs[u],fir);
for(int i=h[u];i;i=e[i].ne)
if(u==fa[e[i].to]&&e[i].to!=hs[u])//对于可能存在的环
dfs2(e[i].to,e[i].to);
}
void cir()
{
int x,y;
memset(vis,0,sizeof(vis));
for(x=1;x<=n;x++)
for(int i=h[x];i;i=e[i].ne)
if(e[i].to!=fa[x]&&fa[e[i].to]!=x)
{
y=e[i].to;
if(de[x]>de[y])
swap(x,y);
for(;y!=x;y=fa[y])
{
q[++cnt]=y;
id[y]=cnt;
vis[y]=1;
b[cnt]=b[cnt-1];
b[cnt].add(tx[y],ty[y]);
}
q[++cnt]=x;
id[x]=cnt;
vis[x]=1;
b[cnt]=b[cnt-1];
b[cnt].add(tx[x],ty[x]);
return;
}
}
int lca(int u,int v)
{
for(;fr[u]!=fr[v];de[fr[u]]>de[fr[v]]?u=fa[fr[u]]:v=fa[fr[v]]);
return de[u]<de[v]?u:v;
}
int main()
{
n=read(),m=read();
if(m<n-1)//为了测试我对题意的理解是否正确……
{
cout<<23333;
return 0;
}
for(int i=1;i<=n;i++)
tx[i]=read(),ty[i]=read();
for(int i=1;i<=m;i++)
{
int x=read(),y=read();
add(x,y);add(y,x);
}
dfs1(1,1,0);
memset(vis,0,sizeof(vis));
if(n==m)
cir();
else
q[cnt=1]=1;
memset(fa,0,sizeof(fa));
for(int i=1;i<=cnt;i++)
{
dfs1(q[i],q[i],0);
dfs2(q[i],q[i]);
}
Q=read();
while(Q--)
{
int x=read(),y=read();
if(rt[x]==rt[y])
{
int lc=lca(x,y);
u=a[x]+a[y]-a[lc]-a[fa[lc]];
printf("%.5lf\n",u.wk());
}
else
{
if(id[rt[x]]>id[rt[y]])
swap(x,y);
u=a[x]-a[rt[x]]+a[y]-a[rt[y]]+b[id[rt[y]]]-b[id[rt[x]]-1];
v=a[x]-a[rt[x]]+a[y]-a[rt[y]]+b[id[rt[x]]]+b[cnt]-b[id[rt[y]]-1];
printf("%.5lf\n",min(u.wk(),v.wk()));
}
}
return 0;
}
bzoj 3528 [Zjoi2014]星系调查【树链剖分+数学】的更多相关文章
- bzoj 3528: [Zjoi2014]星系调查
Description 银河历59451年,在银河系有许许多多已被人类殖民的星系.如果想要在行 星系间往来,大家一般使用连接两个行星系的跳跃星门. 一个跳跃星门可以把 物质在它所连接的两个行星系中互 ...
- bzoj 3528 [ZJOI2014] 星系调查 题解
[原题] 星系调查 [问题描写叙述] 银河历59451年.在银河系有许很多多已被人类殖民的星系.如果想要在行 星系间往来,大家一般使用连接两个行星系的跳跃星门. 一个跳跃星门能够把 物质在它所连接的 ...
- BZOJ 3672[NOI2014]购票(树链剖分+线段树维护凸包+斜率优化) + BZOJ 2402 陶陶的难题II (树链剖分+线段树维护凸包+分数规划+斜率优化)
前言 刚开始看着两道题感觉头皮发麻,后来看看题解,发现挺好理解,只是代码有点长. BZOJ 3672[NOI2014]购票 中文题面,题意略: BZOJ 3672[NOI2014]购票 设f(i)f( ...
- [bzoj 3531][SDOI2014]旅行(树链剖分+动态开点线段树)
题目:http://www.lydsy.com:808/JudgeOnline/problem.php?id=3531 分析: 对于每个颜色(颜色<=10^5)都建立一颗线段树 什么!那么不是M ...
- BZOJ 3083: 遥远的国度 dfs序,树链剖分,倍增
今天再做一天树的题目,明天要开始专攻图论了.做图论十几天之后再把字符串搞搞,区域赛前再把计几看看. 3083: 遥远的国度 Time Limit: 10 Sec Memory Limit: 128 ...
- BZOJ 1146: [CTSC2008]网络管理Network 树链剖分+线段树+平衡树
1146: [CTSC2008]网络管理Network Time Limit: 50 Sec Memory Limit: 162 MBSubmit: 870 Solved: 299[Submit] ...
- bzoj 4034 [HAOI2015] T2(树链剖分,线段树)
4034: [HAOI2015]T2 Time Limit: 10 Sec Memory Limit: 256 MBSubmit: 1536 Solved: 508[Submit][Status] ...
- bzoj 3531 [Sdoi2014]旅行(树链剖分,线段树)
3531: [Sdoi2014]旅行 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 876 Solved: 446[Submit][Status][ ...
- bzoj 2243 [SDOI2011]染色(树链剖分,线段树)
2243: [SDOI2011]染色 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 4637 Solved: 1726[Submit][Status ...
随机推荐
- 免费第三方API平台整合
各大平台免费接口,非常适用 http://developer.51cto.com/art/201412/458778.htm 绝对干货:供个人开发者赚钱免费使用的一些好的API接口http://www ...
- Git学习之常见错误 clone被拒绝
Git学习之常见错误 问题: git clone 时 报错 Permission Denied (权限被拒绝). 解决方法: 需要把本地的公钥上传到服务器. 解决步骤: ①第一步,设置本地的git的用 ...
- Treasure Hunt--poj1066(最短路加判断线段的关系)
http://poj.org/problem?id=1066 题目大意:有n条线段 他们都在这个房间里 最后有一个点代表起始位置 现在想通过墙出去 他只能爆破每个房间的中点的门 问最少的门通 ...
- Linux下使用Curl调用Java的WebService接口
其实只要是标准的WSDL的SOA接口WebService都可以用. 调用方式: 注意:上面的方式不包括加密或者登录的,其实SOA有一套完整的加密方式. curl -H'Content-Type: te ...
- ArcGIS Engine 中的多线程使用
转自原文ArcGIS Engine 中的多线程使用 一直都想写写AE中多线程的使用,但一直苦于没有时间,终于在中秋假期闲了下来.呵呵,闲话不说了,进入正题! 大家都了解到ArcGIS中处理大数据量时速 ...
- 了解kaggle
Kaggle官网 数据挖掘的比赛,主要是特征工程 Kaggle 数据挖掘比赛经验分享 Kaggle 机器学习竞赛冠军及优胜者的源代码汇总 程序化广告交易中的点击率预估
- 一处折腾笔记:Android内嵌html5加入原生微信分享的解决的方法
有一段时间没有瞎折腾了. 这周一刚上班萌主过来反映说:微信里面打开聚客宝.分享功能是能够的(这里是用微信自身的js-sdk实现的).可是在android应用里面打开点击就没反应了:接下来狡猾的丁丁在产 ...
- Android拍照、摄像方向旋转的问题 代码具体解释
近期做了个拍照.摄像的应用.遇到了拍照.摄像的图像相对于现实.翻转了90度.原因:相机这个硬件的角度是横屏的角度,所以会出现都是横屏的. 1.照相.摄影预览图像的正确角度显 示: public sta ...
- Python 之 读取txt文件
本文直接给出三种实现方法,代码例如以下. 方法一: f = open("Proc_Data.txt") # 返回一个文件对象 line = f.readline() # 调用文件的 ...
- CentOS 使用httpd 配置局域网 yum源
1.上传centos iso而且挂载 mkdir /mnt/cdr/iso mount -o /xxx.iso /mnt/cdr 2.使用createrepo来创建repo文件 cd /mnt/cdr ...