It is very easy to use restarts in or-tools
They are handled via a search monitor:
There are two types of restart search monitor:
slv.LubyRestart(scale) # Luby restart sequence
slv.ConstantRestart(limit) # Constant limit
We will also need to randomize the search strategy:
CHOOSE_RANDOM_VAR
for variable selection...ASSIGN_RANDOM_VALUE
for value selectionOr-tools allows to specify easily a fragment selection strategy
We just need to implement a simple python class:
class MyRandomLns(pywrapcp.BaseLns):
def __init__(self, x, size, rand):
pywrapcp.BaseLns.__init__(self, x)
self.size_ = size # Number of vars to relax (e.g.)
self.rand_ = rand # RNG
def InitFragments(self):
pass
def NextFragment(self):
positions =
for pos in positions:
self.AppendToFragment(pos)
return True
__init__
is called when the class is builtOr-tools allows to specify easily a fragment selection strategy
We just need to implement a simple python class:
class MyRandomLns(pywrapcp.BaseLns):
def __init__(self, x, size, rand):
pywrapcp.BaseLns.__init__(self, x)
self.size_ = size # Number of vars to relax (e.g.)
self.rand_ = rand # RNG
def InitFragments(self):
pass
def NextFragment(self):
positions = # random indices of the variables
for pos in positions:
self.AppendToFragment(pos)
return True
NextFragment
specifies the variables to relax Or-tools allows to specify easily a fragment selection strategy
We just need to implement a simple python class:
class MyRandomLns(pywrapcp.BaseLns):
def __init__(self, x, size, rand):
pywrapcp.BaseLns.__init__(self, x)
self.size_ = size # Number of vars to relax (e.g.)
self.rand_ = rand # RNG
def InitFragments(self):
pass
def NextFragment(self):
positions = # random indices of the variables
for pos in positions:
self.AppendToFragment(pos)
return True
InitFragments
is called whenever an improving solution is foundThe tricky part is setting up the LNS process
The tricky part is setting up the LNS process
Solve
, yellow: otherThe tricky part is setting up the LNS process
The tricky part is setting up the LNS process
Assignment
object, stored by a special dec. builderThe tricky part is setting up the LNS process
Solve
needs a DecisionBuilder
as argument...The tricky part is setting up the LNS process
DecisionBuilder
to control the LNS processThe tricky part is setting up the LNS process
The tricky part is setting up the LNS process
The tricky part is setting up the LNS process
SolveOnce
decision builder...The tricky part is setting up the LNS process
The tricky part is setting up the LNS process
Assignment
The tricky part is setting up the LNS process
Let's apply these techniques to our shift-scheduling problem
In the start kit, you will find two templates:
shiftsched_restart.py
, to test a restart strategyshiftsched_lns.py
, to test a LNSFor the "restart" case:
Let's apply these techniques to our shift-scheduling problem
In the start kit, you will find two templates:
shiftsched_restart.py
, to test a restart strategyshiftsched_lns.py
, to test a LNSFor the "LNS" case, the tricky configuration has already been done:
SolutionsLimit
, if you want...Let's apply these techniques to our shift-scheduling problem
In the start kit, you will find two templates:
shiftsched_restart.py
, to test a restart strategyshiftsched_lns.py
, to test a LNSFor the "LNS" case, the tricky configuration has already been done:
Let's apply these techniques to our shift-scheduling problem
In the start kit, you will find two templates:
shiftsched_restart.py
, to test a restart strategyshiftsched_lns.py
, to test a LNSThere are three benchmark instances:
Instance1.json
, with optimum 607Instance2.json
, with optimum 828Instance3.json
, with optimum 1001See how close you can get
shiftsched_dfs.py
is provided