96SEO 2026-02-19 17:18 0
之前我们学习了类中的一些默认成员函数构造函数、析构函数、拷贝构造函数、赋值重载。

今天我们接着学习剩下的取地址运算符重载以及其他关于类和对象的知识。
取地址运算符重载分为两种普通对象的取地址重载和const对象取地址重载。
为了说明这两种取地址重载的区别我们首先引入一个概念const修饰成员函数。
在c中成员函数可以被const修饰修饰时要将const写在成员函数参数列表的后面。
例如
const修饰成员函数的本质是修饰this指针指向的内容它的作用是防止该函数内部对成员变量的值进行修改。
对于一个普通成员函数const对象是无法调用的因为const对象的成员变量不允许被修改而当成员函数被const修饰时就确保了函数内部不会修改成员变量的值const对象就可以调用该函数。
普通对象的取地址重载用于返回普通对象的地址而const对象的取地址重载用于返回const对象的地址。
两种重载函数的区别是前者没有被const修饰后者被const修饰。
这就使得两个函数构成了重载便于不同的对象调用。
我们简单实现一下这两个函数
const//为保证类型匹配返回值也要用const修饰{return
一般情况下编译器自动生成的取地址重载函数我们就可以直接使用不需要显示实现了。
当我们不希望使用者能够获取到对象的地址时可以显示实现取地址重载并将空指针或者野指针作为返回值。
之前我们已经学习了构造函数的特点、使用规则等知识不过构造函数的知识还不止这些接下来我们对之前构造函数的内容进行一些补充。
之前我们在实现构造函数时都是在函数体内部对成员变量赋初值实际上对成员变量进行初始化的方式还有一种初始化列表。
它位于构造函数的参数列表之后函数体大括号之前。
它的使用方式是以冒号开始将需要被初始化的成员以逗号分隔成员之后写一个放在括号当中的值或者表达式用于初始化成员。
例如
初始化列表初始化的顺序与成员在类中的声明顺序一致而与列表中的成员顺序无关。
一下三种变量必须在初始化列表中进行初始化否则会编译报错引用类型的成员变量、const成员变量、不存在默认构造的类类型成员变量。
并不是指我们一定要将该变量显示写在初始化列表中我们也可以使用如下方式
可以看到对于const成员“_c”我们并没有显示在初始化列表中对其进行初始化而是在其声明时为其赋了一个缺省值初值。
这是c11规定的语法该初值是给初始化列表的当初始化列表当中没有显示对一个成员进行初始化时如果声明时有缺省值则会用这个值进行初始化本质也是通过初始化列表初始化只不过并没有显示写出所以程序并不会发生报错。
当然对于普通成员我们也可以在声明时赋缺省值但是相比显示写在初始化列表当中会有一些效率的损耗。
注对类类型的成员变量通过初始化列表进行初始化时本质也是在调用它的构造函数。
如果我们既没有显示地在初始化列表对成员进行初始化也没有在声明时赋缺省值那么对于内置类型的成员当对象被创建时编译器一般不会对其初始化对于自定义类型的成员对象被创建时就会调用它的默认构造函数如果没有默认构造函数就会发生报错。
上述程序中我们创建了一个MyClass类对象m并且将其初始化为1。
我们都知道它的本质是在调用构造函数不过它的运行过程并不是这么简单。
在
”是MyClass类型这个过程中就需要发生类型转换。
程序首先会调用构造函数将“
”构造为MyClass类型的一个临时对象然后将该临时对象拷贝构造给m。
对于这种调用构造函数调用拷贝构造的情况编译器会将其优化为直接调用构造函数所以我们无法感受到类型转换的过程但它的确是存在的。
当我们在构造函数之前加上关键字“
”之后就无法调用该构造函数进行隐式类型转换。
当然如果有合适的构造函数类与类之间也可以发生类型转换。
};//大括号赋值的写法C11以后才支持m.Print();return
在C当中static可以修饰成员变量和成员函数它们在面向对象编程中有着很重要的作用。
用static修饰的成员变量叫做静态成员变量。
静态成员变量必须要在类外进行初始化而不是类中因为在类中给的初值是给初始化列表的而静态成员变量不走初始化列表。
例如
注意静态成员变量存储在静态区中不属于任何一个对象而是被所有对象所共享的使用对象或者类的作用域限定符就可以访问到静态成员变量。
当然既然是成员变量也会受到
是公有成员所以我们直接访问到了该变量。
当静态成员变量是私有成员时该如何访问呢这就需要静态成员函数了。
用static修饰的成员函数称之为静态成员函数静态成员函数与普通成员函数的显著区别是它不存在this指针。
由于静态成员函数不存在this指针所以它也就无法访问到普通成员变量只能访问静态成员变量。
当然如果一个成员函数是非静态的它也可以访问静态成员变量。
接下来我们用静态成员函数来访问私有的静态成员变量
接下来我们运用static修饰成员的知识尝试做一个小练习计算123...n的值要求不能使用乘除法、for、while、if、else、switch、case等关键字及条件判断语句A?B:C。
思路分析要计算123...n的值习惯的思路是循环累加或者使用等差数列求和公式但是由于禁止使用乘除法和循环语句这两种方法就行不通了。
那么递归能否解决呢由于递归一定要有限制条件而if...else语句也被禁用了所以递归也是不行的。
那该怎么办呢我们都知道静态成员变量为所有对象所公有那么我们就可以定义一个静态成员变量x每当一个对象被创建出时x的值就1。
这样我们创建出n个对象并将每一次得到的x值累加到另一个变量中就可以算出最终结果了。
public://在构造函数中操作静态变量使得对象被创建时就会累加MyClass(){x;sum
当类中的成员被设置为私有外部无法访问到时友元就可以突破这种封装使得外部可以访问这些私有成员。
友元可以分为友元函数和友元类我们需要使用友元时在函数或类的声明之前加上关键字
并将其放在另一个类宿主类当中。
此时该函数或类就成为了宿主类的友元。
友元函数只是一种声明并不是说一个函数成为了一个类的友元他就是该类的成员函数了。
友元函数可以在类的任意地方声明并不受public等限定符的限制。
友元类的关系是单向且不可传递的比如A是B的友元但不意味着B是A的友元A是B的友元B是C的友元并不意味着A是C的友元。
不难看出将函数或类声明为友元后就可以访问宿主类的私有或保护成员了。
友元有时虽然提供了便利但是它明显是破坏了类的封装性不符合“高类聚低耦合”的设计原则所以实际开发中不宜多用。
如果一个类A定义在另一个类B当中那么类A就成为了类B的内部类。
内部类与全局定义的类相比它受到外部类的类域和访问限定符限制并且默认是外部类的友元类。
这里要注意内部类是一个类定义在令一个类当中而不是将对象作为一个类的成员不要将两者混淆。
内部类的本质也是一种封装的体现当我们需要让一个类B仅供类A使用那么就可以考虑让B成为A的内部类。
b;//受到类域限制声明时要限定类域b.Print(a);return
注意匿名对象的生命周期只有当前一行当程序运行到下一行时该对象就被销毁。
我们来验证一下
可以看到程序在打印hehe之前就已经调用了析构函数意味着匿名对象已经被销毁。
当我们需要创建一个临时对象并且只使用一次时就可以考虑创建匿名对象。
相比普通对象它能够很大限度地简化代码。
今天我们学习了类和对象相关的新概念和知识例如取地址重载、static修饰成员、友元、内部类等它们对于我们深入学习并理解c的后续内容以及实现对象的相关功能有很大帮助。
如果你觉得博主讲的还不错就请留下一个小小的赞在走哦感谢大家的支持❤❤❤
作为专业的SEO优化服务提供商,我们致力于通过科学、系统的搜索引擎优化策略,帮助企业在百度、Google等搜索引擎中获得更高的排名和流量。我们的服务涵盖网站结构优化、内容优化、技术SEO和链接建设等多个维度。
| 服务项目 | 基础套餐 | 标准套餐 | 高级定制 |
|---|---|---|---|
| 关键词优化数量 | 10-20个核心词 | 30-50个核心词+长尾词 | 80-150个全方位覆盖 |
| 内容优化 | 基础页面优化 | 全站内容优化+每月5篇原创 | 个性化内容策略+每月15篇原创 |
| 技术SEO | 基本技术检查 | 全面技术优化+移动适配 | 深度技术重构+性能优化 |
| 外链建设 | 每月5-10条 | 每月20-30条高质量外链 | 每月50+条多渠道外链 |
| 数据报告 | 月度基础报告 | 双周详细报告+分析 | 每周深度报告+策略调整 |
| 效果保障 | 3-6个月见效 | 2-4个月见效 | 1-3个月快速见效 |
我们的SEO优化服务遵循科学严谨的流程,确保每一步都基于数据分析和行业最佳实践:
全面检测网站技术问题、内容质量、竞争对手情况,制定个性化优化方案。
基于用户搜索意图和商业目标,制定全面的关键词矩阵和布局策略。
解决网站技术问题,优化网站结构,提升页面速度和移动端体验。
创作高质量原创内容,优化现有页面,建立内容更新机制。
获取高质量外部链接,建立品牌在线影响力,提升网站权威度。
持续监控排名、流量和转化数据,根据效果调整优化策略。
基于我们服务的客户数据统计,平均优化效果如下:
我们坚信,真正的SEO优化不仅仅是追求排名,而是通过提供优质内容、优化用户体验、建立网站权威,最终实现可持续的业务增长。我们的目标是与客户建立长期合作关系,共同成长。
Demand feedback