免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
12下一页
最近访问板块 发新帖
查看: 2924 | 回复: 14
打印 上一主题 下一主题

[C++] 这个是怎么回事? 新手求解~~~ [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2014-11-02 14:27 |只看该作者 |倒序浏览
class A{
private:
   int i;

public:
   A(){this->i = 2;}
  void hello(){printf("hello is %d",i);}
void helloworld(){printf("hello world");}
}


int main(){
A *p = new A();
p =NULL;
p -> helloworld();
p ->hello();

return 0;
}
结果:   hello world  
           segment fault :11
为什么第一个函数调用成功。第二个失败? 大概我知道p = NULL 后 数据段清空  在调函数hello()时 数据i已经消失 所以会报 数据段溢出错误。

我的问题:1.    p = NULL    是将指针P指向的堆清0  还是重新指向一个为0的堆?
              2.    p = NULL  后数据段 清零 数据i内存没了。 为什么指针p还能找到 函数并能 压入栈?

论坛徽章:
24
金牛座
日期:2013-08-21 07:55:52射手座
日期:2014-09-29 15:36:14摩羯座
日期:2014-10-07 21:42:57天秤座
日期:2014-10-16 10:03:00午马
日期:2014-10-19 13:40:31酉鸡
日期:2014-11-08 08:44:54处女座
日期:2014-11-10 17:01:21午马
日期:2014-12-06 16:52:38羊年新春福章
日期:2015-02-12 17:41:512015年迎新春徽章
日期:2015-03-04 09:58:1115-16赛季CBA联赛之广夏
日期:2022-10-07 19:41:11巳蛇
日期:2014-09-13 15:29:54
2 [报告]
发表于 2014-11-02 16:53 |只看该作者
本帖最后由 wangspace 于 2014-11-02 16:57 编辑

class 以分号结束
在说你这写C++怎么还用printf ?

论坛徽章:
24
金牛座
日期:2013-08-21 07:55:52射手座
日期:2014-09-29 15:36:14摩羯座
日期:2014-10-07 21:42:57天秤座
日期:2014-10-16 10:03:00午马
日期:2014-10-19 13:40:31酉鸡
日期:2014-11-08 08:44:54处女座
日期:2014-11-10 17:01:21午马
日期:2014-12-06 16:52:38羊年新春福章
日期:2015-02-12 17:41:512015年迎新春徽章
日期:2015-03-04 09:58:1115-16赛季CBA联赛之广夏
日期:2022-10-07 19:41:11巳蛇
日期:2014-09-13 15:29:54
3 [报告]
发表于 2014-11-02 17:16 |只看该作者
你写的代码
A *p = new A();  //创建一个对象,p保存对象在堆中的地址
p = NULL;    //p指针现在为NULL,以不指向你刚创建的对象,你对对象的成员还怎么操作?

论坛徽章:
0
4 [报告]
发表于 2014-11-02 17:23 |只看该作者
回复 3# wangspace


    那为什么还能正常调用p -> helloworld();  正常打印:
      hello world

论坛徽章:
24
金牛座
日期:2013-08-21 07:55:52射手座
日期:2014-09-29 15:36:14摩羯座
日期:2014-10-07 21:42:57天秤座
日期:2014-10-16 10:03:00午马
日期:2014-10-19 13:40:31酉鸡
日期:2014-11-08 08:44:54处女座
日期:2014-11-10 17:01:21午马
日期:2014-12-06 16:52:38羊年新春福章
日期:2015-02-12 17:41:512015年迎新春徽章
日期:2015-03-04 09:58:1115-16赛季CBA联赛之广夏
日期:2022-10-07 19:41:11巳蛇
日期:2014-09-13 15:29:54
5 [报告]
发表于 2014-11-02 17:29 |只看该作者
虽然将P=NULL,但p指针还指向内存中创建对象的原始位置

论坛徽章:
0
6 [报告]
发表于 2014-11-02 17:31 |只看该作者
回复 3# wangspace


    我知道 我这个不是按 C++ 规范编写。

只是想 探寻一下 C++ 内部机制。

指针 指向的堆  分配一段内存。
然后= NULL后,指针指向一个 0的内存 空间。
但是 这时候 问什么 指针P 还能正常 调 函数。

如果函数不包含 类数据,则能正常运行 否则 会crash。

论坛徽章:
24
金牛座
日期:2013-08-21 07:55:52射手座
日期:2014-09-29 15:36:14摩羯座
日期:2014-10-07 21:42:57天秤座
日期:2014-10-16 10:03:00午马
日期:2014-10-19 13:40:31酉鸡
日期:2014-11-08 08:44:54处女座
日期:2014-11-10 17:01:21午马
日期:2014-12-06 16:52:38羊年新春福章
日期:2015-02-12 17:41:512015年迎新春徽章
日期:2015-03-04 09:58:1115-16赛季CBA联赛之广夏
日期:2022-10-07 19:41:11巳蛇
日期:2014-09-13 15:29:54
7 [报告]
发表于 2014-11-02 17:33 |只看该作者
回复 6# changkaizhao
我想还是应该把最基本的搞懂,我也是初学者


   

论坛徽章:
0
8 [报告]
发表于 2014-11-02 17:38 |只看该作者
回复 5# wangspace


    = NULL后 并不是 没变 ,地址会 变为0

论坛徽章:
8
CU大牛徽章
日期:2013-04-17 10:59:39CU大牛徽章
日期:2013-04-17 11:01:45CU大牛徽章
日期:2013-04-17 11:02:15CU大牛徽章
日期:2013-04-17 11:02:36CU大牛徽章
日期:2013-04-17 11:02:58技术图书徽章
日期:2013-12-04 10:48:50酉鸡
日期:2014-01-03 10:32:30辰龙
日期:2014-03-06 15:04:07
9 [报告]
发表于 2014-11-02 17:39 |只看该作者
本帖最后由 shan_ghost 于 2014-11-02 17:48 编辑

1、“堆”泛指“可动态分配的存储区”; 指针 (void *) 0 或 NULL 指向的不是堆。

最好不要在谈论c时滥用堆这个概念。事实上,c/c++相关标准并没有提及堆这个概念。

尤其是对某个概念没有清晰的认识时,更要尽量避免使用它。很容易扯出一堆乱麻的。



2、指针是一个“存储一个地址值”的变量。修改这个变量除了改变指针的指向外,并不会带来什么神奇变化。牢记这一点。
换句话说,p = NULL 后 数据段清空是无稽之谈;它之后,你原来new的那块内存就不再能访问了,但里面的内容却不会消失,那块区域在进程退出前也不再能重用——这也叫“内存泄漏”。


3、调用函数时,只是函数参数入栈,和函数入口地址没什么关系。


总的来说,楼主并没有真正理解指针、堆、栈等等基础概念,建议多看一些比较可靠的书,把基础补一补。就着错误的概念发挥是非常危险的。

事实上,你最好赶紧回去看书。后面的东西,在你补足基础前是不可能真正看懂的。



4、类 和 对象 不是一码事。你可以认为类信息(包括静态变量、成员函数表以及成员函数本身在代码段中的位置等)是全局唯一的;而对象是类的实例,每个实例都会占据特定的一块内存。

所以,任何时候,调用任何类成员函数,都不会有问题;但是,如果传给成员函数的参数值有问题,成员函数当然还是有可能会崩溃。

注意类成员函数有一个默认的隐藏参数,即this。p->hello(),其实在编译器眼里,是hello(p);同样的,在hello里面用到的i,实质上是this->i;对你的代码,也就是p->i。

你的代码把p指向NULL,这就是一个无效指针。所以一旦通过p访问i的内容就会出现段错误——这是执行p->hello()里面的逻辑时出错,并不是调用p->hello()出错了。
而helloworld()并没有去访问p指向的地址,所以执行它并不会崩溃。

你可以通过单步执行验证这一点。

论坛徽章:
59
2015年亚洲杯之约旦
日期:2015-01-27 21:27:392015年亚洲杯之日本
日期:2015-02-06 22:09:41拜羊年徽章
日期:2015-03-03 16:15:432015年辞旧岁徽章
日期:2015-03-03 16:54:152015年迎新春徽章
日期:2015-03-04 09:50:282015元宵节徽章
日期:2015-03-06 15:50:392015年亚洲杯之阿联酋
日期:2015-03-19 17:39:302015年亚洲杯之中国
日期:2015-03-23 18:52:23巳蛇
日期:2014-12-14 22:44:03双子座
日期:2014-12-10 21:39:16处女座
日期:2014-12-02 08:03:17天蝎座
日期:2014-07-21 19:08:47
10 [报告]
发表于 2014-11-02 17:44 |只看该作者
因为
p -> helloworld();
没有访问 p指向的内容,
成员函数地址并没有保存在类中,
不需要通过基地址访问。
但成员的函数变量是保存在类中的。
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP