Lawler的算法实现辅助

时间:2010-03-18 00:44:08

标签: php algorithm scheduling

UPDATE2:

我想我现在明白了:

<?php

/*
 * @name Lawler's algorithm PHP implementation
 * @desc This algorithm calculates an optimal schedule of jobs to be
 *       processed on a single machine (in reversed order) while taking
 *       into consideration any precedence constraints.
 * @author Richard Knop
 *
 */

$jobs = array(1 => array('processingTime' => 2,
                         'dueDate'        => 3),
              2 => array('processingTime' => 3,
                         'dueDate'        => 15),
              3 => array('processingTime' => 4,
                         'dueDate'        => 9),
              4 => array('processingTime' => 3,
                         'dueDate'        => 16),
              5 => array('processingTime' => 5,
                         'dueDate'        => 12),
              6 => array('processingTime' => 7,
                         'dueDate'        => 20),
              7 => array('processingTime' => 5,
                         'dueDate'        => 27),
              8 => array('processingTime' => 6,
                         'dueDate'        => 40),
              9 => array('processingTime' => 3,
                         'dueDate'        => 10));
// precedence constrainst, i.e job 2 must be completed before job 5 etc
$successors = array(2=>5,
                    7=>9);
$n = count($jobs);
$optimalSchedule = array();

for ($i = $n; $i >= 1; $i--) {

    // jobs not required to precede any other job
    $arr = array();
    foreach ($jobs as $k => $v) {

        if (false === array_key_exists($k, $successors)) {
            $arr[] = $k;
        }

    }

    // calculate total processing time
    $totalProcessingTime = 0;
    foreach ($jobs as $k => $v) {
        if (true === array_key_exists($k, $arr)) {
            $totalProcessingTime += $v['processingTime'];
        }
    }

    // find the job that will go to the end of the optimal schedule array
    $min = null;
    $x = 0;
    $lastKey = null;
    foreach($arr as $k) {
        $x = $totalProcessingTime - $jobs[$k]['dueDate'];
        if (null === $min || $x < $min) {
            $min = $x;
            $lastKey = $k;
        }
    }

    // add the job to the optimal schedule array
    $optimalSchedule[$lastKey] = $jobs[$lastKey];
    // remove job from the jobs array
    unset($jobs[$lastKey]);
    // remove precedence constraint from the successors array if needed
    if (true === in_array($lastKey, $successors)) {
        foreach ($successors as $k => $v) {
            if ($lastKey === $v) {
                unset($successors[$k]);
            }
        }
    }

}

// reverse the optimal schedule array and preserve keys
$optimalSchedule = array_reverse($optimalSchedule, true);

// add tardiness to the array
$i = 0;
foreach ($optimalSchedule as $k => $v) {
    $optimalSchedule[$k]['tardiness'] = 0;
    $j = 0;
    foreach ($optimalSchedule as $k2 => $v2) {
        if ($j <= $i) {
            $optimalSchedule[$k]['tardiness'] += $v2['processingTime'];
        }
        $j++;
    }
    $i++;
}

echo '<pre>';
print_r($optimalSchedule);
echo '</pre>';

更新:

所以这里有一些更多来源,我发现Lawler算法的解释:

  • Source 1
  • Source 2
  • Source 3(这是一个非常好的消息来源,但预览中缺少一个关键页面+这本书在亚马逊或其他任何地方都没有,因为它只限于中国 - 如果是我本来会买的那本书已)

这是我在PHP中实现Lawler的算法(我知道......但我已经习惯了):

<?php    

$jobs = array(1, 2, 3, 4, 5, 6);
$jobsSubset = array(2, 5, 6);
$n = count($jobs);
$processingTimes = array(2, 3, 4, 3, 2, 1);
$dueDates = array(3, 15, 9, 7, 11, 20);
$optimalSchedule = array();
foreach ($jobs as $j) {
    $optimalSchedule[] = 0;
}
$dicreasedCardinality = array();
for ($i = $n; $i >= 1; $i--) {

    $x = 0;
    $max = 0;

    // loop through all jobs
    for ($j = 0; $j < $i; $j++) {

        // ignore if $j already is in the $dicreasedCardinality array
        if (false === in_array($j, $dicreasedCardinality)) {

            // if the job has no succesor in $jobsSubset
            if (false === isset($jobs[$j+1])
                || false === in_array($jobs[$j+1], $jobsSubset)) {

                // here I find an array index of a job with the maximum due date
                // amongst jobs with no sucessor in $jobsSubset
                if ($x < $dueDates[$j]) {

                    $x = $dueDates[$j];
                    $max = $j;

                }

            }

        }

    }

    // move the job at the end of $optimalSchedule
    $optimalSchedule[$i-1] = $jobs[$max];

    // decrease the cardinality of $jobs
    $dicreasedCardinality[] = $max;
}

print_r($optimalSchedule);

现在上面的内容会返回一个最佳的时间表:

Array
(
    [0] => 1
    [1] => 1
    [2] => 1
    [3] => 3
    [4] => 2
    [5] => 6
)

这对我来说似乎不对。问题可能在于我对算法的实现,因为我不确定我是否正确理解它。我使用this来源来实现它。

描述有点令人困惑。例如,我不太明白如何定义子集D(我猜它是任意的)。

有人可以帮我解决这个问题吗?我一直试图找到一些简单的算法解释来源,但我找到的所有来源都更复杂(有数学证明等)所以我坚持上面的链接。

是的,这是一个功课,如果不明显的话。

我还有几个星期的时间来解决这个问题,但是我已经花了几天时间试图弄清楚这个算法究竟是如何工作的,但没有成功,所以我认为在那段时间里我不会有任何更亮的。

1 个答案:

答案 0 :(得分:1)

根据您链接的来源,Lawler的算法将“作业i”必须在作业j之前安排“(指定为关系prec)形式的一组约束作为输入。 ,似乎没有在您的代码中显示。如果您可以按任何顺序安排作业,那么Lawler的算法专门针对更简单的Earliest Deadline First算法(一行描述:按截止日期增加的顺序对作业进行排序)。

编辑:让我更具体一点。最初集合D是所有工作。 Lawler在最后期限内多次找到i中的作业D,但受j之后Di内其他作业i的约束的限制(即,Di没有后继者,并从D移除{{1}}。这些作业的安排顺序与拆卸顺序相反。如果没有优先约束,那么这只是EDF。