长方形—C++

编程之美一道简单的热身题,活生生的组合数学例子啊。

题意如下

      在 N 条水平线与 M 条竖直线构成的网格中,放 K 枚石子,每个石子都只能放在网格的交叉点上。问在最优的摆放方式下,最多能找到多少四边平行于坐标轴的长方形,它的四个角上都恰好放着一枚石子。

输入

  输入文件包含多组测试数据。第一行,给出一个整数T(1 ≤ T ≤ 100)为数据组数。接下来依次给出每组测试数据。每组数据为三个用空格隔开的整数 N,M,K。(0 ≤ K ≤ N * M,0 < N, M ≤ 30000)。

输出

  对于每组测试数据,输出一行"Case #X: Y",其中X表示测试数据编号,Y表示最多能找到的符合条件的长方形数量。所有数据按读入顺序从1开始编号。

样例输入

3
3 3 8
4 5 13
7 14 86

样例输出

Case #1: 5
Case #2: 18
Case #3: 1398

解题思路:

  考虑对于一个布满点的网格(x行,Y列),所有的矩形总数为组合数C(2,x)*C(2,y)。但k不一定刚好就能布满整个网格,所以我们先找到在网格中最大能形成的长方形矩阵的长y和宽x,剩余石子为r,则有k = x * y + r , 最大能形成的长方形矩阵最多有C(x,2)*C(y,2)个矩形,剩余石子 r 以一排或一列的形式,靠在大矩形短的一边要注意是否到达边界),则多增加的矩形数为 C(2,r)*y (y为长边的点数)。 枚举x与y,计算出C(x,2)*C(y,2)+ C(r,2)*y的最大值。

 

代码如下:

#include <iostream>
#include <cmath>
#include <algorithm>
using namespace std;

long long Func(long long n)
{
    return n*(n-1)/2;
}
int main()
{
    int T;
    cin >> T;
    for(int cnt = 1; cnt <= T; ++cnt)
    {
        int n, m, k;
        cin >> n >> m >> k;
        if(n > m) swap(n,m);        //n为短边,m为长边(个人习惯~0~)
        long long ans = 0;
        for(int x = 2; x <= n; ++x)   //对x进行枚举(短边)
        {
            int y = k / x;         //可能的最大长边y
            int r = k % x;         //剩余的石子数r
            if( y > m || ( y == m && r) )   //超边界  
                continue;
            ans = max(ans, Func(x)*Func(y) + y*Func(r));
        }
        cout << "Case #" << cnt << ": " << ans << endl;
    }
    return 0;
}

 

 

posted on 2014-04-06 21:51  bbking  阅读(1075)  评论(0编辑  收藏  举报

导航