IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> C++知识库 -> <C++ Primer >学习笔记 第七章 类(上) -> 正文阅读

[C++知识库]<C++ Primer >学习笔记 第七章 类(上)

类的思想是数据抽象和封装,数据抽象是一种依赖于接口和实现分离式编程。

类简介

在C++中通过定义一个类来定义属于自己的数据结构,一个类中定义了一个类型和相关的以及与其关联的一组操作。
类定义在哪里?
我们在操作的时候需要通过使用头文件的方式访问自己应用程序所定义的类,通常头文件以 .h 、.hpp 、.hxx来结尾,使用方法类似于标准库,不过标准库一般不带后缀。编译器一般不关心头文件的形式,但是有的IDE对这个有要求。

类名是什么?
每个类实际上都定义了一个新的类,其类型名就是类名。假定类名为Sales_item.就相当于定义了一个名为Sales_item的类型。有了类名之后就可以像内置类型一样定义类类型变量

Sales_item book;

这句话的含义是book是一个Sales_item类型对象。

定义一个类:

.h文件
#ifndef SALES_DATA_H_INCLUDE
#define SALES_DATA_H_INCLUDE
#endif
#include<string>
#include<iostream>
class Sales_data
{
	
}.c文件
#include <iostream>
#include "Sales_data.h"
int main()
{
	Sales_data book;
	std::cin >> book;
	std :: cout << book<< std::endl;
 } 

什么是成员函数?

Sales_item book1,book2;
if(book1.isbn() == book2.isbn())

book1和book2是类的对象,isbn()是类的成员函数。成员函数定义为类的一部分函数,有的时候也被叫做“方法”。通常以一个类对象的名义来调用成员函数

1.24 输入两个书目的ISBN号 销售册数和售价,输出ISBN 总销售册数 总价格和平均售价(以Z和回车结束)(这里使用的是别人提供的类,具体下载和使用方法见第一章的笔记)

#include <iostream>
#include "Sales_item.h"
int main()
{
		std::cout<<"Please enter the sales record of the same book"<<std::endl;
	Sales_item total;
	if(std::cin>>total)
	{
	
	Sales_item trans;
		while (std::cin>>trans){
			if(total.isbn() == trans.isbn())
			total+=trans;
			else
			{
				std::cout<<total<<std::endl;
				total = trans;
			}
			
		} 
		std::cout<<"ISBN, the number of units sold, the sales amount and the average selling price are"<<std::endl;
		std::cout<<total<<std::endl;
	}
	else{
		std::cerr<<"no data"<<std::endl;
		return -1;
	}
	return 0;
 } 

7.1 定义抽象数据类型

上面的程序中使用的Sales_item中的类是一个抽象数据类,我们通过类的接口来使用这个类,并没有直接访问类的成员,并且也不知道这个类的成员有哪些,这就是抽象数据类型。

this指针和const成员函数

我们在调用成员函数时,其实是在替某个对象调用它。

book.isbn()

如果isbn指向的是 Sales_book的某个成员,则它隐式的指向调用该函数对象的成员。比如调用的isbn()函数返回的是bookNo,实际上返回的是book.bookNo。成员函数通过一个隐式的名为this的参数来调用她访问的那个对象。当利用一个对象调用一个成员函数时,用该对象的地址初始化this,比如上面那个调用,编译器会把book的地址传递给isbn的隐式形参this,因为this指的是这个对象,任何对类成员的直接访问都可以看作this指针的隐式引用。this是一个常量指针,可以在成员函数体内部使用。

这个参数解决了一个问题,确定调用的成员函数返回值所依赖的对象,比如定义了很多类对象都可以调用isbn(),如果没有this我们很难确定此时返回的bookNo究竟是哪个对象的bookNo

this的类型是指向类类型的非常量的常量指针,因此无法使用this指向一个常量,也无法在一个常量对象上调用普通的成员函数。因此当成员函数不会改变this所指的对象时可以利用const修改this指针的类型,提高函数的灵活性。做法是将const放在成员函数的参数列表之后,这样的成员函数被称作常量成员函数

设计一个类

设计一个抽象数据类包含以下几个功能

  • isbn函数,用于返回对象的ISBN编号
  • combine函数,用于将一个类对象加到另外一个上
  • add函数,执行两个对象的加法
  • read函数,将数据读入到类的对象中
  • print函数,讲对象的数值输出在窗口

定义成员函数
定义在类内部的函数时隐式的inline函数,成员函数的声明必须在类的内部,但是成员函数的函数体可以定义在类内部,也可以定义在外部,定义在外部的成员函数不会隐式的inline,因此可以在定义和声明的地方加上inline关键字。

定义在内部的函数

  • isbn函数
    isbn函数的参数列表为空,返回值是一个string对象:
 string isbn() const
 {return bookNo;}

其中const用作修饰隐形this,类本身就是一个作用域,bookNo就是定义在类内的数据成员,类中的函数可以随意使用类内的成员,不需要在意次序问题。

定义在类外部的成员函数

-combine函数

Sales_data& Sales_data::combine(const Sales_data& rhs ){
	units_sold+=rhs.units_sold;     //售出总数=目前对象+以前售出书总数 
	revenue+=rhs.revenue ;          //总收入=目前对象收入+以前的收入
	return *this;//返回对象 
}

定义在外部的成员函数的声明必须和定义相匹配,并且定义在外部的成员函数必须包含他所属的类名,函数的调用如下

total.combine(book);

其中total的地址被绑定在隐式的this参数上,rsh绑定在book上,因此units_sold是total的变量,相当于

total.units_sold
this->units_sold

return 返回的是解引用this以获得该调用函数的对象,换句话说,上面的这个调用返回的是total的引用。

定义相关非成员函数
add、read、print这几个函数从操作上属于类的接口,实际不是,他们是与类相关的非成员函数,一般与类声明在同一个头文件中,定义相关非成员函数需要实用friend关键字,申明成友元。

  • read函数
istream &read(istream &is,Sale_data &item) 
{
	double price=0;
	is>>item.bookNo>>item.units_sold>>price;
	item.revenue=price*item.units_sold;
	return is; 
}
  • print函数
ostream &print(ostream &os,const Sales_data &item)
{
	os<<item.isbn()<<" "<<item.units_sold<<" "
	  <<item.revenue<<" "<<item.avg_price();
	  return os;
}

read和print分别接受一个各自IO类型的引用作为其参数,IO类不能被拷贝所以要用引用传递,并且由于流内容会被改变因此使用的是普通引用。

  • add函数
Salse_data add(const Sales_data &lhs,const Sales_data &rhs)
{
	Sales_data sum =lhs;
	sum.combine(rhs);
	return sum;
}

构造函数

构造函数的作用是初始化对象的数据成员,无论何时只要类的对象被创建,就会执行构造函数。构造函数的名字和类名同名并且构造函数没有返回类型,类可以包含多个构造函数,每个构造函数之间必须在参数数量和参数类型上有所区别。构造函数不能被声明成const。

合成的默认构造函数
当函数没有定义构造函数,类会通过一个默认构造函数初始化,默认构造函数不需要实。编译器创造的构造函数又称为合成的默认构造函数,合成默认构造函数只适用于一些简单的函数。

定义构造函数
还是以Sales_data为例

struct Sale_data {
Sales_data() = default;
Sales_data(const std::string&s):bookNo(s){}
Sales_data(const std::string &s ,unsigned n,double p):bookNo(s),units_sold(n),revenue(p*n){}
Salse_data(std::istream&);
};

Sales_data() = default;
这个默认构造函数没有任何实参,因此是默认构造函数,在C++11的新标准中,如果我们需要编译器帮助我们构造一个默认构造函数,可以通过在参数列表后面接=default来要求编译器生成构造函数。=default可以放在类内部,也可以在外部,在内部认为是内联的。

Sales_data(const std::string&s):bookNo(s){}
:bookNo(s)的含义是构造函数初始值列表,他负责为对象的一个或几个数据成员赋初始值,()或者{}里面是成员的初始值,不同的初始值用逗号隔开
如果构造函数是const或者引用则必须进行初始化
在类的外部定义构造函数
istream为参数的构造函数需要执行一些实际的操作,在他函数体内需要调用read并且给数据成员赋值。

Sales_data::Sales_data(std::istream &is)
//定义了一个名字叫Sales_data的Sales_data类成员,因为类名和成员名相同,所以这个是个构造函数
//初始值列表是空的,但是执行了构造函数,所以仍然被初始化了
{
	read(is,*this);
}

委托构造函数
委托构造函数将自己的一些职责委托给其他构造函数,委托构造函数也是由一个成员列表和一个函数体构成。
C++11 委托构造函数

  C++知识库 最新文章
【C++】友元、嵌套类、异常、RTTI、类型转换
通讯录的思路与实现(C语言)
C++PrimerPlus 第七章 函数-C++的编程模块(
Problem C: 算法9-9~9-12:平衡二叉树的基本
MSVC C++ UTF-8编程
C++进阶 多态原理
简单string类c++实现
我的年度总结
【C语言】以深厚地基筑伟岸高楼-基础篇(六
c语言常见错误合集
上一篇文章      下一篇文章      查看所有文章
加:2022-05-24 17:55:54  更:2022-05-24 17:58:24 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/11 6:07:29-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码