Chinaunix首页 | 论坛 | 博客
  • 博客访问: 1502619
  • 博文数量: 148
  • 博客积分: 2234
  • 博客等级: 大尉
  • 技术积分: 3225
  • 用 户 组: 普通用户
  • 注册时间: 2012-05-17 21:34
个人简介

未来很长。

文章存档

2017年(7)

2016年(4)

2015年(1)

2014年(6)

2013年(31)

2012年(99)

分类: C/C++

2012-05-23 16:11:24

一个二叉树的建立代码如下所示:

点击(此处)折叠或打开

  1. #include<stdio.h>
  2.  #include<stdlib.h>
  3. struct BiTNode {
  4.     char data;
  5.     struct BiTNode* lchild;
  6.     //二叉树的结构 struct BiTNode* rchild;
  7.  };
  8.  typedef struct BiTNode * BiTree ;
  9.  void CreatBiTree( BiTree ); //创建二叉树
  10.  void PreOrderTraverse(BiTree ); // 遍历该二叉树,打印各个节点的值
  11.  void main(){
  12.      BiTree T=NULL; //T为根节点
  13.      CreatBiTree( T ) ;
  14.      PreOrderTraverse( T ) ;
  15.      system("pause");
  16.      return 0;
  17.  }
  18.  //用先序递归过程建立二叉树
  19.  void CreatBiTree( BiTree T)
  20.  {
  21.      char ch;
  22.      ch=getchar();
  23.      if( ch == '*') //如果输入星号则二叉树的节点为空节点
  24.          T=NULL;
  25.      else{
  26.         
  27.          if ( ! ( T=(BiTree)malloc( sizeof(struct BiTNode) ) ) )
  28.          {
  29.              printf("内存分配失败!");
  30.                 return ; //如果输入的不是星号。则为新节点分配内存空间
  31.          }
  32.          else
  33.          {
  34.             
  35.              T->data=ch;
  36.              CreatBiTree(T->lchild); //分配成功则接着建立左子树和右子树
  37.              CreatBiTree(T->rchild);
  38.          }
  39.      }
  40.  }
  41.  void PreOrderTraverse(BiTree T){
  42.      if (T)
  43.      {
  44.          printf("%c",T->data); ;
  45.          if (T->lchild)
  46.              PreOrderTraverse(T->lchild) ; //遍历该树,输出各个节点的值
  47.          if (T->rchild)
  48.              PreOrderTraverse(T->rchild) ;
  49.      }
  50.  }
首先我们一看就知道这有问题,哪儿有问题呢,创建二叉树的时候有问题,看看那传的是什么值,传的是指针,而接受的参数是指针,这无非跟我们普通的函数传的是一个变量接收的还是一个普通变量一样,想一下,当调用函数结束时,它的空间还存在么,那么你传过来的参数的值会给你主函数中返回去吗,显然是不可能的啊,所以修改后的代码如下所示:

点击(此处)折叠或打开

  1. #include<stdio.h>
  2.  #include<stdlib.h>
  3. struct BiTNode {
  4.     char data;
  5.     struct BiTNode* lchild;
  6.     //二叉树的结构 struct BiTNode* rchild;
  7.  };
  8.  typedef struct BiTNode * BiTree ;
  9.  void CreatBiTree( BiTree ); //创建二叉树
  10.  void PreOrderTraverse(BiTree ); // 遍历该二叉树,打印各个节点的值
  11.  void main(){
  12.      BiTree T=NULL; //T为根节点
  13.      CreatBiTree( T ) ;
  14.      PreOrderTraverse( T ) ;
  15.      system("pause");
  16.      return 0;
  17.  }
  18.  //用先序递归过程建立二叉树
  19. void CreatBiTree( BiTree * T)
  20. {

  21.     char ch;

  22.     ch=getchar();
  23.     if( ch == '*') //如果输入星号则二叉树的节点为空节点
  24.         *T=NULL;
  25.     else
  26.     {
  27.         if ( ! ( *T=(BiTree)malloc( sizeof(struct BiTNode) ) ) )
  28.         {
  29.              printf("内存分配失败!");
  30.                 return ; //如果输入的不是星号。则为新节点分配内存空间
  31.         }
  32.         else
  33.         {
  34.             (*T)->data=ch;
  35.             CreatBiTree(&((*T)->lchild)); //分配成功则接着建立左子树和右子树
  36.          CreatBiTree(&((*T)->rchild));
  37.         }
  38.     }
  39. }


  40.  void PreOrderTraverse(BiTree T){
  41.      if (T)
  42.      {
  43.          printf("%c",T->data); ;
  44.          if (T->lchild)
  45.              PreOrderTraverse(T->lchild) ; //遍历该树,输出各个节点的值
  46.          if (T->rchild)
  47.              PreOrderTraverse(T->rchild) ;
  48.      }
  49.  }

二叉树中,个人认为初学者容易犯一下几个错误:

1:创建二叉树,有两类模式:

a: 在main 函数中定义一个BiTree   注意这是个二叉树节点的指针类型;然后将这个参数传递给一个CreatBiTree 函数;在该函数中递归创建二叉树;代码如下:           

点击(此处)折叠或打开

  1. void CreatBiTree( BiTree * T) //注意我这里用的是二级指针;这个是好多人容易犯错误的地方;

  2.          {
  3.                  char ch;

  4.                  ch=getchar();
  5.                 if( ch == '*') //如果输入星号则二叉树的节点为空节点
  6.                     *T=NULL;
  7.                else{
  8.                        if ( ! ( *T=(BiTree)malloc( sizeof(struct BiTNode) ) ) ) {
  9.                                printf("内存分配失败!");
  10.                                 return ; //如果输入的不是星号。则为新节点分配内存空间
  11.                        }
  12.                        else

  13.                       {
  14.                                 (*T)->data=ch;
  15.                                 CreatBiTree(&((*T)->lchild)); //分配成功则接着建立左子树和右子树
  16.                                 CreatBiTree(&((*T)->rchild));
  17.                      }
  18.        }
  19. }

 b:可以直接在CreatBiTree函数中创建二叉树,并返回二叉树的根 指针;

 如:BiTree T=CreatBiTree();代码如下:

         

点击(此处)折叠或打开

  1. BiTree CreatBiTree()

  2.           {

  3.                      char ch;

  4.                      BiTNode *pt=NULL;

  5.                      ch=getchar();

  6.                      if( '*'==ch)

  7.                              return NULL;

  8.                      else {

  9.                               pt=(BiTNode *)malloc(sizeof(?BiTNode ) );

  10.                               pt->data=ch;

  11.                               pt->lchild=CreatBiTree();

  12.                               pt->rchild=CreatBiTree();

  13.                               return pt;

  14.            }


呵呵。。从上边的两个建树过程看来,还是第二个容易理解一点:

不过大部分人,都在疑惑一个问题,那就是在 第一个 (即 A) 的过程中,为什么一定要使用二级指针,而一级为什么就不行呢??

不是说函数中传递指针,在函数中改变指针的值,就是在改变 实参中的数据信息嘛???

额。。。其实吧,上边说的也对,可问题就在这块了。。。问题是,在建立二叉树的过程中,不是改变了形参的值,而是: 改变了形参的指向;而推出该函数后,形参被释放了,那么为形参动态分配的空间,也就没办法释放了,造成了内存泄露问题。。

举个例子吧:


 

点击(此处)折叠或打开

  1. void get_Vale1(char *pt)

  2. {

  3.        pt=(char *)malloc(strlen("sx_liang")+1);

  4.         strcpy(pt,"sx_liang");

  5. }

  6. void get_Vale2(char **pt)


  7. {

  8.        *pt=(char *)malloc(strlen("sx_liang")+1);

  9.         strcpy(*pt,"sx_liang");

  10. }



  11. int main()

  12. {

  13.          char *pt=NULL;

  14.    

  15.          get_Vale1(pt); //看看,这里调用的过程中,传递的是一级指针;

  16.            if( NULL==pt) //额。。。。在这里呢,就会执行if 里边的信息了。。。

  17.           {

  18.                        cout<<"pt is NULL"<<endl;

  19.                        exit(1);

  20.           }

  21.            else

  22.                 cout<<pt<<endl;

  23.                 get_Vale2(&pt); //这里呢,传递的是二级指针,结果呢,就输出了“sx_liang";

  24.          if( NULL==pt)

  25.           {

  26.                        cout<<"pt is NULL

什么原因呢: 咱开始分析一下:

void get_Vale1(char *M)//这里呢,传递了一个指针,此时,M和 实参 pt 的指向内存的同一块空间,都指向了null;

M=(char *)malloc(strlen("sx_liang")+1); //此时呢,给M的重新开辟了一块空间, 而实参pt 没有变化,还指向NULL;

//就是这里,M 和pt 已经没有联系了,彻底变成了两个指向不同的指针;

strcpy(M,"sx_liang");//退出函数时,因为M是局部变量,在栈中分配的空间,那么M自动销毁,而在堆中为M分配的空间呢,泄露了,没有被释放;是个问题!

void get_Vale2(char **M)//这里呢,传递的是二级指针;此时,M 指向实参,而不是和实参一样指向null;*M 此时才和pt相同,都指向了NULL;注意体会一下;

*M=(char *)malloc(strlen("sx_liang")+1);//在这里呢,*M就是 实参pt ,给*M申请空间,就是在为pt 申请空间;注意体会了;

strcpy(*M,"sx_liang");//推出函数后,M销毁,但是它申请的空间没被释放,但是该空间有pt 指向它,不担心释放问题;

额。。。到这里了,应该就能明白为什么 二叉树的创建过程中 A 办法要使用二级指针了吧。。。。一级指针,每次创建的节点,都没有和头指针root 联系起来,当然就没办法打印了, 而二级呢。。。呵呵。。。就OK了。。。。

就是这样了。。。呵呵。。。



阅读(7671) | 评论(2) | 转发(2) |
给主人留下些什么吧!~~

wenjavac2012-05-24 16:43:05

多谢啊,刚还在纠结这个问题。恍然大悟啊

十七岁的回忆2012-05-24 10:20:36

恩恩,博文很思路清晰,可以借鉴一下