C++初阶--类与对象--const成员和日期类的实现

简介: C++初阶--类与对象--const成员和日期类的实现


const 成员

将const修饰的成员函数称之为const成员函数。

在一个成员函数里面,对于this指针指向的对象,是隐藏式的,没有办法用常规的方法去修饰它,所以我们是这样进行修饰的:

注意事项:

对一个日期类的实现

源码

date.h

#define  _CRT_SECURE_NO_WARNINGS 1
#include<iostream>
#include<assert.h>
using namespace std;
class Date
{
private:
  int _year;
  int _month;
  int _day;
public:
  Date(int year=1 , int month=1 , int day=1 );
  Date(const Date& d);
  Date& operator+=(int day);
  Date operator+(int day)const;
  //日期-天数
  Date& operator-=(int day);
  //日期-天数
  Date operator-(int day)const;
  Date& operator++();
  Date operator++(int);
  Date& operator--();
  Date operator--(int);
  //日期-日期
  int operator-(const Date& d)const;
  bool operator>(const Date& d)const;
  bool operator<(const Date& d)const;
  bool operator==(const Date& d)const;
  bool operator>=(const Date& d)const;
  bool operator<=(const Date& d)const;
  void Print()const;
  int GetMonthDay()const;
};

date.cpp

#define  _CRT_SECURE_NO_WARNINGS 1
#include"Date.h"
Date::Date(int year,int month,int day)
{
  _year = year;
  _month = month;
  _day = day;
  if (_year<1 || _month < 1 || _month>12 || _day>GetMonthDay())
  {
    Print();
    cout << "日期非法" << endl;
  }
}
Date::Date(const Date& d)
{
  _year = d._year;
  _month = d._month;
  _day = d._day;
}
void Date::Print()const
{
  cout << _year << "-" << _month << "-" << _day << endl;
}
int Date::GetMonthDay()const
{
  assert(_year >= 1 && _month >= 1 && _month<=12);
  int day=0;
  day+= _day;
  int Array[] = { 0,31,28,31,30,31,30,31,31,30,31,30,31 };
  if (_month == 2 && (_year % 4 == 0 && _year % 100 != 0 || _year % 400 == 0))
  {
    return 29;
  }
  return Array[_month];
}
//d1+=50;
Date& Date::operator+=(int day)
{
  if (day < 0)
  {
    return *this -= (-day);
  }
  _day += day;
  while (_day > GetMonthDay())
  {
    _day -= GetMonthDay();
    _month++;
    if (_month > 12)
    {
      _month = 1;
      _year++;
    }
  }
  return (*this);
}
//d2=d1+100
Date Date::operator+(int day)const
{
  Date tmp(*this);
  tmp += day;
  return tmp;
}
//d1-=100
Date& Date::operator-=(int day)
{
  if (day < 0)
  {
    return *this += (-day);
  }
  _day -= day;
  while (_day <= 0)
  {
    _month--;
    if (_month == 0)
    {
      _month = 12;
      _year--;
    }
    _day += GetMonthDay();
  }
  return (*this);
}
//d1=d2-100
//日期-天数
Date Date::operator-(int day)const
{
  Date tmp(*this);
  tmp -= day;
  return tmp;
}
//++d1
Date& Date::operator++()
{
  *this += 1;
  return *this;
}
//d1++
Date Date::operator++(int)
{
  Date tmp(*this);
  *this += 1;
  return tmp;
}
//--d1
Date& Date::operator--()
{
  *this -= 1;
  return *this;
}
//d1--
Date Date::operator--(int)
{
  Date tmp(*this);
  *this -= 1;
  return tmp;
}
bool Date::operator>(const Date& d)const
{
  if (_year < d._year)
  {
    return false;
  }
  if (_year == d._year && _month < d._month)
  {
    return false;
  }
  if (_year == d._year && _month == d._month && _day < d._day)
  {
    return false;
  }
  return true;
}
bool Date::operator==(const Date& d)const
{
  if (_year == d._year && _month == d._month && _day == d._day)
  {
    return true;
  }
  return false;
}
bool Date::operator>=(const Date& d)const
{
  return *this > d || *this == d;
}
bool Date::operator<(const Date& d)const
{
  return !(*this>d || *this==d);
}
bool Date::operator<=(const Date& d)const
{
  return *this < d || *this == d;
}
int Date::operator-(const Date& d)const
{
  int day = 0;
  Date min = *this;
  Date max = d;
  if (*this>d)
  {
    min = d;
    max = *this;
  }
  while (min < max)
  {
    min++;
    day++;
  }
  return day;
}

解析

这里采用多文件编程的方式,所以在date.cpp中,是在Date类外使用的,需要加上作用域限定符;

对于不改变类对象,也就是this指针指向的对象的,尽量都加上const进行修饰,有一定程度提高效率;

在date.h中,不管成员变量还是成员函数,在类中都只是起到声明的作用,类的成员变量的定义将会在对象中进行定义,类的成员函数都是在date.cpp定义的

构造函数的验证

在我们定义时,我们已经加上了缺省值了,

所以在定义时就不用加上缺省值了,加上反而非产生冲突,使编译器无法辨别要哪个缺省值,产生矛盾;

在定义时我们加上了对日期是否合法进行了判断,但没有强制性退出,只是打印警告;

我们将给出三组数据进行验证:

运算符的重复利用

对于重复的步骤,我们可以运用另一个函数来进行完成:

而这里也是有一定讲究的,对于参数和返回值来说,都会创建一个临时变量对象来拷贝传参的值,而这就表示着要调用拷贝构造函数,所以我们在写程序时,要尽量的少调用函数,提高一定的效率

上述程序中,调用函数次数:

如果我们反过来写的话:

注意:对于+=来说,会改变this指针指向对象的值,所以这里不能使用const来进行修饰;

而+ 到最后会返回一个值给对应的对象,这个对象不是this指针所指向的对象的,所以可以使用const进行修饰;

我们在date.cpp中没有写明=的重载,类会生成一个默认的运算符重载;

前置++与后置++

在C++中,前置++的重载形式与后置++的重载形式是相同的,为了辨别这两种重载形式,后置++重载时在参数中多加一个int类型的参数,但调用函数时该参数是不参与传递的。

前置++就是直接+=1;而后置++会等到语句结束后才加上1,所以先找一个中间值tmp来进行代替,表示还没有+1;

其他运算符的验证

相关文章
|
编译器 C++ 容器
【c++11】c++11新特性(上)(列表初始化、右值引用和移动语义、类的新默认成员函数、lambda表达式)
C++11为C++带来了革命性变化,引入了列表初始化、右值引用、移动语义、类的新默认成员函数和lambda表达式等特性。列表初始化统一了对象初始化方式,initializer_list简化了容器多元素初始化;右值引用和移动语义优化了资源管理,减少拷贝开销;类新增移动构造和移动赋值函数提升性能;lambda表达式提供匿名函数对象,增强代码简洁性和灵活性。这些特性共同推动了现代C++编程的发展,提升了开发效率与程序性能。
540 12
|
人工智能 机器人 编译器
c++模板初阶----函数模板与类模板
class 类模板名private://类内成员声明class Apublic:A(T val):a(val){}private:T a;return 0;运行结果:注意:类模板中的成员函数若是放在类外定义时,需要加模板参数列表。return 0;
280 0
|
存储 编译器 程序员
c++的类(附含explicit关键字,友元,内部类)
本文介绍了C++中类的核心概念与用法,涵盖封装、继承、多态三大特性。重点讲解了类的定义(`class`与`struct`)、访问限定符(`private`、`public`、`protected`)、类的作用域及成员函数的声明与定义分离。同时深入探讨了类的大小计算、`this`指针、默认成员函数(构造函数、析构函数、拷贝构造、赋值重载)以及运算符重载等内容。 文章还详细分析了`explicit`关键字的作用、静态成员(变量与函数)、友元(友元函数与友元类)的概念及其使用场景,并简要介绍了内部类的特性。
452 0
|
编译器 C++
类和对象(中 )C++
本文详细讲解了C++中的默认成员函数,包括构造函数、析构函数、拷贝构造函数、赋值运算符重载和取地址运算符重载等内容。重点分析了各函数的特点、使用场景及相互关系,如构造函数的主要任务是初始化对象,而非创建空间;析构函数用于清理资源;拷贝构造与赋值运算符的区别在于前者用于创建新对象,后者用于已存在的对象赋值。同时,文章还探讨了运算符重载的规则及其应用场景,并通过实例加深理解。最后强调,若类中存在资源管理,需显式定义拷贝构造和赋值运算符以避免浅拷贝问题。
|
编译器 C++
类和对象(下)C++
本内容主要讲解C++中的初始化列表、类型转换、静态成员、友元、内部类、匿名对象及对象拷贝时的编译器优化。初始化列表用于成员变量定义初始化,尤其对引用、const及无默认构造函数的类类型变量至关重要。类型转换中,`explicit`可禁用隐式转换。静态成员属类而非对象,受访问限定符约束。内部类是独立类,可增强封装性。匿名对象生命周期短,常用于临时场景。编译器会优化对象拷贝以提高效率。最后,鼓励大家通过重复练习提升技能!
|
编译器 C++ 开发者
【C++篇】深度解析类与对象(下)
在上一篇博客中,我们学习了C++的基础类与对象概念,包括类的定义、对象的使用和构造函数的作用。在这一篇,我们将深入探讨C++类的一些重要特性,如构造函数的高级用法、类型转换、static成员、友元、内部类、匿名对象,以及对象拷贝优化等。这些内容可以帮助你更好地理解和应用面向对象编程的核心理念,提升代码的健壮性、灵活性和可维护性。
|
编译器 C语言 C++
类和对象的简述(c++篇)
类和对象的简述(c++篇)
|
设计模式 安全 C++
【C++进阶】特殊类设计 && 单例模式
通过对特殊类设计和单例模式的深入探讨,我们可以更好地设计和实现复杂的C++程序。特殊类设计提高了代码的安全性和可维护性,而单例模式则确保类的唯一实例性和全局访问性。理解并掌握这些高级设计技巧,对于提升C++编程水平至关重要。
280 16
|
存储 编译器 C++
类和对象(上)(C++)
本篇内容主要讲解了C++中类的相关知识,包括类的定义、实例化及this指针的作用。详细说明了类的定义格式、成员函数默认为inline、访问限定符(public、protected、private)的使用规则,以及class与struct的区别。同时分析了类实例化的概念,对象大小的计算规则和内存对齐原则。最后介绍了this指针的工作机制,解释了成员函数如何通过隐含的this指针区分不同对象的数据。这些知识点帮助我们更好地理解C++中类的封装性和对象的实现原理。
|
安全 C++
【c++】继承(继承的定义格式、赋值兼容转换、多继承、派生类默认成员函数规则、继承与友元、继承与静态成员)
本文深入探讨了C++中的继承机制,作为面向对象编程(OOP)的核心特性之一。继承通过允许派生类扩展基类的属性和方法,极大促进了代码复用,增强了代码的可维护性和可扩展性。文章详细介绍了继承的基本概念、定义格式、继承方式(public、protected、private)、赋值兼容转换、作用域问题、默认成员函数规则、继承与友元、静态成员、多继承及菱形继承问题,并对比了继承与组合的优缺点。最后总结指出,虽然继承提高了代码灵活性和复用率,但也带来了耦合度高的问题,建议在“has-a”和“is-a”关系同时存在时优先使用组合。
955 6