test/min_mean_cycle_test.cc
author Peter Kovacs <kpeter@inf.elte.hu>
Fri, 07 Aug 2009 14:52:40 +0200
changeset 810 93cd93e82f9b
child 811 1fac515a59c1
permissions -rw-r--r--
Add a detailed test file for MinMeanCycle and fix test_tools.h (#179)
kpeter@810
     1
/* -*- mode: C++; indent-tabs-mode: nil; -*-
kpeter@810
     2
 *
kpeter@810
     3
 * This file is a part of LEMON, a generic C++ optimization library.
kpeter@810
     4
 *
kpeter@810
     5
 * Copyright (C) 2003-2009
kpeter@810
     6
 * Egervary Jeno Kombinatorikus Optimalizalasi Kutatocsoport
kpeter@810
     7
 * (Egervary Research Group on Combinatorial Optimization, EGRES).
kpeter@810
     8
 *
kpeter@810
     9
 * Permission to use, modify and distribute this software is granted
kpeter@810
    10
 * provided that this copyright notice appears in all copies. For
kpeter@810
    11
 * precise terms see the accompanying LICENSE file.
kpeter@810
    12
 *
kpeter@810
    13
 * This software is provided "AS IS" with no warranty of any kind,
kpeter@810
    14
 * express or implied, and with no claim as to its suitability for any
kpeter@810
    15
 * purpose.
kpeter@810
    16
 *
kpeter@810
    17
 */
kpeter@810
    18
kpeter@810
    19
#include <iostream>
kpeter@810
    20
#include <sstream>
kpeter@810
    21
kpeter@810
    22
#include <lemon/smart_graph.h>
kpeter@810
    23
#include <lemon/lgf_reader.h>
kpeter@810
    24
#include <lemon/min_mean_cycle.h>
kpeter@810
    25
#include <lemon/path.h>
kpeter@810
    26
#include <lemon/concepts/digraph.h>
kpeter@810
    27
#include <lemon/concept_check.h>
kpeter@810
    28
kpeter@810
    29
#include "test_tools.h"
kpeter@810
    30
kpeter@810
    31
using namespace lemon;
kpeter@810
    32
kpeter@810
    33
char test_lgf[] =
kpeter@810
    34
  "@nodes\n"
kpeter@810
    35
  "label\n"
kpeter@810
    36
  "1\n"
kpeter@810
    37
  "2\n"
kpeter@810
    38
  "3\n"
kpeter@810
    39
  "4\n"
kpeter@810
    40
  "5\n"
kpeter@810
    41
  "6\n"
kpeter@810
    42
  "7\n"
kpeter@810
    43
  "@arcs\n"
kpeter@810
    44
  "    len1 len2 len3 len4  c1 c2 c3 c4\n"
kpeter@810
    45
  "1 2    1    1    1    1   0  0  0  0\n"
kpeter@810
    46
  "2 4    5    5    5    5   1  0  0  0\n"
kpeter@810
    47
  "2 3    8    8    8    8   0  0  0  0\n"
kpeter@810
    48
  "3 2   -2    0    0    0   1  0  0  0\n"
kpeter@810
    49
  "3 4    4    4    4    4   0  0  0  0\n"
kpeter@810
    50
  "3 7   -4   -4   -4   -4   0  0  0  0\n"
kpeter@810
    51
  "4 1    2    2    2    2   0  0  0  0\n"
kpeter@810
    52
  "4 3    3    3    3    3   1  0  0  0\n"
kpeter@810
    53
  "4 4    3    3    0    0   0  0  1  0\n"
kpeter@810
    54
  "5 2    4    4    4    4   0  0  0  0\n"
kpeter@810
    55
  "5 6    3    3    3    3   0  1  0  0\n"
kpeter@810
    56
  "6 5    2    2    2    2   0  1  0  0\n"
kpeter@810
    57
  "6 4   -1   -1   -1   -1   0  0  0  0\n"
kpeter@810
    58
  "6 7    1    1    1    1   0  0  0  0\n"
kpeter@810
    59
  "7 7    4    4    4   -1   0  0  0  1\n";
kpeter@810
    60
kpeter@810
    61
                        
kpeter@810
    62
// Check the interface of an MMC algorithm
kpeter@810
    63
template <typename GR, typename Value>
kpeter@810
    64
struct MmcClassConcept
kpeter@810
    65
{
kpeter@810
    66
  template <typename MMC>
kpeter@810
    67
  struct Constraints {
kpeter@810
    68
    void constraints() {
kpeter@810
    69
      const Constraints& me = *this;
kpeter@810
    70
kpeter@810
    71
      typedef typename MMC
kpeter@810
    72
        ::template SetPath<ListPath<GR> >
kpeter@810
    73
        ::template SetLargeValue<Value>
kpeter@810
    74
        ::Create MmcAlg;
kpeter@810
    75
      MmcAlg mmc(me.g, me.length);
kpeter@810
    76
      const MmcAlg& const_mmc = mmc;
kpeter@810
    77
      
kpeter@810
    78
      b = mmc.cycle(p).run();
kpeter@810
    79
      b = mmc.findMinMean();
kpeter@810
    80
      b = mmc.findCycle();
kpeter@810
    81
kpeter@810
    82
      v = const_mmc.cycleLength();
kpeter@810
    83
      i = const_mmc.cycleArcNum();
kpeter@810
    84
      d = const_mmc.cycleMean();
kpeter@810
    85
      p = const_mmc.cycle();
kpeter@810
    86
    }
kpeter@810
    87
kpeter@810
    88
    typedef concepts::ReadMap<typename GR::Arc, Value> LM;
kpeter@810
    89
  
kpeter@810
    90
    GR g;
kpeter@810
    91
    LM length;
kpeter@810
    92
    ListPath<GR> p;
kpeter@810
    93
    Value v;
kpeter@810
    94
    int i;
kpeter@810
    95
    double d;
kpeter@810
    96
    bool b;
kpeter@810
    97
  };
kpeter@810
    98
};
kpeter@810
    99
kpeter@810
   100
// Perform a test with the given parameters
kpeter@810
   101
template <typename MMC>
kpeter@810
   102
void checkMmcAlg(const SmartDigraph& gr,
kpeter@810
   103
                 const SmartDigraph::ArcMap<int>& lm,
kpeter@810
   104
                 const SmartDigraph::ArcMap<int>& cm,
kpeter@810
   105
                 int length, int size) {
kpeter@810
   106
  MMC alg(gr, lm);
kpeter@810
   107
  alg.findMinMean();
kpeter@810
   108
  check(alg.cycleMean() == static_cast<double>(length) / size,
kpeter@810
   109
        "Wrong cycle mean");
kpeter@810
   110
  alg.findCycle();
kpeter@810
   111
  check(alg.cycleLength() == length && alg.cycleArcNum() == size,
kpeter@810
   112
        "Wrong path");
kpeter@810
   113
  SmartDigraph::ArcMap<int> cycle(gr, 0);
kpeter@810
   114
  for (typename MMC::Path::ArcIt a(alg.cycle()); a != INVALID; ++a) {
kpeter@810
   115
    ++cycle[a];
kpeter@810
   116
  }
kpeter@810
   117
  for (SmartDigraph::ArcIt a(gr); a != INVALID; ++a) {
kpeter@810
   118
    check(cm[a] == cycle[a], "Wrong path");
kpeter@810
   119
  }
kpeter@810
   120
}
kpeter@810
   121
kpeter@810
   122
// Class for comparing types
kpeter@810
   123
template <typename T1, typename T2>
kpeter@810
   124
struct IsSameType {
kpeter@810
   125
  static const int result = 0;
kpeter@810
   126
};
kpeter@810
   127
kpeter@810
   128
template <typename T>
kpeter@810
   129
struct IsSameType<T,T> {
kpeter@810
   130
  static const int result = 1;
kpeter@810
   131
};
kpeter@810
   132
kpeter@810
   133
kpeter@810
   134
int main() {
kpeter@810
   135
  #ifdef LEMON_HAVE_LONG_LONG
kpeter@810
   136
    typedef long long long_int;
kpeter@810
   137
  #else
kpeter@810
   138
    typedef long long_int;
kpeter@810
   139
  #endif
kpeter@810
   140
kpeter@810
   141
  // Check the interface
kpeter@810
   142
  {
kpeter@810
   143
    typedef concepts::Digraph GR;
kpeter@810
   144
    typedef MinMeanCycle<GR, concepts::ReadMap<GR::Arc, int> > IntMmcAlg;
kpeter@810
   145
    typedef MinMeanCycle<GR, concepts::ReadMap<GR::Arc, float> > FloatMmcAlg;
kpeter@810
   146
    
kpeter@810
   147
    checkConcept<MmcClassConcept<GR, int>, IntMmcAlg>();
kpeter@810
   148
    checkConcept<MmcClassConcept<GR, float>, FloatMmcAlg>();
kpeter@810
   149
  
kpeter@810
   150
    if (IsSameType<IntMmcAlg::LargeValue, long_int>::result == 0)
kpeter@810
   151
      check(false, "Wrong LargeValue type");
kpeter@810
   152
    if (IsSameType<FloatMmcAlg::LargeValue, double>::result == 0)
kpeter@810
   153
      check(false, "Wrong LargeValue type");
kpeter@810
   154
  }
kpeter@810
   155
kpeter@810
   156
  // Run various tests
kpeter@810
   157
  {
kpeter@810
   158
    typedef SmartDigraph GR;
kpeter@810
   159
    DIGRAPH_TYPEDEFS(GR);
kpeter@810
   160
    
kpeter@810
   161
    GR gr;
kpeter@810
   162
    IntArcMap l1(gr), l2(gr), l3(gr), l4(gr);
kpeter@810
   163
    IntArcMap c1(gr), c2(gr), c3(gr), c4(gr);
kpeter@810
   164
    
kpeter@810
   165
    std::istringstream input(test_lgf);
kpeter@810
   166
    digraphReader(gr, input).
kpeter@810
   167
      arcMap("len1", l1).
kpeter@810
   168
      arcMap("len2", l2).
kpeter@810
   169
      arcMap("len3", l3).
kpeter@810
   170
      arcMap("len4", l4).
kpeter@810
   171
      arcMap("c1", c1).
kpeter@810
   172
      arcMap("c2", c2).
kpeter@810
   173
      arcMap("c3", c3).
kpeter@810
   174
      arcMap("c4", c4).
kpeter@810
   175
      run();
kpeter@810
   176
kpeter@810
   177
    checkMmcAlg<MinMeanCycle<GR, IntArcMap> >(gr, l1, c1,  6, 3);
kpeter@810
   178
    checkMmcAlg<MinMeanCycle<GR, IntArcMap> >(gr, l2, c2,  5, 2);
kpeter@810
   179
    checkMmcAlg<MinMeanCycle<GR, IntArcMap> >(gr, l3, c3,  0, 1);
kpeter@810
   180
    checkMmcAlg<MinMeanCycle<GR, IntArcMap> >(gr, l4, c4, -1, 1);
kpeter@810
   181
  }
kpeter@810
   182
kpeter@810
   183
  return 0;
kpeter@810
   184
}