C++类与对象下(个人笔记)

类与对象下

    • 1.构造函数补充
      • 1.1构造函数体赋值
      • 1.2初始化列表
      • 1.3explicit关键字
    • 2.static成员
      • 2.1特性
    • 3.友元
      • 3.1友元函数
      • 3.2友元类
    • 4.内部类
    • 5.匿名对象
    • 6.拷贝对象的一些优化
    • 7.笔试题


1.构造函数补充

1.1构造函数体赋值

在创建对象时,编译器通过调用构造函数,给对象中各个成员变量一个合适的初始值。

//构造函数体中的语句将其称为赋初值,而不能称作初始化。
class Date
{
public:
	Date(int year, int month, int day)
 	{
     	_year = year;
    	 _month = month;
     	_day = day;
 	}
private:
	int _year;
	int _month;
	int _day;
};

1.2初始化列表

初始化列表:以一个冒号开始,接着是一个以逗号分隔的数据成员列表,每个"成员变量"后面跟
一个放在括号中的初始值或表达式

class Date
{
public:
	Date(int year, int month, int day)//初始化列表
     	: _year(year)
     	, _month(month)
     	, _day(day)
 {}
private:
	int _year;
	int _month;
	int _day;
};

注意事项:

  1. 每个成员变量在初始化列表中只能出现一次(初始化只能初始化一次)
  2. 类中包含以下成员,必须放在初始化列表位置进行初始化:
    引用成员变量
    const成员变量
    自定义类型成员(且该类没有默认构造函数时)
class A {
public:
 	A(int a)
 	:_a(a)
 {}
private:
 	int _a;
};

class B {
public:
 	B(int a, int ref)
 	:_aobj(a)
 	,_ref(ref)
 	,_n(10)
 {}
private:
 	A _aobj;  // 没有默认构造函数
 	int& _ref;  // 引用
 	const int _n; // const 
};
  1. 尽量使用初始化列表初始化,对于自定义类型成员变量,一定会先使用初始化列表初始化。
  2. 成员变量在类中声明次序就是其在初始化列表中的初始化顺序,与其在初始化列表中的先后
    次序无关

1.3explicit关键字

就是说本来单参数的的构造函数支持隐式类型转换,但加了explicit就不支持了

Date d1(2022);
d1=2023;//这个就不支持了,因为2023构造一个无名对象,最后用无名对象给d1对象进行赋值

2.static成员

声明为static的类成员称为类的静态成员,用static修饰的成员变量,称之为静态成员变量;用
static修饰的成员函数,称之为静态成员函数。静态成员变量一定要在类外进行初始化

2.1特性

  1. 静态成员为所有类对象所共享,不属于某个具体的对象,存放在静态区
  2. 静态成员变量必须在类外定义,定义时不添加static关键字,类中只是声明
  3. 类静态成员即可用 类名::静态成员 或者 对象.静态成员 来访问
  4. 静态成员函数没有隐藏的this指针,不能访问任何非静态成员
  5. 静态成员也是类的成员,受public、protected、private 访问限定符的限制

静态成员函数可以调用非静态成员函数吗?
静态成员函数可以调用非静态成员函数,但是需要通过对象的引用或指针来访问非静态成员函数。因为静态成员函数没有隐式的this指针,无法直接访问非静态成员。

非静态成员函数可以调用类的静态成员函数吗?
非静态成员函数可以调用类的静态成员函数,因为静态成员函数在类的作用域内是可见的。可以使用类名加作用域解析运算符(::)来调用静态成员函数。

3.友元

突破封装的方式,不宜多用

3.1友元函数

尝试去重载operator<<,然后发现没办法将operator<<重载成成员函数。因为cout的
输出流对象和隐含的this指针在抢占第一个参数的位置。this指针默认是第一个参数也就是左操作
数了。但是实际使用中cout需要是第一个形参对象,才能正常使用。所以要将operator<<重载成
全局函数。
如果硬是要重载成成员函数,则使用起来会非常怪异,a<<cout(那这样子才是输出),a>>cin(这样子才是输入)

class Date
{
public:
	Date(int year, int month, int day)
     	: _year(year)
     	, _month(month)
     	, _day(day)
 {}
// d1 << cout; -> d1.operator<<(&d1, cout); 不符合常规调用
// 因为成员函数第一个参数一定是隐藏的this,所以d1必须放在<<的左侧
	ostream& operator<<(ostream& _cout)
 	{
    	_cout << _year << "-" << _month << "-" << _day << endl;
     	return _cout;
 }
private:
	int _year;
	int _month;
	int _day;
};

友元函数可以直接访问类的私有成员,它是定义在类外部的普通函数,不属于任何类,但需要在
类的内部声明,声明时需要加friend关键字。

	friend ostream& operator<<(ostream& _cout, const Date& d);
 	friend istream& operator>>(istream& _cin, Date& d);

注意:

  1. 友元函数可访问类的私有和保护成员,但不是类的成员函数
  2. 友元函数不能用const修饰
  3. 友元函数可以在类定义的任何地方声明,不受类访问限定符限制
  4. 一个函数可以是多个类的友元函数
  5. 友元函数的调用与普通函数的调用原理相同

3.2友元类

  1. 友元类的所有成员函数都可以是另一个类的友元函数,都可以访问另一个类中的非公有成员。

  2. 友元关系是单向的,不具有交换性。
    比如上述Time类和Date类,在Time类中声明Date类为其友元类,那么可以在Date类中直接访问Time类的私有成员变量,但想在Time类中访问Date类中私有的成员变量则不行。(就是说a是b的朋友,但b不是a的朋友)

  3. 友元关系不能传递(如果C是B的友元, B是A的友元,则不能说明C时A的友元)

  4. 友元关系不能继承(爸爸跟叔叔是朋友,但儿子跟叔叔不是朋友)

// 声明日期类为时间类的友元类,则在日期类中就直接访问Time类中的私有成员变量
 friend class Date; 

4.内部类

概念:如果一个类定义在另一个类的内部,这个内部类就叫做内部类。内部类是一个独立的类,
它不属于外部类,更不能通过外部类的对象去访问内部类的成员。外部类对内部类没有任何优越
的访问权限。
注意:内部类就是外部类的友元类,内部类可以通过外部类的对象参数来访问外部类中的所有成员。但是外部类不是内部类的友元。

特性:

  1. 内部类可以定义在外部类的public、protected、private都是可以的。
  2. 注意内部类可以直接访问外部类中的static成员,不需要外部类的对象/类名。
  3. sizeof(外部类)=外部类,和内部类没有任何关系。

5.匿名对象

int main()
{
 	A aa1;
 	// 不能这么定义对象,因为编译器无法识别下面是一个函数声明,还是对象定义
 	//A aa1();
 	// 但是我们可以这么定义匿名对象,匿名对象的特点不用取名字,
 	// 但是他的生命周期只有这一行,我们可以看到下一行他就会自动调用析构函数
 	A();
 	A aa2(2);
 	return 0;
 }

6.拷贝对象的一些优化

class A {
	public:
	A(int a = 0)
	:_a(a)
	{
		cout << "A(int a)" << endl;
	}
	A(const A& aa)
	:_a(aa._a)
	{
		cout << "A(const A& aa)" << endl;
	}
	A& operator=(const A& aa)
	{
		cout << "A& operator=(const A& aa)" << endl;
		if (this != &aa)
		{
		_a = aa._a;
		}
		return *this;
	}
	~A()
	{
		cout << "~A()" << endl;
	}
	private:
		int _a;
};
	void f1(A aa)
	{}
	A f2()
	{
		A aa;
		return aa;
	}
	int main()
	{
		// 传值传参
		A aa1;
		f1(aa1);
		cout << endl;
		// 传值返回
		f2();
		cout << endl;
		// 隐式类型,连续构造+拷贝构造->优化为直接构造
		f1(1);
		// 一个表达式中,连续构造+拷贝构造->优化为一个构造
		f1(A(2));
		cout << endl;
		// 一个表达式中,连续拷贝构造+拷贝构造->优化一个拷贝构造
		A aa2 = f2();
		cout << endl;
		// 一个表达式中,连续拷贝构造+赋值重载->无法优化
		aa1 = f2();
		cout << endl;
		return 0;
	}

7.笔试题

求1+2+3+…+n
在这里插入图片描述

#include <regex>
class Solution {
public:
    class sum
    {
    public:
        sum()
        {
            _ret+=_i;
            _i++;
        }
    };
    int Sum_Solution(int n)
    {
        sum a[n];
        return _ret;
    }
private:
    static int _i;
    static int _ret;
};
int Solution:: _i=1;
int Solution:: _ret=0;

计算日期到天数转换
在这里插入图片描述

//加上月份的天数,day单独再加
#include <iostream>
using namespace std;

int main()
{
    int year=0;
    int month=0;
    int day=0;
    int sumday=0;
    cin>>year>>month>>day;
    int GetMonthDay[13]={0,31,28,31,30,31,30,31,31,30,31,30,31};
    if((year%4==0&&year%100!=0)||(year%400==0))
    {
        for(int i=1;i<month;i++)
        {
            sumday+=GetMonthDay[i];
        }
        if(month==2)
        {
            cout<<(sumday+day);
            return 0;
        }
        cout<<(sumday+day+1);
    }
    else
    {
        for(int i=1;i<month;i++)
        {
            sumday+=GetMonthDay[i];
        }
        cout<<(sumday+day);
    }
    return 0;
}

日期差值
在这里插入图片描述

//思路:将日期全部转换为天数,然后再相减,切记题目规定,要加一
#include <iostream>
#include <math.h>
using namespace std;
int leap(int year)
{
    if((year%4==0 && year%100!=0) || year%400==0)
    {
        return 1;
    }
    return 0;
}
int main()
{
    int year1,year2,mon1,mon2,day1,day2;
    int day[13]={0,31,28,31,30,31,30,31,31,30,31,30,31};
    scanf("%4d%2d%2d",&year1,&mon1,&day1);
    scanf("%4d%2d%2d",&year2,&mon2,&day2);
    int sum1=0,sum2=0;
    for(int yy=0;yy<year1;yy++)
    {
        if(leap(yy))
        {
            sum1+=366; 
        }
        else
        {
            sum1+=365;
        }
    }
    if(leap(year1))
    {
        day[2]=29;
    }
    else
    {
        day[2]=28;
    }
    for(int mm=1;mm<mon1;mm++)
    {
        sum1+=day[mm];
    }
    sum1+=day1;
    for(int yy=0;yy<year2;yy++)
    {
        if(leap(yy))
        {
            sum2+=366; 
        }
        else
        {
            sum2+=365;
        }
    }
    if(leap(year2))
    {
        day[2]=29;
    }
    else
    {
        day[2]=28;
    }
    for(int mm=1;mm<mon2;mm++)
    {
        sum2+=day[mm];
    }
    sum2+=day2;
    cout<<abs(sum1-sum2)+1<<endl;
    return 0;
}

打印日期
在这里插入图片描述

#include <climits>
#include <iostream>
using namespace std;
class Date
{
public:
    Date(int year = 1)
    : _year(year)
    {}
    int GetMonthDay(int year, int month)//获取天数
    {
        static int arr[13] = { 0, 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31 };
        if ((month == 2) && ((year % 400 == 0) || ((year % 4 == 0) && (year % 100 != 0))))
        {
            return 29;
        }
        else 
        {
            return arr[month];
        }
    }
    void ShowDate(int day)//获取到的天数小于_day就更新日期
    {
        _day=day;
        while(_day>GetMonthDay(_year, _month))
        {
            _day-=GetMonthDay(_year,_month);
            ++_month;
            if(_month>12)
            {
                _month-=12;
                _year++;
            }
        }
        //打印的格式
        if (_month < 10 && _day < 10)
        {
            cout << _year << "-0" << _month << "-0" << _day << endl;
        }
        else if (_month < 10 && _day >= 10)
        {
            cout << _year << "-0" << _month << "-" << _day << endl;
        }
        else if (_month >= 10 && _day < 10)
        {
            cout << _year << "-" << _month << "-0" << _day << endl;
        }
        else
        {
            cout << _year << "-" << _month << "-" << _day << endl;
        }
    }
private:
    int _year=1;
    int _month=1;
    int _day=1;
};
int main()
{
    int year=0;
    int day=0;
    while(cin>>year>>day)
    {
        Date date(year);
        date.ShowDate(day);
    }
    return 0;
}

日期累加
在这里插入图片描述

//这题其实就是实现日期类(参考类与对象中的博客)里的重载operator+=和operator-=,然后就是注意日期的打印格式就完事了
#include <climits>
#include <iostream>
using namespace std;
class Date
{
public:
    Date(int year=1,int month=1,int day=1)
    :_year(year)
    ,_month(month)
    ,_day(day)
    {}

    int GetMonthDay(int year, int month)
    {
        static int monthArray[13] = { 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 monthArray[month];
    }

    Date& operator+=(int day)
    {
        if(day<0)
        {
            return *this-=(-day);
        }
        _day+=day;
        while(_day>GetMonthDay(_year,_month))
        {
            _day-=GetMonthDay(_year, _month);
            _month++;
            if(_month==13)
            {
                _year++;
                _month=1;
            }
        }
        return *this;
    }

    Date& operator-=(int day)
    {
        if(day<0)
        {
            return *this+=(-day);
        }
        _day-=day;
        while(_day<=0)
        {
            _month--;
            if(_month==0)
            {
                _year--;
                _month=12;
            }
            _day+=GetMonthDay(_year, _month);
        }
        return *this;
    }

    void Print()
    {
        if (_month < 10 && _day < 10)
        {
            cout << _year << "-0" << _month << "-0" << _day << endl;
        }
        else if (_month < 10 && _day >= 10)
        {
            cout << _year << "-0" << _month << "-" << _day << endl;
        }
        else if (_month >= 10 && _day < 10)
        {
            cout << _year << "-" << _month << "-0" << _day << endl;
        }
        else
        {
            cout << _year << "-" << _month << "-" << _day << endl;
        }
    }

    Date& GetYearMonthDay(int addday)
    {
        *this+=addday;
        return *this;
    }

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


int main()
{
    int m=0;
    cin>>m;
    while(m--)
    {
        int year;
        int month;
        int day;
        int addday;
        cin>>year>>month>>day>>addday;
        Date d(year,month,day);
        d.GetYearMonthDay(addday);
        d.Print();
    }
    return 0;
}

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

在数字化转型的背景下,如何构建高效的数据资产管理体系?

在数字化转型的大潮中&#xff0c;数据已成为企业创新发展的重要驱动力。如何高效地管理这些数据资产&#xff0c;不仅关系到企业的日常运营&#xff0c;更直接决定了企业能否在激烈的市场竞争中脱颖而出。对于企业管理者或首席信息官&#xff08;CIO&#xff09;而言&#xff…

大学英语ab级题搜题软件?分享7个支持答案和解析的工具 #笔记#其他

合理利用学习辅助工具和资料&#xff0c;可以帮助大学生更好地组织学习内容、掌握知识点和提升学术水平。 1.智能翻译官 这是一款多语言在线翻译神器&#xff0c;除了最基础的英语以外&#xff0c;还支持日语、德语、俄语、法语等几十种语言文本翻译和拍照翻译&#xff0c;并…

一文搞懂从爬楼梯到最小花费(力扣70,746)

文章目录 题目前知动态规划简介动态规划模版 爬楼梯一、思路二、解题方法三、Code 使用最小花费爬楼梯一、思路二、解题方法三、Code 总结 在计算机科学中&#xff0c;动态规划是一种强大的算法范例&#xff0c;用于解决多种优化问题。本文将介绍动态规划的核心思想&#xff0c…

积木-蓝桥每日真题

0积木 - 蓝桥云课 (lanqiao.cn) 题目描述 小明用积木搭了一个城堡。 为了方便&#xff0c;小明在搭的时候用的是一样大小的正方体积木&#xff0c;搭在了一个n行m列的方格图上&#xff0c;每个积木正好占据方格图的一个小方格。 当然&#xff0c;小明的城堡并不是平面的&#x…

2014最新AI智能系统ChatGPT网站源码+Midjourney绘画网站源码+搭建部署教程文档

一、文章前言 SparkAi创作系统是基于ChatGPT进行开发的Ai智能问答系统和Midjourney绘画系统&#xff0c;支持OpenAI-GPT全模型国内AI全模型。本期针对源码系统整体测试下来非常完美&#xff0c;那么如何搭建部署AI创作ChatGPT&#xff1f;小编这里写一个详细图文教程吧。已支持…

2.SpringBoot利用Thymeleaf实现页面的展示

什么是Thymeleaf&#xff1f; Thymeleaf是一个现代服务器端Java模板引擎&#xff0c;适用于Web和独立环境&#xff0c;能够处理HTML&#xff0c;XML&#xff0c;JavaScript&#xff0c;CSS甚至纯文本。 Thymeleaf的主要目标是提供一种优雅且高度可维护的模板创建方式。为实现这…

代码审计sql注入部分函数绕过方法

目录 1.宽字节注入 2.预编译模式下的sql注入 3无法预编译的 1.3.1. like关键字 1.3.2.不能加单引号 4.相关实战实战 4.1.某个业务网站具有sql注入 4.2.梦想cms代码审计 5.相关参考资料 1.宽字节注入 <?php $dbinit_db(); $db->query("set SET NAMESgbk);…

Thonny 开发环境下使用PICO系列教程2----点亮板载灯3S后熄灭

Thonny 开发环境下使用PICO系列教程2----点亮板载灯3S后熄灭 硬件代码 硬件 链接: 官网地址 参考原理图可以发现&#xff0c;PICO板载灯连接的是GP25引脚 代码 // 板载灯点亮3秒后熄灭 import board //想要控制PICO的引脚就要引入board import time//延迟 from digitali…

【QT入门】Qt自定义控件与样式设计之QPushButton常用qss

往期回顾 【QT入门】Qt自定义控件与样式设计之qss介绍(Qt style sheet)-CSDN博客 【QT入门】 Qt自定义控件与样式设计之qss选择器-CSDN博客 【QT入门】 Qt自定义控件与样式设计之QLineEdit的qss使用-CSDN博客 【QT入门】Qt自定义控件与样式设计之QPushButton常用qss 这里我们主…

数据结构__顺序表

概念及结构 顺序表是用一段物理地址连续的存储单元依次存储数据元素的线性结构&#xff0c;一般情况下采用数组存储。在数组上完成数据的增删查改 需要用到数组&#xff1a;数组的绝对优势&#xff1a;下标的随机访问&#xff08;因为物理空间连续&#xff09; a[i]等…

政安晨【AIGC实践】(一):在Kaggle上部署使用Stable Diffusion

目录 简述 开始 配置 执行 安装完毕&#xff0c;一键运行 结果展示 政安晨的个人主页&#xff1a;政安晨 欢迎 &#x1f44d;点赞✍评论⭐收藏 收录专栏: 人工智能数字虚拟世界实践 希望政安晨的博客能够对您有所裨益&#xff0c;如有不足之处&#xff0c;欢迎在评论区提…

2024.4.8-day12-CSS 常用样式属性和字体图标

个人主页&#xff1a;学习前端的小z 个人专栏&#xff1a;HTML5和CSS3悦读 本专栏旨在分享记录每日学习的前端知识和学习笔记的归纳总结&#xff0c;欢迎大家在评论区交流讨论&#xff01; 文章目录 作业2024.4.8-学习笔记盒子阴影文本阴影透明的vertical-align字体使用 作业 &…

2024年网络安全趋势前瞻:从AI攻击到云安全新挑战

随着2024年开展新的序幕&#xff0c;网络安全领域正面临着前所未有的挑战与机遇&#xff0c;一系列引人注目的趋势和预测逐渐浮出水面。 一、AI技术发展引发的安全问题 近年来&#xff0c;我们见证了AI技术的飞速进步&#xff0c;其中ChatGPT等引领潮流的AI服务成为公众瞩目的…

鸿蒙OS实战开发:【多设备自适应服务卡片】

介绍 服务卡片的布局和使用&#xff0c;其中卡片内容显示使用了一次开发&#xff0c;多端部署的能力实现多设备自适应。 用到了卡片扩展模块接口&#xff0c;[ohos.app.form.FormExtensionAbility] 。 卡片信息和状态等相关类型和枚举接口&#xff0c;[ohos.app.form.formInf…

C++要点细细梳理——trivial:运算符优先级、switch、临时变量默认赋值等

1. 运算符优先级 在C语言中&#xff0c;运算符的优先级决定了在表达式中各个运算符的执行顺序。当一个表达式中有多个运算符时&#xff0c;优先级高的运算符会先被计算。如果两个运算符的优先级相同&#xff0c;那么它们的结合性&#xff08;从左到右或从右到左&#xff09;会决…

【优选算法专栏】专题十六:BFS解决最短路问题(二)

本专栏内容为&#xff1a;算法学习专栏&#xff0c;分为优选算法专栏&#xff0c;贪心算法专栏&#xff0c;动态规划专栏以及递归&#xff0c;搜索与回溯算法专栏四部分。 通过本专栏的深入学习&#xff0c;你可以了解并掌握算法。 &#x1f493;博主csdn个人主页&#xff1a;小…

主从复制、数据持久化 、Redis主从集群、哨兵机制 、Redis分片集群

数据持久化 Redis、主从集群、哨兵机制 Redis分片集群 1、单点 redis 的问题2、主从复制2.1 命令传播 3、Redis的持久化3.1 AOF3.2 RDB&#xff08;默认方式&#xff09;RDB 方式&#xff1a;执行快照时&#xff0c;数据能被修改吗&#xff1f;RDB 方式总结 3.3 RDB 和 AOF 组合…

ORAN C平面 Section Extension 22

ORAN C平面Section扩展22用于ACK/NACK请求。除section type 7外&#xff0c;section扩展22可以用于从O-DU发送到O-RU的所有section type和section扩展。 对于一个section描述&#xff0c;O-DU可以使用section扩展22要求O-RU使用section type 8 C平面消息进行ACK/NACK反馈。关于…

ctfshow web入门 web29-web38

web29 把flag和i屏蔽了 system函数也行但是通常会屏蔽所以我直接用passthru 看看有啥 cat的话要查看源代码 web30 没有意外把这个system屏蔽了没事我不用哈哈哈 ?cpassthru("cat f*"); 然后查看源代码 web31 把空格屏蔽了 某位大佬的题解看到的 %09或者/**/绕过…

代码随想录第34天| 1005.K次取反后最大化的数组和 134. 加油站 135. 分发糖果

1005.K次取反后最大化的数组和 1005. K 次取反后最大化的数组和 - 力扣&#xff08;LeetCode&#xff09; 代码随想录 (programmercarl.com) 贪心算法&#xff0c;这不就是常识&#xff1f;还能叫贪心&#xff1f;LeetCode&#xff1a;1005.K次取反后最大化的数组和_哔哩哔…