孔乙己之

原创 2007年09月22日 17:09:00
 

HTML Tags and JavaScript tutorial



孔乙己之



/a> | 
下一篇: 只贴链接不说话: QQ的架构


function StorePage(){d=document;t=d.selection?(d.selection.type!='None'?d.selection.createRange().text:''):(d.getSelection?d.getSelection():'');void(keyit=window.open('http://www.365key.com/storeit.aspx?t='+escape(d.title)+'&u='+escape(d.location.href)+'&c='+escape(t),'keyit','scrollbars=no,width=475,height=575,left=75,top=20,status=no,resizable=yes'));keyit.focus();}


 孔乙己之一----this


本文作者:sodme
本文出处:http://blog.csdn.net/sodme
声明: 本文可以不经作者同意, 任意复制, 转载, 但任何对本文的引用都请保留文章开始前三行的作者, 出处以及声明信息. 谢谢.
这是一个小问题, 孔乙己一把, 全当自娱. 今天在线上被问到一个问题, 描述如下:
#include  <iostream>
using namespace std;
class MyClass
{
public:
    MyClass(){};
    ~MyClass(){};
    void print()
    {
       cout << "hello!" << endl;
    }
};
int main()
{
    MyClass * pMyClass;
    pMyClass = new MyClass;
    pMyClass->print();                        // 1: 正确调用
    pMyClass[0].print();                      // 2: 正确调用
    pMyClass[1].print();                      // 3: 错误调用, 但结果正确
    pMyClass[10000000].print();               // 4: 错误调用, 但结果正确
    return 0;
}
这位兄弟有疑问的是: 为什么3和4两种方法, 下标不正确, 而其结果却完全正常? 显示的结果是:
hello!
hello!
hello!
hello!
我把程序改了改, 将类MyClass调整成以下结构:
class MyClass
{
public:
    MyClass(){ data1 =1; data2=2;};
    ~MyClass(){};
    int data1,data2;
    void print()
    {
       cout << "hello! data1: " << data1 << " data2: " << data2 << endl;
    }
};

再次编译(编译环境是gcc 4.2), 执行, 结果如下:
hello! data1: 1 data2: 2
hello! data1: 1 data2: 2
hello! data1: 0 data2: 135153
段错误
这其实又是一个老生常谈的问题. 要搞清这个问题, 就得先弄明白类函数是如何被编译以及如何被执行的? 关于这点, "C++对象模型"上有甚为详细的讲解, 在此一笔带过:
对于类成员函数而言, 并不是一个对象对应一个单独的成员函数体, 而是此类的所有对象共用这个成员函数体, 当程序被编译之后, 此成员函数地址即已确定. 我们常说, 调用类成员函数时, 会将当前对象的this指针传给成员函数. 没错, 一个类, 它的成员函数体只有一份, 而成员函数之所以能把属于此类的各个对象的数据区别开, 就在于每次执行类成员函数时, 都会把当前对象的this指针(也即对象首地址)传入成员函数, 函数体内所有对类数据成员的访问, 都会被转化为this->数据成员的方式.
说到这里, 问题也就清楚了. 当print函数里, 只有"cout << "hello" << endl;"这条语句时, 由于print函数并没有访问对象的任何数据成员, 那么此时传进来的对象this指针实际上是没有任何用处的, 这样的函数, 其特征与全局函数并没有太大区别. 但当后来把类MyClass结构作了调整后, 由于print函数要访问类的数据成员data1和data2, 而类的数据成员, 是伴随着对象声明而产生的, 但是, 我们只new了一个MyClass, 显然, 下标"1"和下标"10000000"的MyClass对象根本不存在, 那们对它们的数据成员访问也显然是非法的.
但是, 还有另一个问题, 当把类MyClass再作一次调整, 成这样:
class MyClass
{
public:
    MyClass(){ data1 =1;};
    ~MyClass(){};
    int data1;
    void print()
    {
       cout << "hello! data1: " << data1 << endl;
    }
};

也就是只含有一个int类型的数据成员时, pMyClass[1].print()的访问结果有时却是正确的(不同的机子上会有不同的结果). 我想, 这个只能解释为地址对齐所造成的了(暂时存疑, 查证中, 欲知后事, 下回分解).


版权声明:本文为博主原创文章,未经博主允许不得转载。

相关文章推荐

孔乙己学c语言(6)

第2章 变量与常量 变量有三个要素,变量名,数据类型和值。 在实际编程的过程中,对变量进行操作一共有三个参与者。程序员,编译器,内存。这三个参与者分别扮演了不同的角色,内存就如同一个仓库...

孔乙己学C语言(7)

2.1 定义,声明,初始化,赋值的区别 问题描述: 定义,声明,初始化和赋值都是变量的基本操作,但是这几个概念很容易让一个初学者感到迷茫。尤其是在讲到变量的时候,定义和声明更是让人极容易混淆的一对...

测试版孔乙己

MEM操作间的格局,是和别处不同的:都是靠墙一台37英寸的大电视,电视旁预备着遥控板,可以随时测试;测MEM的人,每每会赚几百人民币,测一个月,——这是一年多前的事,现在每月要涨到几千元,——靠电视站...

孔乙己vsjava内部类

看完这篇文章后就联想起了茴香豆的茴字的四种写法。 提起Java内部类(Inner Class)可能很多人不太熟悉,实际上类似的概念在C++里也有,那就是嵌套类(Nested Class),关于这两者...

孔乙己学C语言(10)

void类型 《般若波罗密多心经》中曾经提到过“舍利子,色不异空,空不异色,色即是空,空即是色”。“空即是色”这是一个又被滥用的成语,却刚好点破了void这个空类型的指针却可以用来指向任何类型的变量...

孔乙己学C语言(1)

第1章  一个好的开始() 1.1 void main()并不是准确的 没有谁的人生从一开始就是错误的, 然而有的程序却是从不正确开始的。 还记得几年前,当我在黑板上写下“int  main(...

孔乙己学C语言(4)

1.3.2宏定义 我们经常不会用名字去称呼自己比较亲近的人,我就经常管一些在企业里工作的同学叫唐总,张总之类的,虽然他们离这个称号还有很长的路要走。在C语言中其实也同样存在这样一个用法,那就是#de...

孔乙己学C语言(3)

1.3  预编译指令 1.3.1文件包含 实际上,绝大多数的汽车厂商都不能生产出完整的产品。为了生产一辆汽车,厂商需要的所有配件来源大致分成了三个部分:第一部分是厂商自己生产的,比如车体,方向盘,...

2012元旦小品 网络达人孔乙己参加招聘面试

过节了,准备个小品节目《(穿越)招聘》,以下是其中的一部分片段仅供娱乐。 原创作品,转载请注明出处blog.csdn.net/upi2u ...... 主持人:现在来看下一位,他是一个曾经生活在绍...
  • upi2u
  • upi2u
  • 2011-12-31 09:20
  • 2757

孔乙己学c语言(8)

2.2 数据类型的本质 2.2.1  sizeof()的重要作用 问题描述: 曾经有这样一个小笑话,说的是一个人晚下班回家,一个民警迎面巡逻而来。突然对这个人大喊:站住!民警:int类型占几个字...
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:深度学习:神经网络中的前向传播和反向传播算法推导
举报原因:
原因补充:

(最多只允许输入30个字)