SoPlex Doxygen Documentation
spxlpfwrite.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 spxlpfwrite.cpp
17  * @brief Write LP as LPF format file.
18  */
19 #define NUM_ENTRIES_PER_LINE 5
20 
21 #include <assert.h>
22 #include <stdio.h>
23 #include <fstream>
24 
25 #include "spxdefines.h"
26 #include "spxlp.h"
27 
28 namespace soplex
29 {
30  namespace {
31 
32  // get the name of a row or construct one
33  const char* getRowName(
34  const SPxLP& p_lp,
35  int p_idx,
36  const NameSet* p_rnames,
37  char* p_buf,
38  int p_num_written_rows)
39  {
40  assert(p_buf != 0);
41  assert(p_idx >= 0);
42  assert(p_idx < p_lp.nRows());
43 
44  if (p_rnames != 0)
45  {
46  DataKey key = p_lp.rId(p_idx);
47 
48  if (p_rnames->has(key))
49  return (*p_rnames)[key];
50  }
51  sprintf(p_buf, "C%d", p_num_written_rows);
52 
53  return p_buf;
54  }
55 
56  // get the name of a column or construct one
57  const char* getColName(
58  const SPxLP& p_lp,
59  int p_idx,
60  const NameSet* p_cnames,
61  char* p_buf)
62  {
63  assert(p_buf != 0);
64  assert(p_idx >= 0);
65  assert(p_idx < p_lp.nCols());
66 
67  if (p_cnames != 0)
68  {
69  DataKey key = p_lp.cId(p_idx);
70 
71  if (p_cnames->has(key))
72  return (*p_cnames)[key];
73  }
74  sprintf(p_buf, "x%d", p_idx);
75 
76  return p_buf;
77  }
78 
79  // write an SVector
80  void writeSVector( const SPxLP& p_lp, ///< the LP
81  std::ostream& p_output, ///< output stream
82  const NameSet* p_cnames, ///< column names
83  const SVector& p_svec ) ///< vector to write
84  {
85  METHOD("writeSVector");
86 
87  char name[16];
88  int num_coeffs = 0;
89 
90  for (int j = 0; j < p_lp.nCols(); ++j)
91  {
92  const Real coeff = p_svec[j];
93  if (coeff == 0)
94  continue;
95 
96  if (num_coeffs == 0)
97  p_output << coeff << " "
98  << getColName(p_lp, j, p_cnames, name);
99  else
100  {
101  // insert a line break every NUM_ENTRIES_PER_LINE columns
102  if (num_coeffs % NUM_ENTRIES_PER_LINE == 0)
103  p_output << "\n\t";
104 
105  if (coeff < 0)
106  p_output << " - " << -coeff;
107  else
108  p_output << " + " << coeff;
109 
110  p_output << " " << getColName(p_lp, j, p_cnames, name);
111  }
112  ++num_coeffs;
113  }
114  }
115 
116  // write the objective
117  void writeObjective( const SPxLP& p_lp, ///< the LP
118  std::ostream& p_output, ///< output stream
119  const NameSet* p_cnames ) ///< column names
120  {
121  METHOD("writeObjective");
122 
123  const int sense = p_lp.spxSense();
124 
125  p_output << ((sense == SPxLP::MINIMIZE) ? "Minimize\n" : "Maximize\n");
126  p_output << " obj: ";
127 
128  const Vector& obj = p_lp.maxObj();
129  DSVector svec(obj.dim());
130  svec.operator=(obj);
131  svec *= Real(sense);
132  writeSVector(p_lp, p_output, p_cnames, svec);
133  p_output << "\n";
134  }
135 
136  // write non-ranged rows
137  void writeRow( const SPxLP& p_lp, ///< the LP
138  std::ostream& p_output, ///< output stream
139  const NameSet* p_cnames, ///< column names
140  const SVector& p_svec, ///< vector of the row
141  const Real& p_lhs, ///< lhs of the row
142  const Real& p_rhs ) ///< rhs of the row
143  {
144  METHOD("writeRow");
145 
146  writeSVector(p_lp, p_output, p_cnames, p_svec);
147 
148  if ( p_lhs == p_rhs )
149  p_output << " = " << p_rhs;
150  else if ( p_lhs <= -infinity )
151  p_output << " <= " << p_rhs;
152  else {
153  assert(p_rhs >= infinity);
154  p_output << " >= " << p_lhs;
155  }
156 
157  p_output << "\n";
158  }
159 
160  // write all rows
161  void writeRows( const SPxLP& p_lp, ///< the LP
162  std::ostream& p_output, ///< output stream
163  const NameSet* p_rnames, ///< row names
164  const NameSet* p_cnames ) ///< column names
165  {
166  METHOD("writeRows");
167  char name[16];
168 
169  p_output << "Subject To\n";
170 
171  int num_written_rows = 0; // num_written_rows > nRows with ranged rows
172  for (int i = 0; i < p_lp.nRows(); ++i)
173  {
174  const Real lhs = p_lp.lhs(i);
175  const Real rhs = p_lp.rhs(i);
176 
177  if (lhs > -infinity && rhs < infinity && lhs != rhs) {
178  // ranged row -> write two non-ranged rows
179  p_output << " " << getRowName(p_lp, i, p_rnames,
180  name, ++num_written_rows) << "_1 : ";
181  writeRow (p_lp, p_output, p_cnames, p_lp.rowVector(i), lhs, infinity);
182 
183  p_output << " " << getRowName(p_lp, i, p_rnames,
184  name, ++num_written_rows) << "_2 : ";
185  writeRow (p_lp, p_output, p_cnames, p_lp.rowVector(i), -infinity, rhs);
186  }
187  else {
188  p_output << " " << getRowName(p_lp, i, p_rnames,
189  name, ++num_written_rows) << " : ";
190  writeRow (p_lp, p_output, p_cnames, p_lp.rowVector(i), lhs, rhs);
191  }
192  }
193  }
194 
195  // write the variable bounds
196  // (the default bounds 0 <= x <= infinity are not written)
197  void writeBounds( const SPxLP& p_lp, ///< the LP to write
198  std::ostream& p_output, ///< output stream
199  const NameSet* p_cnames ) ///< column names
200  {
201  METHOD("writeBounds");
202  char name[16];
203 
204  p_output << "Bounds\n";
205  for (int j = 0; j < p_lp.nCols(); ++j)
206  {
207  const Real lower = p_lp.lower(j);
208  const Real upper = p_lp.upper(j);
209 
210  if (lower == upper) {
211  p_output << " " << getColName(p_lp, j, p_cnames, name)
212  << " = " << upper << '\n';
213  }
214  else if (lower > -infinity) {
215  if (upper < infinity) {
216  // range bound
217  if (lower != 0)
218  p_output << " " << lower << " <= "
219  << getColName(p_lp, j, p_cnames, name)
220  << " <= " << upper << '\n';
221  else
222  p_output << " " << getColName(p_lp, j, p_cnames, name)
223  << " <= " << upper << '\n';
224  }
225  else if (lower != 0)
226  p_output << " " << lower << " <= "
227  << getColName(p_lp, j, p_cnames, name)
228  << '\n';
229  }
230  else if (upper < infinity)
231  p_output << " -Inf <= "
232  << getColName(p_lp, j, p_cnames, name)
233  << " <= " << upper << '\n';
234  else
235  p_output << " " << getColName(p_lp, j, p_cnames, name)
236  << " free\n";
237  }
238  }
239 
240  // write the generals section
241  void writeGenerals( const SPxLP& p_lp, ///< the LP to write
242  std::ostream& p_output, ///< output stream
243  const NameSet* p_cnames, ///< column names
244  const DIdxSet* p_intvars ) ///< integer variables
245  {
246  METHOD("writeGenerals");
247  char name[16];
248 
249  if (p_intvars == NULL || p_intvars->size() <= 0)
250  return; // no integer variables
251 
252  p_output << "Generals\n";
253  for (int j = 0; j < p_lp.nCols(); ++j)
254  if (p_intvars->number(j) >= 0)
255  p_output << " " << getColName(p_lp, j, p_cnames, name) << "\n";
256  }
257  } // anonymous namespace
258 
259 /// Write LP in "LPF File Format".
260 void SPxLP::writeLPF( std::ostream& p_output, ///< output stream
261  const NameSet* p_rnames, ///< row names
262  const NameSet* p_cnames, ///< column names
263  const DIdxSet* p_intvars ) ///< integer variables
264  const
265 {
266  METHOD("writeLPF");
267 
268  writeObjective (*this, p_output, p_cnames);
269  writeRows (*this, p_output, p_rnames, p_cnames);
270  writeBounds (*this, p_output, p_cnames);
271  writeGenerals (*this, p_output, p_cnames, p_intvars);
272  p_output << "End" << std::endl;
273 }
274 
275 } // namespace soplex
276 
277 //-----------------------------------------------------------------------------
278 //Emacs Local Variables:
279 //Emacs mode:c++
280 //Emacs c-basic-offset:3
281 //Emacs tab-width:8
282 //Emacs indent-tabs-mode:nil
283 //Emacs End:
284 //-----------------------------------------------------------------------------
285