P1227 【[JSOI2008]完美的对称】
这道题,先讲一下我的做题思路
这道题的最主要的目的就是算出中心,我下面称为中点。这个中点其实很好算的,我们只需要算出最左下角的坐标和最右上角的坐标,然后用中点坐标公式算出来就ok了,那么这道题就做完了一半
中点坐标公式:
\(x_{mid}=(x_{min}+x_{max})/2\)
\(y_{mid}=(y_{min}+y_{max})/2\)
那么剩下的一半就是如何就算无解的情况了,首先这个中点我们已经算出来了,如果有什么疑惑的可以自己手模一下
对于无解的情况,我们对于\((x_i,y_i)\),若此时中点为\((x_{mid},y_{mid})\),那么这个点关于中点的中心对称之后的点就是\((2*x_{mid}-x_i,2*y_{mid}-y_i)\)
这个时候对于这个中心对称之后的点,我们只需要\(O(n)\)扫一下就可以了
那么给出第一份程序
#include<bits/stdc++.h>
#define MAXN 50000
#define INF 100000
using namespace std;
double n;
struct node{
double x,y;
}a[MAXN];
double minx=INF,miny=INF,maxx=-INF,maxy=-INF;
double sx,sy;
bool find(double x,double y){
for(register int i=1;i<=n;i++){
if(a[i].x==x&&a[i].y==y) return true; //O(n) 扫一下
}
return false;
}
bool check(){
for(register int i=1;i<=n;i++){
double x=a[i].x,y=a[i].y;
if(find(2*sx-x,2*sy-y)==false) return false; //如果找不到中心对称之后的点,说明不行
}
return true;
}
int main(){
scanf("%lf",&n);
for(register int i=1;i<=n;i++){
scanf("%lf%lf",&a[i].x,&a[i].y);
minx=min(minx,a[i].x);
miny=min(miny,a[i].y);
maxx=max(maxx,a[i].x);
maxy=max(maxy,a[i].y);
}
sx=(double)(maxx+minx)*1.0/2;
sy=(double)(maxy+miny)*1.0/2;
//算出中点
if(check()==false){
puts("This is a dangerous situation!");
}else printf("V.I.P. should stay at (%.1f,%.1f).",sx,sy);
return 0;
}
然后就会发现这个程序跑出来慢得离谱,我的是\(2.9s\),因为每一次查找确实需要用掉太多时间,那如何去优化这个时间呢
我们可以对于所有的坐标进行排序,以\(x\)为第一关键字,\(y\)为第二关键字,那么我们的\(check()\)函数只需要循环\(1\)~\((n+1)/2\),然后我们的\(find()\)函数只需要循环\(n\)~\((n+1)/2\),这是运用了中心对称的性质,这样优化之后的程序是\(600s+\)
#include<bits/stdc++.h>
#define MAXN 50000
#define INF 100000
using namespace std;
int n;
struct node{
double x,y;
}a[MAXN];
double minx=INF,miny=INF,maxx=-INF,maxy=-INF;
double sx,sy;
bool cmp(node q,node w){
if(q.x==w.x) return q.y<w.y;
return q.x<w.x;
}
bool find(double x,double y){
for(register int i=n;i>=(n-1)/2-1;i--){ //这里的-1是因为c++是向下取整,多算一位
if(a[i].x==x&&a[i].y==y) return true;
}
return false;
}
bool check(){
for(register int i=1;i<=(n+1)/2+1;i++){ //这里的+1和上面的理由一样
double x=a[i].x,y=a[i].y;
if(find(2*sx-x,2*sy-y)==false) return false;
}
return true;
}
int main(){
scanf("%d",&n);
for(register int i=1;i<=n;i++){
scanf("%lf%lf",&a[i].x,&a[i].y);
minx=min(minx,a[i].x);
miny=min(miny,a[i].y);
maxx=max(maxx,a[i].x);
maxy=max(maxy,a[i].y);
}
sort(a+1,a+1+n,cmp);
sx=(double)(maxx+minx)*1.0/2;
sy=(double)(maxy+miny)*1.0/2;
if(check()==false){
puts("This is a dangerous situation!");
}else printf("V.I.P. should stay at (%.1f,%.1f).",sx,sy);
return 0;
}
P1227 【[JSOI2008]完美的对称】的更多相关文章
- 洛谷——P1227 [JSOI2008]完美的对称
P1227 [JSOI2008]完美的对称 题目描述 在峰会期间,必须使用许多保镖保卫参加会议的各国代表.代表们除了由他自己的随身保镖保护外,组委会还指派了一些其他的特工和阻击手保护他们.为了使他们的 ...
- 洛谷 P1227 [JSOI2008]完美的对称
传送门 题目大意:求一些点集的公共对称中心 题解:对称中心是可以确定的,再判断. 代码: #include<iostream> #include<cstdio> #includ ...
- [JSOI2008]完美的对称 题解
题目大意: 首先我们给定一点A以及对称中心S,点A'是点A以S为对称中心形成的像点,即点S是线段AA'的对称中心. 点阵组(X)以S为中心的像点是由每个点的像点组成的点阵组.X是用来产生对称中心S的, ...
- [JSOI2008] 完美的对称
题目描述 在峰会期间,必须使用许多保镖保卫参加会议的各国代表.代表们除了由他自己的随身保镖保护外,组委会还指派了一些其他的特工和阻击手保护他们.为了使他们的工作卓有成效,使被保卫的人的安全尽可能得到保 ...
- [luoguP1227] [JSOI2008]完美的对称(sort)
传送门 排序! #include <cstdio> #include <iostream> #include <algorithm> #define N 20001 ...
- 【省选水题集Day1】一起来AK水题吧! 题目(更新到B)
题解:http://www.cnblogs.com/ljc20020730/p/6937954.html 水题A: [AHOI2001]质数和分解 题目网址: https://www.luogu.or ...
- 【省选水题集Day1】一起来AK水题吧! 题解(更新到B)
题目:http://www.cnblogs.com/ljc20020730/p/6937936.html 水题A:[AHOI2001]质数和分解 安徽省选OI原题!简单Dp. 一看就是完全背包求方案数 ...
- Redis Sentinel中的机制与原理详解
序言 Redis-Sentinel是Redis官方推荐的高可用性(HA)解决方案.实际上这意味着你可以使用Sentinel模式创建一个可以不用人为干预而应对各种故障的Redis部署. 它的主要功能有以 ...
- Redis Sentinel实现的机制与原理详解
序言 Redis-Sentinel是Redis官方推荐的高可用性(HA)解决方案.实际上这意味着你可以使用Sentinel模式创建一个可以不用人为干预而应对各种故障的Redis部署. 它的主要功能有以 ...
随机推荐
- 什么是 PHP 过滤器?
PHP 过滤器 PHP 过滤器用于验证和过滤来自非安全来源的数据,比如用户的输入. 什么是 PHP 过滤器? PHP 过滤器用于验证和过滤来自非安全来源的数据. 测试.验证和过滤用户输入或自定义数据是 ...
- OKHttp 官方文档【一】
最近工作比较忙,文章更新出现了延时.虽说写技术博客最初主要是写给自己,但随着文章越写越多,现在更多的是写给关注我技术文章的小伙伴们.最近一段时间没有更新文章,虽有工作生活孩子占用了大部分时间的原因,但 ...
- Hadoop学习之第一个MapReduce程序
期望 通过这个mapreduce程序了解mapreduce程序执行的流程,着重从程序解执行的打印信息中提炼出有用信息. 执行前 程序代码 程序代码基本上是<hadoop权威指南>上原封不动 ...
- SpringSceurity(6)---JWT详解
SpringSceurity(6)---JWT详解 在JWT之前我们在做用户认证的时候,基本上会考虑session 和 token,所以在讲jwt之前,我们先来回顾下这个两个 一.传统的session ...
- C++ 中可调用的且有函数功能的东东
第一个:函数 其实函数在声明的时候都有个名字: 这个名字可以看作是是指针,将其直接赋值给函数指针 也可以看作是可取指的对其& 再赋值给函数指针 第二个:函数指针 通过其被赋值的方式 ...
- IDEA必备插件系列-Rainbow Brackets(彩虹括号)
Rainbow Brackets ,就是彩虹括号,各种鲜明颜色的括号 这个一个开源的项目: https://github.com/izhangzhihao/intellij-rainbow-brack ...
- Android ExpandListView的用法(补上昨天的)(今天自习)
今天自习写ExpandListView的作业,昨天没写博客就是去写作业去了. 今天来说昨天内容吧! 其实ExpandListView和ListView的用法大同小异. 首先就是创建一个自己的适配器(现 ...
- 13、Java 异常处理
1.简介 什么是异常?程序运行时,发生的不被期望的事件,它阻止了程序按照程序员的预期正常执行,这就是异常.异常发生时,是任程序自生自灭,立刻退出终止.在Java中即,Java在编译或运行或者运行过程中 ...
- JavaScript重定向
使用JavaScript重定向到其他网页的一些方法: location.href location.replace() location.assign() 语法: window.location.hr ...
- 【模式识别与机器学习】——PCA与Kernel PCA介绍与对比
PCA与Kernel PCA介绍与对比 1. 理论介绍 PCA:是常用的提取数据的手段,其功能为提取主成分(主要信息),摒弃冗余信息(次要信息),从而得到压缩后的数据,实现维度的下降.其设想通过投影矩 ...