从C语言到C++④(第二章_类和对象_上篇)->类->封装->this指针(中)

简介: 从C语言到C++④(第二章_类和对象_上篇)->类->封装->this指针

从C语言到C++④(第二章_类和对象_上篇)->类->封装->this指针(上):https://developer.aliyun.com/article/1513640

3. 类的作用域和实例化

3.1 类定义的两种方式

① 声明和定义全部放在类中

class Student 
{
public:
  void Init(const char* name, int age, int id) 
    {
    strcpy(_name, name);
    _age = age;
    _id = id;
  }
  void Print() 
    {
    cout << _name << " " << _age << " " << _id << endl;
  }
 
private:
  char _name[10];
  int  _age;
  int  _id;
};

注意事项:

成员函数如果在类中定义,就相当于我们在函数前面加了inline,编译器可能会将它当作内联函数来处理。注意,是可能。并不是说一定会成为内联函数,之前讲内联函数的时候我们也说了。

内联函数对编译器来说也只是一个建议。至于到底会不会成为内联是由编译器来决定的。取决于编译器看这个函数符不符合条件,一般一个函数太长(大概是十行左右),或者函数是一个递归,编译器就不会让它成为内联了。

② 声明和定义分离:

比如创建一个学生类

Student.h:

class Student 
{
public:
  void Init(const char* name, int age, int id);
  void Print();
 
private:
  char _name[10];
  int  _age;
  int  _id;
};

Student.cpp:

#include "Student.h"
 
void Student::Init(const char* name, int age, int id) 
{
    strcpy(_name, name);
  _age = age;
  _id = id;
}
 
void Student::Print() 
{
  cout << _name << " " << _age << " " << _id << endl;
}

这函数名前的 : : 是什么?继续往下看。


3.2 类的作用域

类定义了一个新的作用域,类的所有成员都在类的作用域中。

在类外定义成员,需要使用作用域解析符 : : 来指明成员属于哪个类域。

class Student
{
public:
  void PrintPersonInfo();
private:
  char _name[20];
  char _gender[3];
  int _age;
};
// 这里需要指定PrintPersonInfo是属于Person这个类域
void Student::PrintPersonInfo()
{
  cout << _name << " " << _gender << " " << _age << endl;
}

注意到这样写不会报错

class Student
{
public:
  void PrintPersonInfo();
private:
  char _name[20];
  char _gender[3];
  int _age;
};
 
void PrintPersonInfo()//不会和上面的函数冲突,因为两者属于不同的域
{
  //cout << _name << " " << _gender << " " << _age << endl;
    //没注释掉就是未定义的标识符
}

我们再看下函数重载的定义:

函数重载:C++ 允许在同一个作用域中存在同名的函数。

下面三个不同只要满足一个不同,就可以触发函数重载:

① 参数类型不同

② 参数个数不同

③ 参数顺序不同

所以这里不报错也不会发生函数重载


还有像前面说过比如说我们写一个比较正式一点的项目(声明和定义分离)

Stack.h:

class Stack
{
public:
  void Init();
  void Push(int x);
  // ...
 
private:
  int* _array;
  int  _top;
  int  _capacity;
};

Stack.cpp:

#include "Stack.h"
// 这里需要指定 Init 是属于 Stack 这个类域
// 我们使用 :: 来指定
void Stack::Init() 
{
  _array = nullptr;
  _top = _capacity = 0;
}

这里我们平常练习为了方便使用就不使用声明和定义分离的方式了。


3.3 类的实例化

首先要说清楚的是:类本身是没有存储空间的。

通过类建立出对象,即实例化,才会有实际的存储空间。

我们把用类类型创建对象的过程称为 —— 类的实例化。

① 类只是一个像 "图纸" 一样的东西,限定了类有哪些成员。定义出一个类,

并没有分配实际的内存空间来存储它。

② 一个类可以实例化出多个对象,占用实际的物理空间,存储类成员的变量。

举个例子:

实例化对象就像是在现实中使用设计图建造房子,类就像是设计图。

你可以根据这个设计图造出很多栋楼出来。

只有楼建好了里面才能住人,你才能放家具、放东西进去。

设计图能住人吗?当然是不能,因为并没有分配实际的内存空间。

只有你照着设计图去建造,即实例化出对象,占用实际的物理空间,才能住人。


3.4 声明和定义的区别

函数的声明和定义直接看有没有花括号就行了,但变量呢?

对于变量而言,声明和定义的区别就是有没有开辟空间

对于变量而言如果要声明和定义分离在.h文件要+extern关键字变成声明(或者+static)

class Stack
{
public:
  void Init();//没开空间,是声明
  void Push(int x);//没开空间,是声明
  // ...
 
private:
  int* _array;//没开空间,是声明
  int  _top;//没开空间,是声明
  int  _capacity;//没开空间,是声明
};
 
int capacity;//开空间了,是定义
 
void Stack::Init() //开空间了,是定义
{
  _array = nullptr;
  _top = _capacity = 0;
}

从C语言到C++④(第二章_类和对象_上篇)->类->封装->this指针(下):https://developer.aliyun.com/article/1513645?spm=a2c6h.13148508.setting.16.5e0d4f0eApSShM


目录
相关文章
|
3月前
|
安全 C语言 C++
比较C++的内存分配与管理方式new/delete与C语言中的malloc/realloc/calloc/free。
在实用性方面,C++的内存管理方式提供了面向对象的特性,它是处理构造和析构、需要类型安全和异常处理的首选方案。而C语言的内存管理函数适用于简单的内存分配,例如分配原始内存块或复杂性较低的数据结构,没有构造和析构的要求。当从C迁移到C++,或在C++中使用C代码时,了解两种内存管理方式的差异非常重要。
128 26
|
8月前
|
编译器 C++ 开发者
【C++篇】深度解析类与对象(下)
在上一篇博客中,我们学习了C++的基础类与对象概念,包括类的定义、对象的使用和构造函数的作用。在这一篇,我们将深入探讨C++类的一些重要特性,如构造函数的高级用法、类型转换、static成员、友元、内部类、匿名对象,以及对象拷贝优化等。这些内容可以帮助你更好地理解和应用面向对象编程的核心理念,提升代码的健壮性、灵活性和可维护性。
|
8月前
|
编译器 C语言 C++
类和对象的简述(c++篇)
类和对象的简述(c++篇)
|
7月前
|
编译器 C++
类和对象(中 )C++
本文详细讲解了C++中的默认成员函数,包括构造函数、析构函数、拷贝构造函数、赋值运算符重载和取地址运算符重载等内容。重点分析了各函数的特点、使用场景及相互关系,如构造函数的主要任务是初始化对象,而非创建空间;析构函数用于清理资源;拷贝构造与赋值运算符的区别在于前者用于创建新对象,后者用于已存在的对象赋值。同时,文章还探讨了运算符重载的规则及其应用场景,并通过实例加深理解。最后强调,若类中存在资源管理,需显式定义拷贝构造和赋值运算符以避免浅拷贝问题。
|
7月前
|
存储 编译器 C++
类和对象(上)(C++)
本篇内容主要讲解了C++中类的相关知识,包括类的定义、实例化及this指针的作用。详细说明了类的定义格式、成员函数默认为inline、访问限定符(public、protected、private)的使用规则,以及class与struct的区别。同时分析了类实例化的概念,对象大小的计算规则和内存对齐原则。最后介绍了this指针的工作机制,解释了成员函数如何通过隐含的this指针区分不同对象的数据。这些知识点帮助我们更好地理解C++中类的封装性和对象的实现原理。
|
7月前
|
编译器 C++
类和对象(下)C++
本内容主要讲解C++中的初始化列表、类型转换、静态成员、友元、内部类、匿名对象及对象拷贝时的编译器优化。初始化列表用于成员变量定义初始化,尤其对引用、const及无默认构造函数的类类型变量至关重要。类型转换中,`explicit`可禁用隐式转换。静态成员属类而非对象,受访问限定符约束。内部类是独立类,可增强封装性。匿名对象生命周期短,常用于临时场景。编译器会优化对象拷贝以提高效率。最后,鼓励大家通过重复练习提升技能!
|
8月前
|
安全 编译器 C语言
【C++篇】深度解析类与对象(中)
在上一篇博客中,我们学习了C++类与对象的基础内容。这一次,我们将深入探讨C++类的关键特性,包括构造函数、析构函数、拷贝构造函数、赋值运算符重载、以及取地址运算符的重载。这些内容是理解面向对象编程的关键,也帮助我们更好地掌握C++内存管理的细节和编码的高级技巧。
|
8月前
|
存储 程序员 C语言
【C++篇】深度解析类与对象(上)
在C++中,类和对象是面向对象编程的基础组成部分。通过类,程序员可以对现实世界的实体进行模拟和抽象。类的基本概念包括成员变量、成员函数、访问控制等。本篇博客将介绍C++类与对象的基础知识,为后续学习打下良好的基础。
|
11月前
|
存储 C语言
C语言如何使用结构体和指针来操作动态分配的内存
在C语言中,通过定义结构体并使用指向该结构体的指针,可以对动态分配的内存进行操作。首先利用 `malloc` 或 `calloc` 分配内存,然后通过指针访问和修改结构体成员,最后用 `free` 释放内存,实现资源的有效管理。
887 13
|
存储 人工智能 C语言
C语言程序设计核心详解 第八章 指针超详细讲解_指针变量_二维数组指针_指向字符串指针
本文详细讲解了C语言中的指针,包括指针变量的定义与引用、指向数组及字符串的指针变量等。首先介绍了指针变量的基本概念和定义格式,随后通过多个示例展示了如何使用指针变量来操作普通变量、数组和字符串。文章还深入探讨了指向函数的指针变量以及指针数组的概念,并解释了空指针的意义和使用场景。通过丰富的代码示例和图形化展示,帮助读者更好地理解和掌握C语言中的指针知识。
439 4

热门文章

最新文章