传送门

分析

首先把(package)的克数 $Q_{ij}$ 转化成区间 $[\lceil Q_{ij}/(1.1 r_i )\rceil, \lfloor Q_{ij}/(0.9 r_i)\rfloor]$。可以通过将分子、分母同乘10来避免精度问题。注意某个包对应的区间可能为空,比如 $Q_{ij}=13, r_i=10$。然后将每种成分(ingredient)的包对应的区间按左右端点双关键字从小到大排序(用std::pair<int,int>表示区间)。考虑 $n$ 个最小的区间,两两比较,若不相交则将较小的区间删除(该区间对应的包不可能存在于任何 kit 中)。重复此删除过程,直到最小的 $n$ 个区间交集不为空。用这 $n$ 个区间对应的包组成一个 kit。

这个做法的正确性可通过如下性质证明:

对应同一成分的两区间 $a$、$b$,若 $a \subset b$ 则 $a$、$b$ 的左右两端点至少有一个重合。

Implementation

#include <bits/stdc++.h>
using namespace std;
int T;
int cas;

const int N=55;
int r[N];

int q[N][N];


using P=pair<int,int> ;




int main(){
    for(cin>>T; T--; ){
        printf("Case #%d: ", ++cas); // !注意输出格式
        ///////////////////////////////////////
        int  n, p;
        cin >> n >> p;

        priority_queue<P, vector<P>, greater<P>> que[N];

        for(int i=0; i<n; i++)
            cin >> r[i];

        for(int i=0; i<n; i++)
            for(int j=0; j<p; j++){
                cin >> q[i][j];
                int mi=10*q[i][j]/(11*r[i])+bool(10*q[i][j]%(11*r[i]));
                int ma=q[i][j]*10/(9*r[i]);
                if(mi>ma) continue;
                que[i].push({mi, ma});
                // cout << mi <<' ' << ma<<endl;
            }

        // for(int i=0; i<n; i++){
        //     auto x=que[i].top();
        //     cout << x.first<<' '<< x.second << endl;
        // }
        int res=0;

        for(;;){
            bool flag=true;
            for(int i=0; i<n; i++)
                if(que[i].empty()){
                    flag=false;
                    break;
                }

            if(!flag) break;

            bool f=true;
            for(int i=0; i<n; i++){
                for(int j=i+1; j< n; j++){
                    auto a=que[i].top(), b=que[j].top();
                    if(a.first>b.second){
                        que[j].pop();
                        f=false;
                        break;
                    }
                    if(b.first>a.second){
                        que[i].pop();
                        f=false;
                        break;
                    }
                }
                if(f==false) break;
            }

            if(f==false)
                continue;

            res++;

            for(int i=0; i<n; i++)
                que[i].pop();
        }


        cout << res<<endl;
////////////////////////////////////////////////////////
    }
    return 0;

}

相关文章:

  • 2022-12-23
  • 2021-09-16
  • 2022-12-23
  • 2021-11-15
  • 2022-02-12
  • 2022-02-09
  • 2022-01-11
  • 2021-12-16
猜你喜欢
  • 2022-12-23
  • 2021-08-24
  • 2020-04-12
  • 2021-09-20
  • 2021-08-16
  • 2021-12-24
  • 2021-10-19
相关资源
相似解决方案