Location: LEMON/LEMON-main/test/max_clique_test.cc

Load file history
gravatar
kpeter (Peter Kovacs)
Various search limits for the max clique alg (#405)
/* -*- mode: C++; indent-tabs-mode: nil; -*-
*
* This file is a part of LEMON, a generic C++ optimization library.
*
* Copyright (C) 2003-2010
* Egervary Jeno Kombinatorikus Optimalizalasi Kutatocsoport
* (Egervary Research Group on Combinatorial Optimization, EGRES).
*
* Permission to use, modify and distribute this software is granted
* provided that this copyright notice appears in all copies. For
* precise terms see the accompanying LICENSE file.
*
* This software is provided "AS IS" with no warranty of any kind,
* express or implied, and with no claim as to its suitability for any
* purpose.
*
*/
#include <sstream>
#include <lemon/list_graph.h>
#include <lemon/full_graph.h>
#include <lemon/grid_graph.h>
#include <lemon/lgf_reader.h>
#include <lemon/grosso_locatelli_pullan_mc.h>
#include "test_tools.h"
using namespace lemon;
char test_lgf[] =
"@nodes\n"
"label max_clique\n"
"1 0\n"
"2 0\n"
"3 0\n"
"4 1\n"
"5 1\n"
"6 1\n"
"7 1\n"
"@edges\n"
" label\n"
"1 2 1\n"
"1 3 2\n"
"1 4 3\n"
"1 6 4\n"
"2 3 5\n"
"2 5 6\n"
"2 7 7\n"
"3 4 8\n"
"3 5 9\n"
"4 5 10\n"
"4 6 11\n"
"4 7 12\n"
"5 6 13\n"
"5 7 14\n"
"6 7 15\n";
// Check with general graphs
template <typename Param>
void checkMaxCliqueGeneral(Param rule) {
typedef ListGraph GR;
typedef GrossoLocatelliPullanMc<GR> McAlg;
typedef McAlg::CliqueNodeIt CliqueIt;
// Basic tests
{
GR g;
GR::NodeMap<bool> map(g);
McAlg mc(g);
mc.iterationLimit(50);
check(mc.run(rule) == McAlg::SIZE_LIMIT, "Wrong termination cause");
check(mc.cliqueSize() == 0, "Wrong clique size");
check(CliqueIt(mc) == INVALID, "Wrong CliqueNodeIt");
GR::Node u = g.addNode();
check(mc.run(rule) == McAlg::SIZE_LIMIT, "Wrong termination cause");
check(mc.cliqueSize() == 1, "Wrong clique size");
mc.cliqueMap(map);
check(map[u], "Wrong clique map");
CliqueIt it1(mc);
check(static_cast<GR::Node>(it1) == u && ++it1 == INVALID,
"Wrong CliqueNodeIt");
GR::Node v = g.addNode();
check(mc.run(rule) == McAlg::ITERATION_LIMIT, "Wrong termination cause");
check(mc.cliqueSize() == 1, "Wrong clique size");
mc.cliqueMap(map);
check((map[u] && !map[v]) || (map[v] && !map[u]), "Wrong clique map");
CliqueIt it2(mc);
check(it2 != INVALID && ++it2 == INVALID, "Wrong CliqueNodeIt");
g.addEdge(u, v);
check(mc.run(rule) == McAlg::SIZE_LIMIT, "Wrong termination cause");
check(mc.cliqueSize() == 2, "Wrong clique size");
mc.cliqueMap(map);
check(map[u] && map[v], "Wrong clique map");
CliqueIt it3(mc);
check(it3 != INVALID && ++it3 != INVALID && ++it3 == INVALID,
"Wrong CliqueNodeIt");
}
// Test graph
{
GR g;
GR::NodeMap<bool> max_clique(g);
GR::NodeMap<bool> map(g);
std::istringstream input(test_lgf);
graphReader(g, input)
.nodeMap("max_clique", max_clique)
.run();
McAlg mc(g);
mc.iterationLimit(50);
check(mc.run(rule) == McAlg::ITERATION_LIMIT, "Wrong termination cause");
check(mc.cliqueSize() == 4, "Wrong clique size");
mc.cliqueMap(map);
for (GR::NodeIt n(g); n != INVALID; ++n) {
check(map[n] == max_clique[n], "Wrong clique map");
}
int cnt = 0;
for (CliqueIt n(mc); n != INVALID; ++n) {
cnt++;
check(map[n] && max_clique[n], "Wrong CliqueNodeIt");
}
check(cnt == 4, "Wrong CliqueNodeIt");
}
}
// Check with full graphs
template <typename Param>
void checkMaxCliqueFullGraph(Param rule) {
typedef FullGraph GR;
typedef GrossoLocatelliPullanMc<FullGraph> McAlg;
typedef McAlg::CliqueNodeIt CliqueIt;
for (int size = 0; size <= 40; size = size * 3 + 1) {
GR g(size);
GR::NodeMap<bool> map(g);
McAlg mc(g);
check(mc.run(rule) == McAlg::SIZE_LIMIT, "Wrong termination cause");
check(mc.cliqueSize() == size, "Wrong clique size");
mc.cliqueMap(map);
for (GR::NodeIt n(g); n != INVALID; ++n) {
check(map[n], "Wrong clique map");
}
int cnt = 0;
for (CliqueIt n(mc); n != INVALID; ++n) cnt++;
check(cnt == size, "Wrong CliqueNodeIt");
}
}
// Check with grid graphs
template <typename Param>
void checkMaxCliqueGridGraph(Param rule) {
GridGraph g(5, 7);
GridGraph::NodeMap<char> map(g);
GrossoLocatelliPullanMc<GridGraph> mc(g);
mc.iterationLimit(100);
check(mc.run(rule) == mc.ITERATION_LIMIT, "Wrong termination cause");
check(mc.cliqueSize() == 2, "Wrong clique size");
mc.stepLimit(100);
check(mc.run(rule) == mc.STEP_LIMIT, "Wrong termination cause");
check(mc.cliqueSize() == 2, "Wrong clique size");
mc.sizeLimit(2);
check(mc.run(rule) == mc.SIZE_LIMIT, "Wrong termination cause");
check(mc.cliqueSize() == 2, "Wrong clique size");
}
int main() {
checkMaxCliqueGeneral(GrossoLocatelliPullanMc<ListGraph>::RANDOM);
checkMaxCliqueGeneral(GrossoLocatelliPullanMc<ListGraph>::DEGREE_BASED);
checkMaxCliqueGeneral(GrossoLocatelliPullanMc<ListGraph>::PENALTY_BASED);
checkMaxCliqueFullGraph(GrossoLocatelliPullanMc<FullGraph>::RANDOM);
checkMaxCliqueFullGraph(GrossoLocatelliPullanMc<FullGraph>::DEGREE_BASED);
checkMaxCliqueFullGraph(GrossoLocatelliPullanMc<FullGraph>::PENALTY_BASED);
checkMaxCliqueGridGraph(GrossoLocatelliPullanMc<GridGraph>::RANDOM);
checkMaxCliqueGridGraph(GrossoLocatelliPullanMc<GridGraph>::DEGREE_BASED);
checkMaxCliqueGridGraph(GrossoLocatelliPullanMc<GridGraph>::PENALTY_BASED);
return 0;
}