C++ Day04 this指针,友元函数,重载

this指针

概念

谁调用 this 所在的函数 ,this 就存储谁的地址

特点

1, 在当前类的非静态成员函数中调用本类非静态成员时 , 默认有 this 关键字
2, 静态成员函数 , 没有 this 指针。
示例
#include <iostream>
#include <cstring>
using namespace std;
class Stu{
private:char name[50];char sex[10];int age;
public:Stu(){}Stu(char *n,char *s,int a):age(a){strcpy(name,n);strcpy(sex,s);}void test01(){//以下两句代码效果相同//证明本类函数中调用本类成员默认使用this关键字cout << this->name << endl;cout << name << endl;}static void test02(){//报错,因为静态函数中没有this//cout << name << endl;}
};
int main(int argc, char *argv[])
{Stu s("张三","男",18);s.test01();s.test02();return 0;
}

使用场景

1, 局部变量与成员变量重名时 , 使用 this 区分
2, 调用本类其他成员 , 此时 this 可以省略不写
示例
#include <iostream>
#include <cstring>
using namespace std;
class Stu{
private:char name[50];char sex[10];int age;
public:Stu(){}Stu(char *name,char *sex,int age){//当局部变量与成员变量重名,使用this区分strcpy(this->name,name);strcpy(this->sex,sex);this->age = age;}void print_info(){//调用本类成员变量cout << this->name << endl;cout << this->sex << endl;cout << this->age << endl;}void test(){//调用本类成员函数this->print_info();}
};
int main(int argc, char *argv[])
{Stu s("张三","男",18);s.print_info();s.test();return 0;
}

实例

使用 :*this 完成链式编程
#include <iostream>
#include <cstring>
using namespace std;
class Stu{
private:char name[50];char sex[10];int age;
public:Stu(){}Stu(char *name,char *sex,int age){strcpy(this->name,name);strcpy(this->sex,sex);this->age = age;}void print_info(){cout << this->name << endl;cout << this->sex << endl;cout << this->age << endl;}Stu& eat(char *foodName){cout << name << "吃" << foodName << endl;return *this;}
};
int main(int argc, char *argv[])
{Stu s("张三","男",18);s.eat("凉皮").eat("肉夹馍").eat("甑糕");return 0;
}

const修饰成员函数(了解)

特点

        const修饰的成员函数内部不能对成员数据写操作, mutable 修饰的成员数据 除外。

实例

class Stu{
private:char name[50];char sex[10];int age;mutable int score;
public:Stu(){}Stu(char *name,char *sex,int age){strcpy(this->name,name);strcpy(this->sex,sex);this->age = age;}void print_info(){cout << this->name << endl;cout << this->sex << endl;cout << this->age << endl;}Stu& eat(char *foodName){cout << name << "吃" << foodName << endl;return *this;}void test() const{//age = 10;//错误score = 99;//正确}
};

友元函数(重要)

概述

关键字 :friend
可以声明 :
        1,全局函数
        2,成员函数
        3,类
注意 :
        友元打破c++ 的封装性。一般用于运算符重载

全局友元函数

特点

可以访问其友元类的任意成员 , 包括私有成员

步骤

1, 在定义并实例全局函数
2, 在类中声明步骤 1 中的函数为友元函数
3, 步骤 1 中定义的函数 , 可以访问步骤 2 中定义的类中的所有成员

示例

#include <iostream>
#include <cstring>
using namespace std;
class Stu{
friend void test(Stu &stu);
private:char name[50];char sex[10];int age;
public:Stu(){}Stu(char *name,char *sex,int age){strcpy(this->name,name);strcpy(this->sex,sex);this->age = age;}void print_info(){cout << this->name << endl;cout << this->sex << endl;cout << this->age << endl;}
private:void eat(char *foodName){cout << name << "吃" << foodName << endl;}
};
void test(Stu& stu)
{
//调用友元类的私有属性cout << stu.name << endl;cout << stu.sex << endl;cout << stu.age << endl;
//调用友元类的私有函数stu.eat("大嘴巴子");
}
int main(int argc, char *argv[])
{Stu s("张三","男",18);test(s);return 0;
}

成员友元函数

特点

        可以访问其友元类的任意成员, 包括私有成员

注意       

1, 成员函数作为友元 那么成员函数所在的类 必须定义到最上方
2, 成员函数所在的类的所有成员函数 必须在两个类的下方实现

步骤

        1,定义 B , 但不现实
        2,定义成员函数 A1 所在的类 A, 但其中只定义该成员函数 A1
        3,实现 B , 并在其中声明成员函数 A1 为友元函数
        4,实现成员函数

示例

#include <iostream>
#include <cstring>
using namespace std;
//定义B类,但是没有实现
class B;
class A{
public:void test(B& b);
};
class B{
friend void A::test(B& b);
private:int a;
public:B(int a){this->a = a;}
private:void print_B(){cout << "a = " << a << endl;}
};
void A::test(B& b){cout << b.a << endl;b.print_B();}
int main(int argc, char *argv[])
{A a;B b(10);a.test(b);return 0;
}

整个类作为友元函数

特点

        在B 中声明 A B 的友元类 , 此时 A 中任意成员函数中皆可直接访问 B 中的成员

步骤        

1, 定义 B
2, 定义并实现 A , 其中函数只定义不实现
3, 实现 B , 在其中声明 A 类为友元类
4, 实现 A 类中的成员函数

示例

#include <iostream>
#include <cstring>
using namespace std;
//定义B类,但是没有实现
class B;
class A{
public:void test01(B& b);void test02(B& b);
};
class B{
friend class A;
private:int a;
public:B(int a){this->a = a;}
private:void print_B(){cout << "a = " << a << endl;}
};
void A::test01(B& b)
{cout << "test01" << endl;cout << b.a << endl;b.print_B();
}
void A::test02(B& b)
{cout << "test02" << endl;cout << b.a << endl;b.print_B();
}
int main(int argc, char *argv[])
{A a;B b(10);a.test01(b);cout << "--------------------" << endl;a.test02(b);return 0;
}

注意

1, 友元关系不能被继承。
2, 友元关系是单向的,类 A 是类 B 的朋友,但类 B 不一定是类 A 的朋友。
3, 友元关系不具有传递性。类 B 是类 A 的朋友,类 C 是类 B 的朋友,但类 C 不一
定是类 A 的朋友

实例

说明

遥控器类的对象可以操作电视机类对象的成员
所以遥控器类是电视机类的友元类

代码

#include <iostream>
#include <cstring>
using namespace std;
class TV;
class YK{
public:void up(TV& tv);void down(TV& tv);
};
class TV{
friend class YK;
private:int yl;
public:TV(){}TV(int yl){this->yl = yl;}
};
void YK::up(TV &tv)
{tv.yl++;cout << "当前音量:" << tv.yl << endl;
}
void YK::down(TV &tv)
{tv.yl--;cout << "当前音量:" << tv.yl << endl;
}
int main(int argc, char *argv[])
{TV tv(10);YK yk;yk.up(tv);yk.up(tv);yk.up(tv);yk.down(tv);yk.down(tv);yk.down(tv);return 0;
}

string

c++ 字符串类 , 使其字符串操作方便
示例
#include <iostream>
#include <string>
using namespace std;
int main(int argc, char *argv[])
{
string str01 = "hello";
string str02 = str01;//字符串赋值
cout << str01 << endl;//字符串输出
cout << str02 << endl;
str02 = "world";
cout << str01 << endl;
cout << str02 << endl;
string str03 = str01 + str02;//字符串拼接
cout << str03 << endl;
string str04;
cin >> str04;//字符串输入
cout << str04 << endl;
string str05 = "Hi C++";
string str06 = "Hi C++";
string str07 = "Hi C";
cout << (str05 == str06) << endl;//判断字符串内容是否相同
cout << (str05 == str07) << endl;
cout << &str05 << endl;//打印str05地址
cout << &str06 << endl;//打印str06地址
return 0;
}

重载

引入

经源码查看 string 发现其也是一个类
那么为什么 string 的类对象可以使用 >>,<<,+,== 等运算符 , 我们自定义的类不行呢 ?
因为 string 类对运算符进行了重载
那我们如何实现运算符的重载

概述

作用

        是对已有的运算符重新进行定义,赋予其另一种功能,以适应不同的数据类型。

关键字

        operator

语法

        返回值类型 operator+ 运算符 ( 形参列表 )
        {
                函数体
        }
        如:
        >>
        void operator>>(形参列表 )
        {
        }
思路
1 、分析运算符的运算对象的个数
2 、分析运算符左边的运算对象是 自定对象 还是其他
左边:是其他 只能全局函数实现 (必须使用友元)
左边:自定义对象
可以用使用全局函数重载运算符(参数个数 和 运算符对象的个数一致)
也可以使用成员函数重载运算符(参数可以少一个) (推荐)

示例1:重载<<,>>运算符

效果
        使其可以通过<< 输出自定义类型的变量或通过 >> 输入自定义类型变量
分析
<< >> 符号左边为 cout cin 不是自定义对象 , 只能使用全局函数对其进行重载
示例
#include <iostream>
#include <cstring>
using namespace std;
class Data{
public:int x,y,z;Data(){}Data(int a,int b,int c):x(a),y(b),z(c){}
};
//第一个参数为运算符左边的变量
//第二个参数为运算符右边的变量
istream& operator >>(istream& in,Data& d)
{in >> d.x >> d.y >> d.z;return in;
}
ostream& operator <<(ostream& out,Data& d)
{out << "x = " << d.x << "\ty = " << d.y << "\tz = " << d.z << endl;return out;
}
int main(int argc, char *argv[])
{Data d;cin >> d;cout << d << endl;return 0;
}

示例2:重载+运算符

效果

        使用+ 运算符将自定义类型对象的属性一一相加

分析

        +符号左边为自定义类型 , 可以使用全局函数重载也可以使用成员函数中

示例:全局函数重载

#include <iostream>
using namespace std;
class Data{
public:int x,y,z;Data(){}Data(int a,int b,int c):x(a),y(b),z(c){}
};
//第一个参数为运算符左边的变量
//第二个参数为运算符右边的变量
Data* operator +(Data& d1,Data& d2)
{Data *d = new Data();d->x = d1.x + d2.x;d->y = d1.y + d2.y;d->z = d1.z + d2.z;return d;
}
int main(int argc, char *argv[])
{Data d1(1,2,3);Data d2(1,2,3);Data* d3 = d1 + d2;cout << d3->x << d3->y << d3->z << endl;return 0;
}
示例 2: 成员函数重载 + 运算符
#include <iostream>
using namespace std;
class Data{
public:int x,y,z;Data(){}Data(int a,int b,int c):x(a),y(b),z(c){}
//调用该函数的对象为运算符左边的变量
//参数为运算符右边的变量
Data* operator +(Data& d2)
{Data *d = new Data();d->x = this->x + d2.x;d->y = this->y + d2.y;d->z = this->z + d2.z;return d;
}
};
int main(int argc, char *argv[])
{Data d1(1,2,3);Data d2(1,2,3);Data* d3 = d1 + d2;cout << d3->x << d3->y << d3->z << endl;return 0;
}

示例3:重载==运算符

效果
        比较类中成员变量值是否相同
分析
        符号左边为自定义类型, 可以使用全局函数重载也可以使用成员函数中
示例
#include <iostream>
using namespace std;
class Data{
public:int x,y,z;Data(){}Data(int a,int b,int c):x(a),y(b),z(c){}
//调用该函数的对象为运算符左边的变量
//参数为运算符右边的变量bool operator ==(Data& d2){if(this->x == d2.x && this->y == d2.y && this->z == d2.z){return true;}else{return false;}}
};
int main(int argc, char *argv[])
{Data d1(1,2,3);Data d2(1,2,3);Data d3(2,2,3);cout << (d1 == d2) << endl;cout << (d1 == d3) << endl;return 0;
}

示例4:重载++运算符

注意

++ 运算符分为 ++ 在前与 ++ 在后两种所以需要重载两种
当编译器看到 ++a( 前置 ++), 它就调用 operator++(Type& a)( 全局函数 ),operator++
()( 成员函数 )
当编译器看到 a++( 后置 ++), 它就会去调用 operator++(Type& a,int)( 全局函
),operator++(int)( 成员函数 )

示例

#include <iostream>
using namespace std;
class Data{
public:int x,y,z;Data(){}Data(int a,int b,int c):x(a),y(b),z(c){}Data& operator ++()//++前置{++x;++y;++z;return *this;}Data operator ++(int)//++后置{Data old = *this;//记录旧值++x;++y;++z;return old;//返回旧值}
};
ostream& operator <<(ostream& out,Data& d)
{out << d.x << d.y << d.z << endl;return out;
}
int main(int argc, char *argv[])
{Data d1(1,2,3);++d1;cout << d1;Data d2(1,2,3);Data d3 = d2++;cout << d3;cout << d2;return 0;
}

示例5:重载*->

要求

        重载指针运算符实现智能指针

推演

#include <iostream>
using namespace std;
class Data{
private:int x,y,z;
public:Data(){cout << "无参构造函数" << endl;}Data(int a,int b,int c):x(a),y(b),z(c){cout << "有参构造函数" << endl;}~Data(){cout << "析构函数" << endl;}
};
int main(int argc, char *argv[])
{Data *p = new Data();return 0;
}
观察以上代码 , 我们发现创建的对象没有被销毁 , 但是我们在编写代码时经常会忘记销
, 那该怎么办呢 ?
解决方案如下
#include <iostream>
using namespace std;
class Data{
private:
int x,y,z;
public:Data(){cout << "无参构造函数" << endl;}Data(int a,int b,int c):x(a),y(b),z(c){cout << "有参构造函数" << endl;}~Data(){cout << "析构函数" << endl;}
};
class FreeData{
private:Data* p;
public:FreeData(){p = NULL;}FreeData(Data* data){p = data;}~FreeData(){if(p != NULL){delete p;p = NULL;}}
};
int main(int argc, char *argv[])
{FreeData fd(new Data(1,2,3));return 0;
}
现在我们发现 Data 对象可以销毁 , 但是如何调用其对象中的属性呢 ?
方案如下
#include <iostream>
using namespace std;
class Data{
private:int x,y,z;
public:Data(){cout << "无参构造函数" << endl;}Data(int a,int b,int c):x(a),y(b),z(c){cout << "有参构造函数" << endl;}~Data(){cout << "析构函数" << endl;}int getX(){return x;}
};
class FreeData{
private:Data* p;
public:FreeData(){p = NULL;}FreeData(Data* data){p = data;}~FreeData(){if(p != NULL){delete p;p = NULL;}}Data& operator *(){return *p;}Data* operator ->(){return p;}
};
int main(int argc, char *argv[])
{FreeData fd(new Data(1,2,3));cout << (*fd).getX() << endl;cout << fd->getX() << endl;return 0;
}

示例6:重载()

作用

当类对象作为函数调用时,会执行 operator()( 参数列表 ) 函数。
对象作为函数调用
对象名 ( 实参列表 );
一种仿函数

示例

#include <iostream>
using namespace std;
class Data{
friend ostream& operator <<(ostream& out,Data& d);
private:int x,y,z;
public:Data(){cout << "无参构造函数" << endl;}Data(int a,int b,int c):x(a),y(b),z(c){cout << "有参构造函数" << endl;}~Data(){cout << "析构函数" << endl;}void operator ()(int a,int b,int c){this->x += a;this->y += b;this->z += c;}
};
ostream& operator <<(ostream& out,Data& d)
{out << d.x << "\t" << d.y << "\t" << d.z << endl;return out;
}
int main(int argc, char *argv[])
{Data d(1,2,3);d(2,5,8);cout << d;return 0;
}

示例7:重载=

注意
= 重载时,可能会调用类本身的拷贝构造函数。
如果左值是没有创建的对象时 , 会调用拷贝构造函数 .
如果左值是已创建的类对象 , 会执行 = 重载函数 , 实现数据的拷贝
示例
#include <iostream>
using namespace std;
class Data{
friend ostream& operator <<(ostream& out,Data& d);
private:int x,y,z;
public:Data(){cout << "无参构造函数" << endl;}Data(int a,int b,int c):x(a),y(b),z(c){cout << "有参构造函数" << endl;}Data(const Data& d){cout << "执行拷贝构造" << endl;this->x = d.x;this->y = d.y;this->z = d.z;}~Data(){cout << "析构函数" << endl;}void operator =(Data& d){cout << "执行重载=运算符的函数" << endl;this->x = d.x;this->y = d.y;this->z = d.z;}
};
ostream& operator <<(ostream& out,Data& d)
{out << d.x << "\t" << d.y << "\t" << d.z << endl;return out;
}
int main(int argc, char *argv[])
{Data d1(1,2,3);Data d2(3,6,9);d1 = d2;//d1已完成初始化,执行重载的=号运算符Data d3 = d2;//d3未完成初始化,执行拷贝构造return 0;
}

注意:

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

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

相关文章

算法刷题-动态规划2

算法刷题-动态规划2 珠宝的最高价值下降路径最小和 珠宝的最高价值 题目 大佬思路 多开一行使得代码更加的简洁 移动到右侧和下侧 dp[ i ][ j ]有两种情况&#xff1a; 第一种是从上面来的礼物最大价值&#xff1a;dp[ i ][ j ] dp[ i - 1 ][ j ] g[ i ][ j ] 第二种是从左…

握住音乐的法宝 - 简谱

简谱是音乐学习中至关重要的记谱法。除了简谱&#xff0c;还有吉他和钢琴的五线谱&#xff0c;以及我最喜欢的古琴减字谱等其他记谱方式。如果你对音乐感兴趣&#xff0c;一起学习吧&#xff0c;我不允许你是一个文盲。 一、调拍号 “1C 4/4”即为调拍号&#xff0c;调拍号分…

opencv-直方图均衡化

直方图均衡化是一种用于增强图像对比度的图像处理技术。它通过调整图像的灰度级别分布&#xff0c;使得图像中各个灰度级别的像素分布更均匀&#xff0c;从而提高图像的对比度。 在OpenCV中&#xff0c;你可以使用cv2.equalizeHist()函数来进行直方图均衡化。 以下是一个简单…

opencv-2D直方图

cv2.calcHist() 是 OpenCV 中用于计算直方图的函数。它可以计算一维或多维直方图&#xff0c;用于分析图像中像素值的分布。 基本的语法如下&#xff1a; hist cv2.calcHist(images, channels, mask, histSize, ranges[, hist[, accumulate]])参数说明&#xff1a; images:…

[点云分割] 条件欧氏聚类分割

介绍 条件欧氏聚类分割是一种基于欧氏距离和条件限制的点云分割方法。它通过计算点云中点与点之间的欧氏距离&#xff0c;并结合一定的条件限制来将点云分割成不同的区域或聚类。 在条件欧氏聚类分割中&#xff0c;通常会定义以下两个条件来判断两个点是否属于同一个聚类&…

利用Python进行数据分析【送书第六期:文末送书】

&#x1f468;‍&#x1f393;博主简介 &#x1f3c5;云计算领域优质创作者   &#x1f3c5;华为云开发者社区专家博主   &#x1f3c5;阿里云开发者社区专家博主 &#x1f48a;交流社区&#xff1a;运维交流社区 欢迎大家的加入&#xff01; &#x1f40b; 希望大家多多支…

SpringBoot——》配置logback日志文件

推荐链接&#xff1a; 总结——》【Java】 总结——》【Mysql】 总结——》【Redis】 总结——》【Kafka】 总结——》【Spring】 总结——》【SpringBoot】 总结——》【MyBatis、MyBatis-Plus】 总结——》【Linux】 总结——》【MongoD…

目标检测原理

一、什么是目标检测 目标检测的任务是找出图像中所有感兴趣的目标&#xff08;物体&#xff09;&#xff0c;确定他们的类别和位置&#xff0c;是计算机视觉领域的核心问题之一。由于各类物体有不同的外观、形状、姿态&#xff0c;再加上光照、遮挡等因素的干扰&#xff0c;目…

LeetCode207.课程表

看完题我就想&#xff0c;这不就是进程里面的死锁问题嘛&#xff0c;进程1等进程2释放锁&#xff0c;进程2等进程3释放锁&#xff0c;进程3等进程1释放锁&#xff0c;这就造成了死锁。或者是spring中的循环依赖问题&#xff0c;BeanA的初始化需要初始化一个BeanB&#xff0c;Be…

Instant Web API .Net Core Crack

Instant Web API .Net Core 是立即构建即时数据库 Web API&#xff0c;无需编码。在几分钟内生成您的 Web API&#xff0c;以更快地构建应用程序。使用 VS 2022 和 Entity Framework Core 为任何 MS SQL 数据库生成 Web API。 新功能 - 使用 Visual Studio 2022 为 PostgreSQL …

自动化测试学习指南

软件自动化测试的学习步骤 大概步骤如下&#xff1a; 1. 做好手工测试&#xff08;了解各种测试的知识&#xff09;-> 2. 学习编程语言-> 3. 学习Web基础&#xff08;HTML,HTTP,CSS,DOM,Javascript&#xff09;或者 学习Winform -> 4. 学习自动化测试工具 ->5.…

字符串和内存函数(2)

文章目录 2.13 memcpy2.14 memmove2.15 memcmp2.16 memset 2.13 memcpy void* memcpy(void* destination, const void* source, size_t num); 函数memcpy从source的位置开始向后复制num个字节的数据到destination的内存位置。这个函数在遇到 ‘\0’ 的时候并不会停下来。如果so…

AQS和ReentrantLock还能这样理解?

1.公平锁和非公平锁 1.1含义 公平锁:在竞争环境下&#xff0c;先到临界区的线程比后到的线程一定更快地获取得到锁。非公平锁:先到临界区的线程未必比后到的线程更快地获取得到锁。 1.2如何自我实现 公平锁实现&#xff1a;可以把竞争的线程放在一个先进先出的队列上。只要…

你了解Postman 变量吗?

变量是在Postman工具中使用的一种特殊功能&#xff0c;用于存储和管理动态数据。它们可以用于在请求的不同部分、环境或集合之间共享和重复使用值。 Postman变量有以下几种类型&#xff1a; 1、环境变量&#xff08;Environment Variables&#xff09;: 环境变量是在Postman…

2023.11.22 数据仓库2-维度建模

目录 1.数仓建设方案 2.数仓结构图,项目架构图 2.1项目架构图 2.2数仓结构图 3.建模设计 4.维度建模 什么是事实表: 什么是维度表: 数据发展模式y以及对应的模型 5.数仓建设规范 数据库划分规范 表命名规范 表字段类型规范 1.数仓建设方案 ODS: 源数据层(临时存储层) 贴…

防爆智能安全帽、防爆手持终端,防爆智能矿灯守护安全,在煤矿安全生产远程可视化监管中的应用

煤矿安全新守护&#xff1a;如何通过防爆智能装备实现远程可视化监管 煤矿是国民经济的重要支柱产业&#xff0c;但长期以来&#xff0c;安全生产事故的频发一直是困扰煤矿行业发展的严峻问题。安全生产事故不仅危及矿工的生命安全&#xff0c;也对企业和地方经济造成了重大的…

csdn最新最全pytest系列——pytest-xdist插件之多进程运行测试用例|| pytest-parallel插件之多线程运行测试用例

pytest之多进程运行测试用例(pytest-xdist) 前言 平常我们功能测试用例非常多时&#xff0c;比如有1千条用例&#xff0c;假设每个用例执行需要1分钟&#xff0c;如果单个测试人员执行需要1000分钟才能跑完当项目非常紧急时&#xff0c;会需要协调多个测试资源来把任务分成两部…

HPC 集群计算类型的注意事项

HPC 集群计算类型的注意事项 HPC 工作负载在 CPU &#xff0c;内存&#xff0c;网络和存储资源需求方面有不同的要求。 您可以从以下内容开始: 核心计数每个核心的内存网络带宽和等待时间处理器时钟速度 目标是选取返回最佳性价比的计算配置。 HPC 工作负载可以与单个核心作…

Centos8上部署MySQL主从备份

虚拟机环境如下&#xff1a; Node1192.168.1.110Centos8 Node2192.168.1.111Centos8 1.在Node1和Node2上安装数据库&#xff1b; yum install -y mysql* 2.关闭防火墙服务&#xff0c;关闭开启自启&#xff1b; systemctl stop firewalld systemctl disable firewall…

2022年06月 Scratch(二级)真题解析#中国电子学会#全国青少年软件编程等级考试

Scratch等级考试(1~4级)全部真题・点这里 一、单选题(共25题,每题2分,共50分) 第1题 角色初始位置如图所示,下面哪个选项能让角色移到舞台的左下角? A: B: C: D: </