C++ Primer Plus 学习笔记 第十五章 友元类 友元成员函数 其他友元关系 共同的友元 嵌套类

本文主要是介绍C++ Primer Plus 学习笔记 第十五章 友元类 友元成员函数 其他友元关系 共同的友元 嵌套类,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

与友元函数类似。

代码:

tv.h

#ifndef TV_H_
#define TV_H_class Tv
{public:
// 这就是友元类的写法friend class Remote;enum {Off, On};enum {MinVal, MaxVal = 20};enum {Antenna, Cable};enum {TV, DVD};Tv(int s = Off, int mc = 125) : state(s), volume(5),maxchannel(mc), channel(2), mode(Cable), input(TV) {}void onoff() {state = (state == On) ? Off : On;}bool ison() {return state == On;}bool volup();bool voldown();void chanup();void chandown();void set_mode() {mode = (mode == Antenna) ? Cable : Antenna;}void set_input() {input = (input == TV)? DVD : TV;}void settings() const;private:int state;int volume;int maxchannel;int channel;int mode;int input;
};class Remote
{private:int mode;public:Remote(int m = Tv::TV) : mode(m){}bool volup(Tv & t) {return t.volup();}bool voldown(Tv & t) {return t.voldown();}void cnoff(Tv & t) {t.onoff();}void chanup(Tv & t) {t.chanup();}void chandown(Tv & t) {t.chandown();}// 这里涉及到了友元的访问作用域 如果不是友元的话  编译器不允许直接调用对象的私有成员void set_chan(Tv & t, int c) {t.channel = c;}void set_mode(Tv & t) {t.set_mode();}void set_input(Tv & t) {t.set_input();} 
};#endif

这里面那么多的函数 只有1个函数涉及到了友元,那能不能让特定的成员函数成为友元呢? 可以:

class Tv
{friend void Remote::set_chan(Tv & t, int c);.....
};

光这样写还不行  要让编译器事先知道Remote是啥 但是, 又有一个问题 Remote中有涉及到了Tv类,那也要事先让编译器知道Tv类是啥; 这就蛋疼了,咋搞?

使用前向声明

class Tv;

class Remote {....};

class Tv{...};

这样排列后还是有一个问题,就是Remote中很多函数时内联函数,也就是说提前调用了Tv类的函数。那编译器依然找不到对应的函数。咋办?

Remote类中涉及调用Tv类的成员的部分 放到Tv类定义的代码段后面写。就可以了

上示例吧:

#ifndef TV_H_
#define TV_H_// 先声明
class Tv;class Remote
{public:enum {Off, On};enum {MinVal, MaxVal = 20};enum {Antenna, Cable};enum {TV, DVD};private:int mode;public:Remote(int m = TV) : mode(m){}bool volup(Tv & t);bool voldown(Tv & t);void cnoff(Tv & t);void chanup(Tv & t);void chandown(Tv & t);void set_chan(Tv & t, int c);void set_mode(Tv & t);void set_input(Tv & t);
};// 再详细声明
class Tv
{public:
// 特定友元函数friend void Remote::set_chan(Tv & t, int c);enum {Off, On};enum {MinVal, MaxVal = 20};enum {Antenna, Cable};enum {TV, DVD};Tv(int s = Off, int mc = 125) : state(s), volume(5),maxchannel(mc), channel(2), mode(Cable), input(TV) {}void onoff() {state = (state == On) ? Off : On;}bool ison() {return state == On;}bool volup();bool voldown();void chanup();void chandown();void set_mode() {mode = (mode == Antenna) ? Cable : Antenna;}void set_input() {input = (input == TV)? DVD : TV;}void settings() const;private:int state;int volume;int maxchannel;int channel;int mode;int input;
};// 最后定义内联函数
inline bool Remote::volup(Tv & t) {return t.volup();}
inline bool Remote::voldown(Tv & t) {return t.voldown();}
inline void Remote::cnoff(Tv & t) {t.onoff();}
inline void Remote::chanup(Tv & t) {t.chanup();}
inline void Remote::chandown(Tv & t) {t.chandown();}
inline void Remote::set_mode(Tv & t) {t.set_mode();}
inline void Remote::set_input(Tv & t) {t.set_input();}
inline void Remote::set_chan(Tv & t, int c) {t.channel = c;}
#endif

其他友元关系:

如果Tv类成为Remote的友元。那有一点要注意  在Tv类中调用了Remote类的相关成员或成员函数  一定要写在Remote类详细声明的后面。(先来后到 不然编译器无法识别Remote是啥玩意儿,这个跟刚才那个友元成员函数是一个情况)

共同的友元

这个没啥内容 就是弄一个内联函数 然后成为两个或者多个类的友元函数就是了

这里有句话很关键

嵌套类

嵌套类定义格式

嵌套类的作用域:

访问控制:

模板中的嵌套

其实就是内部类啦

形象的表示就是人的器官 也是类 但是是在人这个类的内部

上代码:

queuetp.h

#ifndef QUEUETP_H_
#define QUEUETP_H_template<class Item>
class QueueTP
{private:enum {Q_SIZE = 10};class Node{public:Item item;Node * next;Node(const Item & i) : item(i), next(0) {}};Node * front;Node * rear;int items;const int qsize;QueueTP(const QueueTP & q) : qsize(0) {}QueueTP & operator=(const QueueTP & q) {return *this;}public:QueueTP(int qs = Q_SIZE);~QueueTP();bool isempty() const{return items == 0;}bool isfull() const{return items == qsize;}int queuecount() const{return items;}bool enqueue(const Item & item);bool dequeue(Item & item);
};template<class Item>
QueueTP<Item>::QueueTP(int qs) : qsize(qs)
{front = rear = 0;items = 0;
}template<class Item>
QueueTP<Item>::~QueueTP()
{Node * temp;while(front != 0){temp = front;front = front-> next;delete temp;}
}template<class Item>
bool QueueTP<Item>::enqueue(const Item & item)
{if (isfull())return false;Node * add = new Node(item);items++;if (front == 0)front = add;elserear->next = add;rear = add;return true;
}template<class Item>
bool QueueTP<Item>::dequeue(Item & item)
{if (front == 0)return false;item = front->item;items--;Node * temp = front;front = front -> next;delete temp;if (items == 0)rear = 0;return true;
}#endif

nested.cpp

#include <iostream>
#include "queuetp.h"int main()
{using std::string;using std::cin;using std::cout;QueueTP<string> cs(5);string temp;while(!cs.isfull()){cout << "Please enter your name. You will be ""served in the order of arrival.\n""name: ";getline(cin, temp);cs.enqueue(temp);}cout << "The queue is full. Processing begins!\n";while (!cs.isempty()){cs.dequeue(temp);cout << "Now processing " << temp << "...\n";}return 0;
}

完结

这篇关于C++ Primer Plus 学习笔记 第十五章 友元类 友元成员函数 其他友元关系 共同的友元 嵌套类的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/639633

相关文章

Spring常见错误之Web嵌套对象校验失效解决办法

《Spring常见错误之Web嵌套对象校验失效解决办法》:本文主要介绍Spring常见错误之Web嵌套对象校验失效解决的相关资料,通过在Phone对象上添加@Valid注解,问题得以解决,需要的朋... 目录问题复现案例解析问题修正总结  问题复现当开发一个学籍管理系统时,我们会提供了一个 API 接口去

C++中实现调试日志输出

《C++中实现调试日志输出》在C++编程中,调试日志对于定位问题和优化代码至关重要,本文将介绍几种常用的调试日志输出方法,并教你如何在日志中添加时间戳,希望对大家有所帮助... 目录1. 使用 #ifdef _DEBUG 宏2. 加入时间戳:精确到毫秒3.Windows 和 MFC 中的调试日志方法MFC

Oracle的to_date()函数详解

《Oracle的to_date()函数详解》Oracle的to_date()函数用于日期格式转换,需要注意Oracle中不区分大小写的MM和mm格式代码,应使用mi代替分钟,此外,Oracle还支持毫... 目录oracle的to_date()函数一.在使用Oracle的to_date函数来做日期转换二.日

SpringBoot基于MyBatis-Plus实现Lambda Query查询的示例代码

《SpringBoot基于MyBatis-Plus实现LambdaQuery查询的示例代码》MyBatis-Plus是MyBatis的增强工具,简化了数据库操作,并提高了开发效率,它提供了多种查询方... 目录引言基础环境配置依赖配置(Maven)application.yml 配置表结构设计demo_st

深入理解C++ 空类大小

《深入理解C++空类大小》本文主要介绍了C++空类大小,规定空类大小为1字节,主要是为了保证对象的唯一性和可区分性,满足数组元素地址连续的要求,下面就来了解一下... 目录1. 保证对象的唯一性和可区分性2. 满足数组元素地址连续的要求3. 与C++的对象模型和内存管理机制相适配查看类对象内存在C++中,规

解决mybatis-plus-boot-starter与mybatis-spring-boot-starter的错误问题

《解决mybatis-plus-boot-starter与mybatis-spring-boot-starter的错误问题》本文主要讲述了在使用MyBatis和MyBatis-Plus时遇到的绑定异常... 目录myBATis-plus-boot-starpythonter与mybatis-spring-b

在 VSCode 中配置 C++ 开发环境的详细教程

《在VSCode中配置C++开发环境的详细教程》本文详细介绍了如何在VisualStudioCode(VSCode)中配置C++开发环境,包括安装必要的工具、配置编译器、设置调试环境等步骤,通... 目录如何在 VSCode 中配置 C++ 开发环境:详细教程1. 什么是 VSCode?2. 安装 VSCo

Spring Boot 中整合 MyBatis-Plus详细步骤(最新推荐)

《SpringBoot中整合MyBatis-Plus详细步骤(最新推荐)》本文详细介绍了如何在SpringBoot项目中整合MyBatis-Plus,包括整合步骤、基本CRUD操作、分页查询、批... 目录一、整合步骤1. 创建 Spring Boot 项目2. 配置项目依赖3. 配置数据源4. 创建实体类

SpringBoot嵌套事务详解及失效解决方案

《SpringBoot嵌套事务详解及失效解决方案》在复杂的业务场景中,嵌套事务可以帮助我们更加精细地控制数据的一致性,然而,在SpringBoot中,如果嵌套事务的配置不当,可能会导致事务不生效的问题... 目录什么是嵌套事务?嵌套事务失效的原因核心问题:嵌套事务的解决方案方案一:将嵌套事务方法提取到独立类

C++11的函数包装器std::function使用示例

《C++11的函数包装器std::function使用示例》C++11引入的std::function是最常用的函数包装器,它可以存储任何可调用对象并提供统一的调用接口,以下是关于函数包装器的详细讲解... 目录一、std::function 的基本用法1. 基本语法二、如何使用 std::function