SoPlex Doxygen Documentation
spxmpswrite.cpp
Go to the documentation of this file.
1 /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2 /* */
3 /* This file is part of the class library */
4 /* SoPlex --- the Sequential object-oriented simPlex. */
5 /* */
6 /* Copyright (C) 1996-2012 Konrad-Zuse-Zentrum */
7 /* fuer Informationstechnik Berlin */
8 /* */
9 /* SoPlex is distributed under the terms of the ZIB Academic Licence. */
10 /* */
11 /* You should have received a copy of the ZIB Academic License */
12 /* along with SoPlex; see the file COPYING. If not email to soplex@zib.de. */
13 /* */
14 /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
15 
16 /**@file spxmpswrite.cpp
17  * @brief Write LP as MPS format file.
18  */
19 //#define DEBUGGING 1
20 
21 #include <assert.h>
22 #include <stdio.h>
23 #include <iostream>
24 
25 #include "spxdefines.h"
26 #include "spxlp.h"
27 #include "spxout.h"
28 #include "exceptions.h"
29 
30 namespace soplex
31 {
32 static void writeRecord(
33  std::ostream& os,
34  const char* indicator,
35  const char* name,
36  const char* name1 = 0,
37  const Real value1 = 0.0,
38  const char* name2 = 0,
39  const Real value2 = 0.0)
40 {
41  char buf[81];
42 
43  sprintf(buf, " %-2.2s %-8.8s",
44  (indicator == 0) ? "" : indicator,
45  (name == 0) ? "" : name);
46 
47  os << buf;
48 
49  if (name1 != 0)
50  {
51  sprintf(buf, " %-8.8s %12.9g", name1, value1);
52  os << buf;
53 
54  if (name2 != 0)
55  {
56  sprintf(buf, " %-8.8s %12.9g", name2, value2);
57  os << buf;
58  }
59  }
60  os << std::endl;
61 }
62 
63 static Real getRHS(Real left, Real right)
64 {
65  Real rhsval;
66 
67  if (left > -infinity) /// This includes ranges
68  rhsval = left;
69  else if (right < infinity)
70  rhsval = right;
71  else
72  throw SPxInternalCodeException("XMPSWR01 This should never happen.");
73 
74  return rhsval;
75 }
76 
77 static const char* getRowName(
78  const SPxLP& lp,
79  int idx,
80  const NameSet* rnames,
81  char* buf)
82 {
83  assert(buf != 0);
84  assert(idx >= 0);
85  assert(idx < lp.nRows());
86 
87  if (rnames != 0)
88  {
89  DataKey key = lp.rId(idx);
90 
91  if (rnames->has(key))
92  return (*rnames)[key];
93  }
94  sprintf(buf, "C%d", idx);
95 
96  return buf;
97 }
98 
99 static const char* getColName(
100  const SPxLP& lp,
101  int idx,
102  const NameSet* cnames,
103  char* buf)
104 {
105  assert(buf != 0);
106  assert(idx >= 0);
107  assert(idx < lp.nCols());
108 
109  if (cnames != 0)
110  {
111  DataKey key = lp.cId(idx);
112 
113  if (cnames->has(key))
114  return (*cnames)[key];
115  }
116  sprintf(buf, "x%d", idx);
117 
118  return buf;
119 }
120 
121 /// Write LP in "MPS File Format".
122 /** @note There will always be a BOUNDS section, even if there are no bounds.
123  */
125  std::ostream& p_output, ///< output stream.
126  const NameSet* p_rnames, ///< row names.
127  const NameSet* p_cnames, ///< column names.
128  const DIdxSet* p_intvars) ///< integer variables.
129  const
130 {
131  METHOD("writeMPS");
132 
133  const char* indicator;
134  char name [16];
135  char name1[16];
136  char name2[16];
137  bool has_ranges = false;
138  int i;
139  int k;
140 
141  // --- NAME Section ---
142  p_output << "NAME MPSDATA" << std::endl;
143 
144  // --- ROWS Section ---
145  p_output << "ROWS" << std::endl;
146 
147  for(i = 0; i < nRows(); i++)
148  {
149  if (lhs(i) == rhs(i))
150  indicator = "E";
151  else if ((lhs(i) > -infinity) && (rhs(i) < infinity))
152  {
153  indicator = "E";
154  has_ranges = true;
155  }
156  else if (lhs(i) > -infinity)
157  indicator = "G";
158  else if (rhs(i) < infinity)
159  indicator = "L";
160  else
161  throw SPxInternalCodeException("XMPSWR02 This should never happen.");
162 
163  writeRecord(p_output, indicator, getRowName(*this, i, p_rnames, name));
164  }
165  writeRecord(p_output, "N", "MINIMIZE");
166 
167  // --- COLUMNS Section ---
168  p_output << "COLUMNS" << std::endl;
169 
170  bool has_intvars = (p_intvars != 0) && (p_intvars->size() > 0);
171 
172  for(int j = 0; j < (has_intvars ? 2 : 1); j++)
173  {
174  bool is_intrun = has_intvars && (j == 1);
175 
176  if (is_intrun)
177  p_output << " MARK0001 'MARKER' 'INTORG'"
178  << std::endl;
179 
180  for(i = 0; i < nCols(); i++)
181  {
182  bool is_intvar = has_intvars && (p_intvars->number(i) >= 0);
183 
184  if ( ( is_intrun && !is_intvar)
185  || (!is_intrun && is_intvar))
186  continue;
187 
188  const SVector& col = colVector(i);
189  int colsize2 = (col.size() / 2) * 2;
190 
191  assert(colsize2 % 2 == 0);
192 
193  for(k = 0; k < colsize2; k += 2)
194  writeRecord(p_output, 0,
195  getColName(*this, i, p_cnames, name),
196  getRowName(*this, col.index(k), p_rnames, name1),
197  col.value(k),
198  getRowName(*this, col.index(k + 1), p_rnames, name2),
199  col.value(k + 1));
200 
201  if (colsize2 != col.size())
202  writeRecord(p_output, 0,
203  getColName(*this, i, p_cnames, name),
204  getRowName(*this, col.index(k), p_rnames, name1),
205  col.value(k));
206 
207  if (isNotZero(maxObj(i)))
208  writeRecord(p_output, 0, getColName(*this, i, p_cnames, name),
209  "MINIMIZE", -maxObj(i));
210  }
211  if (is_intrun)
212  p_output << " MARK0001 'MARKER' 'INTEND'"
213  << std::endl;
214  }
215  // --- RHS Section ---
216  p_output << "RHS" << std::endl;
217 
218  i = 0;
219  while(i < nRows())
220  {
221  Real rhsval1 = 0.0;
222  Real rhsval2 = 0.0;
223 
224  for(; i < nRows(); i++)
225  if ((rhsval1 = getRHS(lhs(i), rhs(i))) != 0.0)
226  break;
227 
228  if (i < nRows())
229  {
230  for(k = i + 1; k < nRows(); k++)
231  if ((rhsval2 = getRHS(lhs(k), rhs(k))) != 0.0)
232  break;
233 
234  if (k < nRows())
235  writeRecord(p_output, 0, "RHS",
236  getRowName(*this, i, p_rnames, name1),
237  rhsval1,
238  getRowName(*this, k, p_rnames, name2),
239  rhsval2);
240  else
241  writeRecord(p_output, 0, "RHS",
242  getRowName(*this, i, p_rnames, name1),
243  rhsval1);
244 
245  i = k + 1;
246  }
247  }
248 
249  // --- RANGES Section ---
250  if (has_ranges)
251  {
252  p_output << "RANGES" << std::endl;
253 
254  for(i = 0; i < nRows(); i++)
255  if ((lhs(i) > -infinity) && (rhs(i) < infinity))
256  writeRecord(p_output, "", "RANGE",
257  getRowName(*this, i, p_rnames, name1),
258  rhs(i) - lhs(i));
259  }
260  // --- BOUNDS Section ---
261  p_output << "BOUNDS" << std::endl;
262 
263  for(i = 0; i < nCols(); i++)
264  {
265  // skip variables that do not appear in the objective function or any constraint
266  const SVector& col = colVector(i);
267  if (col.size() == 0 && isZero(maxObj(i)))
268  continue;
269  if (lower(i) == upper(i))
270  {
271  writeRecord(p_output, "FX", "BOUND",
272  getColName(*this, i, p_cnames, name1),
273  lower(i));
274 
275  continue;
276  }
277  if ((lower(i) <= -infinity) && (upper(i) >= infinity))
278  {
279  writeRecord(p_output, "FR", "BOUND",
280  getColName(*this, i, p_cnames, name1));
281  continue;
282  }
283  if (lower(i) != 0.0)
284  {
285  if (lower(i) > -infinity)
286  writeRecord(p_output, "LO", "BOUND",
287  getColName(*this, i, p_cnames, name1),
288  lower(i));
289  else
290  writeRecord(p_output, "MI", "BOUND",
291  getColName(*this, i, p_cnames, name1));
292  }
293 
294  if (has_intvars && (p_intvars->number(i) >= 0))
295  {
296  // Integer variables have default upper bound 1.0, but we should write
297  // it nevertheless since CPLEX seems to assume infinity otherwise.
298  writeRecord(p_output, "UP", "BOUND",
299  getColName(*this, i, p_cnames, name1),
300  upper(i));
301  }
302  else
303  {
304  // Continous variables have default upper bound infinity
305  if (upper(i) < infinity)
306  writeRecord(p_output, "UP", "BOUND",
307  getColName(*this, i, p_cnames, name1),
308  upper(i));
309  }
310  }
311  // --- ENDATA Section ---
312  p_output << "ENDATA" << std::endl;
313 
314  // Output warning when writing a maximisation problem
315  if(spxSense() == SPxLP::MAXIMIZE)
316  {
317  MSG_WARNING( spxout << "XMPSWR03 Warning: objective function inverted when writing maximization problem in MPS file format" << std::endl; )
318  }
319 }
320 
321 } // namespace soplex
322 
323 //-----------------------------------------------------------------------------
324 //Emacs Local Variables:
325 //Emacs mode:c++
326 //Emacs c-basic-offset:3
327 //Emacs tab-width:8
328 //Emacs indent-tabs-mode:nil
329 //Emacs End:
330 //-----------------------------------------------------------------------------
331