QpmtnCmax — различия между версиями
(→Описание алгоритма) |
(переписан псевдокод) |
||
Строка 2: | Строка 2: | ||
{{Задача | {{Задача | ||
|definition= | |definition= | ||
− | Дано | + | Дано <tex>m</tex> станков с разной скоростью выполнения работ, работающих параллельно, и <tex>n</tex> работ. Работа может быть прервана в любой момент и продолжена позже на любой машине. Необходимо минимизировать время выполнения всех работ. |
}} | }} | ||
==Алгоритм построения расписания== | ==Алгоритм построения расписания== | ||
===Описание алгоритма=== | ===Описание алгоритма=== | ||
− | + | Перед выполнением алгоритма, упорядочим все работы по убыванию их времени выполнения:<tex> p_1 \geqslant p_2 \geqslant p_3 \ldots \geqslant p_n</tex>, а все машины в порядке убывания скоростей: <tex> s_1 \geqslant s_2 \geqslant s_3 \ldots \geqslant s_m</tex>. Введем следующие обозначения: | |
− | <tex> P_i = p_1 + \ldots + p_i</tex> | + | *<tex>P_i = p_1 + \ldots + p_i</tex>, <tex>i = 1 \ldots n</tex> {{---}} сумма первых <tex>i</tex> работ |
+ | *<tex>S_j = s_1 + \ldots + s_j</tex>, <tex>j = 1 \ldots m</tex> {{---}} сумма первых <tex>j</tex> станков | ||
− | + | <tex> p_i </tex> {{---}} время выполнения <tex>i</tex>-ой работы, <tex> s_j</tex> {{---}} скорость работы <tex> j </tex>-oй машины. | |
− | |||
− | |||
Необходимое условие для выполнения всех работ в интервале <tex>[0 \ldots T]</tex>: | Необходимое условие для выполнения всех работ в интервале <tex>[0 \ldots T]</tex>: | ||
− | <tex> P_n = p_1 + \ldots + p_n \leqslant s_1T + \ldots + s_mT = S_mT</tex> или <tex>P_n | + | <tex> P_n = p_1 + \ldots + p_n \leqslant s_1T + \ldots + s_mT = S_mT</tex> или <tex>\dfrac{P_n}{S_m} \leqslant T</tex> |
− | Кроме того, должно выполняться условие <tex>P_j | + | Кроме того, должно выполняться условие <tex>\dfrac{P_j}{S_j} \leqslant T</tex> для всех <tex> j = 1 \ldots m - 1 </tex>, так как это нижняя оценка времени выполнения работ <tex> J_1 \ldots J_j</tex>. Исходя из этого получаем нижнюю границу <tex>C_{max}</tex> : |
<tex>C_{max} = \max | <tex>C_{max} = \max | ||
Строка 31: | Строка 30: | ||
===Псевдокод=== | ===Псевдокод=== | ||
− | Функция <tex>\mathrm{level}</tex> | + | Функция <tex>\mathrm{level}</tex> принимает на вход два массива — массив с объемами работ и массив скоростей обработки станков, и возвращает вектор четвёрок, где первый элемент является номером станка, второй — номером работы, а два оставшихся время начала и окончания обработки этой работы на этом станке. |
− | '''function''' level( | + | '''function''' level(p : '''int[]''', s : '''int[]''', n : '''int''', m : '''int''') : '''vector<int, int, int, int>''' |
− | + | '''vector<int, int, int, int>''' ans | |
− | + | '''int''' t = 0 | |
− | + | '''int''' k = n | |
− | + | sort(p) <font color=green> // сортируем время обработки работ по убыванию </font> | |
− | + | sort(s) <font color=green> // сортируем станки по убыванию скоростей </font> | |
− | + | '''while''' k > 0 | |
− | + | '''int[]''' to = assign(p, k, m) <font color=green> // получаем распределение работ по станкам </font> | |
− | + | Найдем минимальное dt1 отличное от нуля такое, что (p[i] - s[to[i]] * dt1) = 0 | |
− | + | Найдем минимальное dt2 отличное от нуля такое, что p[i] > p[j] и (p[i] - s[to[i]] * dt2 = p[j] - s[to[j]] * dt2) | |
− | + | '''int''' dt = min(dt1, dt2) | |
− | + | '''for''' j = 0 '''to''' n - 1 | |
− | + | '''if''' p[j] > 0 | |
− | + | '''if''' to[j] <tex> \neq </tex> -1 <font color=green> // рассматриваем работы которые обрабатываются в данном распределении</font> | |
− | + | ans.push(to[j], j, t, t + dt) | |
− | + | p[j] -= s[to[i]] * dt | |
− | + | '''if''' p[j] == 0 | |
− | + | k-- | |
− | + | t += dt <font color=green> // поиск следующего момента времени, в который нужно будет перераспределить машины/работы </font> | |
− | + | '''return''' ans | |
− | |||
− | |||
+ | Функция <tex>\mathrm{assign}</tex> принимает на вход массив с объемами работ и возвращает массив с распределением работ. | ||
+ | '''function''' assign (p : '''int[]''', n : '''int''', m : '''int''') : '''int[]''' | ||
+ | '''int[]''' to <font color=green> // j работа обрабатывается на to[j] станке </font> | ||
+ | fill(to, -1) | ||
+ | '''int''' i = 0 | ||
+ | '''while''' i < m '''and''' i < n | ||
+ | Находим первый j такой что p[j] максимальный | ||
+ | m[j] = i++ | ||
+ | '''return''' to | ||
===Асимптотика=== | ===Асимптотика=== | ||
Строка 74: | Строка 80: | ||
Будем считать, что в начале алгоритма все работы упорядочены, как было сказано ранее: <tex> p_1(0) \geqslant p_2(0) \geqslant \ldots \geqslant p_n(0) </tex>. Это утверждение не меняется на протяжении всего выполнения алгоритма, для любого момента времени. Получаем: <tex> p_1(t) \geqslant p_2(t) \geqslant \ldots \geqslant p_n(t) </tex>. Докажем что алгоритм составляет расписание в соответствии с этим свойством. Чтобы доказать этот факт, будем считать что в любой момент времени <tex>T</tex> нет простоев машин, когда есть хотя бы одна невыполненная работа. Получаем: | Будем считать, что в начале алгоритма все работы упорядочены, как было сказано ранее: <tex> p_1(0) \geqslant p_2(0) \geqslant \ldots \geqslant p_n(0) </tex>. Это утверждение не меняется на протяжении всего выполнения алгоритма, для любого момента времени. Получаем: <tex> p_1(t) \geqslant p_2(t) \geqslant \ldots \geqslant p_n(t) </tex>. Докажем что алгоритм составляет расписание в соответствии с этим свойством. Чтобы доказать этот факт, будем считать что в любой момент времени <tex>T</tex> нет простоев машин, когда есть хотя бы одна невыполненная работа. Получаем: | ||
− | <tex> T(s_1 + \ldots + s_m) = p_1 + p_2 + \ldots + p_n </tex> или <tex> T = {P_n | + | <tex> T(s_1 + \ldots + s_m) = p_1 + p_2 + \ldots + p_n </tex> или <tex> T = \dfrac{P_n}{S_m} </tex> |
Таким образом необходимая оценка достигается нашим алгоритмом. | Таким образом необходимая оценка достигается нашим алгоритмом. |
Версия 01:39, 8 июня 2016
Задача: |
Дано | станков с разной скоростью выполнения работ, работающих параллельно, и работ. Работа может быть прервана в любой момент и продолжена позже на любой машине. Необходимо минимизировать время выполнения всех работ.
Содержание
Алгоритм построения расписания
Описание алгоритма
Перед выполнением алгоритма, упорядочим все работы по убыванию их времени выполнения:
, а все машины в порядке убывания скоростей: . Введем следующие обозначения:- , — сумма первых работ
- , — сумма первых станков
— время выполнения -ой работы, — скорость работы -oй машины.
Необходимое условие для выполнения всех работ в интервале
:или
Кроме того, должно выполняться условие
для всех , так как это нижняя оценка времени выполнения работ . Исходя из этого получаем нижнюю границу :
Перейдем к описанию алгоритма. Будем назвать
-ом работы невыполненную часть работы в момент времениДалее построим расписание, которое достигает нашей оценки
, с помощью -алгоритма.Псевдокод
Функция
принимает на вход два массива — массив с объемами работ и массив скоростей обработки станков, и возвращает вектор четвёрок, где первый элемент является номером станка, второй — номером работы, а два оставшихся время начала и окончания обработки этой работы на этом станке. function level(p : int[], s : int[], n : int, m : int) : vector<int, int, int, int>
vector<int, int, int, int> ans
int t = 0
int k = n
sort(p) // сортируем время обработки работ по убыванию
sort(s) // сортируем станки по убыванию скоростей
while k > 0
int[] to = assign(p, k, m) // получаем распределение работ по станкам
Найдем минимальное dt1 отличное от нуля такое, что (p[i] - s[to[i]] * dt1) = 0
Найдем минимальное dt2 отличное от нуля такое, что p[i] > p[j] и (p[i] - s[to[i]] * dt2 = p[j] - s[to[j]] * dt2)
int dt = min(dt1, dt2)
for j = 0 to n - 1
if p[j] > 0
if to[j]
-1 // рассматриваем работы которые обрабатываются в данном распределении
ans.push(to[j], j, t, t + dt)
p[j] -= s[to[i]] * dt
if p[j] == 0
k--
t += dt // поиск следующего момента времени, в который нужно будет перераспределить машины/работы
return ans
Функция
принимает на вход массив с объемами работ и возвращает массив с распределением работ.function assign (p : int[], n : int, m : int) : int[] int[] to // j работа обрабатывается на to[j] станке fill(to, -1) int i = 0 while i < m and i < n Находим первый j такой что p[j] максимальный m[j] = i++ return to
Асимптотика
-алгоритм вызывает функцию в самом худшем случае раз. Функция выполняется за . Итоговое время работы .
Доказательство корректности алгоритма
Теорема: |
Расписание, построенное данным алгоритмом, является корректным и оптимальным. |
Доказательство: |
Так как нижняя граница :
то достаточно показать, что составленное расписание достигает этой оценки. Будем считать, что в начале алгоритма все работы упорядочены, как было сказано ранее: . Это утверждение не меняется на протяжении всего выполнения алгоритма, для любого момента времени. Получаем: . Докажем что алгоритм составляет расписание в соответствии с этим свойством. Чтобы доказать этот факт, будем считать что в любой момент времени нет простоев машин, когда есть хотя бы одна невыполненная работа. Получаем:или Таким образом необходимая оценка достигается нашим алгоритмом. Допустим хотя бы одна машина простаивает, в момент когда есть невыполненные работы, получим следующее неравенство для времен окончания работ (обозначим далее как ) на станках , пронумерованных по убыванию скоростей:
Докажем написанное выше неравенство: Предположим, что Пусть для некоторого . Тогда последней работы, выполнявшейся на станке в момент времени (где достаточно мал) меньше, чем последней работы на станке . Пришли к противоречию, так как при распределении, работы с наибольшим выставлялись на самые быстрые станки. = ,где . Чтобы работы завершились в момент времени , необходимо начать их в момент времени 0, поскольку если это не выполняется, то у нас найдется работа , которая начинается позже и заканчивается в . Это означает, что в момент времени начинаются как минимум работ. Пусть первые работ стартовали вместе на всех машинах. Мы получаем , из чего следует, что для любого , удовлетворяющего условию . Таким образом, до момента времени нет простаивающих машин. Пришли к противоречию. Получаем . |
Пример
Пусть у нас есть
работ и станка. Покажем работу алгоритма для данного случая.В начальный момент времени начинаем обрабатывать работы с наибольшим временем выполнения
, и на станках , и соответственно. В момент времени -ой работы и -ой работы совпадает. С этого момента начинаем обрабатывать работы и синхронно на станках: и . В момент времени работа опускается до уровня работы .Работы и выполняем одновременно на одном станке . В момент времени начинаем выполнять первые четыре работы на всех станках одновременно, далее просто добавятся работы и , и все работы закончатся одновременно.См. также
Источники информации
- Peter Brucker. «Scheduling Algorithms» — «Springer», 2006 г. — 124 — 129 стр. — ISBN 978-3-540-69515-8