20190817-T1-LOJ6322「雅礼国庆 2017 Day6」Star Way To Heaven
写这篇题解是因为作者太蒻已经忘了最小生成树了。
<题面>
这个题还真是想不到最小生成树。
$80\%$算法
复杂度:$\Theta(k^2 \log N )$
用了二分答案(明显答案具有单调性)
然后$k^2$暴力判断是否合法。
可以得到80分。
$100\%$算法
复杂度:$\Theta(k^2)$
考虑上面的暴力判断,
如何判断呢?要搜点距,$dfs$
然后我们就可以得到一些东西。
假设现在得到了答案是$ans$
我们考虑它的特性。
在每一个点上以$ans$为半径画圆
那么,一定有一条边上的两条圆是相切的。
如果$ans$变小,那么一定有一个更优解。
如果$ans$更大,那么圆一定会相交导致路径不连续。
我们再找找性质,
发现这两个圆一定在上边界到下边界的路径上,且是路径上最长的边,这也是导致上文路径不连续的原因。
对于一个点,那个圆一定出现在与它相连的最短边上,
因为如果有更长边,更长边会充当一个三角形的最长边,导致路径会受更短的一条边约束。
于是有最小生成树(我考试肯定想不出来QAQ)
在找最小生成树时,就一定可以保证找到加入树的边一定是上文的最短边。
于是直接套用$Prim$顺便维护到下边界的距离,复杂度$\Theta(k^2)$
我本来想优化,于是想用堆:$\Theta(k^2) \Rightarrow \Theta(e \log k)$
于是边数$e=k^2$
$\Theta(k^2) \Rightarrow \Theta(k^2 \log k)$//当我没说
这时有一个问题,不能建图,空间复杂度不可承受。
于是需要使用欧几里得距离最小生成树。
你可能肯定会想,这又是什么玩意?
其实就是最小生成树,只是不建边而是去用点直接计算距离。
所以愉快的AC了
#include <iostream>
#include <cstdio>
#include <cmath>
#define N 6060
#define LF double using namespace std; int pn,hei;
struct x_y{
LF x,y;
}ps[5*N];
LF dis[5*N],ans=0;
bool is_v[N];
inline LF len(const x_y a,const x_y b){
return sqrt((a.x-b.x)*(a.x-b.x)+(a.y-b.y)*(a.y-b.y));
}
void prim(){
for(int i=1;i<=pn;i++){
dis[i]=hei-ps[i].y;
}
for(int i=1;i<=pn+1;i++){
int x=0;
for(int j=1;j<=pn+1;j++){
if((!is_v[j])&&(x==0||dis[j]<dis[x])){
x=j;
}
}
ans=max(ans,dis[x]);
if(x==pn+1){//这里是必写的,因为这个意味着更新结束
return ;
}
is_v[x]=1;
int j;
for(j=1;j<=pn;j++){
{
dis[j]=min(dis[j],len(ps[j],ps[x]));
}
}
dis[pn+1]=min(dis[pn+1],ps[x].y);
}
}
int main(){
int x,y;
scanf("%*d%d%d",&hei,&pn);
for(int i=1;i<=pn;i++){
scanf("%d%d",&x,&y);
ps[i].x=x;
ps[i].y=y;
}
dis[pn+1]=hei;
prim();
printf("%.9lf",ans/2);
}
码量也并不大。
这里顺便写一下两个最小生成树的板板。
1.Kruskal
/*****
Kruskal
*****/
#include<cmath>
#include<cstdio>
#include<iostream>
#include<algorithm>
using namespace std;
bool b[101];
int answer,ans,n,t,f[1001];
struct node
{
int fr,to,ti;
}a[10001];
bool cmp(node xx,node yy)
{
return xx.ti<yy.ti;
}
int find(int x)
{
if (f[x]==x) return x;
else return f[x]=find(f[x]);
}
void merge(int x,int y)
{
int xx=find(x);
int yy=find(y);
if (xx<yy) f[xx]=f[yy];
else f[yy]=f[xx];
}
int main()
{
scanf("%d",&n);
for (int i=1; i<=n; ++i) f[i]=i;
for (int i=1; i<=n; ++i)
for (int j=1; j<=n; ++j)
{
int x;
scanf("%d",&x);
if (x!=0) {a[++t]=(node){i,j,x};}
}
int k=0;
sort(a+1,a+t+1,cmp);
for (int i=1; i<=t; ++i)
{
if (find(a[i].fr)!=find(a[i].to)) {//不在同一个集合中
merge(a[i].fr,a[i].to);//合并
ans+=a[i].ti;//记录最小生成值
k++;//记录边数
}
if (k==n-1) break;//一棵树,n个点,n-1条边
}
printf("%d",ans);
}
2.Prim
/******
Prime
******/
#include<iostream>
#include<cstdio>
#include<cstring>
using namespace std;
int con[101][101],dis[101],num,a;
int main(){
cin>>num;
long long sum=0,min=0x7fffffff;
memset(dis,0x7f,sizeof(dis));
for(int i=1;i<=num;i++){
for(int j=1;j<=num;j++){
scanf("%d",&con[i][j]);
}
}
dis[1]=0;
for(int i=1;i<=num;i++)
dis[i]=con[1][i];
for(int i=1;i<=num;i++){
min=0x7f7f7f7f;
for(int j=1;j<=num;j++){
if(dis[j]!=0&&dis[j]<min){
min=dis[j];//cout<<"i"<<i<<"min"<<min<<endl;
a=j;//cout<<"a"<<a<<endl;
}
}
sum+=dis[a];
dis[a]=0;
for(int j=1;j<=num;j++)
if(dis[j]>con[a][j])
dis[j]=con[a][j];
}
printf("%d\n",sum);
return 0;
}
20190817-T1-LOJ6322「雅礼国庆 2017 Day6」Star Way To Heaven的更多相关文章
- 【loj6307】「雅礼国庆 2017 Day1」Clique 贪心
题目描述 数轴上有 $n$ 个点,第 $i$ 个点的坐标为 $x_i$ 权值为 $w_i$ .两个点 $i,j$ 之间存在一条边当且仅当 $|x_i−x_j|\le w_i+w_j$ . 你需要求出这 ...
- Loj #6307. 「雅礼国庆 2017 Day1」Clique
link: https://loj.ac/problem/6307 最大团转补图的独立集,这样的话只有r[x]<l[y]或者r[y]<l[x],x和y才能连边,所以排序之后乱搞就行了. 需 ...
- 【loj6034】「雅礼集训 2017 Day2」线段游戏
#6034. 「雅礼集训 2017 Day2」线段游戏 内存限制:256 MiB 时间限制:1000 ms 标准输入输出 题目类型:传统 评测方式:Special Judge 上传者: 匿名 题目描述 ...
- LOJ_6045_「雅礼集训 2017 Day8」价 _最小割
LOJ_6045_「雅礼集训 2017 Day8」价 _最小割 描述: 有$n$种减肥药,$n$种药材,每种减肥药有一些对应的药材和一个收益. 假设选择吃下$K$种减肥药,那么需要这$K$种减肥药包含 ...
- 「雅礼集训 2017 Day7」事情的相似度
「雅礼集训 2017 Day7」事情的相似度 题目链接 我们先将字符串建后缀自动机.然后对于两个前缀\([1,i]\),\([1,j]\),他们的最长公共后缀长度就是他们在\(fail\)树上对应节点 ...
- 「雅礼集训 2017 Day2」解题报告
「雅礼集训 2017 Day2」水箱 我怎么知道这种题目都能构造树形结构. 根据高度构造一棵树,在树上倍增找到最大的小于约束条件高度的隔板,开一个 \(vector\) 记录一下,然后对于每个 \(v ...
- 「雅礼集训 2017 Day1」 解题报告
「雅礼集训 2017 Day1」市场 挺神仙的一题.涉及区间加.区间除.区间最小值和区间和.虽然标算就是暴力,但是复杂度是有保证的. 我们知道如果线段树上的一个结点,\(max=min\) 或者 \( ...
- [LOJ 6031]「雅礼集训 2017 Day1」字符串
[LOJ 6031] 「雅礼集训 2017 Day1」字符串 题意 给定一个长度为 \(n\) 的字符串 \(s\), \(m\) 对 \((l_i,r_i)\), 回答 \(q\) 个询问. 每个询 ...
- [LOJ 6030]「雅礼集训 2017 Day1」矩阵
[LOJ 6030] 「雅礼集训 2017 Day1」矩阵 题意 给定一个 \(n\times n\) 的 01 矩阵, 每次操作可以将一行转置后赋值给某一列, 问最少几次操作能让矩阵全为 1. 无解 ...
随机推荐
- hp笔记本在设置VT-x为启用模式后还是无法在VMware上开启CentOS虚拟机
在h笔记本上,将VT-x设置为Enabled模式后,需要断开电源,拆下电池,然后再按住开机按钮10秒钟左右放开,再重新装上电池,接通电源即可.
- Joomla - K2组件(文章管理扩展)
一.下载 K2 进入 https://getk2.org/ ,点击DOWNLOAD K2 下载K2 下载完毕得到一个安装包 二.安装 K2 进入看后台,点击顶栏主菜单 扩展管理 -> 扩展安装 ...
- ios与android设备即时语音互通的录音格式预研说明
本文虽属原创,但是内容都是来自于网络,参考了大家的微博,以及论坛的总结. 在做语音对讲的时候,将会碰到录制语音格式的问题,因为要考虑自己开发设备的支持的格式,还要考虑其他设备操作系统的支持的格式,以及 ...
- bean的使用
前言 Spring最基础的功能就是一个bean工厂,所以本文讲解的是Spring生成bean的种种方法及细节,Spring配置文件的名字是bean.xml,定义几个类: 一个Person类: publ ...
- IDEA中Git的使用(多人合作)
首先我们要简单知道github跟Git的区别.git是版本控制工具, github是一个面向开源及私有软件项目的托管平台,也是程序员交流的地方. 接下来就开始讲怎么多人一起开发. 首先我们先拥有git ...
- js 实现页面局部(或图片)放大功能(vue)
方法: adjustStart1 (e) { e.preventDefault() let event = e.touches if (event.length === 2) { this.style ...
- An invalid property 'jdbcType ' was found in mapping
大概2种原因: 1 放进去的类型与字段的类型不匹配 2 比较变态,xml中=两边不能有空格! 错误示例如下: #{plat,jdbcType = INTEGER}, 去掉空格后: #{plat,jd ...
- mybatis-环境配置-基本案例-和hibernate区别
Mybatis第一天 1. Mybatis介绍 MyBatis 本是apache的一个开源项目iBatis, 2010年这个项目由apache software foundation 迁移到了goo ...
- (Eclipse) 安装Subversion1.82(SVN)插件
简介 :SVN是团队开发的代码管理工具,它使我们得以进行多人在同一平台之下的团队开发. 解决问题:Eclipse下的的SVN插件安装. 学到 :Eclipse下的的SVN插件安装. 资源地 ...
- mysql设置密码登录
参考: https://blog.csdn.net/Light_Breeze/article/details/82070222 https://www.jianshu.com/p/d979df2791 ...