- #include <bits/stdc++.h>
- //#include <iostream>
- //#include <vector>
- //#include <algorithm>
- //#include <tuple>
- using namespace std;
- // 應徵者結構體
- struct Applicant {
- int id, x, y, z;
- Applicant(int id, int x, int y, int z) : id(id), x(x), y(y), z(z) {}
- };
- // 自訂排序方式 (根據單職位能力排序)
- bool cmpX(const Applicant &a, const Applicant &b) { return a.x > b.x; }
- bool cmpY(const Applicant &a, const Applicant &b) { return a.y > b.y; }
- bool cmpZ(const Applicant &a, const Applicant &b) { return a.z > b.z; }
- int main() {
- int n, a, b, c;
- cin >> n >> a >> b >> c;
- vector<Applicant> applicants;
- // 讀取輸入資料
- for (int i = 0; i < n; i++) {
- int x, y, z;
- cin >> x >> y >> z;
- applicants.emplace_back(i, x, y, z);
- }
- // 記錄被選中的應徵者
- vector<bool> selected(n, false);
- int maxPower = 0;
- // **Step 1: 選擇最強的工程師**
- sort(applicants.begin(), applicants.end(), cmpX);
- for (int i = 0; i < a && i < n; i++) {
- maxPower += applicants[i].x;
- selected[applicants[i].id] = true;
- }
- // **Step 2: 選擇最強的清潔工**
- sort(applicants.begin(), applicants.end(), cmpY);
- int cleanCount = 0;
- for (int i = 0; i < n && cleanCount < b; i++) {
- if (!selected[applicants[i].id]) { // 尚未被選為工程師
- maxPower += applicants[i].y;
- selected[applicants[i].id] = true;
- cleanCount++;
- }
- }
- // **Step 3: 選擇最強的小丑**
- sort(applicants.begin(), applicants.end(), cmpZ);
- int clownCount = 0;
- for (int i = 0; i < n && clownCount < c; i++) {
- if (!selected[applicants[i].id]) { // 尚未被選為工程師或清潔工
- maxPower += applicants[i].z;
- clownCount++;
- }
- }
- // **Step 4: 找雙職位的應徵者補充**
- vector<tuple<int, int, int>> dualOptions; // (應徵者ID, 增加的實力, 類型)
- for (int i = 0; i < n; i++) {
- if (!selected[applicants[i].id]) continue;
- int extraPower = 0;
- int type = -1;
- // 這個人已被選擇,但還能補其他職位
- if (cleanCount < b) {
- extraPower = applicants[i].y;
- type = 1; // 增加清潔工
- }
- if (clownCount < c && applicants[i].z > extraPower) {
- extraPower = applicants[i].z;
- type = 2; // 增加小丑
- }
- if (type != -1) {
- dualOptions.emplace_back(applicants[i].id, extraPower, type);
- }
- }
- // 按額外實力排序,選擇最優解
- sort(dualOptions.begin(), dualOptions.end(), [](const tuple<int, int, int> &a, const tuple<int, int, int> &b) {
- return get<1>(a) > get<1>(b);
- });
- for (auto &[id, extra, type] : dualOptions) {
- if (type == 1 && cleanCount < b) { // 增加清潔工
- maxPower += extra;
- cleanCount++;
- } else if (type == 2 && clownCount < c) { // 增加小丑
- maxPower += extra;
- clownCount++;
- }
- }
- // **輸出最大總實力**
- cout << maxPower << endl;
- return 0;
- }
複製代碼 回復 1# may
----------------------------------------------------------
測資
測試資料00(基本測試)
輸入:
3 1 1 1
1 2 3
3 1 2
2 3 1
預期輸出:
9
解釋:
第 2 位當工程師 (3)
第 3 位當清潔工 (3)
第 1 位當小丑 (3)
測試資料01(所有名額皆可滿足)
輸入:
3 2 2 2
1 2 3
3 1 2
2 3 1
預期輸出:
15
解釋:
第 2、3 位當工程師 (3+2)
第 1、3 位當清潔工 (2+3)
第 1、2 位當小丑 (3+2)
測試資料 02(部分名額不足)
輸入:
4 3 3 3
4 5 6
3 2 1
7 8 9
5 6 7
預期輸出:
42
解釋:
第 3、4、1 位當工程師 (7+5+4)
第 3、4、1 位當清潔工 (8+6+5)
第 3、4、1 位當小丑 (9+7+6)
測試資料 03(應徵者能力有重複值)
輸入:
5 2 2 2
5 5 5
5 5 5
5 5 5
5 5 5
5 5 5
預期輸出:
20
解釋:
任意 2 位當工程師 (5+5)
任意 2 位當清潔工 (5+5)
任意 2 位當小丑 (5+5)
測試資料 04(名額小於應徵者數量,需選最強者)
輸入:
6 2 2 2
1 2 3
4 5 6
7 8 9
3 6 9
5 2 1
9 3 2
預期輸出:
38
解釋:
第 6、3 位當工程師 (9+7)
第 3、2 位當清潔工 (8+5)
第 3、4 位當小丑 (9+9)
這些測試資料涵蓋了:
基本情境(測試正確性)。
所有名額皆可滿足(測試最大化選擇)。
部分名額不足(測試名額限制)。
應徵者能力相同(測試公平性)。
名額小於應徵者數量(測試最佳選擇策略)。 |