2008NOIP普及组真题 2. 排座椅

线上OJ:

【08NOIP普及组】排座椅

核心思想:

1、假设第 i 行和第 i+1 行之间有 2 组交头接耳的学生对,则 row[i] = 2,表示第i行 和第i+1行有2组;
2、假设第 i 列和第 i+1 列之间有 5 组交头接耳的学生对,则 col[i] = 5,表示第i列 和第i+1列有5组;
3、所以,在读入每一行交头接耳的数据时(共 d 对数据),如果是在同一行,则更新row[i];如果是在同一列,则更新col[i]
4、对row[i]进行降序排序,挑选最大的k个进行输出(输出时按照id的升序进行输出);同理,对col[i]进行降序排序,挑选最大的L个进行输出(输出时按照id的升序进行输出)

题解代码:
#include <bits/stdc++.h>
#define id second
#define cnt first
using namespace std;

const int N = 1005;

typedef pair<int, int> PII;
PII row[N], col[N];
int m, n, k, l, d;
int x, y, p, o;
int prt[N];

bool cmp(PII a, PII b)
{
    return a.cnt > b.cnt;  // 按照该行(该列)交头接耳的数量进行排序(从大到小)
}

int main()
{
    scanf("%d %d %d %d %d", &m, &n, &k, &l, &d);
    for(int i = 1; i <= m; i++) row[i].id = i;
    for(int i = 1; i <= n; i++) col[i].id = i;

    for(int i = 1; i <= d; i++)
    {
        scanf("%d %d %d %d", &x, &y, &p, &o);
        if(x == p)  y > o ? col[o].cnt++: col[y].cnt++;  // 如果第3列和第2列有交头接耳,则第2列的cnt++
        else if(y == o)  x > p ? row[p].cnt++: row[x].cnt++; // 如果第3行和第2行有交头接耳,则第2行的cnt++
    }

    sort(row + 1, row + 1 + m, cmp); // 对行的cnt进行降序排序
    sort(col + 1, col + 1 + n, cmp); // 对列的cnt进行降序排序

    for(int i = 1; i <= k; i++)  prt[i] = row[i].id;  // 把行cnt最大的k个id挑出来
    sort(prt + 1, prt + 1 + k);  // 对这k个id进行升序排序(输出要求)
    for(int i = 1; i <= k; i++)  printf("%d ", prt[i]);
    printf("\n");

    for(int i = 1; i <= l; i++)  prt[i] = col[i].id;  // 把列cnt最大的l个id挑出来
    sort(prt + 1, prt + 1 + l);  // 对这 L 个id进行升序排序(输出要求)
    for(int i = 1; i <= l; i++)  printf("%d ", prt[i]);

    return 0;
}
  • 5
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值