zoukankan      html  css  js  c++  java
  • UVA

    Description

    Download as PDF

    Problem G - Summits

    Time limit: 8 seconds

    You recently started working for the largest map drawing company in theNetherlands. Part of your job is to determine what the summits in aparticular landscape are. Unfortunately, it is not so easy to determinewhich points are summits and which are not, because we do not want tocall a small hump a summit. For example look at the landscape given bythe sample input.

    We call the points of height 3 summits, since there are no higherpoints. But although the points of height 2, which are to theleft of the summit of height 3, are all higher than or equal totheir immediate neighbours, we do notwant to call them summits, because we can reach a higher point fromthem without going to low (the summits of height 3). In contrast,we do want to call the area of height 2 on the right a summit, sinceif we would want to walk to the summit of height 3, we first have todescend to a point with height 0.

    After the above example, we introduce the concept of a d-summit. Apoint, with height h, is a d-summit if and only if it isimpossible to reach a higher point without going through an area withheight smaller than or equal to h-d.

    The problem is, given a rectangular grid of integer heights and aninteger d, to find the number of d-summits.

    Input

    On the first line one positive number: the number of testcases, atmost 100. After that per testcase:

    • One line with three integers 1 ≤ h ≤ 500, 1 &le w ≤ 500 and 1 ≤ d ≤ 1000000000. h and w are the dimensions of the map. d is as defined in the text.
    • h lines with w integers, where the xth integer on the yth line denotes the height 0 ≤ h ≤ 1000000000 of the point (x, y).

    Output

    Per testcase:

    • One line with the number of summits.

    Sample Input

    1
    6 10 2
    0 0 0 0 0 0 0 0 0 0
    0 1 2 1 1 1 1 0 1 0
    0 2 1 2 1 3 1 0 0 0
    0 1 2 1 3 3 1 1 0 0
    0 2 1 2 1 1 1 0 2 0
    0 0 0 0 0 0 0 0 0 0
    

    Sample Output

    4
    
    The 2007 ACM Northwestern European Programming Contest

    题意:多么费解的题目啊,找顶点,假设一个点是最高的话那么就是顶点。假设不是的话,可是它到比它到的点的路径中假设有<=h-d(h为该点的高度)那么就不能去,那么它就是顶点

    思路:首先找个性质:假设A->B,C->B,假设HA>HC,由于HA-d>HC-d,那么C->A,所以我们先按高度排序,然后逐个BFS,假设它的周围能找到跟它一样高的点。那么这些点都是顶点。假设遇到已经被较高找到的点。那么它就也能够到那个较高的点。那么它就不是顶点

    #include <iostream>
    #include <cstring>
    #include <cstdio>
    #include <vector>
    #include <algorithm>
    #include <queue>
    using namespace std;
    const int MAXN = 510;
    
    struct node {
    	int x, y, h;
    	node(int _x = 0, int _y = 0, int _h = 0) {
    		x = _x;
    		y = _y;
    		h = _h;
    	}
    } arr[MAXN*MAXN];
    int map[MAXN][MAXN];
    int vis[MAXN][MAXN];
    int n, m, d;
    int cnt;
    int dx[4]={1, -1, 0, 0};
    int dy[4]={0, 0, 1, -1};
    queue<node> q;
    
    int cmp(node a, node b) {
    	return a.h > b.h;
    }
    
    void cal() {
    	memset(vis, -1, sizeof(vis));
    	int ans = 0;
    	while (!q.empty())
    		q.pop();
    	for (int i = 0; i < cnt; i++) {
    		node cur = arr[i];
    		if (vis[cur.x][cur.y] != -1)
    			continue;
    		int flag = 1;
    		int bound = cur.h - d;
    		int top = cur.h;
    		q.push(cur);
    		int tot = 1;
    		while (!q.empty()) {
    			cur = q.front();
    			q.pop();
    			vis[cur.x][cur.y] = top;
    			for (int i = 0; i < 4; i++) {
    				int nx = cur.x + dx[i];
    				int ny = cur.y + dy[i];
    				if (nx < 1 || ny < 1 || nx > n || ny > m)
    					continue;
    				if (map[nx][ny] <= bound)
    					continue;
    				if (vis[nx][ny] != -1) {
    					if (vis[nx][ny] != top)
    						flag = 0;
    					continue;
    				}
    				node tmp;
    				tmp.x = nx, tmp.y = ny, tmp.h = map[nx][ny];
    				vis[nx][ny] = top;
    				if (tmp.h == top) 
    					tot++;
    				q.push(tmp);
    			}
    		}
    		if (flag) 
    			ans += tot;
    	}
    	printf("%d
    ", ans);
    }
    
    int main() {
    	int t;
    	scanf("%d", &t);
    	while (t--) {
    		scanf("%d%d%d", &n, &m, &d);
    		cnt = 0;
    		for (int i = 1; i <= n; i++)
    			for (int j = 1; j <= m; j++) {
    				scanf("%d", &map[i][j]);
    				arr[cnt++] = node(i, j, map[i][j]);
    			}
    		sort(arr, arr+cnt, cmp);
    		cal();
    	}
    	return 0;
    }



    版权声明:本文博客原创文章,博客,未经同意,不得转载。

  • 相关阅读:
    托词坚持了170多天,昨天因为晚上打球竟然给忘了
    2013转眼间半年过去了,回顾一下。也看一下计划的实施情况以及下半年的计划
    开始新的板子PCB绘制了。
    致时代前行者:致敬每一个奔腾不息的心灵(转)
    刚才看了年初的计划,增加一部分内容
    五种男人
    哪些行业会用到乐泰胶水?
    第一个python小程序
    一个简单的IPmsg程序源码分析(一)
    关于linux下面printf函数缓冲区问题
  • 原文地址:https://www.cnblogs.com/yxwkf/p/4618995.html
Copyright © 2011-2022 走看看