X-Git-Url: http://git.scottworley.com/slidingtile/blobdiff_plain/1955dd8bceb1bd5beef449ff1ac04c35c2714f31..refs/heads/master:/sliding_tile_lib.cc diff --git a/sliding_tile_lib.cc b/sliding_tile_lib.cc index 51aeded..83d6297 100644 --- a/sliding_tile_lib.cc +++ b/sliding_tile_lib.cc @@ -1,13 +1,16 @@ #include "sliding_tile_lib.h" #include +#include #include +#include #include #include #include #include #include #include +#include #include #include @@ -133,8 +136,10 @@ int Board::distance(const Board& o) const { int Board::distance(const InvertedBoard& invo) const { int dist = 0; for (int i = 0; i < BOARD_SIZE; i++) { - dist += std::abs(i % BOARD_DIM - invo.pos[board[i]] % BOARD_DIM) + - std::abs(i / BOARD_DIM - invo.pos[board[i]] / BOARD_DIM); + if (board[i] != 0) { + dist += std::abs(i % BOARD_DIM - invo.pos[board[i]] % BOARD_DIM) + + std::abs(i / BOARD_DIM - invo.pos[board[i]] / BOARD_DIM); + } } return dist; } @@ -186,7 +191,7 @@ std::ostream& operator<<(std::ostream& os, const Step& step) { return os; } -std::shared_ptr find_path(const std::string& start, const std::string& goal) { +std::shared_ptr find_path(const std::string& start, const std::string& goal, unsigned max_frontier) { std::istringstream iss_start{start}, iss_goal{goal}; Board board_start, board_goal; iss_start >> board_start; @@ -197,39 +202,49 @@ std::shared_ptr find_path(const std::string& start, const std::string& goa if (iss_goal.fail() || !iss_goal.eof()) { throw std::runtime_error("Could not parse the goal board: " + goal); } - return find_path(board_start, board_goal); + return find_path(board_start, board_goal, max_frontier); } -std::shared_ptr find_path(const Board& start, const Board& goal) { +static void show_memory_stats() { + std::ifstream statm{"/proc/self/statm"}; + if (statm.is_open()) { + std::string statm_data; + std::getline(statm, statm_data); + std::cerr << "Memory stats: " << statm_data << std::endl; + } +} + +std::shared_ptr find_path(const Board& start, const Board& goal, unsigned max_frontier) { InvertedBoard invgoal = goal.invert(); - auto heap_greater = [invgoal](const std::shared_ptr& a, const std::shared_ptr& b) { - return a->cost(invgoal) > b->cost(invgoal); - }; - std::priority_queue, - std::vector>, - decltype(heap_greater)> todo(heap_greater); + std::multimap> todo; std::set seen; - seen.emplace(start); - todo.push(std::make_shared(start, nullptr)); + seen.insert(start); + auto start_step = std::make_shared(start, nullptr); + todo.emplace(start_step->cost(invgoal), start_step); while (!todo.empty()) { - if (todo.top()->board == goal) { - return todo.top(); + auto cur = todo.begin()->second; + todo.erase(todo.begin()); + if (cur->board == goal) { + show_memory_stats(); + return cur; } - std::vector> successors = todo.top()->successors(todo.top()); + std::vector> successors = cur->successors(cur); for (const std::shared_ptr& s : successors) { if (seen.find(s->board) == seen.end()) { - seen.emplace(s->board); - todo.push(s); + seen.insert(s->board); + todo.emplace(s->cost(invgoal), s); if (seen.size() % 10000 == 0) { std::cerr << "Examined " << seen.size() << " boards. Tracking " << Step::count << " steps. " << todo.size() << " waiting. Considering paths of length " - << todo.top()->cost(invgoal) << std::endl; + << cur->cost(invgoal) << std::endl; } } } - todo.pop(); + while (todo.size() > max_frontier) { + todo.erase(--todo.end()); + } } throw std::runtime_error("No path from start to goal"); }