传送门:
题意:有n块巧克力,每块巧克力长为a[i],宽为b[i]。有m个盒子,每个盒子长为c[i],宽为d[i]。问能否把n块巧克力放入m个盒子中(每个盒子只能放一块,不能旋转)
分析:首先有一个想法,对巧克力和盒子按长宽排序,再对第i个巧克力挑出所有c[i]>=a[i]的盒子,从中选择一个d[i]>=b[i]的盒子放入巧克力。正着考虑贪心并不能保证最优,我们考虑倒着放。假设我们已经处理好了第n块巧克力,现在来处理第n-1的巧克力。假设c[i]>=a[i]的盒子有k个,由于之前已经去掉了一个,现在还有k-1个盒子,要使得当前巧克力的选择最多(显然只对盒子的宽有要求),第n块巧克力取的必须是满足条件c[i]>=a[i]的宽最小的盒子。
代码:
#include<bits/stdc++.h>
using namespace std;
#define int long long
int a[200005],b[200005],c[200005],d[200005];
// vector<pair<int,int>>ab[200005],cd[200005];//放外面并不会分配空间(push_back才会分配空间)不如放里面
multiset<int>s;
signed main()
{
int n,m;
cin>>n>>m;
for(int i=1;i<=n;i++)
{
cin>>a[i];
}
for(int i=1;i<=n;i++)
{
cin>>b[i];
}
for(int i=1;i<=m;i++)
{
cin>>c[i];
}
for(int i=1;i<=m;i++)
{
cin>>d[i];
}
vector<pair<int,int>>ab(n),cd(m);//分配空间
for(int i=1;i<=n;i++)
{
ab[i-1]=make_pair(a[i],b[i]);
}
for(int i=1;i<=m;i++)
{
cd[i-1]=make_pair(c[i],d[i]);
}
sort(ab.begin(),ab.end());
sort(cd.begin(),cd.end());
int j=m-1;
int f=0;
for(int i=n-1;i>=0;i--)
{
while(cd[j].first>=ab[i].first&&j>=0)
{
s.insert(cd[j].second);
j--;
}
auto p=s.lower_bound(ab[i].second);
if(p==s.end())
{
f=1;
break;
}
s.erase(p);
}
if(f) puts("No");
else puts("Yes");
}
v