POJ 2227 The Wedding Juicer (优先级队列+bfs+dfs)
思路描述来自:http://hi.baidu.com/perfectcai_/item/701f2efa460cedcb0dd1c820
也可以参考黑书P89的积水。
题意:Farmer John有一个碗,形状如下,给你一个w*h平面,每个1*1的位置有一个高度Hij,问用这个碗来装牛奶,最多可以装多少体积。
例如:
555
515
555
这个可以装4体积,只有中间的1位置可以装;
再如:
555
512
555
这个可以装1体积,只有中间的1位置可以装,而装到2时就会开始漏出,因为液体是可以流动的;
分析:
由例子二可以很深刻的理解如果在一个桶边上打一些洞,那么之后这个桶能装的水取决于最低的那个洞的高度,
接着这个思路往下,对于最开始,只要找到这w*h外围的方格放入队列,这是最开始的边缘,取出其中最低的点x,
然后考察它周围4个位置中合法并且可以积水的点中的格子y,高度为Hy。
若y的高度大于x的高度,那么直接把Hy放入队列,并标记为禁止积水;
如果小于,把Hx放入队列,且ans+=Hx-Hy,意味着y位置可以放Hx-Hy体积的水,也标记为禁止积水。
然后再取禁止积水的格子中最低的,即队列中的最小值,继续向四周拓展。
核心思想:找最低边缘,把积过水的格子看成高度更新后的边缘。
由于要取最小高度,所以用到优先级队列,至于如何向四周扩展,我用的是bfs+dfs
#include <iostream>
#include <queue>
#include <stdio.h>
#include <algorithm> using namespace std;
const int maxn=;
int w,h;
long long block[maxn][maxn];
int mark[maxn][maxn]; //标记是否可以积水
long long ans=;
struct Node{
int x,y;
long long height;
bool operator<(const Node tmp)const{
return height>tmp.height;
}
}tmp;
priority_queue<Node> q;
//终于知道为什么AC不了了啊,原来之前的参数height写得是h,与整个board的高度h重复了额
void dfs(int x,int y,long long height){
if(x<||x>h||y<||y>w)
return;
mark[x][y]=;
ans+=height-block[x][y];
//上
if(x->&&mark[x-][y]){
if(height>block[x-][y])
dfs(x-,y,height);
else{
mark[x-][y]=;
tmp.x=x-;
tmp.y=y;
tmp.height=block[x-][y];
q.push(tmp);
}
}
//下
if(x+<=h&&mark[x+][y]){
if(height>block[x+][y])
dfs(x+,y,height);
else{
mark[x+][y]=;
tmp.x=x+;
tmp.y=y;
tmp.height=block[x+][y];
q.push(tmp);
}
}
//左
if(y->&&mark[x][y-]){
if(height>block[x][y-])
dfs(x,y-,height);
else{
mark[x][y-]=;
tmp.x=x;
tmp.y=y-;
tmp.height=block[x][y-];
q.push(tmp);
}
}
//右
if(y+<=w&&mark[x][y+]){
if(height>block[x][y+])
dfs(x,y+,height);
else{
mark[x][y+]=;
tmp.x=x;
tmp.y=y+;
tmp.height=block[x][y+];
q.push(tmp);
}
}
}
int main()
{
Node node;
scanf("%d%d",&w,&h);
for(int i=;i<=h;i++){
for(int j=;j<=w;j++){
mark[i][j]=;
scanf("%I64d",&block[i][j]); //存储高度
if(i==||i==h||j==||j==w){
mark[i][j]=; //表示禁止积水
node.x=i;
node.y=j;
node.height=block[i][j];
q.push(node);
}
}
}
ans=;
//bfs+dfs
while(!q.empty()){
node=q.top();
q.pop();
dfs(node.x,node.y,node.height);
}
cout<<ans<<endl;
return ;
}
下面给出只用队列+bfs的代码,速度稍慢一点:
参考链接:http://www.cnblogs.com/dongsheng/archive/2013/04/25/3043512.html
#include <iostream>
#include <queue>
#include <stdio.h>
#include <algorithm> using namespace std;
const int maxn=;
int w,h;
long long block[maxn][maxn];
int mark[maxn][maxn];
int dir[][] = {-,,,,,,,-};
long long ans=;
struct Node {
int x,y;
long long h;
bool operator<(const Node tmp)const {
return h>tmp.h;
}
} tmp;
priority_queue<Node> q;
long long BFS()
{
long long ans = ;
int i, j;
Node t1, t2;
while(!q.empty())
{
t1 = q.top();
q.pop();
for(int k = ; k < ; ++k)
{
i = t1.x + dir[k][];
j = t1.y + dir[k][];
if(i > && i <= h && j > && j <= w && mark[i][j])
{
if(block[i][j] < t1.h)
{
ans += t1.h - block[i][j];
t2.x = i;
t2.y = j;
t2.h = t1.h;
}
else
{
t2.x = i;
t2.y = j;
t2.h = block[i][j];
}
mark[i][j] = ;
q.push(t2);
}
}
}
return ans;
}
int main() {
Node node; //cin>>w>>h;
while(scanf("%d%d",&w,&h)!=EOF) {
for(int i=; i<=h; i++) {
for(int j=; j<=w; j++) {
mark[i][j]=;
scanf("%I64d",&block[i][j]);
if(i==||i==h||j==||j==w) {
mark[i][j]=; //表示禁止积水
node.x=i;
node.y=j;
node.h=block[i][j];
q.push(node);
}
}
}
ans=;
printf("%I64d\n",BFS());
}
return ;
}
POJ 2227 The Wedding Juicer (优先级队列+bfs+dfs)的更多相关文章
- POJ 3253 Fence Repair 贪心 优先级队列
Fence Repair Time Limit: 2000MS Memory Limit: 65536K Total Submissions: 77001 Accepted: 25185 De ...
- 【POJ 3614 Sunscreen】贪心 优先级队列
题目链接:http://poj.org/problem?id=3614 题意:C头牛去晒太阳,每头牛有自己所限定的spf安全范围[min, max]:有L瓶防晒液,每瓶有自己的spf值和容量(能供几头 ...
- POJ:3083 Children of the Candy Corn(bfs+dfs)
http://poj.org/problem?id=3083 Description The cornfield maze is a popular Halloween treat. Visitors ...
- 【BFS+优先级队列】Rescue
https://www.bnuoj.com/v3/contest_show.php?cid=9154#problem/I [题意] 给定一个n*m的迷宫,A的多个小伙伴R要去营救A,问需要时间最少的小 ...
- POJ 2431 Expedition 贪心 优先级队列
Expedition Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 30702 Accepted: 8457 Descr ...
- ACM/ICPC 之 优先级队列+设置IO缓存区(TSH OJ-Schedule(任务调度))
一个裸的优先级队列(最大堆)题,但也有其他普通队列的做法.这道题我做了两天,结果发现是输入输出太过频繁,一直只能A掉55%的数据,其他都是TLE,如果将输入输出的数据放入缓存区,然后满区输出,可以将I ...
- 算法与数据结构基础 - 堆(Heap)和优先级队列(Priority queue)
堆基础 堆(Heap)是具有这样性质的数据结构:1/完全二叉树 2/所有节点的值大于等于(或小于等于)子节点的值: 图片来源:这里 堆可以用数组存储,插入.删除会触发节点shift_down.shif ...
- 体验Rabbitmq强大的【优先级队列】之轻松面对现实业务场景
说到队列的话,大家一定不会陌生,但是扯到优先级队列的话,还是有一部分同学是不清楚的,可能是不知道怎么去实现吧,其实呢,,,这东西已 经烂大街了...很简单,用“堆”去实现的,在我们系统中有一个订单催付 ...
- Java中的队列Queue,优先级队列PriorityQueue
队列Queue 在java5中新增加了java.util.Queue接口,用以支持队列的常见操作.该接口扩展了java.util.Collection接口. Queue使用时要尽量避免Collecti ...
随机推荐
- 多点触摸画板(MultiTouchCanvas)
这是个简单的支持多点触摸的画板控件, 绘制功能基于WPF InkCanvas,也是我drawTool系列文章的开篇. 阅读该文章后可能产生一些问题: 1. 如果对生成的笔迹对象进行控制 如果要对生成的 ...
- INFORMIX数据库常用命令
INFORMIX数据库常用命令 一.onstat命令集 1.onstat - 说明:查看数据库当前的状态 用法:onstat - 2.onstat -c 说明:查看数据库的配置文件 用法:ons ...
- hdu 3342 Legal or Not
题目连接 http://acm.hdu.edu.cn/showproblem.php?pid=3342 Legal or Not Description ACM-DIY is a large QQ g ...
- golang实现ping命令
// Copyright 2009 The Go Authors. All rights reserved.// Use of this source code is governed by a B ...
- R 语言中文乱码问题
R 语言似乎在WINDOWS平台上对中文的支持不是特别好,似乎是3.1.2的一个BUG. 目前我研究出了一个临时解决方案,你可以将代码编写成一个函数,从而在调用的过程中不必如下繁琐: 1. 先将本地语 ...
- Labview实现幅度信号调制(AM)
Labview实现幅度信号调制(AM) 时域上的表达式: 其中,m(t)是交流信号分量,均值为0,需要被调制的信号,此处选择一个正弦信号,正好满足要求. A0是一个直流分量,表示叠加的直流分量,用加法 ...
- OC学习心得【适合初学者】
一.类和对象 1.OC语言是C语言的扩充,并且OC是iOS和OS X操作系统的编程语言. ①具备完善的面向对象特性: 封装:将现实世界中存在的某个客体的属性与行为绑定在一起,并放置在一个逻辑单元内 继 ...
- IOS用CGContextRef画各种图形(文字、圆、直线、弧线、矩形、扇形、椭圆、三角形、圆角矩形、贝塞尔曲线、图片)
... 首先了解一下CGContextRef: An opaque type that represents a Quartz 2D drawing environment. Graphics Con ...
- 网络服务器带宽Mbps、Mb/s、MB/s有什么区别?10M、100M到底是什么概念?
网络服务器带宽Mbps.Mb/s.MB/s有什么区别?我们经常听到IDC提供的服务器接入带宽是10M独享,或者100M独享,100M共享之类的数据.这的10M.100M到底是什么概念呢? 工具/原料 ...
- Applicationpoolidentity 好有趣哦
前言: 在服务器上搭建一个网站 Windows Server2008 R2 +IIS7. 应用程序池默认选择ApplicationPoolIdentity.为了适应分布式需求,所以将SqlServer ...