Scippy

SCIP

Solving Constraint Integer Programs

heur_shifting.c
Go to the documentation of this file.
1/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2/* */
3/* This file is part of the program and library */
4/* SCIP --- Solving Constraint Integer Programs */
5/* */
6/* Copyright (c) 2002-2025 Zuse Institute Berlin (ZIB) */
7/* */
8/* Licensed under the Apache License, Version 2.0 (the "License"); */
9/* you may not use this file except in compliance with the License. */
10/* You may obtain a copy of the License at */
11/* */
12/* http://www.apache.org/licenses/LICENSE-2.0 */
13/* */
14/* Unless required by applicable law or agreed to in writing, software */
15/* distributed under the License is distributed on an "AS IS" BASIS, */
16/* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. */
17/* See the License for the specific language governing permissions and */
18/* limitations under the License. */
19/* */
20/* You should have received a copy of the Apache-2.0 license */
21/* along with SCIP; see the file LICENSE. If not visit scipopt.org. */
22/* */
23/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
24
25/**@file heur_shifting.c
26 * @ingroup DEFPLUGINS_HEUR
27 * @brief LP rounding heuristic that tries to recover from intermediate infeasibilities and shifts continuous variables
28 * @author Tobias Achterberg
29 */
30
31/*---+----1----+----2----+----3----+----4----+----5----+----6----+----7----+----8----+----9----+----0----+----1----+----2*/
32
34#include "scip/heur_shifting.h"
35#include "scip/pub_heur.h"
36#include "scip/pub_lp.h"
37#include "scip/pub_message.h"
38#include "scip/pub_misc.h"
39#include "scip/pub_var.h"
40#include "scip/scip_branch.h"
41#include "scip/scip_heur.h"
42#include "scip/scip_lp.h"
43#include "scip/scip_mem.h"
44#include "scip/scip_message.h"
45#include "scip/scip_numerics.h"
46#include "scip/scip_prob.h"
48#include "scip/scip_sol.h"
50#include <string.h>
51
52#define HEUR_NAME "shifting"
53#define HEUR_DESC "LP rounding heuristic with infeasibility recovering also using continuous variables"
54#define HEUR_DISPCHAR SCIP_HEURDISPCHAR_ROUNDING
55#define HEUR_PRIORITY -5000
56#define HEUR_FREQ 5
57#define HEUR_FREQOFS 0
58#define HEUR_MAXDEPTH -1
59#define HEUR_TIMING SCIP_HEURTIMING_DURINGLPLOOP
60#define HEUR_USESSUBSCIP FALSE /**< does the heuristic use a secondary SCIP instance? */
61
62#define MAXSHIFTINGS 50 /**< maximal number of non improving shiftings */
63#define WEIGHTFACTOR 1.1
64#define DEFAULT_RANDSEED 31 /**< initial random seed */
65
66
67/* locally defined heuristic data */
68struct SCIP_HeurData
69{
70 SCIP_SOL* sol; /**< working solution */
71 SCIP_RANDNUMGEN* randnumgen; /**< random number generator */
72 SCIP_Longint lastlp; /**< last LP number where the heuristic was applied */
73};
74
75
76/*
77 * local methods
78 */
79
80/** update row violation arrays after a row's activity value changed */
81static
83 SCIP* scip, /**< SCIP data structure */
84 SCIP_ROW* row, /**< LP row */
85 SCIP_ROW** violrows, /**< array with currently violated rows */
86 int* violrowpos, /**< position of LP rows in violrows array */
87 int* nviolrows, /**< pointer to the number of currently violated rows */
88 int* nviolfracrows, /**< pointer to the number of violated rows with fractional candidates */
89 int* nfracsinrow, /**< array with number of fractional variables for every row */
90 SCIP_Real oldactivity, /**< old activity value of LP row */
91 SCIP_Real newactivity /**< new activity value of LP row */
92 )
93{
94 SCIP_Real lhs;
95 SCIP_Real rhs;
96 SCIP_Bool oldviol;
97 SCIP_Bool newviol;
98
99 assert(violrows != NULL);
100 assert(violrowpos != NULL);
101 assert(nviolrows != NULL);
102
103 lhs = SCIProwGetLhs(row);
104 rhs = SCIProwGetRhs(row);
105
106 /* SCIPisFeasLT cannot handle comparing different infinities. To prevent this, we make a case distinction. */
107 if( !(SCIPisInfinity(scip, oldactivity) || SCIPisInfinity(scip, -oldactivity)) )
108 {
109 oldviol = (SCIPisFeasLT(scip, oldactivity, lhs) || SCIPisFeasGT(scip, oldactivity, rhs));
110 }
111 else
112 {
113 oldviol = (SCIPisInfinity(scip, -oldactivity) && !SCIPisInfinity(scip, -lhs)) ||
114 (SCIPisInfinity(scip, oldactivity) && !SCIPisInfinity(scip, rhs));
115 }
116
117 /* SCIPisFeasLT cannot handle comparing different infinities. To prevent this, we make a case distinction. */
118 if( !(SCIPisInfinity(scip, newactivity) || SCIPisInfinity(scip, -newactivity)) )
119 {
120 newviol = (SCIPisFeasLT(scip, newactivity, lhs) || SCIPisFeasGT(scip, newactivity, rhs));
121 }
122 else
123 {
124 newviol = (SCIPisInfinity(scip, -newactivity) && !SCIPisInfinity(scip, -lhs)) ||
125 (SCIPisInfinity(scip, newactivity) && !SCIPisInfinity(scip, rhs));
126 }
127
128 if( oldviol != newviol )
129 {
130 int rowpos;
131 int violpos;
132
133 rowpos = SCIProwGetLPPos(row);
134 assert(rowpos >= 0);
135
136 if( oldviol )
137 {
138 /* the row violation was repaired: remove row from violrows array, decrease violation count */
139 violpos = violrowpos[rowpos];
140 assert(0 <= violpos && violpos < *nviolrows);
141 assert(violrows[violpos] == row);
142 violrowpos[rowpos] = -1;
143
144 /* first, move the row to the end of the subset of violated rows with fractional variables */
145 if( nfracsinrow[rowpos] > 0 )
146 {
147 assert(violpos < *nviolfracrows);
148
149 /* replace with last violated row containing fractional variables */
150 if( violpos != *nviolfracrows - 1 )
151 {
152 violrows[violpos] = violrows[*nviolfracrows - 1];
153 violrowpos[SCIProwGetLPPos(violrows[violpos])] = violpos;
154 violpos = *nviolfracrows - 1;
155 }
156 (*nviolfracrows)--;
157 }
158
159 assert(violpos >= *nviolfracrows);
160
161 /* swap row at the end of the violated array to the position of this row and decrease the counter */
162 if( violpos != *nviolrows - 1 )
163 {
164 violrows[violpos] = violrows[*nviolrows - 1];
165 violrowpos[SCIProwGetLPPos(violrows[violpos])] = violpos;
166 }
167 (*nviolrows)--;
168 }
169 else
170 {
171 /* the row is now violated: add row to violrows array, increase violation count */
172 assert(violrowpos[rowpos] == -1);
173 violrows[*nviolrows] = row;
174 violrowpos[rowpos] = *nviolrows;
175 (*nviolrows)++;
176
177 /* if the row contains fractional variables, swap with the last violated row that has no fractional variables
178 * at position *nviolfracrows
179 */
180 if( nfracsinrow[rowpos] > 0 )
181 {
182 if( *nviolfracrows < *nviolrows - 1 )
183 {
184 assert(nfracsinrow[SCIProwGetLPPos(violrows[*nviolfracrows])] == 0);
185
186 violrows[*nviolrows - 1] = violrows[*nviolfracrows];
187 violrowpos[SCIProwGetLPPos(violrows[*nviolrows - 1])] = *nviolrows - 1;
188
189 violrows[*nviolfracrows] = row;
190 violrowpos[rowpos] = *nviolfracrows;
191 }
192 (*nviolfracrows)++;
193 }
194 }
195 }
196}
197
198/** update row activities after a variable's solution value changed */
199static
201 SCIP* scip, /**< SCIP data structure */
202 SCIP_Real* activities, /**< LP row activities */
203 SCIP_ROW** violrows, /**< array with currently violated rows */
204 int* violrowpos, /**< position of LP rows in violrows array */
205 int* nviolrows, /**< pointer to the number of currently violated rows */
206 int* nviolfracrows, /**< pointer to the number of violated rows with fractional candidates */
207 int* nfracsinrow, /**< array with number of fractional variables for every row */
208 int nlprows, /**< number of rows in current LP */
209 SCIP_VAR* var, /**< variable that has been changed */
210 SCIP_Real oldsolval, /**< old solution value of variable */
211 SCIP_Real newsolval /**< new solution value of variable */
212 )
213{
214 SCIP_COL* col;
215 SCIP_ROW** colrows;
216 SCIP_Real* colvals;
217 SCIP_Real delta;
218 int ncolrows;
219 int r;
220
221 assert(activities != NULL);
222 assert(nviolrows != NULL);
223 assert(0 <= *nviolrows && *nviolrows <= nlprows);
224
225 delta = newsolval - oldsolval;
226 col = SCIPvarGetCol(var);
227 colrows = SCIPcolGetRows(col);
228 colvals = SCIPcolGetVals(col);
229 ncolrows = SCIPcolGetNLPNonz(col);
230 assert(ncolrows == 0 || (colrows != NULL && colvals != NULL));
231
232 for( r = 0; r < ncolrows; ++r )
233 {
234 SCIP_ROW* row;
235 int rowpos;
236
237 row = colrows[r];
238 rowpos = SCIProwGetLPPos(row);
239 assert(-1 <= rowpos && rowpos < nlprows);
240
241 if( rowpos >= 0 && !SCIProwIsLocal(row) )
242 {
243 SCIP_Real oldactivity;
244 SCIP_Real newactivity;
245
246 assert(SCIProwIsInLP(row));
247
248 /* update row activity */
249 oldactivity = activities[rowpos];
250 if( !SCIPisInfinity(scip, -oldactivity) && !SCIPisInfinity(scip, oldactivity) )
251 {
252 newactivity = oldactivity + delta * colvals[r];
253 if( SCIPisInfinity(scip, newactivity) )
254 newactivity = SCIPinfinity(scip);
255 else if( SCIPisInfinity(scip, -newactivity) )
256 newactivity = -SCIPinfinity(scip);
257 activities[rowpos] = newactivity;
258
259 /* update row violation arrays */
260 updateViolations(scip, row, violrows, violrowpos, nviolrows, nviolfracrows, nfracsinrow, oldactivity, newactivity);
261 }
262 }
263 }
264
265 return SCIP_OKAY;
266}
267
268/** returns a variable, that pushes activity of the row in the given direction with minimal negative impact on other rows;
269 * if variables have equal impact, chooses the one with best objective value improvement in corresponding direction;
270 * prefer fractional integers over other variables in order to become integral during the process;
271 * shifting in a direction is forbidden, if this forces the objective value over the upper bound, or if the variable
272 * was already shifted in the opposite direction
273 */
274static
276 SCIP* scip, /**< SCIP data structure */
277 SCIP_SOL* sol, /**< primal solution */
278 SCIP_ROW* row, /**< LP row */
279 SCIP_Real rowactivity, /**< activity of LP row */
280 int direction, /**< should the activity be increased (+1) or decreased (-1)? */
281 SCIP_Real* nincreases, /**< array with weighted number of increasings per variables */
282 SCIP_Real* ndecreases, /**< array with weighted number of decreasings per variables */
283 SCIP_Real increaseweight, /**< current weight of increase/decrease updates */
284 SCIP_VAR** shiftvar, /**< pointer to store the shifting variable, returns NULL if impossible */
285 SCIP_Real* oldsolval, /**< pointer to store old solution value of shifting variable */
286 SCIP_Real* newsolval /**< pointer to store new (shifted) solution value of shifting variable */
287 )
288{
289 SCIP_COL** rowcols;
290 SCIP_Real* rowvals;
291 int nrowcols;
292 SCIP_Real activitydelta;
293 SCIP_Real bestshiftscore;
294 SCIP_Real bestdeltaobj;
295 int c;
296
297 assert(direction == +1 || direction == -1);
298 assert(nincreases != NULL);
299 assert(ndecreases != NULL);
300 assert(shiftvar != NULL);
301 assert(oldsolval != NULL);
302 assert(newsolval != NULL);
303
304 /* get row entries */
305 rowcols = SCIProwGetCols(row);
306 rowvals = SCIProwGetVals(row);
307 nrowcols = SCIProwGetNLPNonz(row);
308
309 /* calculate how much the activity must be shifted in order to become feasible */
310 activitydelta = (direction == +1 ? SCIProwGetLhs(row) - rowactivity : SCIProwGetRhs(row) - rowactivity);
311 assert((direction == +1 && SCIPisPositive(scip, activitydelta))
312 || (direction == -1 && SCIPisNegative(scip, activitydelta)));
313
314 /* select shifting variable */
315 bestshiftscore = SCIP_REAL_MAX;
316 bestdeltaobj = SCIPinfinity(scip);
317 *shiftvar = NULL;
318 *newsolval = 0.0;
319 *oldsolval = 0.0;
320 for( c = 0; c < nrowcols; ++c )
321 {
322 SCIP_COL* col;
323 SCIP_VAR* var;
324 SCIP_Real val;
325 SCIP_Real solval;
326 SCIP_Real shiftval;
327 SCIP_Real shiftscore;
328 SCIP_Bool isinteger;
329 SCIP_Bool isfrac;
330 SCIP_Bool increase;
331
332 col = rowcols[c];
333 var = SCIPcolGetVar(col);
334 val = rowvals[c];
335 assert(!SCIPisZero(scip, val));
336 solval = SCIPgetSolVal(scip, sol, var);
337
338 isinteger = SCIPvarGetType(var) != SCIP_VARTYPE_CONTINUOUS;
339 isfrac = (isinteger && !SCIPisFeasIntegral(scip, solval));
340 increase = (direction * val > 0.0);
341
342 /* calculate the score of the shifting (prefer smaller values) */
343 if( isfrac )
344 shiftscore = increase ? -1.0 / (SCIPvarGetNLocksUpType(var, SCIP_LOCKTYPE_MODEL) + 1.0) :
346 else
347 {
348 int probindex;
349 probindex = SCIPvarGetProbindex(var);
350
351 if( increase )
352 shiftscore = ndecreases[probindex]/increaseweight;
353 else
354 shiftscore = nincreases[probindex]/increaseweight;
355 if( isinteger )
356 shiftscore += 1.0;
357 }
358
359 if( shiftscore <= bestshiftscore )
360 {
361 SCIP_Real deltaobj;
362
363 if( !increase )
364 {
365 /* shifting down */
366 assert(direction * val < 0.0);
367 if( isfrac )
368 shiftval = SCIPfeasFloor(scip, solval);
369 else
370 {
371 SCIP_Real lb;
372
373 assert(activitydelta/val < 0.0);
374 shiftval = solval + activitydelta/val;
375 assert(shiftval <= solval); /* may be equal due to numerical digit erasement in the subtraction */
376 if( SCIPvarIsIntegral(var) )
377 shiftval = SCIPfeasFloor(scip, shiftval);
378 lb = SCIPvarGetLbGlobal(var);
379 shiftval = MAX(shiftval, lb);
380 }
381 }
382 else
383 {
384 /* shifting up */
385 assert(direction * val > 0.0);
386 if( isfrac )
387 shiftval = SCIPfeasCeil(scip, solval);
388 else
389 {
390 SCIP_Real ub;
391
392 assert(activitydelta/val > 0.0);
393 shiftval = solval + activitydelta/val;
394 assert(shiftval >= solval); /* may be equal due to numerical digit erasement in the subtraction */
395 if( SCIPvarIsIntegral(var) )
396 shiftval = SCIPfeasCeil(scip, shiftval);
397 ub = SCIPvarGetUbGlobal(var);
398 shiftval = MIN(shiftval, ub);
399 }
400 }
401
402 if( (SCIPisInfinity(scip, shiftval) && SCIPisInfinity(scip, SCIPvarGetUbGlobal(var)))
403 || (SCIPisInfinity(scip, -shiftval) && SCIPisInfinity(scip, -SCIPvarGetLbGlobal(var)))
404 || SCIPisEQ(scip, shiftval, solval) )
405 continue;
406
407 deltaobj = SCIPvarGetObj(var) * (shiftval - solval);
408 if( shiftscore < bestshiftscore || deltaobj < bestdeltaobj )
409 {
410 bestshiftscore = shiftscore;
411 bestdeltaobj = deltaobj;
412 *shiftvar = var;
413 *oldsolval = solval;
414 *newsolval = shiftval;
415 }
416 }
417 }
418
419 return SCIP_OKAY;
420}
421
422/** returns a fractional variable, that has most impact on rows in opposite direction, i.e. that is most crucial to
423 * fix in the other direction;
424 * if variables have equal impact, chooses the one with best objective value improvement in corresponding direction;
425 * shifting in a direction is forbidden, if this forces the objective value over the upper bound
426 */
427static
429 SCIP* scip, /**< SCIP data structure */
430 SCIP_SOL* sol, /**< primal solution */
431 SCIP_Real minobj, /**< minimal objective value possible after shifting remaining fractional vars */
432 SCIP_VAR** lpcands, /**< fractional variables in LP */
433 int nlpcands, /**< number of fractional variables in LP */
434 SCIP_VAR** shiftvar, /**< pointer to store the shifting variable, returns NULL if impossible */
435 SCIP_Real* oldsolval, /**< old (fractional) solution value of shifting variable */
436 SCIP_Real* newsolval /**< new (shifted) solution value of shifting variable */
437 )
438{
439 SCIP_VAR* var;
440 SCIP_Real solval;
441 SCIP_Real shiftval;
442 SCIP_Real obj;
443 SCIP_Real deltaobj;
444 SCIP_Real bestdeltaobj;
445 int maxnlocks;
446 int nlocks;
447 int v;
448
449 assert(shiftvar != NULL);
450 assert(oldsolval != NULL);
451 assert(newsolval != NULL);
452
453 /* select shifting variable */
454 maxnlocks = -1;
455 bestdeltaobj = SCIPinfinity(scip);
456 *shiftvar = NULL;
457 for( v = 0; v < nlpcands; ++v )
458 {
459 var = lpcands[v];
461
462 solval = SCIPgetSolVal(scip, sol, var);
463 if( !SCIPisFeasIntegral(scip, solval) )
464 {
465 obj = SCIPvarGetObj(var);
466
467 /* shifting down */
469 if( nlocks >= maxnlocks )
470 {
471 shiftval = SCIPfeasFloor(scip, solval);
472 deltaobj = obj * (shiftval - solval);
473 if( (nlocks > maxnlocks || deltaobj < bestdeltaobj) && minobj - obj < SCIPgetCutoffbound(scip) )
474 {
475 maxnlocks = nlocks;
476 bestdeltaobj = deltaobj;
477 *shiftvar = var;
478 *oldsolval = solval;
479 *newsolval = shiftval;
480 }
481 }
482
483 /* shifting up */
485 if( nlocks >= maxnlocks )
486 {
487 shiftval = SCIPfeasCeil(scip, solval);
488 deltaobj = obj * (shiftval - solval);
489 if( (nlocks > maxnlocks || deltaobj < bestdeltaobj) && minobj + obj < SCIPgetCutoffbound(scip) )
490 {
491 maxnlocks = nlocks;
492 bestdeltaobj = deltaobj;
493 *shiftvar = var;
494 *oldsolval = solval;
495 *newsolval = shiftval;
496 }
497 }
498 }
499 }
500
501 return SCIP_OKAY;
502}
503
504/** adds a given value to the fractionality counters of the rows in which the given variable appears */
505static
507 int* nfracsinrow, /**< array to store number of fractional variables per row */
508 SCIP_ROW** violrows, /**< array with currently violated rows */
509 int* violrowpos, /**< position of LP rows in violrows array */
510 int* nviolfracrows, /**< pointer to store the number of violated rows with fractional variables */
511 int nviolrows, /**< the number of currently violated rows (stays unchanged in this method) */
512 int nlprows, /**< number of rows in LP */
513 SCIP_VAR* var, /**< variable for which the counting should be updated */
514 int incval /**< value that should be added to the corresponding array entries */
515 )
516{
517 SCIP_COL* col;
518 SCIP_ROW** rows;
519 int nrows;
520 int r;
521
522 assert(incval != 0);
523 assert(nviolrows >= *nviolfracrows);
524 col = SCIPvarGetCol(var);
525 rows = SCIPcolGetRows(col);
526 nrows = SCIPcolGetNLPNonz(col);
527 for( r = 0; r < nrows; ++r )
528 {
529 int rowlppos;
530 int theviolrowpos;
531
532 rowlppos = SCIProwGetLPPos(rows[r]);
533 assert(0 <= rowlppos && rowlppos < nlprows);
534 assert(!SCIProwIsLocal(rows[r]) || violrowpos[rowlppos] == -1);
535
536 /* skip local rows */
537 if( SCIProwIsLocal(rows[r]) )
538 continue;
539
540 nfracsinrow[rowlppos] += incval;
541 assert(nfracsinrow[rowlppos] >= 0);
542 theviolrowpos = violrowpos[rowlppos];
543
544 /* swap positions in violrows array if fractionality has changed to 0 */
545 if( theviolrowpos >= 0 )
546 {
547 /* first case: the number of fractional variables has become zero: swap row in violrows array to the
548 * second part, containing only violated rows without fractional variables
549 */
550 if( nfracsinrow[rowlppos] == 0 )
551 {
552 assert(theviolrowpos <= *nviolfracrows - 1);
553
554 /* nothing to do if row is already at the end of the first part, otherwise, swap it to the last position
555 * and decrease the counter */
556 if( theviolrowpos < *nviolfracrows - 1 )
557 {
558 violrows[theviolrowpos] = violrows[*nviolfracrows - 1];
559 violrows[*nviolfracrows - 1] = rows[r];
560
561 violrowpos[SCIProwGetLPPos(violrows[theviolrowpos])] = theviolrowpos;
562 violrowpos[rowlppos] = *nviolfracrows - 1;
563 }
564 (*nviolfracrows)--;
565 }
566 /* second interesting case: the number of fractional variables was zero before, swap it with the first
567 * violated row without fractional variables
568 */
569 else if( nfracsinrow[rowlppos] == incval )
570 {
571 assert(theviolrowpos >= *nviolfracrows);
572
573 /* nothing to do if the row is exactly located at index *nviolfracrows */
574 if( theviolrowpos > *nviolfracrows )
575 {
576 violrows[theviolrowpos] = violrows[*nviolfracrows];
577 violrows[*nviolfracrows] = rows[r];
578
579 violrowpos[SCIProwGetLPPos(violrows[theviolrowpos])] = theviolrowpos;
580 violrowpos[rowlppos] = *nviolfracrows;
581 }
582 (*nviolfracrows)++;
583 }
584 }
585 }
586}
587
588
589/*
590 * Callback methods
591 */
592
593/** copy method for primal heuristic plugins (called when SCIP copies plugins) */
594static
595SCIP_DECL_HEURCOPY(heurCopyShifting)
596{ /*lint --e{715}*/
597 assert(scip != NULL);
598 assert(heur != NULL);
599 assert(strcmp(SCIPheurGetName(heur), HEUR_NAME) == 0);
600
601 /* call inclusion method of primal heuristic */
603
604 return SCIP_OKAY;
605}
606
607
608/** initialization method of primal heuristic (called after problem was transformed) */
609static
610SCIP_DECL_HEURINIT(heurInitShifting) /*lint --e{715}*/
611{ /*lint --e{715}*/
612 SCIP_HEURDATA* heurdata;
613
614 assert(strcmp(SCIPheurGetName(heur), HEUR_NAME) == 0);
615 assert(SCIPheurGetData(heur) == NULL);
616
617 /* create heuristic data */
618 SCIP_CALL( SCIPallocBlockMemory(scip, &heurdata) );
619 SCIP_CALL( SCIPcreateSol(scip, &heurdata->sol, heur) );
620 heurdata->lastlp = -1;
621
622 /* create random number generator */
623 SCIP_CALL( SCIPcreateRandom(scip, &heurdata->randnumgen,
625
626 SCIPheurSetData(heur, heurdata);
627
628 return SCIP_OKAY;
629}
630
631/** deinitialization method of primal heuristic (called before transformed problem is freed) */
632static
633SCIP_DECL_HEUREXIT(heurExitShifting) /*lint --e{715}*/
634{ /*lint --e{715}*/
635 SCIP_HEURDATA* heurdata;
636
637 assert(strcmp(SCIPheurGetName(heur), HEUR_NAME) == 0);
638
639 /* free heuristic data */
640 heurdata = SCIPheurGetData(heur);
641 assert(heurdata != NULL);
642 SCIP_CALL( SCIPfreeSol(scip, &heurdata->sol) );
643
644 /* free random number generator */
645 SCIPfreeRandom(scip, &heurdata->randnumgen);
646
647 SCIPfreeBlockMemory(scip, &heurdata);
648 SCIPheurSetData(heur, NULL);
649
650 return SCIP_OKAY;
651}
652
653/** solving process initialization method of primal heuristic (called when branch and bound process is about to begin) */
654static
655SCIP_DECL_HEURINITSOL(heurInitsolShifting)
656{
657 SCIP_HEURDATA* heurdata;
658
659 assert(strcmp(SCIPheurGetName(heur), HEUR_NAME) == 0);
660
661 heurdata = SCIPheurGetData(heur);
662 assert(heurdata != NULL);
663 heurdata->lastlp = -1;
664
665 return SCIP_OKAY;
666}
667
668
669/** execution method of primal heuristic */
670static
671SCIP_DECL_HEUREXEC(heurExecShifting) /*lint --e{715}*/
672{ /*lint --e{715}*/
673 SCIP_HEURDATA* heurdata;
674 SCIP_SOL* sol;
675 SCIP_VAR** lpcands;
676 SCIP_Real* lpcandssol;
677 SCIP_ROW** lprows;
678 SCIP_Real* activities;
679 SCIP_ROW** violrows;
680 SCIP_Real* nincreases;
681 SCIP_Real* ndecreases;
682 int* violrowpos;
683 int* nfracsinrow;
684 SCIP_Real increaseweight;
685 SCIP_Real obj;
686 SCIP_Real bestshiftval;
687 SCIP_Real minobj;
688 int nvars;
689 int nfrac;
690 int nlpcands;
691 int nlprows;
692 int nviolrows;
693 int nviolfracrows;
694 int nprevviolrows;
695 int minnviolrows;
696 int nnonimprovingshifts;
697 int c;
698 int r;
699 SCIP_Longint nlps;
700 SCIP_Longint ncalls;
701 SCIP_Longint nsolsfound;
703
704 assert(strcmp(SCIPheurGetName(heur), HEUR_NAME) == 0);
705 assert(scip != NULL);
706 assert(result != NULL);
707 assert(SCIPhasCurrentNodeLP(scip));
708
709 *result = SCIP_DIDNOTRUN;
710
711 /* only call heuristic, if an optimal LP solution is at hand */
713 return SCIP_OKAY;
714
715 /* only call heuristic, if the LP objective value is smaller than the cutoff bound */
717 return SCIP_OKAY;
718
719 /* get heuristic data */
720 heurdata = SCIPheurGetData(heur);
721 assert(heurdata != NULL);
722
723 /* don't call heuristic, if we have already processed the current LP solution */
724 nlps = SCIPgetNLPs(scip);
725 if( nlps == heurdata->lastlp )
726 return SCIP_OKAY;
727 heurdata->lastlp = nlps;
728
729 /* don't call heuristic, if it was not successful enough in the past */
730 ncalls = SCIPheurGetNCalls(heur);
731 nsolsfound = 10*SCIPheurGetNBestSolsFound(heur) + SCIPheurGetNSolsFound(heur);
733 if( nnodes % ((ncalls/100)/(nsolsfound+1)+1) != 0 )
734 return SCIP_OKAY;
735
736 /* get fractional variables, that should be integral */
737 SCIP_CALL( SCIPgetLPBranchCands(scip, &lpcands, &lpcandssol, NULL, &nlpcands, NULL, NULL) );
738
739 /* only call heuristic, if LP solution is fractional */
740 if( nlpcands == 0 )
741 return SCIP_OKAY;
742
743 *result = SCIP_DIDNOTFIND;
744
745 /* get LP rows */
746 SCIP_CALL( SCIPgetLPRowsData(scip, &lprows, &nlprows) );
747
748 SCIPdebugMsg(scip, "executing shifting heuristic: %d LP rows, %d LP candidates\n", nlprows, nlpcands);
749
750 /* get memory for activities, violated rows, and row violation positions */
751 nvars = SCIPgetNVars(scip);
752 nfrac = nlpcands;
753 SCIP_CALL( SCIPallocBufferArray(scip, &activities, nlprows) );
754 SCIP_CALL( SCIPallocBufferArray(scip, &violrows, nlprows) );
755 SCIP_CALL( SCIPallocBufferArray(scip, &violrowpos, nlprows) );
756 SCIP_CALL( SCIPallocBufferArray(scip, &nfracsinrow, nlprows) );
757 SCIP_CALL( SCIPallocBufferArray(scip, &nincreases, nvars) );
758 SCIP_CALL( SCIPallocBufferArray(scip, &ndecreases, nvars) );
759 BMSclearMemoryArray(nfracsinrow, nlprows);
760 BMSclearMemoryArray(nincreases, nvars);
761 BMSclearMemoryArray(ndecreases, nvars);
762
763 /* get the activities for all globally valid rows;
764 * the rows should be feasible, but due to numerical inaccuracies in the LP solver, they can be violated
765 */
766 nviolrows = 0;
767 for( r = 0; r < nlprows; ++r )
768 {
769 SCIP_ROW* row;
770
771 row = lprows[r];
772 assert(SCIProwGetLPPos(row) == r);
773
774 if( !SCIProwIsLocal(row) )
775 {
776 activities[r] = SCIPgetRowActivity(scip, row);
777 if( SCIPisFeasLT(scip, activities[r], SCIProwGetLhs(row))
778 || SCIPisFeasGT(scip, activities[r], SCIProwGetRhs(row)) )
779 {
780 violrows[nviolrows] = row;
781 violrowpos[r] = nviolrows;
782 nviolrows++;
783 }
784 else
785 violrowpos[r] = -1;
786 }
787 else
788 violrowpos[r] = -1;
789 }
790
791 nviolfracrows = 0;
792 /* calc the current number of fractional variables in rows */
793 for( c = 0; c < nlpcands; ++c )
794 addFracCounter(nfracsinrow, violrows, violrowpos, &nviolfracrows, nviolrows, nlprows, lpcands[c], +1);
795
796 /* get the working solution from heuristic's local data */
797 sol = heurdata->sol;
798 assert(sol != NULL);
799
800 /* copy the current LP solution to the working solution */
802
803 /* calculate the minimal objective value possible after rounding fractional variables */
804 minobj = SCIPgetSolTransObj(scip, sol);
805 assert(minobj < SCIPgetCutoffbound(scip));
806 for( c = 0; c < nlpcands; ++c )
807 {
808 obj = SCIPvarGetObj(lpcands[c]);
809 bestshiftval = obj > 0.0 ? SCIPfeasFloor(scip, lpcandssol[c]) : SCIPfeasCeil(scip, lpcandssol[c]);
810 minobj += obj * (bestshiftval - lpcandssol[c]);
811 }
812
813 /* try to shift remaining variables in order to become/stay feasible */
814 nnonimprovingshifts = 0;
815 minnviolrows = INT_MAX;
816 increaseweight = 1.0;
817 while( (nfrac > 0 || nviolrows > 0) && nnonimprovingshifts < MAXSHIFTINGS )
818 {
819 SCIP_VAR* shiftvar;
820 SCIP_Real oldsolval;
821 SCIP_Real newsolval;
822 SCIP_Bool oldsolvalisfrac;
823 int probindex;
824
825 SCIPdebugMsg(scip, "shifting heuristic: nfrac=%d, nviolrows=%d, obj=%g (best possible obj: %g), cutoff=%g\n",
826 nfrac, nviolrows, SCIPgetSolOrigObj(scip, sol), SCIPretransformObj(scip, minobj),
828
829 nprevviolrows = nviolrows;
830
831 /* choose next variable to process:
832 * - if a violated row exists, shift a variable decreasing the violation, that has least impact on other rows
833 * - otherwise, shift a variable, that has strongest devastating impact on rows in opposite direction
834 */
835 shiftvar = NULL;
836 oldsolval = 0.0;
837 newsolval = 0.0;
838 if( nviolrows > 0 && (nfrac == 0 || nnonimprovingshifts < MAXSHIFTINGS-1) )
839 {
840 SCIP_ROW* row;
841 int rowidx;
842 int rowpos;
843 int direction;
844
845 assert(nviolfracrows == 0 || nfrac > 0);
846 /* violated rows containing fractional variables are preferred; if such a row exists, choose the last one from the list
847 * (at position nviolfracrows - 1) because removing this row will cause one swapping operation less than other rows
848 */
849 if( nviolfracrows > 0 )
850 rowidx = nviolfracrows - 1;
851 else
852 /* there is no violated row containing a fractional variable, select a violated row uniformly at random */
853 rowidx = SCIPrandomGetInt(heurdata->randnumgen, 0, nviolrows-1);
854
855 assert(0 <= rowidx && rowidx < nviolrows);
856 row = violrows[rowidx];
857 rowpos = SCIProwGetLPPos(row);
858 assert(0 <= rowpos && rowpos < nlprows);
859 assert(violrowpos[rowpos] == rowidx);
860 assert(nfracsinrow[rowpos] == 0 || rowidx == nviolfracrows - 1);
861
862 SCIPdebugMsg(scip, "shifting heuristic: try to fix violated row <%s>: %g <= %g <= %g\n",
863 SCIProwGetName(row), SCIProwGetLhs(row), activities[rowpos], SCIProwGetRhs(row));
865
866 /* get direction in which activity must be shifted */
867 assert(SCIPisFeasLT(scip, activities[rowpos], SCIProwGetLhs(row))
868 || SCIPisFeasGT(scip, activities[rowpos], SCIProwGetRhs(row)));
869 direction = SCIPisFeasLT(scip, activities[rowpos], SCIProwGetLhs(row)) ? +1 : -1;
870
871 /* search a variable that can shift the activity in the necessary direction */
872 SCIP_CALL( selectShifting(scip, sol, row, activities[rowpos], direction,
873 nincreases, ndecreases, increaseweight, &shiftvar, &oldsolval, &newsolval) );
874 }
875
876 if( shiftvar == NULL && nfrac > 0 )
877 {
878 SCIPdebugMsg(scip, "shifting heuristic: search rounding variable and try to stay feasible\n");
879 SCIP_CALL( selectEssentialRounding(scip, sol, minobj, lpcands, nlpcands, &shiftvar, &oldsolval, &newsolval) );
880 }
881
882 /* check, whether shifting was possible */
883 if( shiftvar == NULL || SCIPisEQ(scip, oldsolval, newsolval) )
884 {
885 SCIPdebugMsg(scip, "shifting heuristic: -> didn't find a shifting variable\n");
886 break;
887 }
888
889 SCIPdebugMsg(scip, "shifting heuristic: -> shift var <%s>[%g,%g], type=%d, oldval=%g, newval=%g, obj=%g\n",
890 SCIPvarGetName(shiftvar), SCIPvarGetLbGlobal(shiftvar), SCIPvarGetUbGlobal(shiftvar), SCIPvarGetType(shiftvar),
891 oldsolval, newsolval, SCIPvarGetObj(shiftvar));
892
893 /* update row activities of globally valid rows */
894 SCIP_CALL( updateActivities(scip, activities, violrows, violrowpos, &nviolrows, &nviolfracrows, nfracsinrow, nlprows,
895 shiftvar, oldsolval, newsolval) );
896 if( nviolrows >= nprevviolrows )
897 nnonimprovingshifts++;
898 else if( nviolrows < minnviolrows )
899 {
900 minnviolrows = nviolrows;
901 nnonimprovingshifts = 0;
902 }
903
904 /* store new solution value and decrease fractionality counter */
905 SCIP_CALL( SCIPsetSolVal(scip, sol, shiftvar, newsolval) );
906
907 /* update fractionality counter and minimal objective value possible after shifting remaining variables */
908 oldsolvalisfrac = (!SCIPisFeasIntegral(scip, oldsolval) && SCIPvarIsIntegral(shiftvar) && !SCIPvarIsImpliedIntegral(shiftvar));
909 obj = SCIPvarGetObj(shiftvar);
910 if( SCIPvarGetType(shiftvar) != SCIP_VARTYPE_CONTINUOUS && oldsolvalisfrac )
911 {
912 assert(SCIPisFeasIntegral(scip, newsolval));
913 nfrac--;
914 nnonimprovingshifts = 0;
915 minnviolrows = INT_MAX;
916 addFracCounter(nfracsinrow, violrows, violrowpos, &nviolfracrows, nviolrows, nlprows, shiftvar, -1);
917
918 /* the rounding was already calculated into the minobj -> update only if rounding in "wrong" direction */
919 if( obj > 0.0 && newsolval > oldsolval )
920 minobj += obj;
921 else if( obj < 0.0 && newsolval < oldsolval )
922 minobj -= obj;
923 }
924 else
925 {
926 /* update minimal possible objective value */
927 minobj += obj * (newsolval - oldsolval);
928 }
929
930 /* update increase/decrease arrays */
931 if( !oldsolvalisfrac )
932 {
933 probindex = SCIPvarGetProbindex(shiftvar);
934 assert(0 <= probindex && probindex < nvars);
935 increaseweight *= WEIGHTFACTOR;
936 if( newsolval < oldsolval )
937 ndecreases[probindex] += increaseweight;
938 else
939 nincreases[probindex] += increaseweight;
940 if( increaseweight >= 1e+09 )
941 {
942 int i;
943
944 for( i = 0; i < nvars; ++i )
945 {
946 nincreases[i] /= increaseweight;
947 ndecreases[i] /= increaseweight;
948 }
949 increaseweight = 1.0;
950 }
951 }
952
953 SCIPdebugMsg(scip, "shifting heuristic: -> nfrac=%d, nviolrows=%d, obj=%g (best possible obj: %g)\n",
954 nfrac, nviolrows, SCIPgetSolOrigObj(scip, sol), SCIPretransformObj(scip, minobj));
955 }
956
957 /* check, if the new solution is feasible */
958 if( nfrac == 0 && nviolrows == 0 )
959 {
960 SCIP_Bool stored;
961
962 /* check solution for feasibility, and add it to solution store if possible
963 * neither integrality nor feasibility of LP rows has to be checked, because this is already
964 * done in the shifting heuristic itself; however, we better check feasibility of LP rows,
965 * because of numerical problems with activity updating
966 */
967 SCIP_CALL( SCIPtrySol(scip, sol, FALSE, FALSE, FALSE, FALSE, TRUE, &stored) );
968
969 if( stored )
970 {
971 SCIPdebugMsg(scip, "found feasible shifted solution:\n");
973 *result = SCIP_FOUNDSOL;
974 }
975 }
976
977 /* free memory buffers */
978 SCIPfreeBufferArray(scip, &ndecreases);
979 SCIPfreeBufferArray(scip, &nincreases);
980 SCIPfreeBufferArray(scip, &nfracsinrow);
981 SCIPfreeBufferArray(scip, &violrowpos);
982 SCIPfreeBufferArray(scip, &violrows);
983 SCIPfreeBufferArray(scip, &activities);
984
985 return SCIP_OKAY;
986}
987
988
989/*
990 * heuristic specific interface methods
991 */
992
993/** creates the shifting heuristic with infeasibility recovering and includes it in SCIP */
995 SCIP* scip /**< SCIP data structure */
996 )
997{
998 SCIP_HEUR* heur;
999
1000 /* include primal heuristic */
1003 HEUR_MAXDEPTH, HEUR_TIMING, HEUR_USESSUBSCIP, heurExecShifting, NULL) );
1004
1005 assert(heur != NULL);
1006
1007 /* primal heuristic is safe to use in exact solving mode */
1008 SCIPheurMarkExact(heur);
1009
1010 /* set non-NULL pointers to callback methods */
1011 SCIP_CALL( SCIPsetHeurCopy(scip, heur, heurCopyShifting) );
1012 SCIP_CALL( SCIPsetHeurInit(scip, heur, heurInitShifting) );
1013 SCIP_CALL( SCIPsetHeurExit(scip, heur, heurExitShifting) );
1014 SCIP_CALL( SCIPsetHeurInitsol(scip, heur, heurInitsolShifting) );
1015
1016 return SCIP_OKAY;
1017}
1018
SCIP_Real * r
Definition: circlepacking.c:59
#define NULL
Definition: def.h:248
#define SCIP_Longint
Definition: def.h:141
#define SCIP_REAL_MAX
Definition: def.h:158
#define SCIP_Bool
Definition: def.h:91
#define MIN(x, y)
Definition: def.h:224
#define SCIP_Real
Definition: def.h:156
#define TRUE
Definition: def.h:93
#define FALSE
Definition: def.h:94
#define MAX(x, y)
Definition: def.h:220
#define SCIP_CALL(x)
Definition: def.h:355
#define nnodes
Definition: gastrans.c:74
int SCIPgetNVars(SCIP *scip)
Definition: scip_prob.c:2246
#define SCIPdebugMsg
Definition: scip_message.h:78
SCIP_RETCODE SCIPincludeHeurShifting(SCIP *scip)
SCIP_RETCODE SCIPgetLPBranchCands(SCIP *scip, SCIP_VAR ***lpcands, SCIP_Real **lpcandssol, SCIP_Real **lpcandsfrac, int *nlpcands, int *npriolpcands, int *nfracimplvars)
Definition: scip_branch.c:402
SCIP_VAR * SCIPcolGetVar(SCIP_COL *col)
Definition: lp.c:17425
SCIP_Real * SCIPcolGetVals(SCIP_COL *col)
Definition: lp.c:17555
SCIP_ROW ** SCIPcolGetRows(SCIP_COL *col)
Definition: lp.c:17545
int SCIPcolGetNLPNonz(SCIP_COL *col)
Definition: lp.c:17534
SCIP_RETCODE SCIPsetHeurCopy(SCIP *scip, SCIP_HEUR *heur, SCIP_DECL_HEURCOPY((*heurcopy)))
Definition: scip_heur.c:167
SCIP_HEURDATA * SCIPheurGetData(SCIP_HEUR *heur)
Definition: heur.c:1368
SCIP_RETCODE SCIPincludeHeurBasic(SCIP *scip, SCIP_HEUR **heur, const char *name, const char *desc, char dispchar, int priority, int freq, int freqofs, int maxdepth, SCIP_HEURTIMING timingmask, SCIP_Bool usessubscip, SCIP_DECL_HEUREXEC((*heurexec)), SCIP_HEURDATA *heurdata)
Definition: scip_heur.c:122
SCIP_Longint SCIPheurGetNSolsFound(SCIP_HEUR *heur)
Definition: heur.c:1603
SCIP_RETCODE SCIPsetHeurExit(SCIP *scip, SCIP_HEUR *heur, SCIP_DECL_HEUREXIT((*heurexit)))
Definition: scip_heur.c:215
SCIP_Longint SCIPheurGetNBestSolsFound(SCIP_HEUR *heur)
Definition: heur.c:1613
SCIP_Longint SCIPheurGetNCalls(SCIP_HEUR *heur)
Definition: heur.c:1593
SCIP_RETCODE SCIPsetHeurInitsol(SCIP *scip, SCIP_HEUR *heur, SCIP_DECL_HEURINITSOL((*heurinitsol)))
Definition: scip_heur.c:231
void SCIPheurMarkExact(SCIP_HEUR *heur)
Definition: heur.c:1457
SCIP_RETCODE SCIPsetHeurInit(SCIP *scip, SCIP_HEUR *heur, SCIP_DECL_HEURINIT((*heurinit)))
Definition: scip_heur.c:199
const char * SCIPheurGetName(SCIP_HEUR *heur)
Definition: heur.c:1467
void SCIPheurSetData(SCIP_HEUR *heur, SCIP_HEURDATA *heurdata)
Definition: heur.c:1378
SCIP_Bool SCIPhasCurrentNodeLP(SCIP *scip)
Definition: scip_lp.c:87
SCIP_RETCODE SCIPgetLPRowsData(SCIP *scip, SCIP_ROW ***rows, int *nrows)
Definition: scip_lp.c:576
SCIP_LPSOLSTAT SCIPgetLPSolstat(SCIP *scip)
Definition: scip_lp.c:174
SCIP_Real SCIPgetLPObjval(SCIP *scip)
Definition: scip_lp.c:253
#define SCIPallocBufferArray(scip, ptr, num)
Definition: scip_mem.h:124
#define SCIPfreeBufferArray(scip, ptr)
Definition: scip_mem.h:136
#define SCIPfreeBlockMemory(scip, ptr)
Definition: scip_mem.h:108
#define SCIPallocBlockMemory(scip, ptr)
Definition: scip_mem.h:89
SCIP_Real SCIProwGetLhs(SCIP_ROW *row)
Definition: lp.c:17686
SCIP_COL ** SCIProwGetCols(SCIP_ROW *row)
Definition: lp.c:17632
SCIP_Real SCIProwGetRhs(SCIP_ROW *row)
Definition: lp.c:17696
int SCIProwGetNLPNonz(SCIP_ROW *row)
Definition: lp.c:17621
int SCIProwGetLPPos(SCIP_ROW *row)
Definition: lp.c:17895
SCIP_Bool SCIProwIsLocal(SCIP_ROW *row)
Definition: lp.c:17795
SCIP_RETCODE SCIPprintRow(SCIP *scip, SCIP_ROW *row, FILE *file)
Definition: scip_lp.c:2176
const char * SCIProwGetName(SCIP_ROW *row)
Definition: lp.c:17745
SCIP_Real SCIPgetRowActivity(SCIP *scip, SCIP_ROW *row)
Definition: scip_lp.c:2068
SCIP_Bool SCIProwIsInLP(SCIP_ROW *row)
Definition: lp.c:17917
SCIP_Real * SCIProwGetVals(SCIP_ROW *row)
Definition: lp.c:17642
SCIP_RETCODE SCIPcreateSol(SCIP *scip, SCIP_SOL **sol, SCIP_HEUR *heur)
Definition: scip_sol.c:516
SCIP_RETCODE SCIPfreeSol(SCIP *scip, SCIP_SOL **sol)
Definition: scip_sol.c:1252
SCIP_RETCODE SCIPprintSol(SCIP *scip, SCIP_SOL *sol, FILE *file, SCIP_Bool printzeros)
Definition: scip_sol.c:2349
SCIP_RETCODE SCIPtrySol(SCIP *scip, SCIP_SOL *sol, SCIP_Bool printreason, SCIP_Bool completely, SCIP_Bool checkbounds, SCIP_Bool checkintegrality, SCIP_Bool checklprows, SCIP_Bool *stored)
Definition: scip_sol.c:4012
SCIP_RETCODE SCIPlinkLPSol(SCIP *scip, SCIP_SOL *sol)
Definition: scip_sol.c:1295
SCIP_Real SCIPgetSolOrigObj(SCIP *scip, SCIP_SOL *sol)
Definition: scip_sol.c:1892
SCIP_RETCODE SCIPsetSolVal(SCIP *scip, SCIP_SOL *sol, SCIP_VAR *var, SCIP_Real val)
Definition: scip_sol.c:1571
SCIP_Real SCIPgetSolVal(SCIP *scip, SCIP_SOL *sol, SCIP_VAR *var)
Definition: scip_sol.c:1765
SCIP_Real SCIPgetSolTransObj(SCIP *scip, SCIP_SOL *sol)
Definition: scip_sol.c:2005
SCIP_Real SCIPretransformObj(SCIP *scip, SCIP_Real obj)
Definition: scip_sol.c:2132
SCIP_Longint SCIPgetNNodes(SCIP *scip)
SCIP_Longint SCIPgetNLPs(SCIP *scip)
SCIP_Real SCIPgetCutoffbound(SCIP *scip)
SCIP_Real SCIPinfinity(SCIP *scip)
SCIP_Bool SCIPisGE(SCIP *scip, SCIP_Real val1, SCIP_Real val2)
SCIP_Bool SCIPisPositive(SCIP *scip, SCIP_Real val)
SCIP_Real SCIPfeasCeil(SCIP *scip, SCIP_Real val)
SCIP_Real SCIPfeasFloor(SCIP *scip, SCIP_Real val)
SCIP_Bool SCIPisInfinity(SCIP *scip, SCIP_Real val)
SCIP_Bool SCIPisFeasLT(SCIP *scip, SCIP_Real val1, SCIP_Real val2)
SCIP_Bool SCIPisFeasIntegral(SCIP *scip, SCIP_Real val)
SCIP_Bool SCIPisNegative(SCIP *scip, SCIP_Real val)
SCIP_Bool SCIPisFeasGT(SCIP *scip, SCIP_Real val1, SCIP_Real val2)
SCIP_Bool SCIPisEQ(SCIP *scip, SCIP_Real val1, SCIP_Real val2)
SCIP_Bool SCIPisZero(SCIP *scip, SCIP_Real val)
SCIP_COL * SCIPvarGetCol(SCIP_VAR *var)
Definition: var.c:23683
int SCIPvarGetNLocksUpType(SCIP_VAR *var, SCIP_LOCKTYPE locktype)
Definition: var.c:4386
SCIP_Bool SCIPvarIsImpliedIntegral(SCIP_VAR *var)
Definition: var.c:23498
SCIP_Real SCIPvarGetObj(SCIP_VAR *var)
Definition: var.c:23900
SCIP_VARTYPE SCIPvarGetType(SCIP_VAR *var)
Definition: var.c:23453
SCIP_Real SCIPvarGetUbGlobal(SCIP_VAR *var)
Definition: var.c:24142
int SCIPvarGetProbindex(SCIP_VAR *var)
Definition: var.c:23662
const char * SCIPvarGetName(SCIP_VAR *var)
Definition: var.c:23267
SCIP_Bool SCIPvarIsIntegral(SCIP_VAR *var)
Definition: var.c:23490
SCIP_Real SCIPvarGetLbGlobal(SCIP_VAR *var)
Definition: var.c:24120
int SCIPvarGetNLocksDownType(SCIP_VAR *var, SCIP_LOCKTYPE locktype)
Definition: var.c:4328
void SCIPfreeRandom(SCIP *scip, SCIP_RANDNUMGEN **randnumgen)
SCIP_RETCODE SCIPcreateRandom(SCIP *scip, SCIP_RANDNUMGEN **randnumgen, unsigned int initialseed, SCIP_Bool useglobalseed)
int SCIPrandomGetInt(SCIP_RANDNUMGEN *randnumgen, int minrandval, int maxrandval)
Definition: misc.c:10223
static SCIP_RETCODE selectEssentialRounding(SCIP *scip, SCIP_SOL *sol, SCIP_Real minobj, SCIP_VAR **lpcands, int nlpcands, SCIP_VAR **shiftvar, SCIP_Real *oldsolval, SCIP_Real *newsolval)
static SCIP_DECL_HEURINITSOL(heurInitsolShifting)
#define HEUR_TIMING
Definition: heur_shifting.c:59
static void addFracCounter(int *nfracsinrow, SCIP_ROW **violrows, int *violrowpos, int *nviolfracrows, int nviolrows, int nlprows, SCIP_VAR *var, int incval)
static SCIP_DECL_HEUREXIT(heurExitShifting)
#define HEUR_FREQOFS
Definition: heur_shifting.c:57
#define HEUR_DESC
Definition: heur_shifting.c:53
#define HEUR_DISPCHAR
Definition: heur_shifting.c:54
#define HEUR_MAXDEPTH
Definition: heur_shifting.c:58
#define HEUR_PRIORITY
Definition: heur_shifting.c:55
static SCIP_DECL_HEUREXEC(heurExecShifting)
static SCIP_DECL_HEURCOPY(heurCopyShifting)
#define HEUR_NAME
Definition: heur_shifting.c:52
static SCIP_RETCODE updateActivities(SCIP *scip, SCIP_Real *activities, SCIP_ROW **violrows, int *violrowpos, int *nviolrows, int *nviolfracrows, int *nfracsinrow, int nlprows, SCIP_VAR *var, SCIP_Real oldsolval, SCIP_Real newsolval)
static void updateViolations(SCIP *scip, SCIP_ROW *row, SCIP_ROW **violrows, int *violrowpos, int *nviolrows, int *nviolfracrows, int *nfracsinrow, SCIP_Real oldactivity, SCIP_Real newactivity)
Definition: heur_shifting.c:82
#define DEFAULT_RANDSEED
Definition: heur_shifting.c:64
static SCIP_DECL_HEURINIT(heurInitShifting)
#define HEUR_FREQ
Definition: heur_shifting.c:56
static SCIP_RETCODE selectShifting(SCIP *scip, SCIP_SOL *sol, SCIP_ROW *row, SCIP_Real rowactivity, int direction, SCIP_Real *nincreases, SCIP_Real *ndecreases, SCIP_Real increaseweight, SCIP_VAR **shiftvar, SCIP_Real *oldsolval, SCIP_Real *newsolval)
#define HEUR_USESSUBSCIP
Definition: heur_shifting.c:60
#define WEIGHTFACTOR
Definition: heur_shifting.c:63
#define MAXSHIFTINGS
Definition: heur_shifting.c:62
LP rounding heuristic that tries to recover from intermediate infeasibilities and shifts continuous v...
memory allocation routines
#define BMSclearMemoryArray(ptr, num)
Definition: memory.h:130
public methods for primal heuristics
public methods for LP management
public methods for message output
#define SCIPdebug(x)
Definition: pub_message.h:93
public data structures and miscellaneous methods
public methods for problem variables
public methods for branching rule plugins and branching
public methods for primal heuristic plugins and divesets
public methods for the LP relaxation, rows and columns
public methods for memory management
public methods for message handling
public methods for numerical tolerances
public methods for global and local (sub)problems
public methods for random numbers
public methods for solutions
public methods for querying solving statistics
struct SCIP_HeurData SCIP_HEURDATA
Definition: type_heur.h:77
@ SCIP_LPSOLSTAT_OPTIMAL
Definition: type_lp.h:44
@ SCIP_DIDNOTRUN
Definition: type_result.h:42
@ SCIP_DIDNOTFIND
Definition: type_result.h:44
@ SCIP_FOUNDSOL
Definition: type_result.h:56
@ SCIP_OKAY
Definition: type_retcode.h:42
enum SCIP_Retcode SCIP_RETCODE
Definition: type_retcode.h:63
@ SCIP_VARTYPE_CONTINUOUS
Definition: type_var.h:71
@ SCIP_LOCKTYPE_MODEL
Definition: type_var.h:141