洛谷P1378油滴扩展
题目描述
在一个长方形框子里,最多有N(0≤N≤6)个相异的点,在其中任何一个点上放一个很小的油滴,那么这个油滴会一直扩展,直到接触到其他油滴或者框子的边界。
必须等一个油滴扩展完毕才能放置下一个油滴。那么应该按照怎样的顺序在这N个点上放置油滴,才能使放置完毕后所有油滴占据的总体积最大呢?(不同的油滴不会相互融合)
注:圆的面积公式V=pi*r*r,其中r为圆的半径.
输入输出格式
输入格式:
第1行一个整数N。
第2行为长方形边框一个顶点及其对角顶点的坐标,x,y,x’,y’。
接下去N行,每行两个整数xi,yi,表示盒子的N个点的坐标。
以上所有的数据都在[-1000,1000]内。
输出格式:
一行,一个整数,长方形盒子剩余的最小空间(结果四舍五入输出)
输入样例#1:
输入:
2
20 0 10 10
13 3
17 7
输出:
50
分析:
这题一看搜索啊,再一看又是圆又是矩形的,还要考虑面积,当时我就想哇这题好难好难啊,然后在dalao的讲解并带嘲讽发现这题真的还不难.就是处理的地方有些麻烦
先讲讲题意吧
1.我们已知一个矩阵.
2.已知在矩阵里我们有n个坐标,这些坐标代表的是这n个油滴的坐标由题目已知这n个油滴可以随意扩展直到碰到矩形边界或碰到另一个油滴
且当一个油滴的坐标被另一个油滴覆盖那么这个油滴是不合法的即这个油滴能扩展的面积为0.
3.我们需要求出在所有的油滴扩展完毕后矩阵剩下的面积最小.
条件及面积的处理
因为题目要求求出最小的剩下面积即我们需要求出这个圆能扩展的最大半径.
圆的半径处理
我们可以一个圆在矩阵里能取得最大半径即是这个圆心离这个矩阵四边中最小的距离,这是第一步,接下来因为有其他油滴的存在所以油滴能取得最大半径还要被其他油滴限制
怎么找出它能取得最大半径?简单的画个图就能明白一个圆的在被其他圆限制的条件下它能取得最大半径就是这两个圆之间的欧几里得距离在减去另一个圆的半径
更明确一点.
1.先找到这个圆在矩形中能取的最大半径.
int len1=min(abs(xa-x[k]),abs(xb-x[k]));//这里xa,xb,ya,yb为矩阵的对角点坐标
int len2=min(abs(ya-y[k]),abs(yb-y[k]));//因为坐标可能存在负数的情况所以要abs
double r1=min(len1,len2);//取最小
2.在其他油滴约束下能取得最大半径
for(int i=;i<=n;i++)
{
if(vis[i]&&k!=i)//k为当前这个油滴的编号即第几个油滴//这里k!=i当然自己不能和自己比较//vis[i]能比较当然是在我们放有油滴的情况下
{
r1=min(r1,dis(x[i],x[k],y[i],y[k])-r[i]);//dis是我声明的求欧几里得距离的函数r[i]半径这个半径是和我们比较的那个圆的半径这里取min值是和上面的r1取最小至于为什么我在上面说的很清楚了
}
}
3.注意
因为题目说了如果当前这个油滴的坐标被其他油滴覆盖的话那这个油滴它不能扩展即我们将它的半径变为0
for(int i=;i<=n;i++)//所以这里我们需要枚举这n个所有油滴的坐标来判断
{
if(vis[i]&&k!=i)//这里vis,k和上面的是一样的用处
{
if(r[i]>dis(x[i],x[k],y[i],y[k]))//这个地方很容易明白如果这两个点之间的欧几里得距离小于我们要比较的这个圆的半径说明这个圆就被覆盖了
return ;我们返回它的半径为0
}
}
剩下的就没什么可讲的贴代码
代码:
#include <iostream>
#include <cmath>
#include <algorithm>
#include <cstdlib>
using namespace std;
const int N=+;
const double pi=3.141592;
int n,xa,xb,ya,yb;
double x[N],y[N],keay,r[N];
bool vis[N];
double dis(int x,int x1,int y,int y1)//欧几里得
{
return sqrt((x-x1)*(x-x1)+(y-y1)*(y-y1));
}
double R(int k)//寻找这个圆能取得最大半径
{
for(int i=;i<=n;i++)
{
if(vis[i]&&k!=i)//不合法将它的半径变为0
{
if(r[i]>dis(x[i],x[k],y[i],y[k]))
return ;
}
}
int len1=min(abs(xa-x[k]),abs(xb-x[k]));//在矩形里能取得最大半径
int len2=min(abs(ya-y[k]),abs(yb-y[k]));
double r1=min(len1,len2);
for(int i=;i<=n;i++)
{
if(vis[i]&&k!=i)
{
r1=min(r1,dis(x[i],x[k],y[i],y[k])-r[i]);//比较在矩形里能取的最大半径和在其他圆约束下能取的最大半径
}
}
return r1;
}
void dfs(int t,double sum)
{
if(t==n)
{
keay=max(keay,sum);
return;
}
for(int i=;i<=n;i++)
{
if(!vis[i])
{
r[i]=R(i);//计算每个点能取的最大半径
vis[i]=true;
dfs(t+,sum+pi*r[i]*r[i]);
vis[i]=false;//回溯
}
}
}
int main()
{
double S;
cin>>n;
cin>>xa>>ya>>xb>>yb;
S=abs(xb-xa)*abs(yb-ya);//S为矩阵面积
for(int i=;i<=n;i++)
cin>>x[i]>>y[i];
dfs(,);
cout<<(int)(S-keay+0.5);//四舍五入
return ;
}
洛谷P1378油滴扩展的更多相关文章
- 洛谷P1378 油滴扩展(搜索)
洛谷P1378 油滴扩展 直接暴力搜索更新答案就可以了. 时间复杂度为 \(O(n!)\) . #include<stdio.h> #include<stdlib.h> #in ...
- 洛谷 P1378 油滴扩展 改错
P1378 油滴扩展 题目描述 在一个长方形框子里,最多有\(N(0≤N≤6)\)个相异的点,在其中任何一个点上放一个很小的油滴,那么这个油滴会一直扩展,直到接触到其他油滴或者框子的边界.必须等一个油 ...
- 洛谷P1378 油滴扩展
P1378 油滴扩展 题目描述 在一个长方形框子里,最多有N(0≤N≤6)个相异的点,在其中任何一个点上放一个很小的油滴,那么这个油滴会一直扩展,直到接触到其他油滴或者框子的边界.必须等一个油滴扩展完 ...
- 洛谷 P1378 油滴扩展
P1378 油滴扩展 题目描述 在一个长方形框子里,最多有N(0≤N≤6)个相异的点,在其中任何一个点上放一个很小的油滴,那么这个油滴会一直扩展,直到接触到其他油滴或者框子的边界.必须等一个油滴扩展完 ...
- 洛谷 P1378 油滴扩展 Label:搜索
题目描述 在一个长方形框子里,最多有N(0≤N≤6)个相异的点,在其中任何一个点上放一个很小的油滴,那么这个油滴会一直扩展,直到接触到其他油滴或者框子的边界.必须等一个油滴扩展完毕才能放置下一个油滴. ...
- 洛谷1378 油滴扩展 dfs进行回溯搜索
题目链接:https://www.luogu.com.cn/problem/P1378 题目中给出矩形的长宽和一些点,可以在每个点放油滴,油滴会扩展,直到触碰到矩形的周边或者其他油滴的边缘,求出剩余面 ...
- P1378 油滴扩展——搜索小记
P1378 油滴扩展 记得这道题好久以前(好像是上个学期?) 就想做了,但是看着里面的半径边界好像很难处理就没做(主要是当时刚学OI(菜还给自己找借口)): 今天上午一直研究SG函数,做的都自闭了,晚 ...
- [动态规划]P1378 油滴扩展
题目描述 在一个长方形框子里,最多有N(0≤N≤6)个相异的点,在其中任何一个点上放一个很小的油滴,那么这个油滴会一直扩展,直到接触到其他油滴或者框子的边界.必须等一个油滴扩展完毕才能放置下一个油滴. ...
- P1378 油滴扩展
题目描述 在一个长方形框子里,最多有N(0≤N≤6)个相异的点,在其中任何一个点上放一个很小的油滴,那么这个油滴会一直扩展,直到接触到其他油滴或者框子的边界.必须等一个油滴扩展完毕才能放置下一个油滴. ...
随机推荐
- HDU 1060 Leftmost Digit(求N^N的第一位数字 log10的巧妙使用)
Leftmost Digit Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others)To ...
- Unity让带有Rigidbody组件的游戏对象停止运动
Rigidbody rigidbody = transform.GetComponent<Rigidbody>(); rigidbody.velocity = Vector3.zero; ...
- shell脚本中 [-eq] [-ne] [-gt] [-lt] [ge] [le]
-eq //等于 -ne //不等于 -gt //大于 (greater ) -lt //小于 (less) -ge //大于等于 -le //小于等于 在linux 中 命令执行状态:0 为真,其他 ...
- Block代替delegate,尽量使用block,对于有大量的delegate方法才考虑使用protocol实现.
Block代替delegate,尽量使用block,对于有大量的delegate方法才考虑使用protocol实现. 1.Block语法总结及示例如下: //1.普通代码块方式bloc ...
- nodejs+express开发blog(1)
前言 Nodejs是一个年轻的编程框架,充满了活力和无限激情,一直都在保持着快速更新.基于Nodejs的官方Web开发库Express也在同步发展着,每年升级一个大版本,甚至对框架底层都做了大手术.在 ...
- oracle的事务隔离级别和读一致性
oracle提供了三个隔离级别: 1.读提交 ,简而言之只能读取语句开始执行前提交的数据 2.串行,这个好理解,就是事务串行运行,避免经典的三个场景-脏读.不可重复读.幻读. 3.只读,oracle已 ...
- C#判断系统是64位还是32位 支持.net4.0以前的版本
C#判断系统是64位还是32位的时候引用了一串代码,这个代码是从园子里面其他博文中转载过来的,引入自己的项目中发现无法使用,在引用了相应的命名空间之后还是提示: "未能找到类型或命名空间名称 ...
- JS日期去杠,日期转换String转Date
1.巧妙使用split()和join()替换字符串var str = '2014-05-05';var newstr = str.split('-').join("");split ...
- 【PBR的基本配置】
PBR基于策略路由的配置 一:基于报文协议的本地PBR 1:首先进行理论分析:在SW1上利用基于报文报文协议类型的PBR,在sw1与sw3的连接链路上,利用acl制定允许tcp的报文通过3000,并与 ...
- JZOJ 5943. 树
Description