使用.NET Core与Google Optimization Tools实现加工车间任务规划

前一篇文章《使用.NET Core与Google Optimization Tools实现员工排班计划Scheduling》算是一种针对内容的规划,而针对时间顺序任务规划,加工车间的工活儿是一个典型的场景。在加工车间有不同的工活儿,一般称为作业,每种作业都有多道工序,每道工序只能在特定的机器上完成。工序有不同的时长,而且是不能更改先后的。这些作业正是制造车间大规模生产线的任务,比如汽车零件制造。问题就是,工厂需要做一个最优的规划,使得作业严格按工序进行的前提下,消耗的时间最短,这样就保证了生产效率是最佳的。如果想做到最优规划,以下约束必不可少:

1. 在作业中必须要前一道工序完成才能进行下一道工序。

2. 对于一台机器,一次只能支持一个作业中的一道工序的运转。

3. 对于每道工序,一旦开始就必须完整地结束。

案例背景


以下是某个车间的作业情况,job代表作业,(m,p)代表了工序,其中m表示从0开始的机器编号,p代表了这道工序需要消耗的时长。本文假设了一个作业计划:

job 0 =  [(0, 3), (1, 2), (2, 2)]

job 1 =  [(0, 2), (2, 1), (1, 4)]

job 2 =  [(1, 4), (2, 3)]

如上所示,job 0有三道工序,第一道工序在0号机器用掉3个单位时长,第二道在1号机器用掉2个单位时长,第三道在2号机器用掉2个单位时长,以此类推,总共八道工序。

解决方案


有一种解决方案如下图,在一个时间轴上,每道工序有一个开始时间,占据一定的时长代表消耗部分,互相不会重叠,所有工序安置完毕,最长的地方就是整个作业全部完成的时长。

上图给了一个示范,一共消耗12个单位时长,当然这也不是最优的,后面通过编码我们再计算出最优的结果。

定义约束


首先我们将工序时长定义为task(i, j),表示job i的第j道工序,定义ti, j表示task(i, j)开始的时间点。有了这两种定义,按照之间的要求,于是有了如下的关系约束:

1. 连接约束,对于同一个作业,前一道工序加上消耗的时长就是后一道工序。比如,对于作业job 0来说,t0, 2表示第二道工序开始的位置,最多消耗2个单位时长之后,就是第三道工序的位置,即:t0, 2   + 2  ≤  t0, 3。

2. 非连接约束,对于不同的作业,要保证前一道工序完成后才能进行下一道工序。比如在1号的机器上有task(0, 2)和task(2, 1),它们消耗的时长分别是2和4个单位,那么就有:

 t0, 2   + 2  ≤  t2, 1     如果task(0, 2)在task(2, 1)前运行的话

或者

t2, 1   + 4  ≤  t0, 2      如果task(2, 1)在task(0, 2)前运行的话

基于这个关系,前面案例的作业计划的约束关系如图所示:

带箭头的实线表示了每个作业的工序,有连接约束的情况,而虚线表示了非连接约束的情况,实线有箭头是因为每个作业的工序是确定的,而虚线没有箭头也就说明顺序是没有确定的,这也正是我们要通过规划解决的问题。

最终求解目标


如果假定pi, j表示task(i, j)的消耗时长,那么我们要解决的全局问题就是在所有task都完成后,求一个maxi, j  ti, j +  pi, j的最小值,表示生产效率最优的结果。

代码分解


看过本文开头谈到的前一篇文章后,对于项目初始化和相同的基本概念就不再介绍了。

首先定义一些初始化用的值。

// 创建约束求解器.

var solver = new Solver("jobshop");
var machines_count = 3;
var jobs_count = 3;
var all_machines = Enumerable.Range(0, machines_count);
var all_jobs = Enumerable.Range(0, jobs_count);

再定义出所有的工序。MakeFixedDurationIntervalVar就是OR-Tools专门用来创建间隔时间的变量类型。

// 将任务拆分成对应的机器和用时的结构

// job 0 = [(0, 3), (1, 2), (2, 2)]

// job 1 = [(0, 2), (2, 1), (1, 4)]

// job 2 = [(1, 4), (2, 3)]

var machines = new int[][]

{

    new[] { 0, 1, 2 },

    new[] { 0, 2, 1 },

    new[] { 1, 2 }

};


var processing_times = new int[][]

{

    new[] { 3, 2, 2 },

    new[] {2, 1, 4 },

    new[] { 4, 3 }

};


// 计算总用时

var horizon = 0;

foreach (var i in all_jobs)

    horizon += processing_times[i].Sum();


// 创建工序变量

var all_tasks = new Dictionary<(int, int), IntervalVar>();

foreach (var i in all_jobs)

{

    foreach (var j in Enumerable.Range(0, machines[i].Length))

    {

        all_tasks[(i, j)] = solver.MakeFixedDurationIntervalVar(0,

                                                    horizon,

                                                    processing_times[i][j],

                                                    false,

                                                    $"Job_{i}_{j}");

    }

}

然后定义连接约束和非连接约束,MakeDisjunctiveConstraints专门用来创建非连接约束的,StartsAfterEnd专门用来创建连接约束。

// 创建连接的顺序变量及连接关系

var all_sequences = new SequenceVarVector();

//var all_machines_jobs = new List<IntervalVar>();

foreach (var i in all_machines)

{

    var machines_jobs = new IntervalVarVector();

    foreach (var j in all_jobs)

    {

        foreach (var k in Enumerable.Range(0, machines[j].Length))

        {

            if (machines[j][k] == i)

            {

                machines_jobs.Add(all_tasks[(j, k)]);

            }

        }

    }

    var disj = solver.MakeDisjunctiveConstraint(machines_jobs, $"machine {i}");

    all_sequences.Add(disj.SequenceVar());

    solver.Add(disj);

}


// 定义连接约束

foreach (var i in all_jobs)

{

    foreach (var j in Enumerable.Range(0, machines[i].Length - 1))

    {

        solver.Add(all_tasks[(i, j + 1)].StartsAfterEnd(all_tasks[(i, j)]));

    }

}

重点的部分,就是创建求解目标了。MakeMinimize用来求最小值,第二个参数表示每次移动的步长,直到有解为止。

// 创建求解的极值目标

var end_tasks = new IntVarVector();

foreach (var i in all_jobs)

{

    end_tasks.Add(all_tasks[(i, machines[i].Length - 1)].EndExpr().Var());

}

var obj_var = solver.MakeMax(end_tasks);

var objective_monitor = solver.MakeMinimize(obj_var.Var(), 1);


// 创建求解的对象

var sequence_phase = solver.MakePhase(all_sequences.ToArray(), Solver.SEQUENCE_DEFAULT);

var vars_phase = solver.MakePhase(new[] { obj_var.Var() }, Solver.CHOOSE_FIRST_UNBOUND, Solver.ASSIGN_MIN_VALUE);

var main_phase = solver.Compose(new[] { sequence_phase, vars_phase });


最后是显示最优解结果的部分。

// 创建最后一个解决方案

var collector = solver.MakeLastSolutionCollector();


// 添加需要关注的变量

collector.Add(all_sequences.ToArray());

collector.AddObjective(obj_var.Var());


foreach (var i in all_machines)

{

    var sequence = all_sequences[i];

    var sequence_count = sequence.Size();

    for (var j = 0; j < sequence_count; j++)

    {

        var t = sequence.Interval(j);

        collector.Add(t.StartExpr().Var());

        collector.Add(t.EndExpr().Var());

    }

}


// 显示结果

var disp_col_width = 10;

if (solver.Solve(main_phase, new SearchMonitor[] { objective_monitor, collector }))

{

    Console.WriteLine("\nOptimal Schedule Length: {0}\n", collector.ObjectiveValue(0));

    var sol_line = "";

    var sol_line_tasks = "";

    Console.WriteLine("Optimal Schedule\n");


    foreach (var i in all_machines)

    {

        var seq = all_sequences[i];

        sol_line += $"Machine {i}: ";

        sol_line_tasks += $"Machine {i}: ";

        var sequence = collector.ForwardSequence(0, seq);

        var seq_size = sequence.Count;


        foreach (var j in Enumerable.Range(0, seq_size))

        {

            var t = seq.Interval(sequence[j]);

            sol_line_tasks += t.Name().PadRight(disp_col_width, ' ');

        }


        foreach (var j in Enumerable.Range(0, seq_size))

        {

            var t = seq.Interval(sequence[j]);

            var sol_tmp = $"[{collector.Value(0, t.StartExpr().Var())},{collector.Value(0, t.EndExpr().Var())}]";

            sol_line += sol_tmp.PadRight(disp_col_width, ' ');

        }

        sol_line += "\n";

        sol_line_tasks += "\n";

    }

    Console.WriteLine(sol_line_tasks);

    Console.WriteLine("Time Intervals for Tasks\n");

    Console.WriteLine(sol_line);

}

运行后结果如下:

可以看到,这一次求得了最优解,与前面给的示范的结果不一样了,总时长上更少,是11而不是12了。对应的图解是这样:

是不是觉得很有趣,跃跃欲试了!动手做就是最好的开始。

完整代码请阅读原文获取。

相关文章:

  • Google Optimization Tools介绍

  • 使用.Net Core与Google Optimization Tools实现员工排班计划Scheduling

原文地址:http://www.cnblogs.com/BeanHsiang/p/9029177.html


.NET社区新闻,深度好文,欢迎访问公众号文章汇总 http://www.csharpkit.com

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/321302.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

P6047-丝之割【斜率优化,dp】

前言 然而丝之鸽还是没有出 正题 题目链接:https://www.luogu.com.cn/problem/P6047 题目大意 两个平行的线&#xff0c;上面连接着若干条弦&#xff0c;第iii条连接上方的xix_ixi​个下方的yiy_iyi​。 然后每次可以选择一个位置(i,j)(i,j)(i,j)&#xff0c;可以切断任何位…

洛谷P1373 小a和uim之大逃离 动态规划

题解 我们可以先简单的想一种状态&#xff0c;也就是dp[i][j][x][y][t]dp[i][j][x][y][t]dp[i][j][x][y][t]&#xff0c;这是最暴力的。 当t0t 0t0时&#xff0c;表示小a处于(i,j)(i,j)(i,j)位置&#xff0c;其中小a拥有x魔液&#xff0c;uim拥有y的魔液时候的方案总数。t1t …

【模拟】游戏(jzoj 1614)

游戏 题目大意&#xff1a; 有一个n*n的棋盘&#xff0c;有一个坐标在x,y的棋子&#xff0c; 1、2号玩家可以将他向左&#xff0c;向下&#xff0c;向左下&#xff08;45∘45^{\circ}45∘&#xff09;移动若干格&#xff0c;假如他们都是AKIOI聪明绝顶的巨佬&#xff0c;请问…

VSTS + XX云服务器构建netcore+docker持续集成交付部署

持续集成交付部署是什么意思,它给我们带来什么好处&#xff1f;先贴一张图持续集成&#xff08;Continuous Integration&#xff09;持续集成强调开发人员提交了新代码之后&#xff0c;立刻进行构建、&#xff08;单元&#xff09;测试(这个要看情况了是否需要)持续交付&#x…

codeforces E. Game with String 概率

题意 这道题目的叙述不好理解&#xff1a; 给你一个字符串ss,小a和小b都知道。现在小b要把字符串的左边一段移动到最右边,生成一个新的字符串s&#x2032;" role="presentation" style="position: relative;">s′s′&#xff0c;小a只知道s′…

P4593-[TJOI2018]教科书般的亵渎【拉格朗日差值】

正题 题目链接:https://www.luogu.com.cn/problem/P4593 题目大意 场上有若干只怪&#xff0c;最高的为nnn&#xff0c;每个怪血量不同&#xff0c;有mmm个血量不存在。 不停释放亵渎&#xff08;全场打一&#xff0c;如果有怪死亡就再次生效&#xff09;&#xff0c;每次一…

【链表】【树形DP】最大利润(jzoj 1487)

最大利润 题目大意&#xff1a; 有n个车站&#xff0c;每个车站有一定的人数&#xff0c;有n-1条线路连接着这些车站&#xff0c;相邻的车站不能同时有两个餐厅&#xff0c;当在一个车站建立餐厅时&#xff0c;会得到这个车站所有人的monny&#xff08;1个人可以得到1份利润&…

P2617-Dynamic Rankings【树套树】

正题 题目链接:https://www.luogu.com.cn/problem/P2617 题目大意 给出一个序列&#xff0c;要求支持 区间查询第kkk大单点修改 解题思路 区间查询第kkk大需要使用主席树&#xff0c;构建权值线段树的前缀和。考虑如何进行单点修改&#xff0c;在前缀和上进行单点修改就是进…

如何创建一个基于 MSBuild Task 的跨平台的 NuGet 工具包

MSBuild 的 Task 为我们扩展项目的编译过程提供了强大的扩展性&#xff0c;它使得我们可以用 C# 语言编写扩展&#xff1b;利用这种扩展性&#xff0c;我们可以为我们的项目定制一部分的编译细节。NuGet 为我们提供了一种自动导入 .props 和 .targets 的方法&#xff0c;同时还…

codeforces F.F. Teodor is not a liar! 最长不降子序列

题意 给出一堆线段。 询问者每次可以询问一个整数点&#xff0c;回答者告诉询问者这个点被多少根线段包括。 问询问者最多问多少次&#xff0c;还不能确定任意一个整数点都不可能被所有的线段包含。 题解 首先用O(n)的方法计算出来每个点被多少条线段包含。 突破点&#x…

2019.01.23【NOIP普及组】模拟赛C组总结

总结 这次的分数是1001000100300 第一题第一题一开始把题看错了&#xff0c;乱打一通&#xff0c;结果才发现自己打错了&#xff0c;把题看清后&#xff0c;就知道用栈&#xff0c;快速解决。第二题和之前做过的一道题十分相似&#xff0c;就是用前缀和来将求一个范围内的巧克…

P2149-[SDOI2009]Elaxia的路线【最短路】

正题 题目链接:https://www.luogu.com.cn/problem/P2149 题目大意 nnn个点mmm条边的一张无向图&#xff0c;给定两个起点和对应的终点。求两个最短路的最长公共距离 解题思路 首先要求是最短路&#xff0c;我们可以先跑一次第一个起点的SPFASPFASPFA&#xff0c;然后从终点开…

Platform.Uno介绍

编者语&#xff1a;Xamarin国内很多人说缺乏可用的实例&#xff0c;我在写书过程中在完善一些常用场景的例子&#xff0c;希望帮到大家。Build 2018结束一周了&#xff0c;善友问我要不要谈谈Xamarin的一些变化&#xff0c;但碍于时间有限一直没有付诸行动。想想总得写点什么给…

codeforces F.Fibonacci String Subsequences

题意 定义F(x)为F(x-1)与F(x-2)的连接&#xff08;其中F(0) ‘0’,F(1) ‘1’&#xff09;。 给出一个长度不超过100的字符串s&#xff0c;询问s在F(x)的所有子序列中出现了多少次。 题解 数量很大的计数问题&#xff0c;我们首先想到的解决方案就是dp。 我们考虑F(x) F…

【dfs】GCD与LCM(jzoj 1608)

GCD与LCM 题目大意&#xff1a; 给出a,b的最大公因数和最小公倍数&#xff0c;求出符合条件的a,b的最小差值 样例输入 6 36 样例输出 6 数据范围限制 提示 数据说明&#xff1a; 对于50%的数据&#xff0c;1<a<b<10^3。 对于100%的数据&#xff0c;1<a&…

P3889-[GDOI2014]吃【线段树】

正题 题目链接:https://www.luogu.com.cn/problem/P3889 题目大意 nnn个数的序列&#xff0c;mmm次询问&#xff0c;每次给出一个区间[l,r][l,r][l,r]&#xff0c;求在区间内和区间外各选一个数使得他们的gcdgcdgcd最大 解题思路 首先没有修改且没有要求强制在线&#xff0c;…

ASP.NET Core amp; Docker 实战经验分享

一.前言最近一直在研究和实践ASP.NET Core、Docker、持续集成。在ASP.NET Core 和 Dcoker结合下遇到了一些坑&#xff0c;在此记录和分享&#xff0c;希望对大家有一些帮助。二.中间镜像我前面写过一个 《ASP.NET Core & Docker 零基础持续集成 》的教程。里面我们通过持续…

codeforces E. Picking Strings 构造

题目链接 Picking String 题意 给出字符串S和T&#xff0c;1e5个询问&#xff0c;每次询问S的一段区间是否能转变成T的一段区间。 转变方式&#xff1a; A>BCA>BCB>ACB>ACC>ABC>ABAAAAAA可以消除 题解 我们从以上四个条件出发推导出更加精华的条件 B>…

P4169-[Violet]天使玩偶/SJY摆棋子【CDQ分治】

正题 题目链接:https://www.luogu.com.cn/problem/P4169 题目大意 nnn个点&#xff0c;然后每次操作 加一个新的点询问一个点更近的点 解题思路 定义tit_iti​表示第几个操作&#xff0c;开始就有的点tit_iti​为000 假设最近的点在左上角&#xff0c;那么有要求ti<tj,xi…

【Floyed】【匈牙利算法】导弹(jzoj 1610)

题目大意&#xff1a; 有n个城市&#xff0c;有一部分是A国的&#xff0c;有一部分是B国的&#xff08;小于A国的&#xff09;&#xff0c;A国每个城市都有一枚导弹&#xff08;只有一枚&#xff09;&#xff0c;炸毁别的城市的时间是到这个城市的距离&#xff0c;请问A国最快…