#include <iostream>
#include <random>
#include <set>
#include <algorithm>
using namespace std;
random_device rd;
mt19937 gen(rd());
const int TL = 100, PROB_DE_OCORRENCIA = 2, PRIZE = 400;
int get_row_size(int N);
int get_col_size(int M);
vector<vector<int>> create_graph(int N, int M);
int get_evaluation(vector<vector<int>> graph, vector<pair<int, int>> solution);
vector<vector<pair<int, int>>> genetic(vector<vector<int>> graph, int N, int M);
vector<pair<int, int>> get_random_solution(int N, int M);
pair<vector<pair<int, int>>, vector<pair<int, int>>> choose_parents(vector<vector<pair<int, int>>> population);
vector<pair<int, int>> get_free_connections(vector<vector<int>> incomplete_graph, int N, int M);
pair<vector<pair<int, int>>, vector<pair<int, int>>> crossover(vector<vector<pair<int, int>>> parents, int N, int M);
vector<pair<int, int>> mutate(vector<pair<int, int>> solution, int N, int M);
int main() {
int N = 5, M = 5;
vector<vector<int>> graph = create_graph(N, M);
for(int i = 0, par = 2; i < N; ++i, par += 2) {
for(int j = 0, impar = 1; j < M; ++j, impar += 2) {
cout<<graph[par][impar]<<' ';
}
cout<<'\n';
}
vector<vector<pair<int, int>>> ans = genetic(graph, N, M);
int count = 0;
for(vector<pair<int, int>> v : ans) {
cout<<"Solution "<<++count<<':'<<'\n';
for(pair<int, int> vv : v) {
cout<<vv.first<<' '<<vv.second<<'\n';
}
cout<<'\n';
}
return 0;
}
int get_row_size(int N) {
return 3 * N + 5;
}
int get_col_size(int M) {
return 3 * M + 5;
}
vector<vector<int>> create_graph(int N, int M) {
vector<vector<int>> graph(get_row_size(N), vector<int>(get_col_size(M)));
uniform_int_distribution<> get_random(1, 100);
for(int i = 0, par = 2; i < N; ++i, par += 2) {
for(int j = 0, impar = 1; j < M; ++j, impar += 2) {
graph[par][impar] = get_random(gen);
}
}
return graph;
}
int get_evaluation(vector<vector<int>> graph, vector<pair<int, int>> solution) {
int somatorio_dos_pesos = 0;
set<int> vertices;
for(auto [vertice1, vertice2] : solution) {
somatorio_dos_pesos += graph[vertice1][vertice2];
vertices.emplace(vertice1), vertices.emplace(vertice2);
}
const int numero_de_vertices = vertices.size();
return numero_de_vertices * PRIZE - somatorio_dos_pesos;
}
vector<vector<pair<int, int>>> genetic(vector<vector<int>> graph, int N, int M) {
uniform_int_distribution<> get_random(1, 100);
const int MAX = 5 * max(N, M);
const int HALF = MAX / 2;
vector<vector<pair<int, int>>> generation;
for(int i = 0; i <= MAX; ++i) {
generation.push_back( get_random_solution(N, M) );
}
for(int i = 1, t = 1; t < TL; ++i, ++t) {
for(int j = 1; j <= HALF; ++j) {
const auto [parent1, parent2] = choose_parents( generation );
const vector<vector<pair<int, int>>> parents = {parent1, parent2};
auto [child1, child2] = crossover(parents, N, M);
// auto [child1, child2] = pair<vector<pair<int, int>>, vector<pair<int, int>>>{parents[0], parents[1]};
if(get_random(gen) < PROB_DE_OCORRENCIA) {
child1 = mutate(child1, N, M);
}
if(get_random(gen) < PROB_DE_OCORRENCIA) {
child2 = mutate(child2, N, M);
}
generation.push_back(child1);
generation.push_back(child2);
}
const auto sort_comparator = [&](vector<pair<int, int>> a, vector<pair<int, int>> b) -> bool {
return get_evaluation(graph, a) > get_evaluation(graph, b);
};
sort(generation.begin(), generation.end(), sort_comparator);
generation.resize( MAX );
}
return generation;
}
vector<pair<int, int>> get_random_solution(int N, int M) {
vector<pair<int, int>> solution;
vector<int> free_verticesN, free_verticesM, vertex_used(max(get_row_size(N), get_col_size(M)), 0);
for(int i = 1, val = 2; i < N; ++i, val += 2) {
free_verticesN.push_back(val);
}
for(int i = 0, val = 1; i < M; ++i, val += 2) {
free_verticesM.push_back(val);
}
for(int i = 1; i <= min(N, M); ++i) {
uniform_int_distribution<> N_get_random(0, free_verticesN.size()-1);
uniform_int_distribution<> M_get_random(0, free_verticesM.size()-1);
const int random_index_n = N_get_random(gen);
const int random_index_m = M_get_random(gen);
const int vertex1 = free_verticesN[random_index_n];
const int vertex2 = free_verticesM[random_index_m];
solution.push_back({ vertex1, vertex2 });
vertex_used[vertex1]++;
vertex_used[vertex2]++;
if(vertex_used[vertex1] == 2) free_verticesN.erase(free_verticesN.begin() + random_index_n);
if(vertex_used[vertex2] == 2) free_verticesM.erase(free_verticesM.begin() + random_index_m);
}
return solution;
}
pair<vector<pair<int, int>>, vector<pair<int, int>>> choose_parents(vector<vector<pair<int, int>>> population) {
uniform_int_distribution<> get_random(0, population.size() - 1);
const int rand_parent1 = get_random(gen);
const int rand_parent2 = get_random(gen);
return { population[rand_parent1], population[rand_parent2] };
}
vector<pair<int, int>> get_free_connections(vector<vector<int>> incomplete_graph, int N, int M) {
vector<pair<int, int>> free_connections;
for(int i = 0, par = 2; i < N; ++i, par += 2) {
for(int j = 0, impar = 1; j < M; ++j, impar += 2) {
if(incomplete_graph[i][j] == false) {
free_connections.emplace_back(par, impar);
}
}
}
return free_connections;
}
pair<vector<pair<int, int>>, vector<pair<int, int>>> crossover(vector<vector<pair<int, int>>> parents, int N, int M) {
//// Child 1 = Half(Parent1) + Half2(Parent2)
//// Child 2 = Half(Parent2) + Half2(Parent1)
// probabilidade de troca = 50%;
uniform_int_distribution<> get_random(0, 1);
const int swap_ = get_random(gen);
if(swap_) {
swap(parents[0], parents[1]);
}
vector<vector<vector<int>>> graph(2, vector<vector<int>>(get_row_size(N), vector<int>(get_col_size(M), false)));
vector<vector<pair<int, int>>> connections(2);
int num_of_connections[2] = {0};
const int HALF = min(N, M) / 2;
for(int i = 0; i < HALF; ++i) {
for(int parent = 0; parent < 2; ++parent) {
const auto [vertex1, vertex2] = parents[parent][i];
const int child = parent;
graph[child][vertex1][vertex2] = true;
connections[child].emplace_back(vertex1, vertex2);
++num_of_connections[child];
}
}
for(int i = HALF; i < min(N, M); ++i) {
for(int parent = 0; parent < 2; ++parent) {
const auto [vertex1, vertex2] = parents[parent][i];
const int child = parent;
if(graph[child][vertex1][vertex2] == false) {
graph[child][vertex1][vertex2] = true;
connections[child].emplace_back(vertex1, vertex2);
++num_of_connections[child];
}
}
}
vector<pair<int, int>> free_connections[2];
free_connections[0] = get_free_connections(graph[0], N, M);
free_connections[1] = get_free_connections(graph[1], N, M);
for(int child = 0; child < 2; ++child) {
vector<int> vertex_used(max(get_row_size(N), get_col_size(M)));
while(num_of_connections[child] < min(N, M)) {
uniform_int_distribution<> get_random_index(0, free_connections[child].size()-1);
int random_index, vertex1, vertex2;
bool valid_edge = false;
while(valid_edge == false) {
random_index = get_random_index(gen);
pair<int, int> vertices = free_connections[child][random_index];
vertex1 = vertices.first, vertex2 = vertices.second;
if(vertex_used[vertex1] == false && vertex_used[vertex2] == false) {
connections[child].emplace_back(free_connections[child][random_index]);
valid_edge = true;
++num_of_connections[child];
}
free_connections[child].erase(free_connections[child].begin() + random_index);
}
}
}
return pair<vector<pair<int, int>>, vector<pair<int, int>>>(connections[0], connections[1]);
}
vector<pair<int, int>> mutate(vector<pair<int, int>> solution, int N, int M) {
vector<vector<int>> graph(get_row_size(N), vector<int>(get_col_size(M), false));
for(auto [vertex1, vertex2] : solution) {
graph[vertex1][vertex2] = true;
}
vector<pair<int, int>> free_edges = get_free_connections(graph, N, M);
uniform_int_distribution<> get_random_index_of_used_edges(0, solution.size()-1);
uniform_int_distribution<> get_random_index_of_free_edges(0, free_edges.size()-1);
const int random_index_of_used_edges = get_random_index_of_used_edges(gen);
const int random_index_of_free_edges = get_random_index_of_free_edges(gen);
solution.erase(solution.begin() + random_index_of_used_edges);
solution.emplace_back(free_edges[random_index_of_free_edges]);
return solution;
}
#include <iostream>
#include <random>
#include <set>
#include <algorithm>

using namespace std;

random_device rd;
mt19937 gen(rd());

const int TL = 100, PROB_DE_OCORRENCIA = 2, PRIZE = 400;

int get_row_size(int N);
int get_col_size(int M);
vector<vector<int>> create_graph(int N, int M);
int get_evaluation(vector<vector<int>> graph, vector<pair<int, int>> solution);
vector<vector<pair<int, int>>> genetic(vector<vector<int>> graph, int N, int M);
vector<pair<int, int>> get_random_solution(int N, int M);
pair<vector<pair<int, int>>, vector<pair<int, int>>> choose_parents(vector<vector<pair<int, int>>> population);
vector<pair<int, int>> get_free_connections(vector<vector<int>> incomplete_graph, int N, int M);
pair<vector<pair<int, int>>, vector<pair<int, int>>> crossover(vector<vector<pair<int, int>>> parents, int N, int M);
vector<pair<int, int>> mutate(vector<pair<int, int>> solution, int N, int M);

int main() {
  int N = 5, M = 5;

  vector<vector<int>> graph = create_graph(N, M);
  for(int i = 0, par = 2; i < N; ++i, par += 2) {
    for(int j = 0, impar = 1; j < M; ++j, impar += 2) {
      cout<<graph[par][impar]<<' ';
    }
    cout<<'\n';
  }
  vector<vector<pair<int, int>>> ans = genetic(graph, N, M);
  int count = 0;
  for(vector<pair<int, int>> v : ans) {
    cout<<"Solution "<<++count<<':'<<'\n';
    for(pair<int, int> vv : v) {
      cout<<vv.first<<' '<<vv.second<<'\n';
    }
    cout<<'\n';
  }

  return 0;
}

int get_row_size(int N) {
  return 3 * N + 5;
}

int get_col_size(int M) {
  return 3 * M + 5;
}

vector<vector<int>> create_graph(int N, int M) {
  vector<vector<int>> graph(get_row_size(N), vector<int>(get_col_size(M)));
  uniform_int_distribution<> get_random(1, 100);
  for(int i = 0, par = 2; i < N; ++i, par += 2) {
    for(int j = 0, impar = 1; j < M; ++j, impar += 2) {
      graph[par][impar] = get_random(gen);
    }
  }
  return graph;
}

int get_evaluation(vector<vector<int>> graph, vector<pair<int, int>> solution) {
  int somatorio_dos_pesos = 0;
  set<int> vertices;
  for(auto [vertice1, vertice2] : solution) {
    somatorio_dos_pesos += graph[vertice1][vertice2];
    vertices.emplace(vertice1), vertices.emplace(vertice2);
  }
  const int numero_de_vertices = vertices.size();
  return numero_de_vertices * PRIZE - somatorio_dos_pesos;
}

vector<vector<pair<int, int>>> genetic(vector<vector<int>> graph, int N, int M) {
  uniform_int_distribution<> get_random(1, 100);
  const int MAX = 5 * max(N, M);
  const int HALF = MAX / 2;

  vector<vector<pair<int, int>>> generation;
  for(int i = 0; i <= MAX; ++i) {
    generation.push_back( get_random_solution(N, M) );
  }
  for(int i = 1, t = 1; t < TL; ++i, ++t) {
    for(int j = 1; j <= HALF; ++j) {
      const auto [parent1, parent2] = choose_parents( generation );
      const vector<vector<pair<int, int>>> parents = {parent1, parent2};
      auto [child1, child2] = crossover(parents, N, M);
      // auto [child1, child2] = pair<vector<pair<int, int>>, vector<pair<int, int>>>{parents[0], parents[1]};

      if(get_random(gen) < PROB_DE_OCORRENCIA) {
        child1 = mutate(child1, N, M);
      }
      if(get_random(gen) < PROB_DE_OCORRENCIA) {
        child2 = mutate(child2, N, M);
      }
      generation.push_back(child1);
      generation.push_back(child2);
    }

    const auto sort_comparator = [&](vector<pair<int, int>> a, vector<pair<int, int>> b) -> bool {
      return get_evaluation(graph, a) > get_evaluation(graph, b);
    };

    sort(generation.begin(), generation.end(), sort_comparator);
    
    generation.resize( MAX );
  }
  return generation;
}

vector<pair<int, int>> get_random_solution(int N, int M) {
  vector<pair<int, int>> solution;
  vector<int> free_verticesN, free_verticesM, vertex_used(max(get_row_size(N), get_col_size(M)), 0);

  for(int i = 1, val = 2; i < N; ++i, val += 2) {
    free_verticesN.push_back(val);
  }
  for(int i = 0, val = 1; i < M; ++i, val += 2) {
    free_verticesM.push_back(val);
  }

  for(int i = 1; i <= min(N, M); ++i) {
    uniform_int_distribution<> N_get_random(0, free_verticesN.size()-1);
    uniform_int_distribution<> M_get_random(0, free_verticesM.size()-1);
    const int random_index_n = N_get_random(gen);
    const int random_index_m = M_get_random(gen);

    const int vertex1 = free_verticesN[random_index_n];
    const int vertex2 = free_verticesM[random_index_m];

    solution.push_back({ vertex1, vertex2 });

    vertex_used[vertex1]++;
    vertex_used[vertex2]++;
    if(vertex_used[vertex1] == 2) free_verticesN.erase(free_verticesN.begin() + random_index_n);
    if(vertex_used[vertex2] == 2) free_verticesM.erase(free_verticesM.begin() + random_index_m);
  }
  return solution;
}

pair<vector<pair<int, int>>, vector<pair<int, int>>> choose_parents(vector<vector<pair<int, int>>> population) {
  uniform_int_distribution<> get_random(0, population.size() - 1);
  const int rand_parent1 = get_random(gen);
  const int rand_parent2 = get_random(gen);
  return { population[rand_parent1], population[rand_parent2] };
}

vector<pair<int, int>> get_free_connections(vector<vector<int>> incomplete_graph, int N, int M) {

  vector<pair<int, int>> free_connections;

  for(int i = 0, par = 2; i < N; ++i, par += 2) {
    for(int j = 0, impar = 1; j < M; ++j, impar += 2) {
      if(incomplete_graph[i][j] == false) {
        free_connections.emplace_back(par, impar);
      }
    }
  }

  return free_connections;
}

pair<vector<pair<int, int>>, vector<pair<int, int>>> crossover(vector<vector<pair<int, int>>> parents, int N, int M) {
  //// Child 1 = Half(Parent1) + Half2(Parent2)
  //// Child 2 = Half(Parent2) + Half2(Parent1)

  // probabilidade de troca = 50%;
  uniform_int_distribution<> get_random(0, 1);
  const int swap_ = get_random(gen);
  if(swap_) {
    swap(parents[0], parents[1]);
  }

  vector<vector<vector<int>>> graph(2, vector<vector<int>>(get_row_size(N), vector<int>(get_col_size(M), false)));
  vector<vector<pair<int, int>>> connections(2);
  int num_of_connections[2] = {0};
  const int HALF = min(N, M) / 2;

  for(int i = 0; i < HALF; ++i) {
    for(int parent = 0; parent < 2; ++parent) {
      const auto [vertex1, vertex2] = parents[parent][i];
      const int child = parent;
      graph[child][vertex1][vertex2] = true;
      connections[child].emplace_back(vertex1, vertex2);
      ++num_of_connections[child];
    }
  }
  for(int i = HALF; i < min(N, M); ++i) {
    for(int parent = 0; parent < 2; ++parent) {
      const auto [vertex1, vertex2] = parents[parent][i];
      const int child = parent;
      if(graph[child][vertex1][vertex2] == false) {
        graph[child][vertex1][vertex2] = true;
        connections[child].emplace_back(vertex1, vertex2);
        ++num_of_connections[child];
      }
    }
  }
  
  vector<pair<int, int>> free_connections[2];
  free_connections[0] = get_free_connections(graph[0], N, M);
  free_connections[1] = get_free_connections(graph[1], N, M);

  for(int child = 0; child < 2; ++child) {
    vector<int> vertex_used(max(get_row_size(N), get_col_size(M)));
    while(num_of_connections[child] < min(N, M)) {
      uniform_int_distribution<> get_random_index(0, free_connections[child].size()-1);
      int random_index, vertex1, vertex2;
      bool valid_edge = false;
      while(valid_edge == false) {
        random_index = get_random_index(gen);
        pair<int, int> vertices = free_connections[child][random_index];
        vertex1 = vertices.first, vertex2 = vertices.second;
        if(vertex_used[vertex1] == false && vertex_used[vertex2] == false) {
          connections[child].emplace_back(free_connections[child][random_index]);
          valid_edge = true;
          ++num_of_connections[child];
        }
        free_connections[child].erase(free_connections[child].begin() + random_index);
      }
    }
  }

  return pair<vector<pair<int, int>>, vector<pair<int, int>>>(connections[0], connections[1]);
}

vector<pair<int, int>> mutate(vector<pair<int, int>> solution, int N, int M) {
  vector<vector<int>> graph(get_row_size(N), vector<int>(get_col_size(M), false));
  for(auto [vertex1, vertex2] : solution) {
    graph[vertex1][vertex2] = true;
  }
  vector<pair<int, int>> free_edges = get_free_connections(graph, N, M);
  
  uniform_int_distribution<> get_random_index_of_used_edges(0, solution.size()-1);
  uniform_int_distribution<> get_random_index_of_free_edges(0, free_edges.size()-1);
  const int random_index_of_used_edges = get_random_index_of_used_edges(gen);
  const int random_index_of_free_edges = get_random_index_of_free_edges(gen);
  solution.erase(solution.begin() + random_index_of_used_edges);
  solution.emplace_back(free_edges[random_index_of_free_edges]);
  return solution;
}
