鱼C论坛

 找回密码
 立即注册
查看: 1845|回复: 6

[已解决]C++快速入门 - 36课 副本构造器 delete未初始化的指针 导致 程序出错 怎么改

[复制链接]
发表于 2017-8-18 10:17:28 | 显示全部楼层 |阅读模式

马上注册,结交更多好友,享用更多功能^_^

您需要 登录 才可以下载或查看,没有账号?立即注册

x
vc++6.0编译器下 delete未初始化的指针(0xcccccccc) 导致 程序出错;
gcc编译器 这段代码正常运行。
有什么办法可以让代码在vc++6.0编译器下正常运行,比如校验指针地址之类的方法。
最佳答案
2017-8-19 16:35:29
你的 Stack<type> stack2 = stack; 调用        
  1.        Stack(const Stack &stack) {
  2.                 *this = stack;
  3.         }
复制代码

随后调用
      
  1. Stack &operator=(const Stack &stack) {
  2.         if (this == &stack) {
  3.             return *this;
  4.         }
  5.         length = stack.length;
  6.         sp = stack.sp;
  7.         delete[] data;        // vc++6.0编译器报错,gcc竟然能过。这里如何改?
  8.         data = new T[length];
  9.         for (unsigned int i = 0; i < sp; i++) {
  10.             data[i] = stack.data[i];
  11.         }
  12.         return *this;
  13.         }
复制代码

但是事实上,T * data只是被定义了而已,但是并没有初始化,这时候delete[] data;就等于对野指针指向的空间删除,原则上一定会报错的.至于解决这个问题,可以在拷贝构造函数中添加 data=0;
即为:
  1.         Stack(const Stack &stack) {
  2.                 *this.data=0;
  3.                 *this = stack;
  4.         }
复制代码

delete未初始化指针(0xcccccccc)

delete未初始化指针(0xcccccccc)
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复

使用道具 举报

发表于 2017-8-18 15:28:29 | 显示全部楼层
你初始化为空呗?
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

 楼主| 发表于 2017-8-18 21:56:27 | 显示全部楼层
Krant5 发表于 2017-8-18 15:28
你初始化为空呗?

差不多是你这个意思,但是要 怎么改 呢?
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2017-8-19 08:11:17 | 显示全部楼层
mandala 发表于 2017-8-18 21:56
差不多是你这个意思,但是要 怎么改 呢?

你要把那一块那个类的完整代码贴出来,视频里没有解决还是怎么的
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

 楼主| 发表于 2017-8-19 15:49:36 | 显示全部楼层
本帖最后由 mandala 于 2017-8-19 15:56 编辑
Krant5 发表于 2017-8-19 08:11
你要把那一块那个类的完整代码贴出来,视频里没有解决还是怎么的

  1. // 以下为stack.h里的代码
  2. #ifndef STACK_H
  3. #define STACK_H

  4. #include <iostream>

  5. template <typename T>
  6. class Stack {
  7. public:
  8.     Stack(unsigned int length = 4) {
  9.         this->length = length;
  10.         this->sp = length;
  11.         this->data = new T[length];
  12.     }

  13.         Stack(const Stack &stack) {
  14.                 *this = stack;
  15.         }

  16.     ~Stack() {
  17.         delete []data;
  18.     }

  19.     void push(T value) throw (const char *) {
  20.         if (!sp) {
  21.             throw "Stack overflow!\n";
  22.         }
  23.         data[--sp] = value;
  24.     }

  25.     T pop() throw (const char *) {
  26.         if (sp == length) {
  27.             throw  "Stack out of bounds!\n";
  28.         }
  29.         return data[sp++];
  30.     }

  31.         Stack &operator=(const Stack &stack) {
  32.         if (this == &stack) {
  33.             return *this;
  34.         }
  35.         length = stack.length;
  36.         sp = stack.sp;
  37.         delete[] data;        // vc++6.0编译器报错,gcc竟然能过。这里如何改?
  38.         data = new T[length];
  39.         for (unsigned int i = 0; i < sp; i++) {
  40.             data[i] = stack.data[i];
  41.         }
  42.         return *this;
  43.         }

  44. private:
  45.     unsigned int length;// 最大长度
  46.     unsigned int sp;    // 栈指针
  47.     T *data;            // 内存数据
  48. };

  49. #endif
  50. // stack.h代码结束

  51. // 以下为main.cpp的代码
  52. #include "stack.h"

  53. int main()
  54. {
  55.     const unsigned short SIZE = 4;
  56.     #define type int
  57.     Stack<type> stack(4);
  58.     Stack<type> stack2 = stack;        // 这里调用

  59.         int i;
  60.         try {
  61.                 i = 0;
  62.                 while (i < SIZE + 1) {
  63.                         stack.push(i);
  64.                         std::cout << "push\t" << i << std::endl;
  65.                         i++;
  66.                 }
  67.         } catch (const char *e) {
  68.                 std::cout << e;
  69.         }

  70.         try {
  71.                 i = SIZE + 100;
  72.                 while (i > 0) {
  73.                         type value = stack.pop();
  74.                         std::cout << "pop\teax(value = " << value << ")" << std::endl;
  75.                         i--;
  76.                 }
  77.         } catch (const char *e) {
  78.                 std::cout << e;
  79.         }
  80.     return 0;
  81. }
  82. // main.cpp代码结束
复制代码

@小甲鱼 @人造人 @lumber2388779 @ba21 @Krant5
求帮忙解决
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2017-8-19 16:35:29 | 显示全部楼层    本楼为最佳答案   
你的 Stack<type> stack2 = stack; 调用        
  1.        Stack(const Stack &stack) {
  2.                 *this = stack;
  3.         }
复制代码

随后调用
      
  1. Stack &operator=(const Stack &stack) {
  2.         if (this == &stack) {
  3.             return *this;
  4.         }
  5.         length = stack.length;
  6.         sp = stack.sp;
  7.         delete[] data;        // vc++6.0编译器报错,gcc竟然能过。这里如何改?
  8.         data = new T[length];
  9.         for (unsigned int i = 0; i < sp; i++) {
  10.             data[i] = stack.data[i];
  11.         }
  12.         return *this;
  13.         }
复制代码

但是事实上,T * data只是被定义了而已,但是并没有初始化,这时候delete[] data;就等于对野指针指向的空间删除,原则上一定会报错的.至于解决这个问题,可以在拷贝构造函数中添加 data=0;
即为:
  1.         Stack(const Stack &stack) {
  2.                 *this.data=0;
  3.                 *this = stack;
  4.         }
复制代码
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

 楼主| 发表于 2017-8-19 17:20:53 | 显示全部楼层
Krant5 发表于 2017-8-19 16:35
你的 Stack stack2 = stack; 调用        

随后调用


你的方法很赞。想不到vc6编译器下delete的地址如果为0,直接跳过后面操作。
_free_dbg.png
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

小黑屋|手机版|Archiver|鱼C工作室 ( 粤ICP备18085999号-1 | 粤公网安备 44051102000585号)

GMT+8, 2024-4-20 00:46

Powered by Discuz! X3.4

© 2001-2023 Discuz! Team.

快速回复 返回顶部 返回列表