贪心算法带有期限的单机作业调度问题
时间: 2023-12-07 10:38:41 浏览: 287
贪心算法带有期限的单机作业调度问题是指在单处理器上,有n个作业需要调度,每个作业有一个期限和一个惩罚,如果在期限之前完成,则不会受到惩罚,否则会受到相应的惩罚。现在需要将这n个作业调度到单处理器上,使得惩罚最小。
解决这个问题的贪心策略是按照作业的期限从小到大排序,然后依次将作业调度到处理器上。如果当前时间已经超过了某个作业的期限,则放弃这个作业,否则将这个作业调度到处理器上。
以下是Python代码实现:
```python
def job_scheduling(jobs):
# 按照期限从小到大排序
jobs = sorted(jobs, key=lambda x: x[1])
n = len(jobs)
# 初始化时间和惩罚
time, penalty = 0, 0
for i in range(n):
# 如果当前时间加上当前作业的时间小于等于期限,则调度这个作业
if time + jobs[i][0] <= jobs[i][1]:
time += jobs[i][0]
penalty += jobs[i][2]
return penalty
```
其中,jobs是一个列表,每个元素是一个三元组,分别表示作业的时间、期限和惩罚。函数返回的是最小惩罚。
相关问题
用c语言贪心法写出下面题目 带有期限的作业排序要解决的是操作系统中单机、无资源约束且每个作业可在等量时间内完成的作业调度问题,形式化描述为: ①只能在一台机器上处理n个作业,每个作业i=1,...,n均可在单位时间内完成; ②每个作业i都有一个期限值d i >0,(d i 是整数); ③当且仅当作业i在它的截止期限前被完成时获得p i >0的效益; 问题的可行解是这n个作业的一个子集合J。J中的每个作业都能在各自的截止期限之前完成,产生一个作业效益之和∑p i 。具有最大效益值的可行解就是最优解。 要求用更快的作业排序算法(利用集合树方法)实现。 输入格式: 第一行为一个不超过200的正整数n,表示作业的个数; 接下来的n行,每行两个正整数(中间用空格隔开),表示每个作业i的截止期限d i 和按期完成产生的效益p i 。
很高兴为您解答这道题目!以下是用C语言实现带有期限的作业排序的贪心算法代码:
```c
#include <stdio.h>
#include <stdlib.h>
typedef struct {
int deadline;
int profit;
} Job;
int cmp(const void* a, const void* b) {
return ((Job*)b)->profit - ((Job*)a)->profit;
}
int main() {
int n;
scanf("%d", &n);
Job jobs[n];
for (int i = 0; i < n; i++) {
scanf("%d %d", &jobs[i].deadline, &jobs[i].profit);
}
// 按照效益从大到小排序
qsort(jobs, n, sizeof(Job), cmp);
int max_deadline = 0;
for (int i = 0; i < n; i++) {
if (jobs[i].deadline > max_deadline) {
max_deadline = jobs[i].deadline;
}
}
int schedule[max_deadline];
for (int i = 0; i < max_deadline; i++) {
schedule[i] = -1;
}
int total_profit = 0;
for (int i = 0; i < n; i++) {
for (int j = jobs[i].deadline - 1; j >= 0; j--) {
if (schedule[j] == -1) {
schedule[j] = i;
total_profit += jobs[i].profit;
break;
}
}
}
printf("最大效益值为:%d\n", total_profit);
return 0;
}
```
该算法的时间复杂度为O(n^2),其中n为作业的个数。如果使用更快的排序算法,可以将时间复杂度降为O(nlogn)。
用c语言更快速的作业排序算法(集合树)的方法实现下面问题 用c语言贪心法写出下面题目并给出详细注释 带有期限的作业排序要解决的是操作系统中单机、无资源约束且每个作业可在等量时间内完成的作业调度问题,形式化描述为: ①只能在一台机器上处理n个作业,每个作业i=1,...,n均可在单位时间内完成; ②每个作业i都有一个期限值d i >0,(d i 是整数); ③当且仅当作业i在它的截止期限前被完成时获得p i >0的效益; 问题的可行解是这n个作业的一个子集合J。J中的每个作业都能在各自的截止期限之前完成,产生一个作业效益之和∑p i 。具有最大效益值的可行解就是最优解。 要求用更快的作业排序算法(利用集合树方法)实现。 输入格式: 第一行为一个不超过200的正整数n,表示作业的个数; 接下来的n行,每行两个正整数(中间用空格隔开),表示每个作业i的截止期限d i 和按期完成产生的效益p i
以下是使用C语言集合树方法解决带有期限的作业排序问题的代码和注释:
```c
#include <stdio.h>
#include <stdlib.h>
// 作业结构体
typedef struct job {
int deadline; // 截止期限
int profit; // 收益
} Job;
// 按照收益从大到小排序
int cmp(const void* a, const void* b) {
return ((Job*)b)->profit - ((Job*)a)->profit;
}
// 集合树节点结构体
typedef struct node {
int left; // 区间左端点
int right; // 区间右端点
int maxDeadline; // 区间内的最大截止期限
} Node;
// 创建集合树
Node* buildTree(int left, int right) {
Node* node = (Node*)malloc(sizeof(Node));
node->left = left;
node->right = right;
node->maxDeadline = 0; // 初始值为0
if (left == right) {
return node;
}
int mid = (left + right) / 2;
node->left = buildTree(left, mid);
node->right = buildTree(mid + 1, right);
return node;
}
// 更新集合树节点信息
void update(Node* node, int i, int deadline) {
if (node->left == node->right) {
// 如果节点是叶子节点,更新最大截止期限
node->maxDeadline = deadline;
return;
}
int mid = (node->left + node->right) / 2;
if (i <= mid) {
// 如果i在左子树,递归更新左子树
update(node->left, i, deadline);
} else {
// 如果i在右子树,递归更新右子树
update(node->right, i, deadline);
}
// 更新当前节点的最大截止期限
node->maxDeadline = node->left->maxDeadline > node->right->maxDeadline ? node->left->maxDeadline : node->right->maxDeadline;
}
// 查询集合树节点信息
int query(Node* node, int left, int right) {
if (node->left == left && node->right == right) {
// 如果区间完全覆盖当前节点,返回最大截止期限
return node->maxDeadline;
}
int mid = (node->left + node->right) / 2;
if (right <= mid) {
// 如果区间在左子树,递归查询左子树
return query(node->left, left, right);
} else if (left > mid) {
// 如果区间在右子树,递归查询右子树
return query(node->right, left, right);
} else {
// 如果区间跨越左右子树,分别递归查询左子树和右子树并返回较大值
int maxLeft = query(node->left, left, mid);
int maxRight = query(node->right, mid + 1, right);
return maxLeft > maxRight ? maxLeft : maxRight;
}
}
int main() {
int n;
scanf("%d", &n);
// 创建作业数组
Job* jobs = (Job*)malloc(sizeof(Job) * n);
for (int i = 0; i < n; i++) {
scanf("%d %d", &jobs[i].deadline, &jobs[i].profit);
}
// 按照收益从大到小排序
qsort(jobs, n, sizeof(Job), cmp);
// 创建集合树
Node* root = buildTree(1, n);
int totalProfit = 0;
// 遍历每个作业
for (int i = 0; i < n; i++) {
int deadline = jobs[i].deadline;
int profit = jobs[i].profit;
// 查询能够完成该作业的最早时间点
int maxDeadline = query(root, 1, deadline);
if (maxDeadline < deadline) {
// 如果没有时间完成该作业,直接跳过
continue;
}
// 更新集合树的节点信息
update(root, maxDeadline, deadline);
// 累加收益
totalProfit += profit;
}
printf("最大收益为:%d\n", totalProfit);
free(jobs);
free(root);
return 0;
}
```
注释已经在代码中给出,简单说明一下算法的思路:
1. 将所有作业按照收益从大到小排序。
2. 创建一个集合树,每个节点记录区间内的最大截止期限。
3. 遍历每个作业,查询能够完成该作业的最早时间点。
4. 如果没有时间完成该作业,直接跳过;否则,更新集合树的节点信息,并累加收益。
5. 计算最大收益。
这个算法的时间复杂度为O(nlogn),比贪心算法更快。
阅读全文