Skip to content

Binarization

Form

y=Bin(x)={1,x0,¬x

Discrete

Constants

m=max(x)

Additional Variables

y{0,1}: logical value of the polynomial.

Basic Formula

y=y

Mathematical Model

s.t.myxyx

That is:

y={1,x10,x=0

Continuous (Non-Precise)

Constants

m=max(x)

Additional Variables

b[0,1]: normalized value of the polynomial.

y{0,1}: logical value of the polynomial.

Basic Formula

y=y

Mathematical Model

s.t.x=mbybϵyb

That is:

y={1,xϵ0,x=0

Continuous (Precise)

Basic Formula

y=Ulp(x)

Ulp(x) can refer to Unit Linear Piecewise Function, using the sampling points {(0,0),(pϵ,0),(p,1),(max(x),1)}.

That is:

y={1,xp0,0x<p

Example

Discrete

kotlin
import kotlinx.coroutines.*
import fuookami.ospf.kotlin.utils.math.*
import fuookami.ospf.kotlin.core.frontend.variable.*
import fuookami.ospf.kotlin.core.frontend.expression.polynomial.*
import fuookami.ospf.kotlin.core.frontend.expression.symbol.linear_function.*
import fuookami.ospf.kotlin.core.frontend.inequality.*
import fuookami.ospf.kotlin.core.frontend.model.mechanism.*
import fuookami.ospf.kotlin.core.backend.plugins.scip.*

val x = UIntVar("x")
x.range.leq(UInt64.two)
val bin = BinaryzationFunction(x, name = "bin")
val solver = ScipLinearSolver()

val model1 = LinearMetaModel()
model1.add(x)
model1.add(bin)
model1.minimize(bin)
val result1 = runBlocking { solver(model1) }
assert(result1.value!!.obj eq Flt64.zero)

val model2 = LinearMetaModel()
model2.add(x)
model2.add(bin)
model2.maximize(bin)
val result2 = runBlocking { solver(model2) }
assert(result2.value!!.obj eq Flt64.one)

val model3 = LinearMetaModel()
model3.add(x)
model3.add(bin)
model3.addConstraint(x eq Flt64.zero)
model3.maximize(bin)
val result3 = runBlocking { solver(model3) }
assert(result3.value!!.obj eq Flt64.zero)

val model4 = LinearMetaModel()
model4.add(x)
model4.add(bin)
model4.addConstraint(x eq Flt64.zero)
model4.minimize(bin)
val result4 = runBlocking { solver(model4) }
assert(result4.value!!.obj eq Flt64.zero)

Continuous

kotlin
import kotlinx.coroutines.*
import fuookami.ospf.kotlin.utils.math.*
import fuookami.ospf.kotlin.core.frontend.variable.*
import fuookami.ospf.kotlin.core.frontend.expression.polynomial.*
import fuookami.ospf.kotlin.core.frontend.expression.symbol.linear_function.*
import fuookami.ospf.kotlin.core.frontend.inequality.*
import fuookami.ospf.kotlin.core.frontend.model.mechanism.*
import fuookami.ospf.kotlin.core.backend.plugins.scip.*

val x = URealVar("x")
x.range.leq(Flt64.two)
val bin = BinaryzationFunction(
    x,
    name = "bin"
)
val solver = ScipLinearSolver()

val model1 = LinearMetaModel()
model1.add(x)
model1.add(bin)
model1.minimize(bin)
val result1 = runBlocking { solver(model1) }
assert(result1.value!!.obj eq Flt64.zero)

val model2 = LinearMetaModel()
model2.add(x)
model2.add(bin)
model2.maximize(bin)
val result2 = runBlocking { solver(model2) }
assert(result2.value!!.obj eq Flt64.one)

val model3 = LinearMetaModel()
model3.add(x)
model3.add(bin)
model3.addConstraint(x eq Flt64.zero)
model3.maximize(bin)
val result3 = runBlocking { solver(model3) }
assert(result3.value!!.obj eq Flt64.zero)

val model4 = LinearMetaModel()
model4.add(x)
model4.add(bin)
model4.addConstraint(x eq Flt64(0.3))
model4.maximize(bin)
val result4 = runBlocking { solver(model4) }
assert(result4.value!!.obj eq Flt64.one)

Complete implementation reference:

Complete example reference: