一、类声明

//类是一种用户自定义类型,声明形式:
class 类名称
{
   public:
             公有成员(外部接口)
   private:
             私有成员
   protected:
             保护成员
};

在关键字public后面声明,它们是类与外部的接口,任何外部函数都可以访问公有类型数据和函数。
在关键字private后面声明,只允许本类中的函数访问,而类外部的任何函数都不能访问。
在关键字protected后面声明,与private类似,其差别表现在继承与派生时对派生类的影响不同。

 C++ Code 
1
2
3
4
5
6
7
8
9
10
11
 
class Clock
{
public:
    void Display();
    void Init(int hour, int minute, int second);

private:
    int hour_;
    int minute_;
    int second_;
};

假设定义了一个Clock 类,因为成员是private的,那么 Clock ck;  ck.hour_ = 12; 是错误的,对此定义一个public 的void SetHour(int hour) 来设置hour_ 的值。

二、内联成员函数、成员函数的重载及其缺省参数

这里有内联函数的概念。成员函数也可以是内联的,若在类内部实现,inline 关键字可加可不加;在类外部实现,需加inline,

如 inline void Clock::SetHour(int hour) { } 。实际上即使加了inline也不一定宏展开,比如遇到switch,for 语句的时候就往往不会。

此外,成员函数也像一般函数那样可以重载,也可以有缺省参数,参考这里

三、类与结构体

class与struct的区别:在未指定访问权限时,class默认的是私有的,struct默认是公有的,

struct Test
{
    int X;//公有的
     ...
};

此外,Test 可以独立作为一个tag,而不像C语言那样需要 struct Test 作为一个类型。

四、隐含的 this 指针

成员函数有一个隐含的附加形参,即指向该对象的指针,这个隐含的形参叫做this指针(编译器自动传递)

使用this指针保证了每个对象可以拥有不同数值的数据成员,但处理这些成员的代码可以被所有对象共享

成员函数是只读的代码,由所有对象共享,并不占对象的存储空间,因为this指针指向当前对象,所以成员函数可以区分它所作用的对象是哪一个。

(哪个对象调用了this所在的函数,this就代表哪个对象)

?

再来看一道经典的题目:

 C++ Code 
1
2
3
4
5
6
7
8
9
10
11
12
 
class A
{
public:
    int m;
    void print()
    {
        cout << "A" << endl;
    }
};

A *pa = 0;
pa->print();

可以理解为如下的C代码:

 C++ Code 
1
2
3
4
5
6
 
void print(A *this)
{
    cout << "A" << endl;
}
A *pa = 0;
print_A(pa);

相当于成员函数传递的this指针为0,那调用会出错吗? 肯定是正确输出"A" 的,因为this为0 表示没有对某个对象进行操作,而print里面确实没有对某

个对象成员进行操作,所以是可以运行的。

五、类作用域、前向声明

(1)、每个类都定义了自己的作用域称为类作用域,类作用域中说明的标识符只在类中可见。除了类作用域,还有块作用域、文件作用域、函数原型作用域、函数作用域,举个例子:

 C++ Code 
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
 
#include <iostream>
using namespace std;

class Test
{
public:
    int num_;
};

//num_ = 20;        Error,num_的作用域在类内部
int num_ = 20;      // num_的作用域是文件作用域,与类中的num_是不同的作用域

int add(int a, int b);  // a, b两个标识符的作用域为函数原型作用域

int main(void)
{
    int num_ = 30;      // num_为块作域
    {
        int num_ = 100; // num_为块作域
    }

cout << num_ << endl;
    cout <<::num_ << endl;
    return 0;
}

int add(int a, int b)   // 形参a与b也算是块作用域
{
    return a + b;
}

int test()
{
LABEL1: //函数作用域
    cout << "label1" << endl;
    goto LABEL3;
LABEL2:
    cout << "label2" << endl;
    goto LABEL1;
LABEL3:
    cout << "label3" << endl;
    goto LABEL2;
}

(2)、C++中类必须先定义,才能够实例化。两个类需要相互引用头文件形成一个“环形”引用时会出错。这时候需要用到前向声明,前向声明的类不能实例,但可以定义指针或引用。

 C++ Code 
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
 
#ifndef _B_H_
#define _B_H_

class A;

class B
{
public:
    B(void);
    ~B(void);

void fun(A &a)
    {

}

A *a_;      // 前向声明的类不能实例化对象
};

#endif // _B_H_

 C++ Code 
1
2
3
4
5
6
7
8
9
10
11
12
13
14
 
#ifndef _A_H_
#define _A_H_

#include "B.h"
class A
{
public:
    A(void);
    ~A(void);

B b_;
};

#endif // _A_H_

六、嵌套类、局部类

(1)、嵌套类

外围类需要使用嵌套类对象作为底层实现,并且该嵌套类只用于外围类的实现,且同时可以对用户隐藏该底层实现。

从作用域的角度看,嵌套类被隐藏在外围类之中,该类名只能在外围类中使用。如果在外围类之外的作用域使用该类名时,需要加名字限定。

嵌套类中的成员函数可以在它的类体外定义。

嵌套类的成员函数对外围类的私有成员没有访问权,反之亦然。

嵌套类仅仅只是语法上的嵌入

(2)、局部类

类也可以定义在函数体内,这样的类被称为局部类(local class)。局部类只在定义它的局部域内可见。

局部类的成员函数必须被定义在类体中。

局部类中不能有静态成员,关于类中的静态成员和静态成员函数以后再谈。

 C++ Code 
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
 
#include <iostream>
using namespace std;

class Outer
{
public:
    class Inner
    {
    public:
        void Fun();
        //{
        //  cout<<"Inner::Fun ..."<<endl;
        //}
    };
public:
    Inner obj_;
    void Fun()
    {
        cout << "Outer::Fun ..." << endl;
        obj_.Fun();
    }
};

void Outer::Inner::Fun()
{
    cout << "Inner::Fun ..." << endl;
}

void Fun()
{
    class LocalClass
    {
    public:
        int num_;
        void Init(int num)
        {
            num_ = num;
        }
        void Display()
        {
            cout << "num=" << num_ << endl;
        }

//static int num2_; // 局部类内部不能定义静态成员
    };

LocalClass lc;
    lc.Init(10);
    lc.Display();
}

int main(void)
{
    Outer o;
    o.Fun();

Outer::Inner i;
    i.Fun();

Fun();
    //LocalClass lc;        Error,局部类只能在定义它的函数体中使用
    return 0;
}

七、PIMPL 技法

来看下面的示例:

 C++ Code 
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
 
// file y.h
#include "x.h"
class Y
{
    void Fun();
    X x_;
};
// file y.cpp
#include "y.h"
void Y::Fun
{
    return x_.Fun();
}
// file main.cpp
#include “y.h”
int main(void)
{
    Y y;
    y.Fun();
}

上面程序存在的问题是:

1、引入更多的头文件,降低编译速度

2、在编译期如果X的大小改变了,y.cpp 和 main.cpp 都得重新编译;在运行期,如果X有子类,也不能使用多态虚函数。

3、假设y.cpp 编译成动态库给main.cpp 使用,当X的大小变化,动态库需要重新编译,此时main.cpp 因为有定义对象y ,故也需要

重新编译。

下面介绍一种PIMPL 技法,有人也把它当作一种设计模式:

PIMPL(private implementation或pointer to implementation)也称为handle/body idiom

PIMPL背后的思想是把客户与所有关于类的私有部分的知识隔离开。避免其它类知道其内部结构

降低编译依赖、提高重编译速度

接口和实现分离

降低模块的耦合度

编译期

运行期

提高了接口的稳定程度

对于库的使用,方法不用改变

对于库的编译,动态库的变更,客户程序不用重新编译

修改后的程序:

 C++ Code 
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
 
// file y.h
class X;
class Y
{
    Y();
    ~Y();
    void Fun();
    X *px_;
};
// file y.cpp
#include "x.h"
Y::Y() : px_( new X ) {}
Y::~Y()
{
    delete px_;
    px_ = 0;
}
void Y::Fun()
{
    return px_->Fun();
}
//  file main.cpp
#include "y.h"
int main(void)
{
    Y y;
    y.Fun();
}

即Y 内部成员是X* 指针,在32位系统上,指针大小固定为4个字节,即使X大小改变,也不影响Y。如果X 有子类,通过基类指针px_

还可以实现虚函数多态。

参考:

C++ primer 第四版
Effective C++ 3rd
C++编程规范

类声明、类作用域、前向声明、this指针、嵌套类、PIMPL 技法 等的更多相关文章

  1. C++ 类 & 对象-C++ 内联函数-C++ this 指针-C++ 类的静态成员

    C++ 内联函数 C++ 内联函数是通常与类一起使用.如果一个函数是内联的,那么在编译时,编译器会把该函数的代码副本放置在每个调用该函数的地方. 对内联函数进行任何修改,都需要重新编译函数的所有客户端 ...

  2. C++中类的前向声明

    概念 可以声明一个类而不是定义它; class Screen; 这个声明被称为"前向声明".在声明之后,定义之前,类screen是一个不完全类型,即已知Screen是一个类型,但不 ...

  3. C++拾遗(十三)友元和嵌套类

    友元类 使用友元的场合: 1.两个类既不是is-a关系也不是has-a关系,但是两个类之间又需要有联系,且一个类能访问另一个类的私有成员和保护成员. 2.一个类需要用到另外多个类的私有成员. C++p ...

  4. C++学习之嵌套类和局部类

    C++学习之嵌套类和局部类 局部类 在一个函数体内定义的类称为局部类. 局部类中只能使用它的外围作用域中的对象和函数进行联系,因为外围作用域中的变量与该局部类的对象无关.在定义局部类时需要注意:局部类 ...

  5. 嵌套类,PIMPL

    body, table{font-family: 微软雅黑; font-size: 10pt} table{border-collapse: collapse; border: solid gray; ...

  6. final与 static的区别;static代码块以及嵌套类介绍

    本篇文章主要分为两个模块进行介绍:1.final,staic,static final之间的异同:2. static 模块:3.嵌套类的概念 1.final,staic,static final之间的 ...

  7. C++ 嵌套类使用(三)

    如果嵌套类型和其外部类型之间的关系需要成员可访问性语义,需要使用C++嵌套类,嵌套类型不应针对其声明类型以外的类型执行任务,而C++局部类允许类.结构和接口被分成多个小块儿并存储在不同的源文件中,这样 ...

  8. C++ 嵌套类使用(一)

    一.嵌套类 在一个类的内部定义另一个类,我们称之为嵌套类(nested class),或者嵌套类型.之所以引入这样一个嵌套类,往往是因为外围类需要使用嵌套类对象作为底层实现,并且该嵌套类只用于外围类的 ...

  9. C++学习笔记(十二):类继承、虚函数、纯虚函数、抽象类和嵌套类

    类继承 在C++类继承中,一个派生类可以从一个基类派生,也可以从多个基类派生. 从一个基类派生的继承称为单继承:从多个基类派生的继承称为多继承. //单继承的定义 class B:public A { ...

随机推荐

  1. 在WPF中实现玻璃模糊效果

    在WPF中实现玻璃模糊效果还是比较简单的,主要方式如下: 添加一个Rectangle或其它控件作为玻璃放到顶部图层 将底部图像作为Brush(大多数的时候用VisualBrush)填充到Rectang ...

  2. BZOJ 1032 JSOI 2007 祖码Zuma 区间DP

    题目大意:依照祖玛的玩法(任意选颜色),给出一段区间.问最少用多少个球可以把全部颜色块都消除. 思路:把输入数据依照连续的块处理.保存成颜色和数量.然后用这个来DP.我们知道,一个单独的块须要两个同样 ...

  3. 强悍的javascript手势库

    /** * Toucher * git:https://github.com/cometwo/Toucher-1 */ "use strict"; (function (root, ...

  4. 判断终端是ios还是安卓的一些妙用(附加微信分享图标修改)

    最近遇到一个项目 要求有两个icon(就是下载地址 下载安卓的apk  和ios的安装包) 一开始的方案是 什么设备都显示这两个icon 但是后来老大说这样不好   安卓用户给他下载ios 也不行  ...

  5. 三款工作流引擎比较:WWF、netBPM 和 ccflow

    下面将对目前比较主流的三款工作流进行介绍和比较,然后通过三款流程引擎分别设计一个较典型的流程来给大家分别演示这三款创建流程的过程.这三款工作流程引擎分别是 Windows Workflow Found ...

  6. 【IntellJ IDEA】idea上所有代码都报错了

    可能会碰到蓝屏,内存溢出重启idea等特殊情况. 重新打开idea后发现原本的代码全都报错了 正确的解决方法: 方法很简单 执行idea工具栏上下面的菜单: File -> Invalidate ...

  7. 打造android万能上拉下拉刷新框架——XRefreshView (二)

    打造Android万能上拉下拉刷新框架--XRefreshView(一) 打造Android万能上拉下拉刷新框架--XRefreshView(三) 一.前言 自从上次发表了打造android万能上拉下 ...

  8. javascript比较两个时间大小

    //var yourtime=document.getElementById('begin_time').value; var yourtime='2010-12-10 11:12'; yourtim ...

  9. Android 下文件cannot execute - Permission denied

    安卓下执行交叉编译的可执行文件发现提示不允许. 原因是mount的方式问题,root后运行 su mount -o rw,remount /mnt/sdcard 就可以了 mount -o rw,re ...

  10. 6)Linux程序设计入门--消息管理

    )Linux程序设计入门--消息管理 前言:Linux下的进程通信(IPC) Linux下的进程通信(IPC) POSIX无名信号量 System V信号量 System V消息队列 System V ...