问题 F: 分盒子(经典)

问题 F: 分盒子
时间限制: 1 Sec 内存限制: 128 MB

[提交][状态][讨论版]
题目描述
国际妇女节马上就要到了!波利卡正在为假期做准备。

商店里有n个糖果盒出售。这个盒子里有糖果。

Polycarp希望为k个女孩准备最多的礼物。每一份礼物将由两个盒子组成。女孩们应该能够平等地分享每一份礼物,所以一份礼物(在一对盒子里)中的糖果总数应该能被k整除。换句话说,如果di+dj能被k整除,那么两个盒子i和j (i≠j)可以组合作为礼物。

Polycarp能提供多少个盒子?当然,每个盒子只能是一件礼物的一部分。Polycarp不能“部分”使用盒子,也不能在盒子之间重新分配糖果。

输入
输入的第一行包含两个整数n和k(1≤n≤2⋅105,1 ≤k≤110)——箱子数量和号码的女孩。

输入的第二行包含n个整数d1,d2,…,dn(1≤di≤109),其中di为第i个盒子中的糖果数量。

多组数据输入

输出
打印一个整数——Polycarp可以作为礼物赠送的盒子的最大数量。

样例输入
7 2
1 2 2 3 2 4 10
8 2
1 2 2 3 2 4 6 10
7 3
1 2 2 3 2 4 5
样例输出
6
8
4
提示

第一组可以选(2,3),(5,6),(1,4),可以使用6个盒子,所以答案为6
/*
之前做过 一个 统计 一个素数对和等于n 的素数对的个数 的题目,和这题有异曲同工之妙。

凑一对和为某个值的对数,就要像办法拆分分别统计可以凑成一对的单个元素的个数。
这题,要使一对和能整除k。
先对数据预处理,每个数对k取余以简化计算,取余的结果同数组mp数组统计个数。
最终只要统计的mp[i],mp[k-i]有值就来计算答案。
特别考虑mp[0],和 i==k-i 的情况
*/
ac_code:

#include 
using namespace std;
int mp[115];
 int main()
 {
     int n,k;
     while(~scanf("%d%d",&n,&k))
     {
         int x;
         memset(mp,0,sizeof(mp));
         for(int i = 0; i < n; i++)
         {
             scanf("%d",&x);
             mp[x%k]++;
         }
         int ans = mp[0]/2;
         for(int i = 1; i <= k/2; i++)
         {
             if(mp[i]&&mp[k-i])
             {
                 if(i != k-i) ans += min(mp[i],mp[k-i]);
                 else ans += mp[i]/2;
             }
         }
         printf("%d\n",ans*2);
     }
     return 0;
 }

你可能感兴趣的:(思维题)