bzoj 3528: [Zjoi2014]星系调查
Description
银河历59451年,在银河系有许许多多已被人类殖民的星系。如果想要在行
Input
第一行是N,M,分别表示这个银河系内的行星系的个数
Output
总共Q行,每一行一个实数,表示花花第i次需要计算的答
Sample Input
3 4
5 6
1 3
4 4
3 3
2 4
1 2
1 3
2 3
2 4
3 5
5 6
3
3 6
2 4
4 6
Sample Output
0.66667
0.00000
1.67544
一道良心的必修一压轴题+树链剖分
通过一系列漫长的推导+勇敢的展开,最后只需要维护
推导的话主要是运用两次主元法(点到直线距离公式要记得),最后由求根公式即可,推导以后补:
设最后的那条直线的解析式为y=kx+b,设n为路径上的点数
由点到直线的距离公式:

带入暴算并把b当做主元:

分子是一个二次函数形式,可以计算出整个式子取最小值时的b值(二次函数顶点)

令:

则 b=yar-xar;
在把b带入以k为主元:
可以把分子变为一个关于k的二次函数形式:

令:


则 暴力计算可得到



所以

乘过去:

由于有解,所以

所以:

易知在Ans取较小的那个零点的时候为所求答案,由求根公式计算即可;
不容易啊
因为图连通且不超过n条边,所以为树或者一个基环树
树的话直接树链剖分,维护的东西直接用树上前缀和即可: x+y-lca-fa[lca];
基环树的话,把环看做根节点后就是树了,如果lca为环的话分别跳到环上,
环上的按照某种时针顺序预处理前缀和,然后分别走两条路统计即可.
如果lca不是环就正常树的即可
#include<iostream>
#include<cstdio>
#include<cstring>
#include<cmath>
#include<algorithm>
#define RG register
using namespace std;
const int N=300050;
int n,m,cnt,tot,root[N],head[N],to[N],nxt[N],fa[N],dep[N],size[N],son[N],top[N],cir[N],rt[N];
int tx[N],ty[N],vis[N];
int gi()
{
int x=0,flag=1;
char ch=getchar();
while(ch<'0'||ch>'9'){if(ch=='-') flag=-1;ch=getchar();}
while(ch>='0'&&ch<='9') x=x*10+ch-'0',ch=getchar();
return x*flag;
}
struct data{
int pre[6];
void update(RG int u,RG int v){
pre[0]++;pre[1]+=u;pre[2]+=u*u;
pre[3]+=v;pre[4]+=v*v;pre[5]+=u*v;
}
double query(){
RG double n=pre[0],totx=pre[1],toty=pre[3],totx2=pre[2],toty2=pre[4],totxy=pre[5];
RG double xar=totx/n,yar=toty/n;
RG double a=totx2-2*xar*totx+n*xar*xar;
RG double b=2*yar*totx+2*xar*toty-2*totxy-2*n*xar*yar;
RG double c=toty2-2*yar*toty+n*yar*yar;
RG double A=4.0,B=-4*(a+c),C=4*a*c-b*b;
return (-B-sqrt(B*B-4*A*C))/2/A;
}
}a[N],b[N];
data operator +(data x,data y){
for(RG int i=0;i<6;i++) x.pre[i]+=y.pre[i]; return x;
}
data operator -(data x,data y){
for(RG int i=0;i<6;i++) x.pre[i]-=y.pre[i]; return x;
}
void lnk(RG int x,RG int y){
to[++tot]=y;nxt[tot]=head[x];head[x]=tot;
}
inline void dfs1(int x,int f){
vis[x]=1;size[x]=1;son[x]=0;rt[x]=f;
a[x]=a[fa[x]]; a[x].update(tx[x],ty[x]);
for(RG int i=head[x];i;i=nxt[i]){
RG int y=to[i];
if (!vis[y]&&y!=fa[x]){
dep[y]=dep[x]+1;fa[y]=x;
dfs1(y,f);size[x]+=size[y];
if(size[y]>size[son[x]]) son[x]=y;
}
}
}
inline void dfs2(RG int x,RG int f){
top[x]=f;
if(son[x]) dfs2(son[x],f);
for (RG int i=head[x]; i; i=nxt[i]){
RG int y=to[i];
if(x==fa[y] && y!=son[x]) dfs2(y,y);
}
}
inline int lca(RG int x,RG int y){
while(top[x]!=top[y]){
if(dep[top[x]]<dep[top[y]]) swap(x,y);
x=fa[top[x]];
}
if(dep[x]<dep[y]) swap(x,y);
return y;
}
inline void work(){
memset(vis,0,sizeof(vis));
for(RG int x=1;x<=n;x++)
for(RG int i=head[x];i;i=nxt[i]){
if(fa[x]!=to[i]&&fa[to[i]]!=x){
RG int y=to[i];
if(dep[x]>dep[y]) swap(x,y);
for(;y!=x;y=fa[y]){
root[++cnt]=y;cir[y]=cnt;vis[y]=1;
b[cnt]=b[cnt-1];b[cnt].update(tx[y],ty[y]);
}
root[++cnt]=x;cir[x]=cnt;vis[x]=1;
b[cnt]=b[cnt-1];b[cnt].update(tx[x],ty[x]);
return;
}
}
}
int main(){
n=gi(),m=gi();RG int x,y;
for(RG int i=1;i<=n;i++) tx[i]=gi(),ty[i]=gi();
for(RG int i=1;i<=m;i++){
x=gi(),y=gi();
lnk(x,y); lnk(y,x);
}
dfs1(1,1);
memset(vis,0,sizeof(vis));
if(n==m) work();else root[cnt=1]=1;
memset(fa,0,sizeof(fa));
for(RG int i=1;i<=cnt;i++){
dfs1(root[i],root[i]); dfs2(root[i],root[i]);
}
RG int Q=gi();data u,v;
for(RG int i=1;i<=Q;i++){
x=gi(),y=gi();
if (rt[x]==rt[y]){
u=a[x]+a[y]-a[lca(x,y)]-a[fa[lca(x,y)]];
printf("%.5f\n",u.query());
} else{
if (cir[rt[x]]>cir[rt[y]]) swap(x,y);
u=a[x]-a[rt[x]]+a[y]-a[rt[y]]+b[cir[rt[y]]]-b[cir[rt[x]]-1];
v=a[x]-a[rt[x]]+a[y]-a[rt[y]]+b[cir[rt[x]]]+b[cnt]-b[cir[rt[y]]-1];
printf("%.5f\n",min(u.query(),v.query()));
}
}
return 0;
}
bzoj 3528: [Zjoi2014]星系调查的更多相关文章
- bzoj 3528 [ZJOI2014] 星系调查 题解
[原题] 星系调查 [问题描写叙述] 银河历59451年.在银河系有许很多多已被人类殖民的星系.如果想要在行 星系间往来,大家一般使用连接两个行星系的跳跃星门. 一个跳跃星门能够把 物质在它所连接的 ...
- bzoj 3528 [Zjoi2014]星系调查【树链剖分+数学】
参考:https://www.cnblogs.com/zhuohan123/p/3698852.html 首先,根据点到直线距离公式 \[ d=\frac{kx_0-y_0+b}{\sqrt{k^{2 ...
- BZOJ3528: [Zjoi2014]星系调查
唉,看到这题直接想起自己的Day1,还是挺难受的,挺傻一题考试的时候怎么就没弄出来呢…… 这两天CP让我给他写个题解,弄了不是很久就把这个题给弄出来了,真不知道考试的时候在干嘛. 明天就出发去北京了, ...
- 题解 洛谷 P3340 【[ZJOI2014]星系调查】
根据题意,发现题目中的图,其实就是一颗树或者是一颗基环树,每个节点上有一个点对\((x,y)\),每次询问为给定端点,找一条直线到端点间的所有点的距离之和最小. 设这条直线为\(y=kx+b\),根据 ...
- 「ZJOI2014」星系调查
「ZJOI2014」星系调查 本题核心在于快速求XPs 的线性假设相斥度. 点\((x1,y1)\)到直线\(y=kx+b\)的距离的平方为\(\displaystyle {(kx1+b-y1)^2} ...
- BZOJ 2768: [JLOI2010]冠军调查 最小割
2768: [JLOI2010]冠军调查 题目连接: http://www.lydsy.com/JudgeOnline/problem.php?id=2768 Description 一年一度的欧洲足 ...
- 【bzoj 3786】星系探索
Description 物理学家小C的研究正遇到某个瓶颈. 他正在研究的是一个星系,这个星系中有n个星球,其中有一个主星球(方便起见我们默认其为1号星球),其余的所有星球均有且仅有一个依赖星球.主星球 ...
- BZOJ 3527: [Zjoi2014]力
Description 求 \(E_i=\sum _{j=0}^{i-1} \frac {q_j} {(i-j)^2}-\sum _{j=i+1}^{n-1} \frac{q_j} {(i-j)^2} ...
- bzoj 2768: [JLOI2010]冠军调查
#include<cstdio> #include<iostream> #define M 100000 #include<cstring> using names ...
随机推荐
- 第六届蓝桥杯软件类省赛题解C++/Java
第六届蓝桥杯软件类省赛题解C++/Java 1[C++].统计不含4的数字统计10000至99999中,不包含4的数值个数.答:暴力循环范围内所有数字判断一下就是了,答案是52488 1[Java]. ...
- css选择器的优先级问题
当我们写页面的时候,不知道你会不会产生这样的问题,为什么我给他添加的这条样式分明已经选择到我要给的元素了,但是他的样式并没有生效,那是为什么呢? 定义的属性有冲突时,浏览器会选择用那一套样式呢,下面来 ...
- Flex布局学习笔记
任何元素都可以使用Flex布局,包括行内元素 display: flex; display: inline-flex使用Flex布局之后,子元素的float, clear, vertical-alig ...
- 《Linux命令行与shell脚本编程大全》第二十二章 gawk进阶
gawk是一门功能丰富的编程语言,你可以通过它所提供的各种特性来编写好几程序处理数据. 22.1 使用变量 gawk编程语言支持两种不同类型的变量: 内建变量和自定义变量 22.1.1 内建变量 ga ...
- centos7 无法启动网络(service network restart)错误解决办法
centos7 无法启动网络(service network restart)错误解决办法: (以下方法均为网上COPY,同时感谢原博主分享) systemctl status network.ser ...
- 探究PHP底层
探究PHP底层 1.PHP是什么? PHP 指的是我们从外面看到的一套完整的系统.这听起来有点糊涂,但其实并不复杂(PHP4 内部结构图).从功能上来分:我们可以分为三部分: 1. 解释器部分(Z ...
- Swift 线程安全数组
欢迎大家前往腾讯云社区,获取更多腾讯海量技术实践干货哦~ 作者:BigNerdCoding 有并发的地方就存在线程安全问题,尤其是对于 Swift 这种还没有内置并发支持的语言来说线程安全问题更为突出 ...
- Js 作用域与作用域链与执行上下文不得不说的故事 ⁄(⁄ ⁄•⁄ω⁄•⁄ ⁄)⁄
最近在研究Js,发现自己对作用域,作用域链,活动对象这几个概念,理解得不是很清楚,所以拜读了@田小计划大神的博客与其他文章,受益匪浅,写这篇随笔算是自己的读书笔记吧~. 作用域 首先明确一个概念,js ...
- Jarvis OJ- [XMAN]level2/3_x64-Writeup——64位简单栈溢出
两道64位栈溢出,思路和之前的32位溢出基本一致,所以放在一起 在这两道中体现的32位和64位的主要区别在于函数参数传递的方式 在32位程序运行中,函数参数直接压入栈中 调用函数时栈的结构为:调用函数 ...
- PHP中的会话控制
了解HTTP(超文本传输协议)可以知道,它采用请求与响应的模式,最大的特点就是无连接无状态. 无连接:每次连接仅处理一个客户端的请求,得到服务器响应后,连接就结束了 无状态:每个请求都是独立的,服务器 ...