C++ 类和对象 日期类的实现

news2024/11/29 8:52:13

作者:@小萌新
专栏:@初阶C++
作者简介:大二学生 希望能和大家一起进步
本篇博客目标: 完成Date类的实现 梳理剩下两个默认函数
在这里插入图片描述
好困 跑个步去
睡醒啦! 继续肝

日期类的实现

  • 本章目标
  • 一. 日期类的实现
    • 1.1 Getmonthday的实现
    • 1.2 构造函数和打印函数
    • 1.3 日期类的运算函数
      • == 符号
      • > 符号
      • <= 符号
      • < 符号
      • >= 符号
    • 1.4 日期类的加减天数的实现
      • +=天数
      • +天数
      • -=天数
      • -天数
    • 1.5 自增自减
      • 前置++ 前置--
      • 后置++ 后置--
    • 1.6 日期减日期
  • 二. 普通对象 const对象取地址
  • 总结

本章目标

  1. 掌握日期类的实现
  2. 了解剩下两个默认函数

一. 日期类的实现

对于日期类来说 其成员变量包括年 月 日这三个

它的通常操作有

日期加天数 计算多少天后是什么时间 是周几

日期减天数 计算多少天前是什么时间 是周几

日期减日期 计算两个日期之间相差多少天 相差多少周

日期加日期没有什么意义 这里不做实现

我们都知道 每个月的天数都不尽相同 并且还有闰年这个影响因素 所以说我们首先要实现一个Getmonthday的函数
它的主要作用是得到某年某月的具体天数

1.1 Getmonthday的实现

思路分析: 首先每个月的天数都不同 我们可以创建一个数组 来填上所有月数对应的天数 类似于这

样子

	int monthDayArr[13] = { 0, 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31 };

当然 我们这里的月份肯定是1~12月 所以为了更严谨可以在前面加个断言

		assert(month > 0 && month < 13);

当然还有一个影响月天数的原因就是是否是闰年

所以说还需要再写以一个判断是否是润年的函数

四年润 百年不润 四百年润

bool isleapyear(int year)
{
	if ((year%4==0 && year%100 !=0) || (year%400==0))
	{
		return true;
	}
	else
	{
		return false;
	}
}

然后我们整体代码表示如下

public:
	int Getmonthday(int year,int month)
	{
		assert(month > 0 && month < 13);
		int monthDayArr[13] = { 0, 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31 };
		// 判断是否是闰年 如果是闰年 二月的天数就是28 
		if (month==2 && isleapyear(year))
		{
			return 29;
		}
		// 否则就返回这个月的天数
		else
		{
			return monthDayArr[month];
		}
	}

想想看 还有没有什么值得优化的地方

这个获取月份的数组 我们是不是要经常使用啊

每次都要创建销毁数组未免也太浪费内存了 我们可以用static关键字修饰下这个数组 将其中的内容

存放到静态区来进行更好的资源管理

1.2 构造函数和打印函数

我们实现了得到月份功能后迫不及待想试验一下了

那么试验前我们首先要对对象进行初始化 然后打印其数据看看是否正确

这两步在前面的博客中已经做过详细讲解 这里不再赘述

构造函数代码如下

	Date(int year = 1,int month =1,int day =1)
	{
		// 判断年月日输入是否正确 
		assert(year >= 1);
		assert(month >= 1 && month <= 12);
		assert(day >= 0 && day <= 31);


		// 赋值 这里用this指针也可以
		this->_year = year;
		_month = month;
		_day = day;
	}

打印函数代码如下

	void Print()
	{
		cout << _year << "-" << _month << "-" << _day << endl;
	}

下面我们来测试下三种情况

1 闰年的二月份

在这里插入图片描述

2 非闰年的二月份

在这里插入图片描述
3 错误的日期

在这里插入图片描述
我们可以发现 这三个场景都符合我们的预期

1.3 日期类的运算函数

== 符号

这个很简单 依次判断三个值是否相等就可以

	bool operator == (const Date& d)
	{
		return _year == d._year
			&& _month == d._month
			&& _day == d._day;
	}

> 符号

实现大于号的思路很简单

如果首先判断年 如果年大于就大于

如果年相同判断月 如果月大于就大于

最后判断日 如果日大于就大于

但是我们判断完年之后是否可以直接返回一个bool类型呢?

很显然不可以

这个时候我们换一个思路 如果小于等于就返回false

代码表示如下

	bool operator > (const Date& d)
	{
	bool operator > (const Date& d)
	{

		if (_year>d._year)
		{
			return true;
		}
		if (_month>d._month && _year == d._year)
		{
			return true;
		}
		if (_day<d._day && _month == d._month && _day == d._day)
		{
			return true;
		}
		return false;
	}

<= 符号

这里很简单 不大于不就是小于等于嘛?

所以这里直接上代码

	bool operator <= (const Date& d)
	{
		return !(*this > d);
	}

< 符号

不大于的同时不等于 就是小于

return (*this) <= d && !((*this) == d);

>= 符号

不小于就是大于等于

	bool operator >= (const Date& d)
	{
		return !((*this) < d);
	}

1.4 日期类的加减天数的实现

+=天数

加上天数之后赋值 这个时候我们的返回值要改变

这里有个难点就是我们我们增加的天数万一很多怎么办

万一跨越了月份呢?

万一跨越了年份呢?

这里我们先从最简单的加上一天看起

如果只加上一天 并且加上这一天之后不会超过这个月天数的大小(想想看 怎么知道这个月有多少天)

那么就直接返回就可以了

如果说大于这个月份的天数呢?
是不是就要往后面的月份进位了啊
如果月份大于十二了呢?
是不是就要往后面的年进位了啊

按照这个思路我们来写代码


	Date& operator += (int day)
	{
		assert(day >= 0);
		// 第一步 日期先加上
		_day = _day + day;
		while (_day>Getmonthday(_year,_month))
		{
			_day -= Getmonthday(_year, _month);
			_month += 1;
			if (_month>12)
			{
				_month -= 12;
				_year += 1;
			}
		}
	}

我们来验证下我们的思路

在这里插入图片描述
完全正确

+天数

这里跟+=天数的区别 就是一个改变自身的值 一个不改变自身的值

使用一个中间值就好了

代码表示如下

	Date& operator + (int day)
	{
		// 拷贝构造
		Date ret =(*this);
		ret += day; // 复用
		return ret;
	}

-=天数

和+=天数的思路差不多 转换下几个符号就可以了

代码表示如下

	Date& operator -= (int day)
	{
		assert(day >= 0);
		// 第一步 日期先加上
		_day = _day - day;
		while (_day <= 0)
		{
			_month -= 1;
			if (_month < 1)
			{
				_month = 12;
				_year -= 1;
			}
			_day += Getmonthday(_year, _month);
		}
		return *this;
	}

在这里插入图片描述

-天数

这个思路也很相似 不过多赘述

	Date& operator + (int day)
	{
		// 拷贝构造
		Date ret = (*this);
		ret -= day; // 复用
		return ret;
	}

1.5 自增自减

由于 前置++ 和 后置++的特殊性 我们无法判断哪个是前置 哪个是后置

所以说C++中引入了以一个这样子的标准

C++规定:将括号中带有int的规定为后置++,不带int的为前置++ 。(int后面可以加参数,也可以不加)

其实也就是前置效率高那么一点点 所以C++就改变后置的类型去了

前置++ 前置–

这个很简单 使用下+=1 -=1就可以了

	Date& operator ++ ()
	{
		// 复用+=
		(*this) += 1;

		return *this;
	}

	Date& operator -- ()
	{
		// 复用-=
		(*this) -= 1;

		return *this;
	}

后置++ 后置–

这个也很类似

使用下临时变量 返回临时变量就可以

	Date& operator ++ (int x)
	{
		Date ret = *this;
		ret += 1;
		return ret;
	}

	Date& operator ++ (int x)
	{
		Date ret = *this;
		ret += 1;
		return ret;
	}

1.6 日期减日期

这个实现思路很简单

我们只需要选出两个中的较大值 然后让其中的较小值不停++ (并且设置一个计数器)

等到它们相等的时候就好了

代码表示如下

	int operator - (const Date& d)
	{
		if (*this == d)
		{
			return 0;
		}
		Date min = *this;
		Date max = d;
		int count = 0;
		// 有可能相差天数为负数
		int flag = 1;
		if (*this>d)
		{
			max = *this;
			min = d;
			flag = -1;
		}


		while (!(max==min))
		{
			++min;
			count= count+1;
		}


		return count*flag;

	}

我们可以发现 可以运行
在这里插入图片描述

那么这就是日期计算器的全部内容啦

工程源代码如下

#define _CRT_SECURE_NO_WARNINGS 1
#include<stdio.h>
#include<string.h>
#include<iostream>
#include<assert.h>
using namespace std;


bool isleapyear(int year)
{
	if ((year%4==0 && year%100 !=0) || (year%400==0))
	{
		return true;
	}
	else
	{
		return false;
	}
}

class Date
{
public:
	Date(int year = 1,int month =1,int day =1)
	{
		// 判断年月日输入是否正确 
		assert(year >= 1);
		assert(month >= 1 && month <= 12);
		assert(day >= 0 && day <= 31);


		// 赋值 这里用this指针也可以
		this->_year = year;
		_month = month;
		_day = day;
	}

	void Print()
	{
		cout << _year << "-" << _month << "-" << _day << endl;
	}

	int Getmonthday(int year,int month)
	{
		assert(month > 0 && month < 13);
		static int monthDayArr[13] = { 0, 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31 };
		// 判断是否是闰年 如果是闰年 二月的天数就是28 
		if (month==2 && isleapyear(year))
		{
			return 29;
		}
		// 否则就返回这个月的天数
		else
		{
			return monthDayArr[month];
		}
	}

	bool operator == (const Date& d)
	{
		return _year == d._year
			&& _month == d._month
			&& _day == d._day;
	}


	bool operator > (const Date& d)
	{
		// 我们大于能直接判断嘛?
		// 显然不能 所以说我们这里如果小于就返回false
		/*if (_year > d._year)
		{
			
		}*/
		if (_year<d._year)
		{
			return false;
		}
		if (_month<d._month)
		{
			return false;
		}
		if (_day<d._day)
		{
			return false;
		}
		// 最后还有一种全部相等的情况 
		if (*this == d)
		{
			return false;
		}
		return true;
	}

	bool operator <= (const Date& d)
	{
		return !(*this > d);
	}

	bool operator < (const Date& d)
	{
		return (*this) <= d && !((*this) == d);
	}

	bool operator >= (const Date& d)
	{
		return !((*this) < d);
	}

	Date& operator += (int day)
	{
		assert(day >= 0);
		// 第一步 日期先加上
		_day = _day + day;
		while (_day>Getmonthday(_year,_month))
		{
			_day -= Getmonthday(_year, _month);
			_month += 1;
			if (_month>12)
			{
				_month -= 12;
				_year += 1;
			}
		}
		return *this;
	}

	Date& operator + (int day)
	{
		// 拷贝构造
		Date ret =(*this);
		ret += day; // 复用
		return ret;
	}

	Date& operator -= (int day)
	{
		assert(day >= 0);
		// 第一步 日期先加上
		_day = _day - day;
		while (_day <= 0)
		{
			_month -= 1;
			// 这里首先要判断month是否越界 
			if (_month < 1)
			{
				_month = 12;
				_year -= 1;
			}
			_day += Getmonthday(_year, _month);
		}
		return *this;
	}

	Date& operator - (int day)
	{
		// 拷贝构造
		Date ret = (*this);
		ret -= day; // 复用
		return ret;
	}

	Date& operator ++ ()
	{
		// 复用+=
		(*this) += 1;

		return *this;
	}

	Date& operator -- ()
	{
		// 复用-=
		(*this) -= 1;

		return *this;
	}

	Date& operator ++ (int x)
	{
		Date ret = *this;
		ret += 1;
		return ret;
	}

	Date& operator -- (int x)
	{
		Date ret = *this;
		ret -= 1;
		return ret;
	}

	int operator - (const Date& d)
	{
		if (*this == d)
		{
			return 0;
		}
		Date min = *this;
		Date max = d;
		int count = 0;
		// 有可能相差天数为负数
		int flag = 1;
		if (*this>d)
		{
			max = *this;
			min = d;
			flag = -1;
		}


		while (!(max==min))
		{
			++min;
			count= count+1;
		}


		return count*flag;

	}

private:
	int _year;
	int _month;
	int _day;
};






int main()
{

	Date d1(2001,7,5);
	Date d2(2022,11,4);
	//d1.Print();
	int ret = d1.Getmonthday(2001,2);
	cout << ret << endl;
	//d1 -= 1000;
	//d1.Print();
	int ret = d1 - d2;
	cout << ret << endl;



	return 0;
}

大佬们想到什么有趣的功能也可以加上去

二. 普通对象 const对象取地址

class Date
{
public :
    Date* operator&()
    {
        return this;
    }
    const Date* operator&()const
    {
        return this;
    }
private :
    int _year ; // 年
    int _month ; // 月
    int _day ; // 日
};

这里稍微了解下就好 基本不会用到这两个操作符

总结

在这里插入图片描述

本文主要讲解日期类的实现以及两个简单的默认构造函数
由于作者水平有限 出现错误在所难难免 希望大佬们看到之后能及时指正
如果本文帮助到了你 别忘了一键三连啊
阿尼亚 哇酷哇酷!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/3402.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

CNN的实现与可视化

CNN的实现 我们已经实现了卷积层和池化层&#xff0c;现在来组合这些层&#xff0c;搭建进行手写数字识别的CNN。如下图所示&#xff0c;网络的构成是“Convolution - ReLU - Pooling -Affine - ReLU - Affine - Softmax”&#xff0c;我们将它实现为名为SimpleConvNet的类。 …

R语言—向量

向量&#xff08;vector&#xff09; R 语言最基本的数据结构是向量。类似于数学上的集合的概念&#xff0c;由一个或多个元素构成。向量其实是用于存储数值型、字符型、或逻辑型数据的一维数组。 创建向量 c()函数 > a <- 1 #给a赋值1 > a #显示a的值 [1] 1 …

【C++初阶】类和对象终极篇

文章目录一.加const修饰this指针二.cout<<自定义类型的对象的实现1.深入理解cout和<<2流插入和流提取3.友元函数的来源a.作为ostream成员函数b.作为全局函数c.作为Date类的成员函数d.作为Date类的友元函数三.再谈构造函数之初始化列表四.隐式类型转换&explicit…

Linux命令从入门到实战----文件目录类

文章目录pwd显示当前工作路径的绝对路径ls列出目录的内容cd切换目录mkdir 创建一个新的目录删除一个空的目录touch创建新文件cp复制文件或目rm删除文件或目录mv移动文件与目录&#xff0c;重命名文件cat查看文件内容&#xff0c;创建新文件more文件内容分屏查看less分屏显示文件…

【C语言】字符串、字符数组

目录 写在开头 正文 一、字符串的本质 二、输入函数scanf和gets 三、输出函数printf和puts 四、字符串的长度——strlen 五、字符串的复制——strcpy 六、字符串的比较函数 七、实战练习 八、二维字符数组——字符串数组 写在最后 写在开头 看了标题&#xff0c;是…

Kotlin编程实战——类与对象(05)

一 概述 类与继承属性和字段接口(interface )函数式&#xff08;SAM&#xff09;接口可见性修饰符扩展数据类(data class)密封类泛型嵌套类与内部类枚举类对象表达式与对象声明类型别名内联类(inline class)委托委托属性 二 类与继承 类继承(open override )抽象类(abstract)…

MySQL去重中 distinct 和 group by 的区别

今天在写业务需要对数据库重复字段进行去重时&#xff0c;因为是去重&#xff0c;首先想到的是distinct关键字。于是一小时过去了。。。。&#xff08;菜鸟一个&#xff0c;大家轻点骂&#xff09; 我把问题的过程用sql语句演示给大家演示一下 首先我使用的是mybatis-plus&am…

数据结构之哈希表

文章目录 一、概念二、哈希冲突三、如何解决哈希冲突&#xff1f; 1.哈希函数设计2.负载因子调节3.闭散列4.开散列&#xff08;哈希桶&#xff09;四、模拟实现哈希桶总结一、概念 顺序结构以及平衡树中&#xff0c;元素与其存储位置之间没有对应的关系&#xff0c;因此在查找一…

C++多态学习笔记

C多态学习笔记一、多态概述二、多态的作用三、多态发生的三个条件四、多态实现的原理五、接口的定义六、模板方法模式七、虚析构函数和纯虚析构函数7.1 虚析构函数7.2 纯虚析构函数八、重写重载重定义九、父类引用子类对象一、多态概述 同一个操作作用于不同的对象&#xff0c;…

2014年848数据结构真题复习

求k频度K0; for&#xff08;i1;i<n;i&#xff09; 假如是1——8&#xff0c;执行了9次&#xff0c;8次有效&#xff0c;最后一次无效for&#xff08;ji;j<n;j&#xff09;k 我的理解&#xff1a;假设n为8我们看k频度实际上就是看内圈for的有效循环次数第一轮是1——8 八次…

基础算法 第七课——归并排序

文章目录导言归并排序的概念步骤说明逐步分析STEP1STEP2STEP3STEP4STEP5STEP6STEP0总结导言 这&#xff0c;是一篇现学现卖的文章。因为&#xff0c;我根本没学过归并排序。所以&#xff0c;这篇文章&#xff0c;绝对能让您学懂归并。如果不懂&#xff0c;那我就再学一遍&…

KVM Forum 2022应该关注的话题

1. QEMU 和 KVM 自动性能基准测试 QEMU & KVM Automated Performance Benchmarking SUSE - Dario Faggioli, SUSE SUSE正在开发一个框架&#xff0c;用于对虚拟化工作负载进行自动性能基准测试。它是围绕着MMTests&#xff08;已经在Linux内核社区使用了几年&#xff09;建…

2022-Java 后端工程师面试指南 -(SSM)

前言 种一棵树最好的时间是十年前&#xff0c;其次是现在 Tips 面试指南系列&#xff0c;很多情况下不会去深挖细节&#xff0c;是小六六以被面试者的角色去回顾知识的一种方式&#xff0c;所以我默认大部分的东西&#xff0c;作为面试官的你&#xff0c;肯定是懂的。 上面的…

Mybatis之foreach

文章目录一、foreach属性二、使用foreach批量删除(法一)1.接口2.mapper文件3.测试类4.运行结果三、使用foreach批量删除(法二)1.mapper文件四、使用foreach批量插入1.接口2.mapper文件3.测试类4.运行结果一、foreach属性 collection&#xff1a;指定数组或者集合 item&#xf…

FPGA时序约束01——基本概念

前言1. 越来越多的时序问题 随着FPGA时钟频率加快与其实现的逻辑功能越来越复杂&#xff0c;开发者遇到的问题很多时候不再是代码逻辑的问题&#xff0c;而是时序问题。一些开发者可能有这样的经历&#xff0c;一个模块在100MHz时钟运行没问题&#xff0c;而将时钟频率改为150…

【仿牛客网笔记】 Spring Boot进阶,开发社区核心功能-事务管理

添加评论中会用到事务管理。 解决的程度不同&#xff0c;层级不同。我们一般选择中间的级别。 选择时既能满足业务的需要&#xff0c;又能保证业务的安全性&#xff0c;在这样的前提下我们追求一个更高的性能。 第一类丢失更新 图中是没有事务隔离的情况 第二类丢失更新 脏…

需求工程方法的学习

作业要求&#xff1a;总结尽可能多的需求工程的方法和技术&#xff0c;要求归纳总结各种方法的适用场景、优缺点等。说明&#xff1a;其中需求工程包括需求获取、需求分析、规格说明、验证、管理等。只要是用于需求工程相关的技术和方法都可以算。 软件需求工程划分为需求开发…

Linux 中 man手册中函数后面括号数字释义

文章目录简介参考资料简介 Linux手册页项目记录了用户空间程序使用的Linux内核和C库接口。 用man手册查看系统命令&#xff0c;系统调用&#xff0c;glibc函数时&#xff0c;会发现其后面会有个括号&#xff0c;括号里面是一个数字&#xff0c;比如&#xff1a; access(2), …

一文了解Spring框架

目录 SpringBoot VS Servlet Spring是什么&#xff1f; loC&#xff1a;控制反转 DI 创建一个Spring项目 创建一个Spring IOC容器 注册Bean对象 获取Bean对象 注意事项&#xff1a; 类注解 为什么有这么多类注解&#xff1f; 注册与注入 方法注解 Bean Spr…

《R语言数据分析》2022-2023第一学期课程分析报告

1 (30分)基本操作题 1.1 (10分) 请写出下面问题的R代码 1.(2分)安装并加载gtools扩展包。 install.packages(“gtools”) library(gtools) 2.(2分)查看当前已经加载的所有包。 as.data.frame(installed.packages())$Package 3.(2分)查看gtools包的帮助网页。 ?gtools…