ismcsolver is a header-only C++14 library providing information set Monte Carlo tree search (ISMCTS) algorithms. Monte Carlo tree search is a (game) AI decision making algorithm noted for its applicability to many different games of perfect information, requiring no domain-specific knowledge and only little information about a game's state. ISMCTS is an elegant extension of this technique to imperfect information games, where not all information is visible to all players, possibly combined with factors of randomness.
Strengths of ISMCTS include:
- Being an anytime algorithm: it can be run as short or as long as desired, more iterations will simply increase the amount of information available for the decision, improving its strength;
- It makes efficient use memory and CPU time by only building a single information tree per search, in which a node merely represents an available move for a given player at a point in the game. This means that if the same move is available at this point in multiple different game states, it will map to the same node, making it easier to find moves that are likely to be good in a variety of circumstances;
- It lends itself well to parallel execution, which is widely available on modern systems and allows faster processing.
This code was originally part of a Qt game I wrote to explore game AI. I eventually wanted to test it with a simpler game, so I extracted it and also decided to port it to standard C++ to make it more useful any future projects. It now provides class templates that allow applying the algorithm to a generic game using a generic type of move.
This section and the next only give a few short examples of using the library. For a full reference to the class templates, please visit the GitHub Page of this repository.
First and foremost, your game (or engine) class should implement the abstract interface ISMCTS::Game<Move>
, replacing the template parameter Move
with the data type (or class) representing a player's move. The library provides two solver class templates, ISMCTS::SOSolver<Move>
and ISMCTS::MOSolver<Move>
. Their differences are explained below, but they should be instantiated with the same Move
type to obtain an object that can select moves for the AI player(s), e.g.
#include <ismcts/sosolver.h>
// ...
void MyGame::doAIMove()
{
ISMCTS::SOSolver<MyMove> solver;
this->doMove(solver(*this));
}
Several simple games are included in the test/common directory. Their main purpose is to test the compilation and functioning of the library code, but they also serve as example implementations of the ISMCTS::Game
interface.
The basic algorithm can be applied, modified and executed in different ways. This section lists what is currently implemented.
These are two variants of the algorithm, respectively Single-Observer and Multiple-Observer. The former is implemented in ISMCTS::SOSolver
and, as its name implies, observes the game from only one perspective: that of the player conducting the search. It only builds a tree for that player and treats all opponent moves as fully observable. While that is indeed the case in many games, some additionally feature actions that are hidden from the other players. MO-ISMCTS, implemented in ISMCTS::MOSolver
, is intended for such games; it builds a tree for each player to model the presence of this extra hidden information.
Several approaches to parallel tree searching exist, see e.g. Parallelization of Information Set Monte Carlo Tree Search by Nick Sephton and the authors of ISMCTS.
These are implemented using std::async
and can be used by providing the optional second ExecutionPolicy
parameter to either class template. For example, the following instantiates a RootParallel SOSolver
for some game where int
represents a move:
ISMCTS::SOSolver<int, ISMCTS::RootParallel> solver;
Three policies are defined in execution.h:
ISMCTS::Sequential
: no multithreading, the default;ISMCTS::RootParallel
: each system thread searches a separate tree structure. Statistics from the root of each tree are then combined to find the overall best move. This method is the fastest, as it avoids synchronisation issues and the overhead of combining results is minimal. The downside is that the individual trees are not searched as deeply, which negatively impacts the quality of the decision;ISMCTS::TreeParallel
: the threads share a single tree structure, combining the depth of a sequential search with improved speed. However, it is slower than root parallelisation, because threads will sometimes compete for access to the same node. The impact depends on the number of threads and characteristics of the game, though the tree will typically branch out quickly, mitigating the issue.
Each solver type has two constructors; one that sets the search operator to iterate a fixed number of times, the other instead letting it search for a fixed length of time (a std::chrono::duration<double>
). Both the mode of operation and the length of the search can be changed after instantiation. Durations can be conveniently created by including <chrono>
and using the std::chrono_literals
, e.g.:
#include <chrono>
// ...
using namespace std::chrono_literals;
ISMCTS::SOSolver<int> solver {5ms};
Due to the nature of header-only libraries, no installation is technically necessary. You can use the headers in three ways:
- Using any build system:
Simply copy the
include
folder to your project and configure your own build. In this case, you must ensure that build targets using a solver class are linked to an implementation of<thread>
, e.g. by specifying-lpthread
; - Using CMake:
- Clone this repository into a subdirectory of your project, e.g. ext/ismcsolver, and add it to your main CMakeLists.txt:
add_subdirectory(ext/ismcsolver)
- Install the headers to a system location and use
find_package()
:find_package(ismcsolver REQUIRED)
add_executable(foo foo.cpp) target_link_libraries(foo ismcsolver)
- Clone this repository into a subdirectory of your project, e.g. ext/ismcsolver, and add it to your main CMakeLists.txt:
Support for C++14 features is required to compile the library code; most reasonably recent compilers and development platforms provide this. The following combinations are automatically tested and should be regarded as minimum versions:
- Linux: GCC 7 and Clang 7;
- Apple Xcode 10.2;
- Microsoft Visual Studio 2017.
The type Move
specified for the game and sequential solver templates must be a TrivialType that is EqualityComparable. Solvers with root parallelisation use a std::map
to evaluate visit counts, so the Move
must additionally be LessThanComparable.
Game implementations will need a source of random numbers. Because std::rand
does not guarantee a good quality sequence and may not be thread safe, modern code should use the <random>
header. The objects representing generator engines are large (5000 bytes) and not practical as class members. A function like the following is the most convenient way to provide thread-safe random number generators:
#include <random>
// Definition as a free function, could also be a class member
std::mt19937 & prng()
{
std::mt19937 static thread_local prng {std::random_device{}()};
return prng;
}
This sets up one pseudorandom Mersenne Twister engine per thread, each seeded with a different random number from a non-deterministic system source if available. The object itself simply generates random unsigned 32 bit values and is more useful as a parameter to other algorithms and function objects:
// Randomly shuffle elements in a range:
std::shuffle(cards.begin(), cards.end(), prng());
// Generate a single random number on an interval:
std::uniform_int_distribution<> singleDie {1, 6};
auto result = singleDie(prng());
This project is licensed under the MIT License, see the LICENSE file for details.