你需要采用前序遍历的方式,将一个二叉树转换成一个由括号和整数组成的字符串。空节点则用一对空括号 "()" 表示。而且你需要省略所有不影响字符串与原始二叉树之间的一对一映射关系的空括号对。
示例 1:
输入: 二叉树: [1,2,3,4]
1
/ \
2 3
/
4
输出: "1(2(4))(3)"
解释: 原本将是“1(2(4)())(3())”,
在你省略所有不必要的空括号对之后,
它将是“1(2(4))(3)”。
示例 2:
输入: 二叉树: [1,2,3,null,4]
1
/ \
2 3
\
4
输出: "1(2()(4))(3)"
解释: 和第一个示例相似,
除了我们不能省略第一个对括号来中断输入和输出之间的一对一映射关系。
思路分析:
这道题虽然说看着复杂,但是处理起来依旧是递归子问题的方式,我们可以分为:首先把根转换为字符串,再去把左子树转换为字符串,最后再把右子树转换为字符串,需要注意的是:
①如果左子树为空,右子树不为空时,“()”不能省略,相反,如果左子树不为空,右子树为空时,括号则可省略;
②一开始给要放转换为字符串的字符串数组,记得一定要把数组的第一个元素给初始化为“\0”;
③还有就是在转换之前我们要先拼一个“(” 进去,转换完之后,再拼一个“)” 进去。
具体代码如下:
/**
* Definition for a binary tree node.
* struct TreeNode {
* int val;
* struct TreeNode *left;
* struct TreeNode *right;
* };
*/
char valstr[30]={0};
void _tree2str(struct TreeNode* t,char* str){
if(t==NULL)
{
return NULL;
}
sprintf(valstr,"%d",t->val);//把数字转换为字符串,好多人可能一开始想到使用itoa函数,
//但是itoa是用POSIX(一套可移植的接口)定义的,是非标准化的,容易出问题,所以就用sprintf来代替,建议少用itoa
strcat(str,valstr);
//走到这里,根部分已转换完毕
if(t->left==NULL)
{
if(t->right==NULL)
{
return;
}
else
{
strcat(str,"()");//当左边为空时,括号不能省略
}
}
else//左边不为空
{
strcat(str,"(");
_tree2str(t->left,str);
strcat(str,")");
}
//走到这里表示左子树已转换完
if(t->right==NULL)
{
return;
}
else
{
strcat(str,"(");
_tree2str(t->right,str);
strcat(str,")");
}
}
char str[1000000];
char* tree2str(struct TreeNode* t) {
str[0]='\0';
_tree2str(t,str);
return str;
}
注意:有的同学可能不太了解sprintf这个函数。这里我简单说明一下:sprintf的参数是一个可变参数列表,这里参数的含义是:
int sprintf(char *str), const char *format(序列化为某种格式), ...)
详细解析可参考:https://blog.csdn.net/qq_42270373/article/details/82749992
可变参数列表的讲解可参考:https://blog.csdn.net/qq_42270373/article/details/81172490