【C++核心】运算符重载和文件操作-程序员宅基地

技术标签: c++  java  前端  有Java编程基础学习C++  

一、运算符重载

  1. 运算符重载概念:对已有的运算符重新进行定义,赋予其另一种功能,以适应不同的数据类型。
  2. 运算符重载的位置:可以在全局函数中实现或者在成员函数中实现。
  3. 运算符重载可以发生函数重载。
  4. 对于内置的数据类型的表达式的的运算符是不可能改变的。
  5. 不要滥用运算符重载。

1. 加号运算符重载

加号运算符重载用于实现两个自定义数据类型相加的运算。

class Person {
    
public:
	Person() {
    };
	Person(int a, int b)
	{
    
		this->m_A = a;
		this->m_B = b;
	}
	//成员函数实现 + 号运算符重载
	Person operator+(const Person& p) {
    
		Person temp;
		temp.m_A = this->m_A + p.m_A;
		temp.m_B = this->m_B + p.m_B;
		return temp;
	}


public:
	int m_A;
	int m_B;
};

//全局函数实现 + 号运算符重载
//Person operator+(const Person& p1, const Person& p2) {
    
//	Person temp(0, 0);
//	temp.m_A = p1.m_A + p2.m_A;
//	temp.m_B = p1.m_B + p2.m_B;
//	return temp;
//}

//运算符重载 可以发生函数重载 
Person operator+(const Person& p2, int val)  
{
    
	Person temp;
	temp.m_A = p2.m_A + val;
	temp.m_B = p2.m_B + val;
	return temp;
}

void test() {
    

	Person p1(10, 10);
	Person p2(20, 20);

	//成员函数方式本质上相当于Person p3 = p2.operator+(p1)
	//全局函数方式本质上相当于Person p3 = operator+(p1,p2)
	Person p3 = p2 + p1;  
	cout << "mA:" << p3.m_A << " mB:" << p3.m_B << endl;


	Person p4 = p3 + 10; //相当于 operator+(p3,10)
	cout << "mA:" << p4.m_A << " mB:" << p4.m_B << endl;

}

int main() {
    

	test();

	system("pause");

	return 0;
}

2. 左移运算符重载

  1. 左移运算符重载可以输出自定义数据类型。重载左移运算符配合友元可以实现输出自定义数据类型。
  2. 左移运算符重载只能用全局函数实现,无法用成员函数实现,因为成员函数实现的重载简化写法为p << cout而不是cout << p
  3. 注意:ostream对象只能有一个。
  4. 个人理解:运算符重载的意思其实就是该运算符在碰到【对此运算符进行重载的类的对象】以后,能够以预先定义的方式继续运行。
class Person {
    
	friend ostream& operator<<(ostream& out, Person& p);

public:

	Person(int a, int b)
	{
    
		this->m_A = a;
		this->m_B = b;
	}

	//成员函数 实现不了  p << cout 不是我们想要的效果
	//void operator<<(Person& p){
    
	//}

private:
	int m_A;
	int m_B;
};

//全局函数实现左移重载
//ostream对象只能有一个
ostream& operator<<(ostream& out, Person& p) {
    
	out << "a:" << p.m_A << " b:" << p.m_B;
	return out;
}

void test() {
    

	Person p1(10, 20);

	cout << p1 << "hello world" << endl; //链式编程
}

int main() {
    

	test();

	system("pause");

	return 0;
}

3. 递增运算符重载

  1. 递增运算符重载实现自己的整型数据。
  2. 前置递增返回引用,后置递增返回值。
  3. 后++需要在参数列表里面加上一个intint代表一个占位参数。

class MyInteger {
    

	friend ostream& operator<<(ostream& out, MyInteger myint);

public:
	MyInteger() {
    
		m_Num = 0;
	}
	//前置++
	MyInteger& operator++() {
    
		//先++
		m_Num++;
		//再返回
		return *this;
	}

	//后置++
	MyInteger operator++(int) {
    
		//先返回
		MyInteger temp = *this; //记录当前本身的值,然后让本身的值加1,但是返回的是以前的值,达到先返回后++;
		m_Num++;
		return temp;
	}

private:
	int m_Num;
};


ostream& operator<<(ostream& out, MyInteger myint) {
    
	out << myint.m_Num;
	return out;
}


//前置++ 先++ 再返回
void test01() {
    
	MyInteger myInt;
	cout << ++myInt << endl;
	cout << myInt << endl;
}

//后置++ 先返回 再++
void test02() {
    

	MyInteger myInt;
	cout << myInt++ << endl;
	cout << myInt << endl;
}

int main() {
    

	test01();
	//test02();

	system("pause");

	return 0;
}

4. 赋值运算符重载

C++编译器默认会给类添加四个函数,除了构造函数、析构函数和拷贝构造函数外,还会添加赋值运算符operator=,用来对属性进行值拷贝。此时,如果类中有属性指向堆区,做赋值操作时也会出现深浅拷贝问题。

如果重载operator==函数,保险起见,最好在深拷贝之前先进行释放。

class Person
{
    
public:

	Person(int age)
	{
    
		//将年龄数据开辟到堆区
		m_Age = new int(age);
	}

	//重载赋值运算符 
	Person& operator=(Person &p)
	{
    
		if (m_Age != NULL)
		{
    
			delete m_Age;
			m_Age = NULL;
		}
		//编译器提供的代码是浅拷贝
		//m_Age = p.m_Age;

		//提供深拷贝 解决浅拷贝的问题
		m_Age = new int(*p.m_Age);

		//返回自身
		return *this;
	}


	~Person()
	{
    
		if (m_Age != NULL)
		{
    
			delete m_Age;
			m_Age = NULL;
		}
	}

	//年龄的指针
	int *m_Age;

};


void test01()
{
    
	Person p1(18);

	Person p2(20);

	Person p3(30);

	p3 = p2 = p1; //赋值操作

	cout << "p1的年龄为:" << *p1.m_Age << endl;

	cout << "p2的年龄为:" << *p2.m_Age << endl;

	cout << "p3的年龄为:" << *p3.m_Age << endl;
}

int main() {
    

	test01();

	//int a = 10;
	//int b = 20;
	//int c = 30;

	//c = b = a;
	//cout << "a = " << a << endl;
	//cout << "b = " << b << endl;
	//cout << "c = " << c << endl;

	system("pause");

	return 0;
}

5. 关系运算符重载

关系运算符重载可以让两个自定义类型对象进行对比操作。

class Person
{
    
public:
	Person(string name, int age)
	{
    
		this->m_Name = name;
		this->m_Age = age;
	};

	bool operator==(Person & p)
	{
    
		if (this->m_Name == p.m_Name && this->m_Age == p.m_Age)
		{
    
			return true;
		}
		else
		{
    
			return false;
		}
	}

	bool operator!=(Person & p)
	{
    
		if (this->m_Name == p.m_Name && this->m_Age == p.m_Age)
		{
    
			return false;
		}
		else
		{
    
			return true;
		}
	}

	string m_Name;
	int m_Age;
};

void test01()
{
    
	//int a = 0;
	//int b = 0;

	Person a("孙悟空", 18);
	Person b("孙悟空", 18);

	if (a == b)
	{
    
		cout << "a和b相等" << endl;
	}
	else
	{
    
		cout << "a和b不相等" << endl;
	}

	if (a != b)
	{
    
		cout << "a和b不相等" << endl;
	}
	else
	{
    
		cout << "a和b相等" << endl;
	}
}


int main() {
    

	test01();

	system("pause");

	return 0;
}

6. 函数调用运算符重载

  1. 函数调用运算符()也可以重载。
  2. 由于重载后使用的方式非常像函数的调用,因此称为仿函数。
  3. 仿函数没有固定写法,非常灵活。
class MyPrint
{
    
public:
	void operator()(string text)
	{
    
		cout << text << endl;
	}

};
void test01()
{
    
	//重载的()操作符 也称为仿函数
	MyPrint myFunc;
	myFunc("hello world");
}


class MyAdd
{
    
public:
	int operator()(int v1, int v2)
	{
    
		return v1 + v2;
	}
};

void test02()
{
    
	MyAdd add;
	int ret = add(10, 10);
	cout << "ret = " << ret << endl;

	//匿名对象调用  
	cout << "MyAdd()(100,100) = " << MyAdd()(100, 100) << endl;
}

int main() {
    

	test01();
	test02();

	system("pause");

	return 0;
}

二、文件操作

程序运行时产生的数据都属于临时数据,程序一旦运行结束都会被释放,通过文件可以将数据持久化。C++中对文件操作需要包含头文件 fstream

文件类型分为两种:

  1. 文本文件:文件以文本的ASCII码形式存储在计算机中。
  2. 二进制文件:文件以文本的二进制形式存储在计算机中,用户一般不能直接读懂它们。

文件的打开方式如下(后续会说明如何使用):

打开方式 解释
ios::in 为读文件而打开文件
ios::out 为写文件而打开文件
ios::ate 初始位置:文件尾
ios::app 追加方式写文件
ios::trunc 如果文件存在先删除,再创建
ios::binary 二进制方式

注意:文件打开方式可以配合使用,利用|操作符。比如:用二进制方式写文件ios::binary | ios:: out

1. 文本文件

1.1 写文件

  1. 包含头文件:#include <fstream\>
  2. 创建流对象:ofstream ofs;
  3. 打开文件:ofs.open("文件路径",打开方式);
  4. 写数据:ofs << "写入的数据";
  5. 关闭文件:ofs.close();

注意:如果文件路径没有说明具体路径,则会放在当前源文件的同级目录下。

#include <fstream>

void test01()
{
    
	ofstream ofs;
	ofs.open("test.txt", ios::out);

	ofs << "姓名:张三" << endl;
	ofs << "性别:男" << endl;
	ofs << "年龄:18" << endl;

	ofs.close();
}

int main() {
    

	test01();

	system("pause");

	return 0;
}

1.2 读文件

读文件步骤如下:

  1. 包含头文件:#include <fstream\>
  2. 创建流对象:ifstream ifs;
  3. 打开文件(需要额外判断文件是否打开成功):ifs.open("文件路径",打开方式);
  4. 读数据四种方式读取
  5. 关闭文件:ifs.close();
#include <fstream>
#include <string>
void test01()
{
    
	ifstream ifs;
	ifs.open("test.txt", ios::in);

	if (!ifs.is_open())
	{
    
		cout << "文件打开失败" << endl;
		return;
	}

	//第一种方式
	//char buf[1024] = { 0 };
	//while (ifs >> buf)
	//{
    
	//	cout << buf << endl;
	//}

	//第二种
	//char buf[1024] = { 0 };
	//while (ifs.getline(buf,sizeof(buf)))
	//{
    
	//	cout << buf << endl;
	//}

	//第三种
	//string buf;
	//while (getline(ifs, buf))
	//{
    
	//	cout << buf << endl;
	//}

	char c;
	while ((c = ifs.get()) != EOF)
	{
    
		cout << c;
	}

	ifs.close();


}

int main() {
    

	test01();

	system("pause");

	return 0;
}

2. 二进制文件

如果要以二进制的方式对文件进行读写操作,那么打开方式要指定为ios::binary

2.1 写文件

  1. 二进制方式写文件主要利用流对象调用成员函数write
  2. 函数原型 :ostream& write(const char * buffer,int len);。其中,字符指针buffer指向内存中一段存储空间,len是读写的字节数。
  3. 步骤和写文本文件的步骤大致相同。
#include <fstream>
#include <string>

class Person
{
    
public:
	char m_Name[64];
	int m_Age;
};

//二进制文件  写文件
void test01()
{
    
	//1、包含头文件

	//2、创建输出流对象
	ofstream ofs("person.txt", ios::out | ios::binary);
	
	//3、打开文件
	//ofs.open("person.txt", ios::out | ios::binary);

	Person p = {
    "张三"  , 18};

	//4、写文件
	ofs.write((const char *)&p, sizeof(p));

	//5、关闭文件
	ofs.close();
}

int main() {
    

	test01();

	system("pause");

	return 0;
}

2.2 读文件

  1. 二进制方式读文件主要利用流对象调用成员函数read
  2. 函数原型:istream& read(char *buffer,int len);。其中,字符指针buffer指向内存中一段存储空间,len是读写的字节数。
  3. 步骤和读文本文件的步骤大致相同。
#include <fstream>
#include <string>

class Person
{
    
public:
	char m_Name[64];
	int m_Age;
};

void test01()
{
    
	ifstream ifs("person.txt", ios::in | ios::binary);
	if (!ifs.is_open())
	{
    
		cout << "文件打开失败" << endl;
	}

	Person p;
	ifs.read((char *)&p, sizeof(p));

	cout << "姓名: " << p.m_Name << " 年龄: " << p.m_Age << endl;
}

int main() {
    

	test01();

	system("pause");

	return 0;
}
版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/weixin_44446626/article/details/137749726

智能推荐

攻防世界_难度8_happy_puzzle_攻防世界困难模式攻略图文-程序员宅基地

文章浏览阅读645次。这个肯定是末尾的IDAT了,因为IDAT必须要满了才会开始一下个IDAT,这个明显就是末尾的IDAT了。,对应下面的create_head()代码。,对应下面的create_tail()代码。不要考虑爆破,我已经试了一下,太多情况了。题目来源:UNCTF。_攻防世界困难模式攻略图文

达梦数据库的导出(备份)、导入_达梦数据库导入导出-程序员宅基地

文章浏览阅读2.9k次,点赞3次,收藏10次。偶尔会用到,记录、分享。1. 数据库导出1.1 切换到dmdba用户su - dmdba1.2 进入达梦数据库安装路径的bin目录,执行导库操作  导出语句:./dexp cwy_init/[email protected]:5236 file=cwy_init.dmp log=cwy_init_exp.log 注释:   cwy_init/init_123..._达梦数据库导入导出

js引入kindeditor富文本编辑器的使用_kindeditor.js-程序员宅基地

文章浏览阅读1.9k次。1. 在官网上下载KindEditor文件,可以删掉不需要要到的jsp,asp,asp.net和php文件夹。接着把文件夹放到项目文件目录下。2. 修改html文件,在页面引入js文件:<script type="text/javascript" src="./kindeditor/kindeditor-all.js"></script><script type="text/javascript" src="./kindeditor/lang/zh-CN.js"_kindeditor.js

STM32学习过程记录11——基于STM32G431CBU6硬件SPI+DMA的高效WS2812B控制方法-程序员宅基地

文章浏览阅读2.3k次,点赞6次,收藏14次。SPI的详情简介不必赘述。假设我们通过SPI发送0xAA,我们的数据线就会变为10101010,通过修改不同的内容,即可修改SPI中0和1的持续时间。比如0xF0即为前半周期为高电平,后半周期为低电平的状态。在SPI的通信模式中,CPHA配置会影响该实验,下图展示了不同采样位置的SPI时序图[1]。CPOL = 0,CPHA = 1:CLK空闲状态 = 低电平,数据在下降沿采样,并在上升沿移出CPOL = 0,CPHA = 0:CLK空闲状态 = 低电平,数据在上升沿采样,并在下降沿移出。_stm32g431cbu6

计算机网络-数据链路层_接收方收到链路层数据后,使用crc检验后,余数为0,说明链路层的传输时可靠传输-程序员宅基地

文章浏览阅读1.2k次,点赞2次,收藏8次。数据链路层习题自测问题1.数据链路(即逻辑链路)与链路(即物理链路)有何区别?“电路接通了”与”数据链路接通了”的区别何在?2.数据链路层中的链路控制包括哪些功能?试讨论数据链路层做成可靠的链路层有哪些优点和缺点。3.网络适配器的作用是什么?网络适配器工作在哪一层?4.数据链路层的三个基本问题(帧定界、透明传输和差错检测)为什么都必须加以解决?5.如果在数据链路层不进行帧定界,会发生什么问题?6.PPP协议的主要特点是什么?为什么PPP不使用帧的编号?PPP适用于什么情况?为什么PPP协议不_接收方收到链路层数据后,使用crc检验后,余数为0,说明链路层的传输时可靠传输

软件测试工程师移民加拿大_无证移民,未受过软件工程师的教育(第1部分)-程序员宅基地

文章浏览阅读587次。软件测试工程师移民加拿大 无证移民,未受过软件工程师的教育(第1部分) (Undocumented Immigrant With No Education to Software Engineer(Part 1))Before I start, I want you to please bear with me on the way I write, I have very little gen...

随便推点

Thinkpad X250 secure boot failed 启动失败问题解决_安装完系统提示secureboot failure-程序员宅基地

文章浏览阅读304次。Thinkpad X250笔记本电脑,装的是FreeBSD,进入BIOS修改虚拟化配置(其后可能是误设置了安全开机),保存退出后系统无法启动,显示:secure boot failed ,把自己惊出一身冷汗,因为这台笔记本刚好还没开始做备份.....根据错误提示,到bios里面去找相关配置,在Security里面找到了Secure Boot选项,发现果然被设置为Enabled,将其修改为Disabled ,再开机,终于正常启动了。_安装完系统提示secureboot failure

C++如何做字符串分割(5种方法)_c++ 字符串分割-程序员宅基地

文章浏览阅读10w+次,点赞93次,收藏352次。1、用strtok函数进行字符串分割原型: char *strtok(char *str, const char *delim);功能:分解字符串为一组字符串。参数说明:str为要分解的字符串,delim为分隔符字符串。返回值:从str开头开始的一个个被分割的串。当没有被分割的串时则返回NULL。其它:strtok函数线程不安全,可以使用strtok_r替代。示例://借助strtok实现split#include <string.h>#include <stdio.h&_c++ 字符串分割

2013第四届蓝桥杯 C/C++本科A组 真题答案解析_2013年第四届c a组蓝桥杯省赛真题解答-程序员宅基地

文章浏览阅读2.3k次。1 .高斯日记 大数学家高斯有个好习惯:无论如何都要记日记。他的日记有个与众不同的地方,他从不注明年月日,而是用一个整数代替,比如:4210后来人们知道,那个整数就是日期,它表示那一天是高斯出生后的第几天。这或许也是个好习惯,它时时刻刻提醒着主人:日子又过去一天,还有多少时光可以用于浪费呢?高斯出生于:1777年4月30日。在高斯发现的一个重要定理的日记_2013年第四届c a组蓝桥杯省赛真题解答

基于供需算法优化的核极限学习机(KELM)分类算法-程序员宅基地

文章浏览阅读851次,点赞17次,收藏22次。摘要:本文利用供需算法对核极限学习机(KELM)进行优化,并用于分类。

metasploitable2渗透测试_metasploitable2怎么进入-程序员宅基地

文章浏览阅读1.1k次。一、系统弱密码登录1、在kali上执行命令行telnet 192.168.26.1292、Login和password都输入msfadmin3、登录成功,进入系统4、测试如下:二、MySQL弱密码登录:1、在kali上执行mysql –h 192.168.26.129 –u root2、登录成功,进入MySQL系统3、测试效果:三、PostgreSQL弱密码登录1、在Kali上执行psql -h 192.168.26.129 –U post..._metasploitable2怎么进入

Python学习之路:从入门到精通的指南_python人工智能开发从入门到精通pdf-程序员宅基地

文章浏览阅读257次。本文将为初学者提供Python学习的详细指南,从Python的历史、基础语法和数据类型到面向对象编程、模块和库的使用。通过本文,您将能够掌握Python编程的核心概念,为今后的编程学习和实践打下坚实基础。_python人工智能开发从入门到精通pdf

推荐文章

热门文章

相关标签