写这个程序的目的是学习数据结构的时候方便调试,学习起来也比较直观。

这个是我测试SplayTree时候的gif

STEP 1

新建一个头文件,命名为DrawATree.hh, 将以下内容复制进去

#ifndef DRAWTREE_HH
#define DRAWTREE_HH #include <ostream>
#include <sstream>
#include <iostream>
#include <cmath>
#include <algorithm> namespace
{
#define lChild l_child_ //使用前将 l_child_ 更换为 自己树节点的左孩子的名字
#define rChild r_child_ //使用前将 r_child_ 更换为 自己树节点的右孩子的名字
#define data data_ //使用前将 data_ 更换为 自己树节点的数据的变量名
#define MAXN (1000) //这棵树的节点上限
#define PERID (2) //打印两个节点的横坐标间隔
unsigned int SUM; //统计当前遍历到第几个节点 #ifdef _WIN32
void clear() {system("cls"); }
#elif __linux__
void clear() { system("clear"); }
#endif // 将光标移动到 (X,Y)
std::string AXIS(int X, int Y) {
std::stringstream ss;
ss<< "\033["<< Y << ";" <<X<<"H";
return ss.str();
} struct DrawNode{
int x,y,dataSize;
}axisArray[MAXN]; //计算节点数据输出的长度
template <typename TreePtr>
int dataSize(TreePtr const& root) {
std::stringstream ss;
ss << (root->data);
return (ss.str()).length();
} //中序遍历, 从左往右画节点(不连线)
//横坐标通过全局变量SUM和上一个节点数据的输出长度算出
//纵坐标通过递归深度判断
//PERID 是两个节点间隔长度
template <typename TreePtr>
void buildDrawTree (TreePtr const& root, int deep) {
if(!root) return; //判断空节点,如果你的节点判空和我不一样,这里也要改, 比如我之前的判断空节点是(root->height_== -1). if(root->lChild) buildDrawTree(root->lChild, deep+1); axisArray[SUM] = (struct DrawNode){axisArray[SUM-1].x+axisArray[SUM-1].dataSize+PERID, deep, dataSize(root)};
std::cout << AXIS(axisArray[SUM].x, axisArray[SUM].y) << root->data;
++SUM; if(root->rChild) buildDrawTree(root->rChild, deep+1); } template <typename TreePtr>
void Draw (TreePtr const& t) { //画树函数
clear(); //清屏
SUM = 1;
int maxy = 0; buildDrawTree<TreePtr> (t, 2); //每个结点画出来 //画节点间连线,因为画的节点不会太多,所以就写了n^2的算法,比较好实现
//每个节点只有一个父节点,所以画出每个节点和自己父节点的连线即可
for(int i=1; i<SUM; i++) {
//x,y是子节点的坐标,p是父节点的axisArray数组的下标, px,py是父节点的坐标;
int x = axisArray[i].x, y = axisArray[i].y, p=0, px=0, py=y-1; if(y==1) continue; // 根结点没有父节点,跳过 for(int j=1; j<SUM; j++) { //循环找父节点
if(i==j) continue;
if((!p || abs(axisArray[j].x-x) < abs(px-x)) && axisArray[j].y+1 == y)
p = j, px = axisArray[j].x;
} int s = (2*x+axisArray[i].dataSize)>>1;
std::cout << AXIS(s,py) << '+';
if(s<px)
for(int i=s+1; i<px; i++) std::cout << AXIS(i,py) << '-';
else
for(int i=px+axisArray[p].dataSize;i<s; i++) std::cout << AXIS(i,py) << '-';
maxy = std::max(maxy, y);
}
std::cout << AXIS(1,maxy+1); //打印完把光标移到最下边.
// getchar();
}
} // namespace
#endif

STEP 2

修改头文件DrawATree.hh 第12,13,14,51行代码,如果需要的话. 代码中有注释说明怎么修改.

STEP 3

测试你的源代码.

 要保证你的节点数据可以用cout<<输出

#include "DrawATree.hh"
#include "YourTree.hh" int main() {
YourTreeType t;
Draw(t.root()); // 这里要传入你的根结点的指针!
}

实例:

树的头文件: Tree.hh

typedef struct TreeNode Node;
typedef Node* PtrNode; struct TreeNode{
char c_;
PtrNode l_child_, r_child_;
TreeNode(char c) : c_(c), l_child_(0), r_child_(0) {}
void Insert(PtrNode p, bool left);
}; void Node::Insert(PtrNode p, bool left) {
if(left) l_child_ = p;
else r_child_ = p;
}

源代码:

#include "Tree.hh"
#include "DrawATree.hh"
#define left (1)
#define right (0) int main() {
PtrNode root = new TreeNode('a');
root->Insert(new TreeNode('b'), left);
root->Insert(new TreeNode('c'), right);
Draw(root);
getchar();
}

效果:

C++ 命令行窗口打印二叉树(图形)的更多相关文章

  1. 【Python】解决Python脚本 在cmd命令行窗口运行时,中文乱码问题

    问题描述 python2.X,代码中指定了UTF-8,但是在cmd命令行窗口时,打印的中文仍然会乱码 在python3不存在该问题 运行结果: 原因 搜索得知,中文windows默认的输出编码为gbk ...

  2. <VS>MFC程序显示命令行窗口

    编写MFC程序时,想打印出调试信息,使用cout后,发现程序并没有像想象中那样自动弹出命令行窗口,要输出的信息也没地方去查看.百度后知道要手动调出命令行窗口,才可以看到输出的信息.   百度上介绍了两 ...

  3. 解决PowerShell命令行窗口中不显示光标的问题

    不知道什么原因,在有些系统上打开PowerShell命令行窗口后,光标无法显示.这种情况在Windows Server 2008/2012.Windows 8/9/10上都出现过,估计是由于某些系统软 ...

  4. 在windows中,如何使用cmd命令行窗口正确显示编码为utf-8格式的文字

    在windows中,如何使用cmd命令行窗口正确显示编码为utf-8格式的文字呢? 正确的步骤如下: 1, 打开cmd命令行窗口 2, 输入命令 >chcp 65001 数字65001代表的是c ...

  5. MySQL 遇到的问题:在服务里找不到自己的 MySQL,以及在命令行窗口中运行服务出现的问题。

    1.用数据库的时候在服务里找不到自己的 MySQL ,于是就想用命令行窗口去运行. ①.在开始里,键入 cmd ,打开命令行窗口. ②.输入:mysql -u root -p 回车,这时会提示请输入密 ...

  6. C#中隐式操作CMD命令行窗口

    原文:C#中隐式操作CMD命令行窗口 MS的CMD命令行是一种重要的操作界面,一些在C#中不那么方便完成的功能,在CMD中几个简单的命令或许就可以轻松搞定,如果能在C#中能完成CMD窗口的功能,那一定 ...

  7. 在windows命令行窗口下执行:查看所有的端口占用情况

    开始--运行--cmd 进入命令提示符 输入netstat -ano 即可看到所有连接的PID 之后在任务管理器中找到这个PID所对应的程序如果任务管理器中没有PID这一项,可以在任务管理器中选&qu ...

  8. Windows 桌面和文件夹的右键->打开命令行窗口

    Windows 桌面和文件夹的右键->打开命令行窗口 1.先按下shift,再点鼠标右键运行CMD,(不是管理员权限) 上图是我已经加了右键的,并且 系统设置了  ps代替cmd,所以是“在此处 ...

  9. 命令行窗口中用telnet测试HTTP协议

    1. 命令行窗口中用telnet测试HTTP协议 HTTP消息是由普通ASCII文本组成.消息包括消息头和数据体部分.消息头以行为单位,每行以CRLF(回车和换行)结束,消息头结束后,额外增加一个CR ...

随机推荐

  1. wireshark的过滤命令

    1.ip.addr == 192.168.1.1 这种是目标地址和源地址都是 后面指定的IP

  2. LIS 问题 二分查找优化

    按n=5,a-{4,2,3,1,5}为例 dp的值依次是: INF INF INF INF INF 4     INF INF INF INF 2     INF INF INF INF 2      ...

  3. vue实现tab选项卡切换

    上代码: <template>   <div class="push">     //点击按钮     <div class="tab&qu ...

  4. 高级I/O

    1.阻塞问题的几种解决模型 1.1 非阻塞模型:类似于死循环,需要适当加休眠,防止cpu占用太高:效率较低 1.2 线程异步:效率最高的一种解决方案 1.3 I/O多路转接与复用  select(低) ...

  5. icloud/onenote/onedrive/microoutlook/百度云账号都是怎么回事(未完成)

    在用一些跨机器跨平台软件的时候,一旦换了电脑或者手机,我经常出现蒙的状态,不知道怎么同步或者使用了,在这里总结一下. 一.icloud账号 当我们买iphone或者macbook的时候,一开机它会让你 ...

  6. 对Java8新的日期时间类的学习(二)

    示例11 在Java中如何判断某个日期是在另一个日期的前面还是后面 这也是实际项目中常见的一个任务.你怎么判断某个日期是在另一个日期的前面还是后面,或者正好相等呢?在Java 8中,LocalDate ...

  7. jQuery插件开发小结

    jQuery插件开发规范 1. 使用闭包 (function($) { // Code goes here })(jQuery); 这是来自jQuery官方的插件开发规范要求,使用这种编写方式有什么好 ...

  8. mysqli存储过程

    <?php$link = mysqli_connect('localhost','root','','chinatupai');  $sql = "call getEmail('000 ...

  9. jQuery ajax中的参数含义

    所有options均可选,下面简要说明每个option 1.async 默认为true,即请求为异步请求,这也是ajax存在的意义.但同时也可以将这个参数设置为false,实现同步请求.(同步请求会锁 ...

  10. 关于KMP算法的重大发现

    之前写KMP模板的时候,nx[i]代表最大的一个x,使s[1,x-1]是s[1,i-1]的后缀.(方法1) 然而网上还有另一种方法求nx数组,nx[i]表示最大的一个x,使s[1,x]是s[1,i]的 ...