valarray 包含对象成员的类(cpp14章)

C++代码重用   

1.公有继承可以实现 

2.包含、私有继承、保护继承用于实现has-a关系,即新的类将包含另一个类的对象

(使用这样类成员:本身是另外一个类对象称为包含 (组合或层次化)。)

3.函数模板、类模板

valarray类构造函数举例 

double gap[5] = { 3.1,3.5,3.8,2.9,3.3 };
valarray<double> v1;   //创建1个double类型的空数组
valarray<int> v2(8);   //创建8个int类型数组
valarray<int> v3(10,8); //创建8个int类型数组,数组中每个数都是10
valarray<double> v4(gap,4);//取出gap数组的前四个元素用于填充v4数组
valarray<int> v5 = { 20,32,17,9 };//C++ 11中

类方法举例:

operator[]() : 访问各个元素
size() : 返回包含的元素数
sum() : 返回所有元素的总和
max() : 返回最大的元素
min(): 返回最小的元素

举例:每个学生的录入考试成绩  (has_a关系,学生有姓名,也有一组考试成绩)

        Valarray类 包含了string类和valarray<double>类

       用string对象表示学生的名字,valarray<double>表示考试的分数

       将其声明为私有,意味着Valarray类的成员函数可以使用string和valarray<double>类的公有接口来访问和修改name和scores对象。但类外不可这么做,只能通过Valarray类的公有接口访问name和scores。通常被描述为:Valarray类获得了其成员对象的实现,但没有继承接口。

1.代码:(用包含的方法)

valarray.h

#ifndef VALARRAY_H_
#define VALARRAY_H_
#include <iostream>
#include <string>
#include <valarray>
using namespace std;

//14章  14.1 valarray包含成员对象的类
class Student
{
private:
	typedef valarray<double> ArrayDb;
	string name;
	ArrayDb scores;//valarray<double> ArrayDb 
public:
	Student():name("Null student"),scores(){} //成员初始化列表
	explicit Student(const string&s):name(s), scores() {}  //explicit关闭隐式转换,使其只能显调用
	explicit Student(int n) :name("Nully"), scores(n) {}
	Student(const string&s,int n) :name(s), scores(n) {}
	Student(const string&s, const ArrayDb &a) :name(s), scores(a) {}
	Student(const string&s, const double *pd,int n) :name(s), scores(pd,n) {}
	~Student(){}

	double Average() const; //平均成绩 不可修改
	const string &Name() const;
	double &operator[](int n); //stu[0]=100;
	double operator[](int n) const;//a=stu[0]

	friend istream &operator >>(istream &is, Student &stu);//友元函数重载输入输出运算符
	friend istream &getline(istream &is, Student &stu);
	friend ostream &operator<<(ostream &os, Student &stu);
};


#endif // !VALARRAY_H_

valarray.cpp

#include "valarray.h"

double Student::Average() const
{
	if (scores.size() > 0)
		return  scores.sum() / scores.size();
	else
	    return 0.0;
}

const string & Student::Name() const
{
	return name;
}

double & Student::operator[](int n)
{
	return scores[n];
}

double Student::operator[](int n) const
{
	return scores[n];
}

istream & operator>>(istream & is, Student & stu)
{
	is >> stu.name;
	return is;
}

istream & getline(istream & is, Student & stu)
{
	getline(is, stu.name);
	return is;
}

ostream & operator<<(ostream & os, Student & stu)
{
	os << "Scores for" << stu.name << ":" << endl;//显示学生的姓名和各科分数
	int i;
	int lim = stu.scores.size();
	if (lim > 0)
	{
		for ( i = 0; i < lim; i++)
		{
			os << stu.scores[i] << " ";
			if (i % 5 == 4)
				os << endl;
		}
		if (i % 5 != 0)
			os << endl;
	}
	else
		os << "Empty array" << endl;
	return os;
	  
}


main.cpp

#include <iostream>
#include "valarray.h"

using namespace std;
const int pupils = 3; //人数
const int quizzes = 5;//每个人都有5门成绩
void set(Student &sa, int n);
int main()
{
	Student ada[pupils] = { Student(quizzes),Student(quizzes) ,Student(quizzes) };
	int i;
	for (i = 0; i < pupils; i++)
		set(ada[i], quizzes);
	
	cout << "\n Student List:" << endl;
	for (i = 0; i < pupils; i++)
		cout << ada[i].Name() << endl;//显示每个学生的姓名

	cout << "\n Result List:" << endl;
	for (i = 0; i < pupils; i++)
		cout << ada[i];
	cout << "Average:" << ada[i].Average() << endl;
	return 0;
}

void set(Student &sa, int n)
{
	cout << "Please enter the student's name:";
	getline(cin, sa);
	cout << "Please enter:" << n << "quiz scores:" << endl;
	for (int i = 0; i < n; i++)
		cin >> sa[i];
	while (cin.get() != '\n');
}



运行结果:

2.利用私有继承的方法

使用私有继承,基类的公有方法将成为派生类的私有方法。

包含和私有继承的不同

1.包含提供了2个被显示命名的对象成员(name和scores),而私有继承提供了2个无名的子对象成员。

2.私有继承的构造函数使用初始化成员列表语法,它使用类名而不是成员名来标识构造函数。

包含:使用成员名name和scores

Student(const string&s, const double *pd,int n) :name(s), scores(pd,n) {}

私有继承:使用string和ArrayDb

Student(const string&s, const double *pd,int n) :string(s), ArrayDb(pd,n) {}

修改包含代码使其变为私有继承的方式:
1.访问基类的方法

包含:使用valarray的 方法,如size()和sum(),使用对象名调用方法

double Student::Average() const
{
	if (scores.size() > 0)
		return  scores.sum() / scores.size();
	else
	    return 0.0;
}

私有继承(只能在派生类的方法中使用基类的方法):使用类名和作用域与作用域运算符来调用基类的方法

double Student::Average() const
{
	if (ArrayDb::size() > 0) //访问基类的方法   使用类名+作用域解析运算符
		return  ArrayDb::sum() / ArrayDb::size();
	else
	    return 0.0;
}

2.访问基类对象

包含:name为string类对象

const string & Student::Name() const
{
	return name;
}

私有继承:string对象没有名称。Student类的代码如何访问string对象?使用强制类型转换

const string & Student::Name() const
{
	return (const string &)*this; //访问基类对象 通过强制类型转换
}

*this为调用方法的对象。返回一个引用,该引用指向用于该方法的Student对象中的继承而来的string对象。

3.访问基类的友元函数

类名显示限定函数名不适用于友元函数,因为友元函数不是成员函数,不属于类

包含:

ostream & operator<<(ostream & os, Student & stu)
{
	os << "Scores for" << stu.name << ":" << endl;//显示学生的姓名和各科分数
....
}

私有继承:使用显示转换  为基类调用正确的函数

ostream & operator<<(ostream & os, const Student & stu)
{
	os << "Scores for" << (string &)stu << ":" << endl;//显示学生的姓名和各科分数
.....
}

整体代码:

valarray.h

#ifndef VALARRAY_H_
#define VALARRAY_H_
#include <iostream>
#include <string>
#include <valarray>
using namespace std;

//14章  14.2私有继承
class Student:private string,private valarray<double> //私有继承
{
private:
	typedef valarray<double> ArrayDb;
//	string name;
//	ArrayDb scores;//valarray<double> ArrayDb 
public:
	Student():string("Null student"), ArrayDb(){} //成员初始化列表
	explicit Student(const string&s):string(s), ArrayDb() {}
	explicit Student(int n) :string("Nully"), ArrayDb(n) {}
	Student(const string&s,int n) :string(s), ArrayDb(n) {}
	Student(const string&s, const ArrayDb &a) :string(s), ArrayDb(a) {}
	Student(const string&s, const double *pd,int n) :string(s), ArrayDb(pd,n) {}
	~Student(){}

	double Average() const; //平均成绩 不可修改
	const string &Name() const;
	double &operator[](int n); //stu[0]=100;
	double operator[](int n) const;//a=stu[0]

	friend istream &operator >>(istream &is, Student &stu);
	friend istream &getline(istream &is, Student &stu);
	friend ostream &operator<<(ostream &os,const  Student &stu);
};


#endif // !VALARRAY_H_

valarray.cpp

#include "valarray.h"

double Student::Average() const
{
	if (ArrayDb::size() > 0) //访问基类的方法   使用类名+作用域解析运算符
		return  ArrayDb::sum() / ArrayDb::size();
	else
	    return 0.0;
}

const string & Student::Name() const
{
	return (const string &)*this; //访问基类对象 通过强制类型转换
}

double & Student::operator[](int n)
{
	return ArrayDb::operator[](n);
}

double Student::operator[](int n) const
{
	return ArrayDb::operator[](n); //访问基类的方法   使用类名+作用域解析运算符
}

istream & operator>>(istream & is, Student & stu)
{
	is >> (string &)stu;
	return is;
}

istream & getline(istream & is, Student & stu)
{
	getline(is, (string &)stu);
	return is;
}

ostream & operator<<(ostream & os, const Student & stu)
{
	os << "Scores for" << (string &)stu << ":" << endl;//显示学生的姓名和各科分数
	int i;
	int lim = stu.ArrayDb::size(); 
	if (lim > 0)
	{
		for ( i = 0; i < lim; i++)
		{
			os << stu.ArrayDb::operator[](i) << " ";
			if (i % 5 == 4)
				os << endl;
		}
		if (i % 5 != 0)
			os << endl;
	}
	else
		os << "Empty array" << endl;
	return os;
	  
}


main.cpp

#include <iostream>
#include "valarray.h"

using namespace std;
const int pupils = 3; //人数
const int quizzes = 5;//每个人都有5门成绩
void set(Student &sa, int n);
int main()
{
	Student ada[pupils] = { Student(quizzes),Student(quizzes) ,Student(quizzes) };
	int i;
	for (i = 0; i < pupils; i++)
		set(ada[i], quizzes);
	
	cout << "\n Student List:" << endl;
	for (i = 0; i < pupils; i++)
		cout << ada[i].Name() << endl;//显示每个学生的姓名

	cout << "\n Result List:" << endl;
	for (i = 0; i < pupils; i++)
		cout << ada[i];
	cout << "Average:" << ada[i].Average() << endl;
	return 0;
}

void set(Student &sa, int n)
{
	cout << "Please enter the student's name:";
	getline(cin, sa);
	cout << "Please enter:" << n << "quiz scores:" << endl;
	for (int i = 0; i < n; i++)
		cin >> sa[i];
	while (cin.get() != '\n');
}



结果:

3.使用私有继承的场景:

1.假设类包含保护成员(可以是数据成员也可以是成员函数),则这样的成员在派生类中是可用的,但在继承层次外是不可用的。使用组合将这样的类包含在另一个类中,则后者将不是派生类,而是继承层次之外,因此不能访问保护成员。  但是通过私有继承得到的将是派生类,因此能够访问保护成员。

2.当需要重新定义虚函数。派生类可以重新定义虚函数,但包含不可以,使用私有继承,重新定义的函数将只能在类中使用,而不是公有的。

通常:应使用包含来建立has-a关系;如果新类需要访问原有类的保护成员,或需要重定义一个虚函数,则使用私有继承。

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值