Game.cpp 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388
  1. #include <iostream>
  2. #include <climits>
  3. #include <chrono>
  4. #include "Game.h"
  5. #include "Types.h"
  6. std::array<Vector, 8> Game::neighbours;
  7. std::array<Vector, 9 * 5> Game::fieldVectors;
  8. long int Game::selectionTime = 0;
  9. long int Game::moveTime = 0;
  10. int Game::turns = 0;
  11. int Game::moves = 0;
  12. Game::Game() : active(-1, -1), mustTake(false) {
  13. // sets the basic game up:
  14. // 0 - start game
  15. // 1 - player vs ai
  16. // 0 - player starts
  17. std::cout << "0\n1\n0\n";
  18. // allows for each neighbor iterating
  19. neighbours[0].set(1, 0);
  20. neighbours[1].set(0, 1);
  21. neighbours[2].set(-1, 0);
  22. neighbours[3].set(0, -1);
  23. neighbours[4].set(1, 1);
  24. neighbours[5].set(-1, 1);
  25. neighbours[6].set(1, -1);
  26. neighbours[7].set(-1, -1);
  27. // allows for each for all fields
  28. uint index = 0;
  29. for(uint y = 0; y < 5; y++) {
  30. for(uint x = 0; x < 9; x++) {
  31. fieldVectors[index++].set(x, y);
  32. }
  33. }
  34. }
  35. void Game::readLine() {
  36. String line;
  37. while(true) {
  38. // read input by each character
  39. char c = std::cin.get();
  40. if(c == '\n') {
  41. // parse the command if the line ended
  42. if(parseLine(line)) {
  43. return;
  44. }
  45. // clear the buffer for the next command
  46. line.clear();
  47. } else if(c != '\r') {
  48. // line carriages are ignored
  49. line.append(c);
  50. }
  51. }
  52. }
  53. bool Game::parseLine(const String& line) {
  54. // identify when the server expects an answer or a new game field is available
  55. if(line == " 0 1 2 3 4 5 6 7 8") {
  56. fields.read();
  57. fields.print(active);
  58. } else if(line == "Please enter origin x-axis") {
  59. makeSelection();
  60. } else if(line == "Please enter destination x-axis") {
  61. makeMove();
  62. } else if(line == "Please enter wether you want to Withdraw or Approach [W/A]") {
  63. takeStone();
  64. } else if(line == "Do you want to continue with your turn [Y/N]?") {
  65. // we always want to continue if possible
  66. std::cout << "Y\n";
  67. } else if(line == "************************Player 1 won!**********************" ||
  68. line == "************************Player 2 won!**********************") {
  69. // stop the game if anybody wins - the server does not do that
  70. std::cerr << line << "\n";
  71. //std::cerr << "Turns: " << turns << "\n";
  72. //std::cerr << "Moves: " << moves << "\n";
  73. //std::cerr << "Total selection time: " << selectionTime << "\n";
  74. //std::cerr << "Total move time: " << moveTime << "\n";
  75. std::cerr << turns << "," << moves << "," << selectionTime << "," << moveTime << "\n";
  76. return true;
  77. }
  78. return false;
  79. }
  80. bool Game::isInRange(const Vector& v) const {
  81. return v.x >= 0 && v.x <= 8 && v.y >= 0 && v.y <= 4;
  82. }
  83. bool Game::areNeighbors(const Vector& from, const Vector& to) const {
  84. // fields are not neighbors with themselves
  85. if(from == to) {
  86. return false;
  87. }
  88. Vector diff = from - to;
  89. // fields with a bigger distance than one cannot be neighbors
  90. if(diff.x < -1 || diff.x > 1 || diff.y < -1 || diff.y > 1) {
  91. return false;
  92. }
  93. // either left, right, upper or lower neighbor
  94. if(diff.x == 0 || diff.y == 0) {
  95. return true;
  96. }
  97. // check if the field has diagonal connections (both coords are either even or uneven numbers)
  98. return (from.x & 1) == (from.y & 1);
  99. }
  100. uint Game::removeLine(const Vector& from, const Vector& to, Fields::State state) {
  101. // walk along the line and removes opponent stones
  102. Vector current = to;
  103. Vector unit = to - from;
  104. // counter for removed stones
  105. uint rank = 0;
  106. while(true) {
  107. current += unit;
  108. // stop removing on wrong stones or the end of the game field
  109. if(!isInRange(current) || !fields.hasState(current, state)) {
  110. return rank;
  111. }
  112. fields.setState(current, Fields::EMPTY);
  113. rank++;
  114. }
  115. }
  116. uint Game::getRank(const Vector& from, const Vector& to, Fields::State state) const {
  117. // same as removeLine but the removable stones are only counted
  118. Vector current = to;
  119. Vector unit = to - from;
  120. uint rank = 0;
  121. while(true) {
  122. current += unit;
  123. if(!isInRange(current) || !fields.hasState(current, state)) {
  124. return rank;
  125. }
  126. rank++;
  127. }
  128. }
  129. uint Game::getStoneTakes(const Vector& from, const Vector& to) const {
  130. // an invalid turn cannot take any stones
  131. if(!isInRange(to) || !fields.hasState(to, Fields::EMPTY) || !areNeighbors(from, to) ||
  132. lastLocations.contains(to) || (to - from) == direction) {
  133. return 0;
  134. }
  135. // the turn must take stones
  136. Fields::State enemy = (fields.hasState(from, Fields::BLACK) ? Fields::WHITE : Fields::BLACK);
  137. if(getRank(from, to, enemy) == 0 && getRank(to, from, enemy) == 0) {
  138. return 0;
  139. }
  140. // actually do the turn on a copied game, store the rank (= taken stones)
  141. Game copy = *this;
  142. uint rank = copy.move(from, to);
  143. // recursively check all neighbors on the copied game for another move, take the best move
  144. uint maxRank = 0;
  145. for(Vector& m : neighbours) {
  146. uint rank = copy.getStoneTakes(to, to + m);
  147. if(rank > maxRank) {
  148. maxRank = rank;
  149. }
  150. }
  151. return rank + maxRank;
  152. }
  153. uint Game::getStoneFreedom(const Vector& from, Fields::State state) const {
  154. // no freedom if the stone is not yours
  155. if(!fields.hasState(from, state)) {
  156. return 0;
  157. }
  158. // count directions the stone can move to
  159. uint counter = 0;
  160. for(const Vector& m : neighbours) {
  161. Vector to = from + m;
  162. if(isInRange(to) && areNeighbors(from, to) && fields.hasState(to, Fields::EMPTY)) {
  163. counter++;
  164. }
  165. }
  166. return counter;
  167. }
  168. uint Game::getFreedom(Fields::State state) const {
  169. uint sum = 0;
  170. for(const Vector& v : fieldVectors) {
  171. sum += getStoneFreedom(v, state);
  172. }
  173. return sum;
  174. }
  175. uint Game::move(const Vector& from, const Vector& to) {
  176. // remember direction to block the same move
  177. direction = to - from;
  178. // remember the old location to block moving to it again
  179. lastLocations.add(from);
  180. // actually from the stone
  181. fields.setState(to, fields.getState(from));
  182. fields.setState(from, Fields::EMPTY);
  183. // get the enemy by inverting the goal state
  184. Fields::State enemy = (fields.hasState(to, Fields::BLACK) ? Fields::WHITE : Fields::BLACK);
  185. // compare quality of the two sides, take better side
  186. uint rankA = getRank(from, to, enemy);
  187. uint rankB = getRank(to, from, enemy);
  188. if(rankA >= rankB) {
  189. return removeLine(from, to, enemy);
  190. }
  191. return removeLine(to, from, enemy);
  192. }
  193. void Game::makeSelection() {
  194. // add statistics
  195. ssize_t time = getNanos();
  196. turns++;
  197. // new turn starts - reset previously blocked turns
  198. lastLocations.clear();
  199. direction.set(0, 0);
  200. // remember if the following turn must take a stone
  201. mustTake = isTakingPossible();
  202. // calculate rank for all possible selections, take best selection
  203. Vector selection(0, 0);
  204. int maxRank = INT_MIN;
  205. for(Vector& to : fieldVectors) {
  206. if(!fields.hasState(to, Fields::EMPTY)) {
  207. continue;
  208. }
  209. // scan neighbors of empty fields for white stones
  210. for(Vector& m : neighbours) {
  211. Vector from = to + m;
  212. // prevent invalid stone locations
  213. if(!isInRange(from) || !fields.hasState(from, Fields::WHITE) || !areNeighbors(from, to)) {
  214. continue;
  215. }
  216. // prevent non taking turns if taking is a must
  217. if(mustTake && getRank(from, to, Fields::BLACK) + getRank(to, from, Fields::BLACK) == 0) {
  218. continue;
  219. }
  220. // calculate rank and potentially store the new max value
  221. int rank = getQuality(from, to);
  222. if(rank > maxRank) {
  223. maxRank = rank;
  224. selection = from;
  225. }
  226. }
  227. }
  228. // mark the stone as active
  229. active = selection;
  230. // add selection time to statistics, do not measure IO
  231. selectionTime += getNanos() - time;
  232. // send the selection to the server
  233. std::cerr << "Selecting (" << selection.x << ", " << selection.y << ")\n";
  234. // print the selection and the current game field for logging purposes
  235. fields.print(active);
  236. std::cout << selection.x << "\n" << selection.y << "\n";
  237. }
  238. void Game::makeMove() {
  239. // add statistics
  240. ssize_t time = getNanos();
  241. moves++;
  242. // check all neighbors of active selection and choose the move with the highest rank
  243. Vector maxTo(0, 0);
  244. int maxRank = INT_MIN;
  245. for(Vector& m : neighbours) {
  246. Vector to = active + m;
  247. // prevent invalid moves
  248. if(!isInRange(to) || !fields.hasState(to, Fields::EMPTY) || lastLocations.contains(to) ||
  249. !areNeighbors(active, to) || m == direction) {
  250. continue;
  251. }
  252. // if this is not the first move or a must take move - prevent non taking move
  253. if(!lastLocations.isEmpty() || mustTake) {
  254. int rank = getRank(active, to, Fields::BLACK) + getRank(to, active, Fields::BLACK);
  255. if(rank == 0) {
  256. continue;
  257. }
  258. }
  259. // calculate rank and potentially store the new max value
  260. int rank = getQuality(active, to);
  261. if(rank > maxRank) {
  262. maxRank = rank;
  263. maxTo = to;
  264. }
  265. }
  266. // remember the old location to prevent moving to it
  267. lastLocations.add(active);
  268. // remember the direction to prevent moving in it again
  269. direction = maxTo - active;
  270. // mark the location as active
  271. active = maxTo;
  272. // add selection time to statistics, do not measure IO
  273. moveTime += getNanos() - time;
  274. // send the move to the server
  275. std::cerr << "Move to (" << maxTo.x << ", " << maxTo.y << ")\n";
  276. // log the move
  277. std::cout << maxTo.x << "\n" << maxTo.y << "\n";
  278. }
  279. void Game::takeStone() {
  280. // calculate the better taking side ...
  281. uint rankA = getRank(active, active + direction, Fields::BLACK);
  282. uint rankB = getRank(active + direction, active, Fields::BLACK);
  283. // and send it to the server
  284. if(rankA >= rankB) {
  285. std::cout << "W\n";
  286. } else {
  287. std::cout << "A\n";
  288. }
  289. }
  290. int Game::getQuality(const Vector& from, const Vector& to) const {
  291. // base quality is how much stones this turn can take (+ following moves)
  292. int quality = getStoneTakes(from, to);
  293. // actually do the turn on a copy
  294. Game copy = *this;
  295. copy.move(from, to);
  296. // reset previous location and direction to simulate an opponent turn
  297. copy.lastLocations.clear();
  298. copy.direction.set(0, 0);
  299. // calculate white and black freedom on the potential new board
  300. int whiteFreedom = copy.getFreedom(Fields::WHITE);
  301. int blackFreedom = copy.getFreedom(Fields::BLACK);
  302. // calculates the strongest opponent turn
  303. uint maxRank = copy.getBestEnemyRank();
  304. // opponent count is important in the endgame
  305. int black = countBlack();
  306. if(black >= 8) {
  307. black = 1;
  308. }
  309. // finalize the quality formula
  310. return quality - maxRank * black + (whiteFreedom - blackFreedom) / 4;
  311. }
  312. int Game::countBlack() const {
  313. int counter = 0;
  314. for(Vector& v : fieldVectors) {
  315. counter += fields.hasState(v, Fields::BLACK);
  316. }
  317. return counter;
  318. }
  319. bool Game::isTakingPossible() const {
  320. // scan all fields for possible taking moves
  321. for(Vector& to : fieldVectors) {
  322. if(!fields.hasState(to, Fields::EMPTY)) {
  323. continue;
  324. }
  325. // check if any white neighbored stone can move to the found empty stone
  326. for(Vector& m : neighbours) {
  327. Vector from = to + m;
  328. if(!isInRange(from) || !areNeighbors(from, to) || !fields.hasState(from, Fields::WHITE)) {
  329. continue;
  330. }
  331. // return true if the white stone can take black stones in any direction
  332. if(getRank(from, to, Fields::BLACK) + getRank(to, from, Fields::BLACK) > 0) {
  333. return true;
  334. }
  335. }
  336. }
  337. return false;
  338. }
  339. uint Game::getBestEnemyRank() const {
  340. uint maxRank = 0;
  341. // scan all fields for best rank
  342. for(Vector& to : fieldVectors) {
  343. if(!fields.hasState(to, Fields::EMPTY)) {
  344. continue;
  345. }
  346. // scan neighbors of found empty fields
  347. for(Vector& m : neighbours) {
  348. Vector from = to + m;
  349. // prevent invalid moves
  350. if(!isInRange(from) || !fields.hasState(from, Fields::BLACK)) {
  351. continue;
  352. }
  353. // calculate and store the potentially better rank
  354. uint rank = getStoneTakes(from, to);
  355. if(rank > maxRank) {
  356. maxRank = rank;
  357. }
  358. }
  359. }
  360. return maxRank;
  361. }
  362. long int Game::getNanos() {
  363. return std::chrono::high_resolution_clock::now().time_since_epoch().count();
  364. }