sort: 将 std::sort 排序算法替换为 sqrt_sort_sort_ins 算法

不需要通过转换为 vector 再调用 std::sort 排序。
自对弈时长由 52s 缩短到 45s。

排序算法代码来源:
https://github.com/swenson/sort
This commit is contained in:
Calcitem 2019-10-05 13:45:43 +08:00 committed by CalciteM Team
parent d9758fcca5
commit 6d81a994d8
6 changed files with 3394 additions and 15 deletions

View File

@ -60,6 +60,7 @@ HEADERS += \
src/base/hashMap.h \ src/base/hashMap.h \
src/base/memmgr.h \ src/base/memmgr.h \
src/base/misc.h \ src/base/misc.h \
src/base/sort.h \
src/base/stack.h \ src/base/stack.h \
src/base/thread.h \ src/base/thread.h \
src/ai/search.h \ src/ai/search.h \

View File

@ -451,6 +451,7 @@
<ClInclude Include="src\base\HashNode.h" /> <ClInclude Include="src\base\HashNode.h" />
<ClInclude Include="src\base\memmgr.h" /> <ClInclude Include="src\base\memmgr.h" />
<ClInclude Include="src\base\misc.h" /> <ClInclude Include="src\base\misc.h" />
<ClInclude Include="src\base\sort.h" />
<ClInclude Include="src\base\stack.h" /> <ClInclude Include="src\base\stack.h" />
<QtMoc Include="src\base\thread.h" /> <QtMoc Include="src\base\thread.h" />
<ClInclude Include="src\base\zobrist.h" /> <ClInclude Include="src\base\zobrist.h" />

View File

@ -126,6 +126,9 @@
<ClInclude Include="src\base\memmgr.h"> <ClInclude Include="src\base\memmgr.h">
<Filter>base</Filter> <Filter>base</Filter>
</ClInclude> </ClInclude>
<ClInclude Include="src\base\sort.h">
<Filter>base</Filter>
</ClInclude>
</ItemGroup> </ItemGroup>
<ItemGroup> <ItemGroup>
<CustomBuild Include="debug\moc_predefs.h.cbt"> <CustomBuild Include="debug\moc_predefs.h.cbt">

View File

@ -33,6 +33,14 @@
#include "types.h" #include "types.h"
#include "option.h" #include "option.h"
#define SORT_NAME nodep
#define SORT_TYPE AIAlgorithm::Node*
#define SORT_CMP(x, y) (-AIAlgorithm::nodeCompare((x), (y)))
player_t gSideToMove;
#include "sort.h"
using namespace CTSL; using namespace CTSL;
// 用于检测重复局面 (Position) // 用于检测重复局面 (Position)
@ -286,6 +294,58 @@ bool AIAlgorithm::nodeGreater(const Node *first, const Node *second)
#endif #endif
} }
int AIAlgorithm::nodeCompare(const Node * first, const Node * second)
{
int ret = 0;
if (gSideToMove == PLAYER_BLACK) {
if (first->value > second->value) {
ret = 1;
goto out;
}
if (first->value < second->value) {
ret = -1;
goto out;
}
if (first->value == second->value) {
if (!first->pruned && second->pruned) {
ret = 1;
goto out;
} else if (first->pruned && !second->pruned) {
ret = -1;
goto out;
}
}
}
if (gSideToMove == PLAYER_WHITE) {
if (first->value < second->value) {
ret = 1;
goto out;
}
if (first->value > second->value) {
ret = -1;
goto out;
}
if (first->value == second->value) {
if (!first->pruned && second->pruned) {
ret = 1;
goto out;
} else if (first->pruned && !second->pruned) {
ret = -1;
goto out;
}
}
}
out:
return ret;
}
void AIAlgorithm::sortMoves(Node *node) void AIAlgorithm::sortMoves(Node *node)
{ {
// 这个函数对效率的影响很大,排序好的话,剪枝较早,节省时间,但不能在此函数耗费太多时间 // 这个函数对效率的影响很大,排序好的话,剪枝较早,节省时间,但不能在此函数耗费太多时间
@ -297,34 +357,29 @@ void AIAlgorithm::sortMoves(Node *node)
//#define DEBUG_SORT //#define DEBUG_SORT
#ifdef DEBUG_SORT #ifdef DEBUG_SORT
for (int i = 0; i < node->childrenSize; i++) { for (int i = 0; i < node->childrenSize; i++) {
loggerDebug("* [%d] %x = %d\n", i, &(node->children[i]), node->children[i]->value); loggerDebug("* [%d] %p: %d = %d (%d)\n",
i, &(node->children[i]), node->children[i]->move, node->children[i]->value, !node->children[i]->pruned);
} }
loggerDebug("\n"); loggerDebug("\n");
#endif #endif
// TODO: 暂时使用 std::sort 排序, 后续需实现自己的排序函数 #define NODE_PTR_SORT_FUN(x) nodep_##x
vector<Node *> vec; gSideToMove = tempGame.position.sideToMove; // TODO: 暂时用全局变量
vec.reserve(NODE_CHILDREN_SIZE);
for (int i = 0; i < node->childrenSize; i++) { // 此处选用排序算法
vec.push_back(node->children[i]); NODE_PTR_SORT_FUN(sqrt_sort_sort_ins)(node->children, node->childrenSize);
}
std::stable_sort(vec.begin(), vec.end(), cmp);
for (int i = 0; i < node->childrenSize; i++) {
node->children[i] = vec[i];
}
#ifdef DEBUG_SORT #ifdef DEBUG_SORT
if (tempGame.position.sideToMove == PLAYER_BLACK) { if (tempGame.position.sideToMove == PLAYER_BLACK) {
for (int i = 0; i < node->childrenSize; i++) { for (int i = 0; i < node->childrenSize; i++) {
loggerDebug("+ [%d] %x = %d\n", i, &(node->children[i]), node->children[i]->value); loggerDebug("+ [%d] %p: %d = %d (%d)\n",
i, &(node->children[i]), node->children[i]->move, node->children[i]->value, !node->children[i]->pruned);
} }
} else { } else {
for (int i = 0; i < node->childrenSize; i++) { for (int i = 0; i < node->childrenSize; i++) {
loggerDebug("- [%d] %x = %d\n", i, &(node->children[i]), node->children[i]->value); loggerDebug("- [%d] %p: %d = %d (%d)\n",
i, &(node->children[i]), node->children[i]->move, node->children[i]->value, !node->children[i]->pruned);
} }
} }
loggerDebug("\n----------------------------------------\n"); loggerDebug("\n----------------------------------------\n");

View File

@ -100,6 +100,46 @@ public:
#endif /* TRANSPOSITION_TABLE_ENABLE */ #endif /* TRANSPOSITION_TABLE_ENABLE */
hash_t hash; // 哈希值 hash_t hash; // 哈希值
#endif /* DEBUG_AB_TREE */ #endif /* DEBUG_AB_TREE */
bool operator < (const Node &other)
{
if (value < other.value) {
return true;
}
if (value == other.value) {
if (pruned && !other.pruned) {
return true;
}
}
return false;
}
bool operator > (const Node &other)
{
if (value > other.value) {
return true;
}
if (value == other.value) {
if (!pruned && other.pruned) {
return true;
}
}
return false;
}
bool operator == (const Node &other)
{
if (value == other.value &&
pruned == other.pruned) {
return true;
}
return false;
}
}; };
#ifdef MEMORY_POOL #ifdef MEMORY_POOL
@ -139,6 +179,7 @@ public:
#ifdef MEMORY_POOL #ifdef MEMORY_POOL
static bool nodeLess(const Node *first, const Node *second); static bool nodeLess(const Node *first, const Node *second);
static bool nodeGreater(const Node *first, const Node *second); static bool nodeGreater(const Node *first, const Node *second);
static int nodeCompare(const Node *first, const Node *second);
#endif // MEMORY_POOL #endif // MEMORY_POOL
#ifdef ENDGAME_LEARNING #ifdef ENDGAME_LEARNING

3278
src/base/sort.h Normal file

File diff suppressed because it is too large Load Diff