用c++或c 做一个深度遍历的 棋谱树,我用来 做围棋棋谱的教学,要求节省内存、效率高,便于保存(棋谱)和拷贝棋谱

为了实现一个高效且节省内存的围棋棋谱树,我们可以使用如下思路和方法:

    1.    数据结构设计:

    •    使用一个树结构,每个节点表示棋局的一步(包括坐标和落子的颜色)。

    •    每个节点存储子节点的指针,表示该步之后的可能变化。

    •    为了节省内存,可以使用指针列表或数组管理子节点。

    •    节点还应包括对父节点的指针,便于回溯。

    2.    效率优化:

    •    使用紧凑的结构体存储棋局信息(例如,用一个 char 存储坐标,用一个 bool 存储黑白方)。

    •    使用内存池管理节点,减少频繁的动态内存分配开销。

    •    通过深度优先遍历(DFS)进行棋谱操作。

    3.    便于保存和拷贝:

    •    使用序列化和反序列化方法将棋谱存储为文件或字符串。

    •    提供简单的接口来拷贝整个棋谱树。

 

以下是一个示例实现:

 

代码实现

 

#include <iostream>

#include <vector>

#include <string>

#include <memory>

#include <fstream>

 

// 棋谱节点结构

struct MoveNode {

    int x, y;               // 落子坐标

    bool isBlack;           // 是否为黑子

    std::vector<MoveNode*> children; // 子节点

    MoveNode* parent;       // 父节点

 

    // 构造函数

    MoveNode(int x, int y, bool isBlack, MoveNode* parent = nullptr)

        : x(x), y(y), isBlack(isBlack), parent(parent) {}

 

    ~MoveNode() {

        for (auto child : children) {

            delete child;

        }

    }

};

 

// 棋谱树类

class GoTree {

private:

    MoveNode* root; // 根节点

 

public:

    // 构造函数

    GoTree() {

        root = new MoveNode(-1, -1, false); // 虚拟根节点

    }

 

    ~GoTree() {

        delete root;

    }

 

    // 添加棋谱节点

    MoveNode* addMove(MoveNode* parent, int x, int y, bool isBlack) {

        auto* newNode = new MoveNode(x, y, isBlack, parent);

        parent->children.push_back(newNode);

        return newNode;

    }

 

    // 深度优先遍历(打印棋谱)

    void printTree(MoveNode* node, int depth = 0) {

        if (node->x != -1 && node->y != -1) { // 跳过虚拟根节点

            std::cout << std::string(depth * 2, ' ')

                      << (node->isBlack ? "Black" : "White") << " (" << node->x << ", " << node->y << ")\n";

        }

        for (auto* child : node->children) {

            printTree(child, depth + 1);

        }

    }

 

    // 保存棋谱到文件

    void saveTree(MoveNode* node, std::ostream& os) {

        if (node->x != -1 && node->y != -1) {

            os << (node->isBlack ? "B" : "W") << "," << node->x << "," << node->y << "\n";

        }

        for (auto* child : node->children) {

            saveTree(child, os);

        }

        os << "END\n"; // 用END标记一个节点的子节点结束

    }

 

    // 从文件加载棋谱

    MoveNode* loadTree(std::istream& is, MoveNode* parent = nullptr) {

        std::string line;

        MoveNode* currentNode = parent ? parent : root;

        while (std::getline(is, line)) {

            if (line == "END") {

                return currentNode; // 子节点结束

            }

            char color;

            int x, y;

            if (sscanf(line.c_str(), "%c,%d,%d", &color, &x, &y) == 3) {

                bool isBlack = (color == 'B');

                currentNode = addMove(currentNode, x, y, isBlack);

                loadTree(is, currentNode);

            }

        }

        return currentNode;

    }

 

    MoveNode* getRoot() {

        return root;

    }

};

 

int main() {

    GoTree tree;

 

    // 构建棋谱

    auto* move1 = tree.addMove(tree.getRoot(), 3, 3, true);   // 黑3,3

    auto* move2 = tree.addMove(move1, 16, 16, false);         // 白16,16

    tree.addMove(move2, 4, 4, true);                          // 黑4,4

 

    // 打印棋谱

    std::cout << "棋谱树:\n";

    tree.printTree(tree.getRoot());

 

    // 保存棋谱

    std::ofstream outFile("goboard.txt");

    tree.saveTree(tree.getRoot(), outFile);

    outFile.close();

 

    // 加载棋谱

    std::ifstream inFile("goboard.txt");

    GoTree loadedTree;

    loadedTree.loadTree(inFile);

    inFile.close();

 

    // 打印加载后的棋谱

    std::cout << "\n加载后的棋谱树:\n";

    loadedTree.printTree(loadedTree.getRoot());

 

    return 0;

}

 

功能说明

    1.    数据结构:

    •    每一步棋局存储为一个 MoveNode,包含坐标、落子颜色、子节点和父节点。

    •    使用 std::vector 存储子节点,可以动态扩展子节点数量。

    2.    操作与接口:

    •    addMove:添加棋谱节点。

    •    printTree:深度优先遍历打印棋谱树。

    •    saveTree 和 loadTree:实现棋谱的保存和加载。

    3.    文件格式:

    •    保存为简单的 CSV 格式:B,3,3 表示黑棋落在 (3,3),END 表示子节点结束。

 

优点

    •    节省内存:使用紧凑的节点结构体和动态内存管理。

    •    效率高:支持快速的棋谱构建、保存和加载。

    •    可扩展性:可以根据需求增加功能,如支持回溯、分支删除等。

 

你可以根据实际需求进一步优化,比如将 std::vector 替换为固定大小的数组以进一步节省内存。

 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/890921.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

理想的以太网网络故障排查工具:LinkXpert M3

在排查铜缆、光纤和Wi-Fi以太网故障时&#xff0c;通常需要多种昂贵的工具&#xff0c;操作起来会很复杂。因此&#xff0c;我们推出了LinkXpert M3 —— 一个专为铜缆、光纤和Wi-Fi以太网网络故障排查而设计的工具。它功能强大、体积小巧、性价比高&#xff0c;无疑是您值得信…

Git如何设置和修改当前分支跟踪的上游分支

目录 前言 背景 设置当前分支跟踪的上游分支 当前分支已有关联&#xff0c;删除其关联&#xff0c;重新设置上游 常用的分支操作 参考资料 前言 仅做学习记录&#xff0c;侵删 背景 在项目开发过程中&#xff0c;从master新建分支时&#xff0c;会出现没有追踪的上游分…

专业版pycharm与服务器连接

一、先连接服务器 先创建配置&#xff1a; 名字随便取一个&#xff1a; 点击测试连接测试是否连接成功&#xff1b; 二、添加解释器 添加解释器&#xff0c;这个解释器是最开始在xshell中创建好的虚拟环境&#xff0c;具体虚拟环境创建可参考这篇&#xff1a;AutoDL服务器深…

LabVIEW如何学习FPGA开发

FPGA&#xff08;现场可编程门阵列&#xff09;开发因其高性能、低延迟的特点&#xff0c;在实时控制和高速数据处理领域具有重要地位。LabVIEW FPGA模块为开发者提供了一个图形化编程平台&#xff0c;降低了FPGA开发的门槛。本篇文章将详细介绍LabVIEW FPGA开发的学习路径&…

ISDP010_基于DDD架构实现收银用例主成功场景

信息系统开发实践 &#xff5c; 系列文章传送门 ISDP001_课程概述 ISDP002_Maven上_创建Maven项目 ISDP003_Maven下_Maven项目依赖配置 ISDP004_创建SpringBoot3项目 ISDP005_Spring组件与自动装配 ISDP006_逻辑架构设计 ISDP007_Springboot日志配置与单元测试 ISDP008_SpringB…

中学数学:一个函数值计算题

在数学的领域中&#xff0c;函数是一种描述变量之间关系的桥梁&#xff0c;它能够揭示出看似复杂现象背后的简洁规律。通过函数&#xff0c;我们可以预测、分析并解决实际问题。在这张图片中&#xff0c;我们看到了一位数学爱好者手写的解题过程&#xff0c;它展示了如何巧妙地…

#渗透测试#漏洞挖掘#红蓝攻防#常见未授权访问漏洞汇总

免责声明 本教程仅为合法的教学目的而准备&#xff0c;严禁用于任何形式的违法犯罪活动及其他商业行为&#xff0c;在使用本教程前&#xff0c;您应确保该行为符合当地的法律法规&#xff0c;继续阅读即表示您需自行承担所有操作的后果&#xff0c;如有异议&#xff0c;请立即停…

基于Oauth2的SSO单点登录---前端

Vue-element-admin 是一个基于 Vue.js 和 Element UI 的后台管理系统框架&#xff0c;提供了丰富的组件和功能&#xff0c;可以帮助开发者快速搭建现代化的后台管理系统。 一、基本知识 &#xff08;一&#xff09;Vue-element-admin 的主要文件和目录 vue-element-admin/ |--…

华为 AI Agent:企业内部管理的智能变革引擎(11/30)

一、华为 AI Agent 引领企业管理新潮流 在当今数字化飞速发展的时代&#xff0c;企业内部管理的高效性与智能化成为了决定企业竞争力的关键因素。华为&#xff0c;作为全球领先的科技巨头&#xff0c;其 AI Agent 技术在企业内部管理中的应用正掀起一场全新的变革浪潮。 AI Ag…

RustDesk内置ID服务器,Key教程

RustDesk内置ID服务器&#xff0c;Key教程 首先需要准备一个域名&#xff0c;并将其指定到你的 rustdesk 服务器 ip 地址上&#xff0c;这里编译采用的是Github Actions &#xff0c;说白了是就workflows&#xff0c;可以创建一些自动化的工作流程&#xff0c;例如代码的检查&a…

Wend看源码-Java-集合学习(List)

摘要 本篇文章深入探讨了基于JDK 21版本的Java.util包中提供的多样化集合类型。在Java中集合共分类为三种数据结构&#xff1a;List、Set和Queue。本文将详细阐述这些数据类型的各自实现&#xff0c;并按照线程安全性进行分类&#xff0c;分别介绍非线程安全与线程安全的实现方…

阿里云新用户服务器配置

创建和链接实例 创建实例&#xff0c;点击左侧标签栏总的实例&#xff0c; 找到链接帮助 根据帮助中的ip信息&#xff0c;然后启用vscode的ssh链接 ctrlp选择配置&#xff0c;输入公网的ip即可 passwd修改root密码 安装conda 参考 https://blog.csdn.net/adreammaker/arti…

五金产品视觉检测

五金产品种类繁多&#xff0c;且与我们的日常生活紧密有关&#xff0c;依照加工工艺的不同&#xff0c;五金产品有压铸件&#xff0c;五金冲压件&#xff0c;铸件等&#xff0c;无论是哪种加工方式&#xff0c;产品总会存在各式各样的问题&#xff0c;今天我们就五金产品的缺陷…

拼多多纠偏,能否实现买卖平权?

科技新知 原创作者丨江蓠 编辑丨蕨影 当曾将仅退款、运费险作为标配的电商平台们开始听到商家诉求&#xff0c;有意优化营商环境&#xff0c;作为“仅退款”服务发起者的拼多多也坐不住了。 在推出一揽子减免计划讨好中小商家之后&#xff0c;拼多多近期被传正在内测精选用户…

XGPT用户帮助手册

文章目录 20242024.12.27 摘要 本文介绍如何使用XGPT软件, XGPT融合了当前最先进的人工智能技术&#xff0c;并专为国内用户优化。 2024 2024.12.27 XGPT v1正式发布, 特色功能: 具备图像文本多模态处理功能包含GPT等最先进模型国内可访问 B站视频介绍 图1 XGPT v1 快照

低代码开源项目Joget的研究——Joget7社区版安装部署

大纲 环境准备安装必要软件配置Java配置JAVA_HOME配置Java软链安装三方库 获取源码配置MySql数据库创建用户创建数据库导入初始数据 配置数据库连接配置sessionFactory编译下载tomcat启动下载aspectjweaver移动jw.war文件编写脚本运行 测试参考资料 Joget&#xff0c;作为一款开…

后端开发如何高效使用 Apifox?

Apifox 是一个 API 协作开发平台&#xff0c;后端、前端、测试都可以使用 Apifox 来提升团队的工作效率。对于后端开发者而言&#xff0c;Apifox 的核心功能主要包括四个模块&#xff1a;调用 API、定义 API、开发与调试 API 以及生成 API 文档。本文将详细介绍后端开发人员如何…

flask后端开发(11):User模型创建+注册页面模板渲染

目录 一、数据库创建和配置信息1.新建数据库2.数据库配置信息3.User表4.ORM迁移 二、注册页面模板渲染1.导入静态文件2.蓝图注册路由 一、数据库创建和配置信息 1.新建数据库 终端中 CREATE DATABASE zhiliaooa DEFAULT CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci;2…

极客说|微软新模型:Phi-4 来了

作者&#xff1a;魏新宇 - 微软 AI 全球黑带高级技术专家 「极客说」 是一档专注 AI 时代开发者分享的专栏&#xff0c;我们邀请来自微软以及技术社区专家&#xff0c;带来最前沿的技术干货与实践经验。在这里&#xff0c;您将看到深度教程、最佳实践和创新解决方案。关注「极客…

redis相关数据类型介绍

当然&#xff0c;Redis 作为一个高性能的键值存储系统&#xff0c;提供了多种数据类型来支持不同的应用场景。 1. String&#xff08;字符串&#xff09; • 定义&#xff1a;Redis 最基本的数据类型&#xff0c;用于存储字符串值。 • 操作&#xff1a;SET、GET、INCR、DECR、…