C++(学习)2024.9.20

server/2024/9/24 8:28:34/

目录

C++面向对象的基础知识

this指针

概念

功能

1.类内调用成员

2.区分重名的成员变量和局部变量

3. 链式调用

static%E5%85%B3%E9%94%AE%E5%AD%97-toc" style="margin-left:40px;">static关键字

1.静态局部变量

2.静态成员变量

3.静态成员函数

4.单例设计模式

const%E5%85%B3%E9%94%AE%E5%AD%97-toc" style="margin-left:40px;">const关键字

const%E4%BF%AE%E9%A5%B0%E6%88%90%E5%91%98%E5%87%BD%E6%95%B0-toc" style="margin-left:80px;">1.const修饰成员函数

const%E4%BF%AE%E9%A5%B0%E5%AF%B9%E8%B1%A1-toc" style="margin-left:80px;">2.const修饰对象

const%E4%BF%AE%E9%A5%B0%E6%88%90%E5%91%98%E5%8F%98%E9%87%8F-toc" style="margin-left:80px;">3.const修饰成员变量

const%E4%BF%AE%E9%A5%B0%E5%B1%80%E9%83%A8%E5%8F%98%E9%87%8F-toc" style="margin-left:80px;">4.const修饰局部变量

运算符重载

友元

1.概念

2.友元函数

3.友元类

4.友元成员函数


C++面向对象的基础知识

this指针

概念

this指针是一个特殊的指针,指向当前类对象的首地址。

成员函数(包括构造函数和析构函数)中都有this指针,因此this指针只能在类内使用。实际上this指针指向的就是当前运行的成员函数所绑定的对象。

#include <iostream>
using namespace std;
class Test
{
public:void test_this(){cout << this << endl;}
};
int main()
{Test t1;cout << &t1 << endl;t1.test_this();Test t2;cout << &t2 << endl;t2.test_this();Test *t3 = new Test;cout << t3 << endl;t3->test_this();delete t3;return 0;
}

功能

1.类内调用成员

成员(成员变量+成员函数)必须由对象调用。类中成员的调用都依赖于this指针的,通常由编译器自动添加。

#include <iostream>
using namespace std;
class Test
{
private:string name;
public:Test(string n){this->name = n;}string get_name(){return this->name;}
};
int main()
{Test t1("zhangsan");cout << t1.get_name() << endl;return 0;
}
2.区分重名的成员变量和局部变量
#include <iostream>
using namespace std;
class Test
{
private:string name;
public:Test(string name){this->name = name;  // 通过this指针在函数体中进行区分}string get_name(){return name;}
};
int main()
{Test t1("zhangsan");cout << t1.get_name() << endl;return 0;
}
3. 链式调用

支持链式调用的成员函数特点:
●当一个成员函数的返回值是当前类型的引用时,往往表示这个函数支持链式调用。
●return后面是*this。

#include <iostream>
using namespace std;
class Test
{
private:int val = 0;
public:Test &add(int i){val += i;return *this;}int get_val(){return val;}
};
int main()
{Test t1;t1.add(1);t1.add(2);t1.add(100);cout << t1.get_val() << endl;   // 103// 链式调用Test t2;cout << t2.add(2).add(3).add(200).get_val() << endl;return 0;
}

static%E5%85%B3%E9%94%AE%E5%AD%97">static关键字

1.静态局部变量

使用static修饰局部变量,这样的变量就是静态局部变量。
静态局部变量在第一次调用时创建,直到程序结束后销毁,同一个类的所有对象共用这一份静态局部变量。

#include <iostream>
using namespace std;
class Test
{
public:void func(){int a = 1;static int b = 1;cout << "a=" << ++a << " " << &a << endl;cout << "b=" << ++b << " " << &b << endl;}
};
int main()
{Test t1;t1.func();t1.func();t1.func();Test t2;t2.func();Test *t3 = new Test;t3->func();return 0;
}

注意:a可能会在同一个内存地址反复创建销毁。

2.静态成员变量

        使用static修饰成员变量,这样的变量就是静态成员变量。

        需要在类内声明,类外初始化。
        一个类的所有对象共用一份静态成员变量。虽然静态成员变量可以使用对象调用的,但是更建议直接使用类名调用。所以静态成员变量可以脱离对象使用,在程序开始运行时就开辟内存空间,程序执行结束后销毁。
        更推荐使用类名直接调用,代码的可读性更高。

#include <iostream>
using namespace std;
class Test
{
public:int a = 1;static int b;
};
// 静态成员变量,类外初始化
int Test::b = 1;int main()
{cout << Test::b << &Test::b << endl;Test t1;cout << t1.a++ << &t1.a << endl;cout << t1.b++ << &t1.b << endl;cout << " --------------- " << endl;Test t2;cout << t2.a++ << &t2.a << endl;cout << t2.b++ << &t2.b << endl; return 0;
}

3.静态成员函数

使用static修饰成员函数,这样的函数就是静态成员函数。
与静态成员变量相似的有:    
        都可以通过类名直接调用,也可以通过对象进行调用,也可以脱离对象使用。
        静态成员函数没有this指针,不能再静态成员函数中调用同类中其他非静态成员,但是静态成员函数可以调用静态成员。

#include <iostream>
using namespace std;
class Test
{
public:void func0(){//func1();    // 非静态成员函数可以调用静态成员函数cout << "非静态成员函数" << endl;}static void func1(Test &t){Test t2;t2.func0();t.func0();//func0();     // 错误 静态成员函数,不能调用非静态成员函数cout << "静态成员函数1" << endl;}
};int main()
{Test t1;Test t;t1.func1(t);return 0;
}

        静态成员函数内调用当前类的非静态成员需要通过参数将对象传递进来,也可以在函数内创建对象进行调用。

4.单例设计模式

        设计模式是一套被反复使用、多人知晓、经过分类的、代码设计经验的总结。通常用于一些面向对象的语言,如:JAVA、C++、C#等。这里以一个简化版本的单例设计模式为例,讲解static的实际使用。

#include <iostream>
using namespace std;
class Singleton
{
private:Singleton(){}Singleton(const Singleton&);static Singleton *instance; // 静态成员变量
public:static Singleton* get_instance()    // 静态成员函数{if(instance == NULL){instance = new Singleton;}return instance;}static void delete_instance(){if(instance != NULL){delete instance;instance = NULL;}}
};
Singleton *Singleton::instance = NULL;
int main()
{Singleton *s1 =  Singleton::get_instance();Singleton *s2 =  Singleton::get_instance();cout << s1 << endl;cout << s2 << endl;return 0;
}

const%E5%85%B3%E9%94%AE%E5%AD%97">const关键字

const%E4%BF%AE%E9%A5%B0%E6%88%90%E5%91%98%E5%87%BD%E6%95%B0">1.const修饰成员函数

const修饰的成员函数,表示常成员函数。
特性:
(1)可以调用成员变量,但是不能修改成员变量的值
(2)不能调用非const的修饰的成员函数,哪怕这个函数并没有修改成员变量。

建议只要成员函数不修改成员变量就使用const修饰,例如show、get等等。

#include <iostream>
using namespace std;
class Demo
{
private:int a;
public:Demo(int a){this->a = a;}void func0(){cout << "hello" << endl;}int get_demo()const{return a;}void test()const    // 常成员函数{//a++;    // 错误 const修饰的成员函数,不能修改成员变量cout << a << endl;//func0();    // 错误,const修饰的成员函数,不能调用非const修饰的成员函数get_demo();}
};
int main()
{Demo demo(1);demo.test();cout << demo.get_demo() << endl;return 0;
}

const%E4%BF%AE%E9%A5%B0%E5%AF%B9%E8%B1%A1">2.const修饰对象

const修饰的对象被称为常量对象,这种对象的成员变量值无法被修改,也无法调用非const的成员函数。

#include <iostream>
using namespace std;
class Demo
{
private:int a;
public:int b = 20;Demo(int a){this->a = a;}void func0(){cout << "hello" << endl;}int get_demo()const{return a;}void test()const     // 常成员函数{//a++;     // 错误 const修饰的成员函数,不能修改成员变量cout << a << endl;//func0();     // 错误,const修饰的成员函数,不能调用非const修饰的成员函数get_demo();}};int main()
{const Demo demo(1);    //Demo const demo(1);cout << demo.get_demo() << endl; //demo.func0();       // 错误,const修饰对象,无法调用非const修饰的成员函数demo.test();cout << demo.b << endl;//demo.b = 10;     // 错误,const修饰的对象,无法修改成员变量return 0;
}

const%E4%BF%AE%E9%A5%B0%E6%88%90%E5%91%98%E5%8F%98%E9%87%8F">3.const修饰成员变量

const修饰的成员变量为常成员变量,表示该成员变量的值无法被修改。
常成员变量有两种初始化方式:
(1)声明后直接赋值
(2)构造初始化列表

上述两种方式同时使用时,以构造初始化列表为准。

#include <iostream>
using namespace std;
class Demo
{
private:const int a = 1;const int b = 2;const int c = 3;
public:Demo(int a,int b,int c):a(a),b(b),c(c){}void show(){cout << a << " " << b << " " << c << endl;}
};int main()
{Demo d1(10,20,30);d1.show();return 0;
}

const%E4%BF%AE%E9%A5%B0%E5%B1%80%E9%83%A8%E5%8F%98%E9%87%8F">4.const修饰局部变量

const修饰局部变量,表示该局部变量不可被修改。 这种方式常用于引用参数。

#include <iostream>
using namespace std;class Demo
{
private:const int a = 1;const int b = 2;const int c = 3;
public:Demo(int a,int b,int c):a(a),b(b),c(c){}void show(){cout << a << " " << b << " " << c << endl;}void test(const int &a1){a1++;const int d = 10;}
};int main()
{int a = 1;Demo d1(10,20,30);d1.show();d1.test(a);return 0;
}

运算符重载

友元

1.概念

        类实现了数据的隐藏和封装,类的数据成员一般定义为私有成员,仅能通过类的成员函数才能读写。如果数据成员定义为公共的,则又破坏了封装性。但是再某些情况下,需要频繁的读写类的数据成员,特别是在对某些成员函数多次调用时,由于参数传递,类型检查和安全性检查都需要时间的开销,而影响程序的运行效率。

友元有三种实现方式:

        1.友元函数
        2.友元类
        3.友元成员函数

        友元在于提高程序的运行效率,但是,它破坏了类的封装性和隐藏性,是的非成员函数能够访问类的私有成员,导致程序维护性变差,因此使用友元要慎重。

2.友元函数

        友元函数不属于任何一个类,是一个类外的函数,但是需要在类内进行“声明”。虽然友元函数不是类中的函数,但是却可以访问类中的所有成员(包括私有成员)

#include <iostream>
using namespace std;
class Test
{
private:int a;
public:Test(int i):a(i){}void show(){cout << a << " " << &a << endl;}// 友元函数,类内声明friend void and_test(Test &t);
};// 友元函数
void and_test(Test &t)
{cout << ++t.a << " " << &t.a << endl;
}int main()
{Test t1(1);and_test(t1);t1.show();return 0;
}

友元函数的使用注意以下几点:
1.友元函数没有this指针
2.友元函数的“声明”可以放置到类中的任何位置,不受权限修饰符的影响。
3.一个友元函数理论上可以访问多个类,只需要再各个类中分别“声明”。

3.友元类

        当一个类成为了另一个类Test的朋友时,类Test的所有成员都可以被类B访问,此时类B就是类Test的友元类。

#include <iostream>
using namespace std;
class Test
{
private:int a;
public:Test(int i):a(i){}void show(){cout << a << " " << &a << endl;}// 友元类,类内声明friend class B;
};
class B
{
public:void and_test(Test &t){++t.a;cout << t.a << " " << &t.a << endl;}void and_test1(Test &t){cout << ++t.a << " " << &t.a << endl;}
};
int main()
{Test t1(1);B b;b.and_test(t1);b.and_test1(t1);t1.show();return 0;
}

友元类的使用需要注意以下几点:
1.友元关系不能被继承
2.友元关系不具有交换性(比如:类B声明为类Test的友元,类B可以访问类Test中的成员,但是类Test不能访问类B中的私有成员,如果需要访问,需要将类Test声明成类B的友元)
互为友元代码,需要类内声明,类外实现。

#include <iostream>
using namespace std;
class Cat;
class Test
{
private:int a;
public:Test(int i):a(i){}void test(Cat &c);friend class Cat;
};
class Cat
{
private:int b;
public:Cat(int i):b(i){}void test1(Test &t);friend class Test;
};
void Test::test(Cat &c)
{cout <<c.b<<endl;
}
void Cat::test1(Test &t)
{cout <<t.a++<<endl;
}
int main()
{Test t(44);Cat c(12);c.test1(t);return 0;
}

4.友元成员函数

        使类B中的成员函数成为类Test的友元成员函数,这样类B的该成员函数就可以访问类Test的所有成员了。

#include <iostream>
using namespace std;// 第四步:声明被访问的类
class Test;
class B
{
public:// 第二步:声明友元成员函数(类内声明,类外实现)void and_test(Test &t);
};class Test
{
private:int a;
public:Test(int i):a(i){}void show(){cout << a << " " << &a << endl;}// 友元成员函数,第一步:确定友元成员函数的格式并声明friend void B::and_test(Test &t);
};// 第三步:类外定义友元成员函数
void B::and_test(Test &t)
{++t.a;cout << t.a << " " << &t.a << endl;
}int main()
{Test t1(1);B b;b.and_test(t1);t1.show();return 0;
}

http://www.ppmy.cn/server/121256.html

相关文章

GNU链接器(LD):输入分区和输出分区介绍

0 参考资料 GNU-LD-v2.30-中文手册.pdf GNU linker.pdf1 前言 一个完整的编译工具链应该包含以下4个部分&#xff1a; &#xff08;1&#xff09;编译器 &#xff08;2&#xff09;汇编器 &#xff08;3&#xff09;链接器 &#xff08;4&#xff09;lib库 在GNU工具链中&…

[vulnhub] Hackademic.RTB1

第一次打靶机&#xff0c;思路看的红队笔记 https://www.vulnhub.com/entry/hackademic-rtb1,17/ 环境&#xff1a;kali Linux - 192.168.75.131&#xff0c;靶机 - 192.168.75.132 主机发现和端口扫描 扫描整个网络有哪台机子在线&#xff0c;不进行端口扫描 nmap -sP 192.16…

漏洞挖掘 | Selenium Grid 中的 SSRF

Selenium 网格框架上的基本服务器端请求伪造 最近&#xff0c;我正在阅读漏洞文章看到Peter Jaric写的一篇 Selenium Grid 文章&#xff1b;他解释了 Selenium Grid 框架上缺乏身份验证和安全措施强化的问题。 在网上进行了更多搜索&#xff0c;我发现 Selenium Grid 开箱即用…

Webpack 常见配置项

1. entry 指定一个或多个入口点&#xff0c;Webpack 从这里开始构建依赖图。 entry: {main: ./src/index.js,admin: ./src/admin.js }2. output 指定输出文件的路径和名称。 output: {filename: [name].bundle.js,path: path.resolve(__dirname, dist),publicPath: /assets…

通过深度学习识别情绪

通过深度学习识别情绪&#xff08;Emotion Recognition using Deep Learning&#xff09;是一项结合多模态数据的技术&#xff0c;旨在通过分析人类的面部表情、语音语调、文本内容等特征来自动识别情绪状态。情绪识别在人机交互、健康监测、教育、娱乐等领域具有广泛的应用。 …

共享行业使用第三方支付分账系统优势

在共享行业中&#xff0c;选择最适合的分账模式需要考虑多个因素&#xff0c;包括安全性、透明度、灵活性、合规性以及成本效益等。以下是对几种常见分账模式的简要评估&#xff1a; 银行分账系统&#xff1a; 优点&#xff1a;信誉高、安全性好、资金可追踪性强。 缺点&…

【机器学习】——支持向量机

文章目录 支持向量机&#xff08;Support Vector Machine, SVM&#xff09;概述SVM 的工作原理线性不可分数据&#xff1a;软间隔与核技巧SVM 的数学形式SVM 的优势SVM 的缺点SVM 的应用 支持向量机&#xff08;Support Vector Machine, SVM&#xff09;概述 支持向量机&#…

自学前端的正确姿势是...

师傅带进门&#xff0c;修行在个人。 在前端自学成才的道路上&#xff0c;有些人走的很快&#xff0c;有些人却举步维艰。 为什么会这样子呢&#xff1f;因为他们没有掌握自学前端的正确姿势。 在介绍应该要怎样自学前端之前&#xff0c;首先来看下&#xff0c;自学前端容易…