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

news2024/11/25 20:51:10

类与对象下

    • 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;
}

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

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

相关文章

掌握Linux虚拟网络设备:从基础到应用的全面指南

在现代计算环境中&#xff0c;尤其是云计算☁️、容器化&#x1f4e6;和微服务架构&#x1f3d7;️大行其道的时代&#xff0c;了解和掌握Linux虚拟网络设备变得极为重要。本文将深入探讨Linux虚拟网络设备的世界&#xff0c;带你了解它们是什么、包含哪些类型、为什么需要它们…

《深入浅出Spring Boot 3.x》正式出版了一周

各位&#xff0c;我编写的《深入浅出Spring Boot 3.x》已经正式发版了。 目前在京东已经开始销售了&#xff0c;希望有需要的朋友多多支持哦。 尽量采用Java 8后的语法编写&#xff0c;采用JDK 17去使用Jarkata EE 9。 相关内容如下&#xff1a;

(虚拟DOM)前端八股文修炼Day10

一 虚拟 DOM 是什么 虚拟 DOM (Virtual DOM) 本质上是真实 DOM 的一个轻量级的 JavaScript 表示形式。它是一个在内存中的抽象&#xff0c;用于描述真实 DOM 的结构和内容。虚拟 DOM 提供了一种机制&#xff0c;允许开发者通过操作 JavaScript 对象来间接更新页面&#xff0c;…

kafka(四)——生产者流程分析(c++)

前言 kafka生产者负责将数据发布到kafka集群的主题&#xff1b;kafka生产者消息发送方式有两种&#xff1a; 同步发送异步回调发送 流程 流程说明&#xff1a; Kafka Producer整体可看作是一个异步处理操作&#xff1b;消息发送过程中涉及两个线程&#xff1a;main线程和se…

一文读懂RISC-V与ARM

RISC-V和ARM是近年来备受关注的两种处理器架构。RISC-V是一种基于精简指令集计算(RISC)原理的开源指令集架构(ISA)&#xff0c;而ARM是一种专有ISA&#xff0c;由于其长期存在于嵌入式系统和移动设备中&#xff0c;已成为嵌入式系统和移动设备的主导选择。市场以及多年积累的信…

【网络】什么是RPC

RPC 是Remote Procedure Call的缩写&#xff0c;译为远程过程调用。是一个计算机通信协议。 1、为什么需要远程调用 在如何给女朋友解释什么是分布式这一篇文章中介绍过&#xff0c;为了提升饭店的服务能力&#xff0c;饭店从一开始只有一个负责所有事情的厨师发展成有厨师、切…

前端二维码工具小程序产品使用说明书

一、产品概述 前端二维码工具小程序是一款便捷实用的二维码生成与识别工具&#xff0c;通过本小程序&#xff0c;用户可以轻松根据文本或链接生成二维码&#xff0c;并支持扫一扫功能识别二维码内容&#xff0c;同时提供复制识别内容的功能。此外&#xff0c;本小程序还具备美…

如何使用Java和RabbitMQ实现延迟队列(方式二)?

前言 昨天写了一篇关于Java和RabbitMQ使用插件实现延迟队列功能的文章&#xff0c;今天来讲下另外一种方式&#xff0c;不需要RabbitMQ的插件。 前期准备&#xff0c;需要安装好docker、docker-compose的运行环境。 需要安装RabbitMQ的可以看下面这篇文章。 如何使用PHP和R…

React - 请你说一说setState是同步的还是异步的

难度级别:中高级及以上 提问概率:70% 在React项目中,使用setState可以更新状态数据,而不能直接使用为this.state赋值的方式。而为了避免重复更新state数据,React首先将state添加到状态队列中,此时我们可以通过shouldComponentUpdate这个钩…

WWDC24定档6月 | 崩坏3将推Mac系统版 苹果AI启航 visionOS 2.0将系数登场WWDC24

这几天又有一件苹果用户圈大事发生了&#xff01;WWDC24正式定档&#xff0c;将在6月10日-14日召开&#xff0c;届时一众软件系统&#xff0c;包括iOS18&#xff0c;iPadOS&#xff0c;WatchOS&#xff0c;VisionOS等等&#xff0c;都将迎来更新。另外就是手游崩坏3官宣&#x…

vector的使用和底层模拟实现

爱吃喵的鲤鱼 个人主页 文章目录 前言一、pandas是什么&#xff1f;二、使用步骤 1.引入库2.读入数据总结 前言 我们已经 学习了string在来实现vector会发现他们两的结构很像&#xff0c;而string只支持存储字符串&#xff0c;vector支持任意类型&#xff1b; 一、vector是什么…

3. Django 初探路由

3. 初探路由 一个完整的路由包含: 路由地址, 视图函数(或者视图类), 可选变量和路由命名. 本章讲述Django的路由编写规则与使用方法, 内容分为: 路由定义规则, 命名空间与路由命名, 路由的使用方式.3.1 路由定义规则 路由称为URL (Uniform Resource Locator, 统一资源定位符)…

-bash: cd: /etc/hadoop: 没有那个文件或目录

解决办法&#xff1a;source /etc/profile 运行 source /etc/profile 命令会重新加载 /etc/profile 文件中的配置&#xff0c;这样做的目的是使任何更改立即生效&#xff0c;而不需要注销并重新登录用户。通常&#xff0c;/etc/profile 文件包含系统范围的全局 Shell 配置&…

电商社交新零售:创新引领新趋势,变革新零售思维格局-亿发

新零售O2O模式是如何颠覆传统零售商业模式&#xff1f; 传统电商出现瓶颈&#xff1a; 传统电商在发展过程中逐渐出现了瓶颈&#xff0c;主要表现在市场竞争激烈、用户获取成本上升、用户黏性下降等问题。传统电商往往只能通过价格竞争或促销活动来吸引用户&#xff0c;而这种…

hexo接入github Discussions评论系统

评论存储仓 可以是你的博客项目的(github)仓库&#xff0c;也可以单独新建一个评论存储仓库。 我的博客项目在gitee上&#xff0c;就以新建存储仓为例&#xff1a; 使用Discussions评论系统必须开通Discussions模块&#xff01; 安装giscus插件 https://github.com/apps/…

数据仓库发展历史与架构演进

从1990年代Bill Inmon提出数据仓库概念后经过四十多的发展&#xff0c;经历了早期的PC时代、互联网时代、移动互联网时代再到当前的云计算时代&#xff0c;但是数据仓库的构建目标基本没有变化&#xff0c;都是为了支持企业或者用户的决策分析&#xff0c;包括运营报表、企业营…

vscode中vue插件

在Visual Studio Code (VSCode) 中&#xff0c;有许多插件可以帮助Vue开发者提高工作效率和代码质量。以下是一些针对Vue开发的必备VSCode插件&#xff0c;结合了多篇搜索结果中的信息&#xff0c;以提供详尽的介绍。 Volar Volar是Vue.js开发者的官方推荐插件&#xff0c;专门…

python基于opencv实现数籽粒

千粒重是一个重要的农艺性状&#xff0c;通过对其的测量和研究&#xff0c;我们可以更好地理解作物的生长状况&#xff0c;优化农业生产&#xff0c;提高作物产量和品质。但数籽粒数目是一个很繁琐和痛苦的过程&#xff0c;我们现在用一个简单的python程序来数水稻籽粒。代码的…

React之基础项目搭建

前言 React的生态系统非常庞大&#xff0c;拥有大量的第三方库和工具&#xff0c;如React Native&#xff08;用于构建原生移动应用&#xff09;、Next.js&#xff08;用于构建服务器渲染应用&#xff09;、Create React App&#xff08;用于快速搭建React应用的脚手架&#x…

【机器学习300问】62、若想将逻辑回归用于多分类有哪些常见做法?

逻辑回归算法在设计之初是用于二分类问题的&#xff0c;但若想把它用在多分类上也不是不行&#xff0c;这得看你具体面临的多分类问题是什么样的&#xff08;问题的定义&#xff09;。不同的问题就有不同的应对之策&#xff1a; 一、一对一 &#xff08;1&#xff09;方法的原…