標題:
202406潛力2-會議室租用MeetingRoom_找不到judge
[打印本頁]
作者:
may
時間:
2024-10-20 16:29
標題:
202406潛力2-會議室租用MeetingRoom_找不到judge
[attach]20011[/attach]
大明經營一間專門租用共享空間的公司,由於需求太過熱門,目前只剩下一間會議室可供出租。已知還有 N 位客戶在等候租用,第 i位客戶需要租用時間ai到 bi(包含 ai和 bi)並會支付 ci元。公司希望在使用會議室的時間不衝突的情況之下選擇一些客戶使得收入最大化。 例如:有N = 3位客戶,需要會議室的時段與支付費用如下表所示。最大化收入的策略為選擇第1和3位客戶,收入為6+3=9元。
[attach]20336[/attach]
輸入格式
第一列有一個正整數 N ( N ≤ 2×105),表示有 N 位客戶。接下來 N 列,每列有三個正整數 ai, bi, ci ( ai ≤ bi ≤ 106, ci ≤ 104),兩個正整數之間以一個空白為間
隔,表示第 i 位客戶要從時段 ai到 bi (包含 ai和 bi) 支付 ci元租用會議室。
輸出格式
請輸出一個非負整數,表示最大收入。
[attach]20337[/attach]
評分說明
此題目測資分成四組,每組測資有多筆測試資料,需答對該組所有測試資
料才能獲得該組分數,各組詳細限制如下。
第一組 (20 分):N ≤ 10。
第二組 (20 分):所有的 ci = 1。
第三組 (30 分):N ≤ 103。
第四組 (30 分):無特別限制。
作者:
may
時間:
2024-12-23 21:55
#include<cstdio>
#include<cstring>
#include<vector>
#include<algorithm>
using namespace std;
const int N = 200000 + 5;
struct Case {
int l, r, w;
};
int solve(int n, const int l[N], const int r[N], const int w[N])
{
static Case v[N];
static int dp[N];
for (int i = 0; i < n; ++i)
{
v[i] = (Case){l[i], r[i], w[i]};
}
sort(v, v + n, [](const auto& lhs, const auto& rhs) {
return lhs.r < rhs.r;
});
dp[0] = v[0].w;
for (int i = 1; i < n; ++i)
{
int cur;
int pos = lower_bound(v, v + i, (Case){0, v[i].l, 0}, [](const auto& lhs, const auto& rhs) {
return lhs.r < rhs.r;
}) - v - 1;
if (pos < 0)
{
cur = v[i].w;
}
else
{
cur = dp[pos] + v[i].w;
}
dp[i] = max(dp[i - 1], cur);
}
return dp[n - 1];
}
int main()
{
int n;
static int l[N], r[N], w[N];
scanf("%d", &n);
for (int i = 0; i < n; ++i)
{
scanf("%d%d%d", &l[i], &r[i], &w[i]);
}
printf("%d\n", solve(n, l, r, w));
return 0;
}
複製代碼
作者:
may
時間:
2024-12-23 22:09
//#include <iostream>
//#include <vector>
//#include <algorithm>
#include <bits/stdc++.h>
using namespace std;
// 定義結構存儲客戶資料
struct Customer {
int start, end, profit;
};
// 比較函數:按結束時間排序
bool compareByEnd(const Customer &a, const Customer &b) {
return a.end < b.end;
}
// 二分搜尋:找到不重疊的最後一個客戶
int findLastNonConflict(const vector<Customer> &customers, int index) {
int low = 0, high = index - 1;
while (low <= high) {
int mid = (low + high) / 2;
if (customers[mid].end < customers[index].start) {
if (customers[mid + 1].end < customers[index].start) {
low = mid + 1;
} else {
return mid;
}
} else {
high = mid - 1;
}
}
return -1;
}
int main() {
int N;
cin >> N;
vector<Customer> customers(N);
for (int i = 0; i < N; ++i) {
cin >> customers[i].start >> customers[i].end >> customers[i].profit;
}
// 按結束時間排序
sort(customers.begin(), customers.end(), compareByEnd);
// 動態規劃陣列
vector<long long> dp(N);
dp[0] = customers[0].profit;
for (int i = 1; i < N; ++i) {
// 包含當前客戶的收益
long long includeProfit = customers[i].profit;
int last = findLastNonConflict(customers, i);
if (last != -1) {
includeProfit += dp[last];
}
// 不包含當前客戶的收益
dp[i] = max(dp[i - 1], includeProfit);
}
// 輸出最大收益
cout << dp[N - 1] << endl;
return 0;
}
/*
輸入:
3
1 4 6
2 7 8
5 6 3
輸出:
9
--------
6
2 3 3
3 4 2
4 5 2
6 7 3
2 6 8
1 4 6
*/
複製代碼
--------------------------------------------------------
先備知識:1.struct 2.vector 3.vector排序 4.二分搜 5動態規劃
程式解釋
輸入處理:
讀取客戶數量 N 和每位客戶的租用時間及收益。
排序:
根據每位客戶的結束時間 bi 對客戶進行排序。
二分搜尋:
使用 findLastNonConflict 找到對應客戶中不與當前客戶重疊的最後一個客戶。
動態規劃:
計算包含或不包含當前客戶的收益,並取兩者的最大值。
輸出結果:
最終結果即為 dp[N-1],即最大收益。
時間複雜度
排序:
歡迎光臨 種子論壇 | 高雄市資訊培育協會學員討論區 (http://istak.org.tw/seed/)
Powered by Discuz! 7.2