APS开源源码解读: 排程工具 optaplanner II

上篇

排产,原则上也就是分配时间,分配资源;保证资源日历约束,保证工艺路线约束。我们看一下如何实现optaplanner 优化的

  • 定义一个move, 一个move可能改变了分配到的资源,也可能改变了一个资源上的顺序。改变即意味着优化的可能
  • 原本分配的资源,把后工序移到前面来。前后各算一个score. 通过socreDirector中定义的beforeVariableChanged/afterVariableChanged
  • 再更新另一个资源上的链,同时前后继续scoreDirector触发
  • 另一个shadow variable: Start time,通过StartTimeUpdatingVariableListener触发改变。这里的改变保证了不同资源上的job满足工艺路线的约束

模型

如何建模对一个优化问题非常关键

Chain Structure:

The previousAllocation can now be either a Resource (anchor) or another Allocation

Resource becomes a shadow variable automatically updated based on the chain

This creates a clean chain: Resource -> Allocation1 -> Allocation2 -> ...

Move Simplification:

No need for separate resource change moves

Single ChainChangeMove handles both resource changes and sequence changes

When moving to a different resource, just set the previous to be the resource

When moving within same resource, set previous to be another allocation

Benefits:

Simpler code structure

Fewer variables to maintain

More efficient move generation

Better encapsulation of the chaining logic

Automatic resource updates through shadow variable

过程

本文我们继续试图学习optaplanner来进行排产规划。

When a move changes an allocation's resource and/or sequence:

When OptaPlanner/timefold selects a move:

  1. Check if move is doable (isMoveDoable)
  • Verify resource compatibility
  • Check for circular dependencies
  • Validate sequence constraints
  1. Execute the move (doMove)
  • Update primary planning variables
  • Shadow variable listener triggers
  • Start/end times cascade update
  1. Score calculation
  • Check hard constraints
    • Resource conflicts
    • Job dependencies
    • Resource compatibility
  • Evaluate soft constraints
    • Makespan
    • Setup times
    • Resource preferences
  1. Accept/Reject move based on score
  • If better: keep change
  • If worse: maybe keep (based on meta-heuristic)
    The previousAllocation shadow variable is updated first

This triggers the AnchorShadowVariable (resource) to update

The StartTimeUpdatingVariableListener then recalculates timing

Chain Update Process: Anchor首先跟着genuine variable变,如果anchor变了,则其后续的链条都会检查更新

When an allocation moves to a new resource:

a. It disconnects from its old chain (previous/next allocations are relinked)

b. Inserts into new resource's chain at specified position

c. Updates shadow variables (resource, start time) for moved allocation

d. Recursively updates all downstream allocations in both chains

Impact on Route Dependencies:

The StartTimeUpdatingVariableListener ensures timing consistency

When start time changes, it propagates updates down the chain

If jobs have dependencies across resources, you'll need additional constraints

Can add precedence constraints between related jobs

例如

Before:

Resource1 -> Allocation1 -> Allocation2 -> Allocation3

Resource2 -> Allocation4 -> Allocation5
After:

Resource1 -> Allocation1 -> Allocation3

Resource2 -> Allocation4 -> Allocation2 -> Allocation5

The system will:

Update Allocation2's previousAllocation to Allocation4

Update Allocation2's resource to Resource2

Recalculate start times for Allocation2 and all subsequent allocations

Update Allocation3's previousAllocation to Allocation1

Recalculate start times in Resource1's chain

细节

当一个move发生的时候:

  1. isMoveDoable() is checked
  2. doMove() is called if move is doable
  3. Score is calculated after each change
  4. Move is accepted/rejected based on score

During doMove()

  1. beforeVariableChanged() notifies ScoreDirector
  2. Chain updates occur
  3. afterVariableChanged() notifies ScoreDirector
  4. Shadow variables update automatically
  5. Score calculator runs automatically

Score calculation triggers:

  1. After every genuine variable change
  2. After shadow variables update
  3. After variable listeners complete their updates
  4. Before move acceptance/rejection decision
java 复制代码
// Move implementation for changing position in chain
public class ChainChangeMove extends AbstractMove<JobShopSchedule> {
    private final Allocation allocation;
    private final Object newPrevious; // Can be Resource or Allocation
    
    @Override
    protected void doMoveOnGenuineVariables(ScoreDirector<JobShopSchedule> scoreDirector) {
        Object oldPrevious = allocation.getPreviousAllocation();
        Allocation nextAllocation = findNextAllocation(allocation);
        
        // Remove allocation from old chain position
        if (nextAllocation != null) {
            scoreDirector.beforeVariableChanged(nextAllocation, "previousAllocation");
            nextAllocation.setPreviousAllocation(oldPrevious);
            scoreDirector.afterVariableChanged(nextAllocation, "previousAllocation");
        }
        
        // Insert allocation at new chain position
        Allocation newNext = null;
        if (newPrevious instanceof Allocation) {
            newNext = findNextAllocation((Allocation) newPrevious);
        }
        
        // Update the moved allocation's previous
        scoreDirector.beforeVariableChanged(allocation, "previousAllocation");
        allocation.setPreviousAllocation(newPrevious);
        scoreDirector.afterVariableChanged(allocation, "previousAllocation");
        
        // Update the next allocation's previous if it exists
        if (newNext != null) {
            scoreDirector.beforeVariableChanged(newNext, "previousAllocation");
            newNext.setPreviousAllocation(allocation);
            scoreDirector.afterVariableChanged(newNext, "previousAllocation");
        }
    }
}

ScoreDirector manages this process

scoreDirector.beforeVariableChanged()

→ Make changes

→ scoreDirector.afterVariableChanged()

→ Shadow updates

→ Score calculation

→ Move acceptance

java 复制代码

For chain moves specifically:

a. Original chain score is calculated

b. Move changes are applied

c. New chain score is calculated

d. Downstream impacts are scored

e. Total impact determines move acceptance

相关推荐
希忘auto2 天前
详解Redis的常用命令
redis·1024程序员节
yaosheng_VALVE2 天前
探究全金属硬密封蝶阀的奥秘-耀圣控制
运维·eclipse·自动化·pyqt·1024程序员节
dami_king2 天前
SSH特性|组成|SSH是什么?
运维·ssh·1024程序员节
一个通信老学姐7 天前
专业125+总分400+南京理工大学818考研经验南理工电子信息与通信工程,真题,大纲,参考书。
考研·信息与通信·信号处理·1024程序员节
sheng12345678rui7 天前
mfc140.dll文件缺失的修复方法分享,全面分析mfc140.dll的几种解决方法
游戏·电脑·dll文件·dll修复工具·1024程序员节
huipeng9268 天前
第十章 类和对象(二)
java·开发语言·学习·1024程序员节
earthzhang20218 天前
《深入浅出HTTPS》读书笔记(19):密钥
开发语言·网络协议·算法·https·1024程序员节
爱吃生蚝的于勒9 天前
计算机基础 原码反码补码问题
经验分享·笔记·计算机网络·其他·1024程序员节
earthzhang20219 天前
《深入浅出HTTPS》读书笔记(20):口令和PEB算法
开发语言·网络协议·算法·https·1024程序员节
一个通信老学姐10 天前
专业140+总分410+浙江大学842信号系统与数字电路考研经验浙大电子信息与通信工程,真题,大纲,参考书。
考研·信息与通信·信号处理·1024程序员节