描述

写一个MyString 类,使得下面程序的输出结果是:

1. abcd-efgh-abcd-

2. abcd-

3.

4. abcd-efgh-

5. efgh-

6. c

7. abcd-

8. ijAl-

9. ijAl-mnop

10. qrst-abcd-

11. abcd-qrst-abcd- uvw xyz

about

big

me

take

abcd

qrst-abcd-

要 求:MyString类必须是从C++的标准类string类派生而来。提示1:如果将程序中所有 "MyString" 用"string" 替换,那么题目的程序中除了最后两条语句编译无法通过外,其他语句都没有问题,而且输出和前面给的结果吻合。也就是说,MyString类对 string类的功能扩充只体现在最后两条语句上面。提示2: string类有一个成员函数 string substr(int start,int length); 能够求从 start位置开始,长度为length的子串

程序:

    #include <iostream>
#include <cstring>
#include <cstdlib>
using namespace std; // 在此处补充你的代码 int CompareString( const void * e1, const void * e2) {
MyString * s1 = (MyString * ) e1;
MyString * s2 = (MyString * ) e2;
if( *s1 < *s2 ) return -;
else if( *s1 == *s2 ) return ;
else if( *s1 > *s2 ) return ;
}
int main() {
MyString s1("abcd-"),s2,s3("efgh-"),s4(s1);
MyString SArray[] = {"big","me","about","take"};
cout << "1. " << s1 << s2 << s3<< s4<< endl;
s4 = s3; s3 = s1 + s3;
cout << "2. " << s1 << endl;
cout << "3. " << s2 << endl;
cout << "4. " << s3 << endl;
cout << "5. " << s4 << endl;
cout << "6. " << s1[] << endl;
s2 = s1; s1 = "ijkl-";
s1[] = 'A' ;
cout << "7. " << s2 << endl;
cout << "8. " << s1 << endl;
s1 += "mnop";
cout << "9. " << s1 << endl;
s4 = "qrst-" + s2;
cout << "10. " << s4 << endl;
s1 = s2 + s4 + " uvw " + "xyz";
cout << "11. " << s1 << endl;
qsort(SArray,,sizeof(MyString), CompareString);
for( int i = ;i < ;++i )
cout << SArray[i] << endl;
//输出s1从下标0开始长度为4的子串
cout << s1(,) << endl;
//输出s1从下标为5开始长度为10的子串
cout << s1(,) << endl;
return ;
}

思路:

好了这题已经成功恶心我了一整天,明天还有个魔兽的题,今晚写完了放松会儿,明天继续接受挑战。。

作为一道“继承和派生”单元的作业编程题,的确是把这块知识玩到极致了,就拿这个题来说,非常全面综合的考察了对string类继承的方方面面

有些部分的思路在代码中已经做了注释,还有些没说的单独拿出来说一下:

关于operator+的运算符重载,首先在C++官方文档上获取到operator+的3个可能会被这题用到的重载函数

string operator+ (const string& lhs, const string& rhs);//a
string operator+ (const string& lhs, const char* rhs);//b
string operator+ (const char* lhs, const string& rhs);//c

然后我们再来看这题中出现过+的地方

(1)

s3 = s1 + s3;

先不说‘=’,单看+操作符,完全可以调用父类中的a方法实现

(2)

s4 = "qrst-" + s2;

同样不看‘=’,这条语句+的操作可以通过父类中c方法实现

(3)

s1 = s2 + s4 + " uvw " + "xyz";

该语句可以通过b方法实现

OK,既然三个操作在父类中都有相应的成员函数可以实现,我们为什么还要自己折腾去重写呢?就是因为它的返回值!在这个程序的(1)(2)(3)这三条语句中,每一个在进行完加法操作后并没有结束,而是接着进行了赋值的操作——这就是关键点,接受他们+返回值的是一个MyString类的对象,也就是一个子类,而如果我们通过父类的已有operator+去调用他们的结果,返回值是一个父类即string类的对象,而父类的对象是无法赋值到子类上去的。因此我们要重写父类中的这三个operator+,其实也只需要将他们的返回值做出相应的改变就好。

然后再说一下string类中的c_str()函数,先看下C++Reference中对它的定义

const char* c_str() const noexcept;

The pointer returned points to the internal array currently used by the string object to store the characters that conform its value.

说白了就是把string中存储的内容给放到一个char数组中,然后返回这个数组的首地址,const char* 类型

这里由于我们重写operator+的返回值要求是MyString类型的,所以在return的时候要构建MyString类的对象,而它的有参构造函数就只能接受const char*类型的参数,所以我们有必要做出这么一步转换。

main函数的第一行s4(s1)一开始的时候我对这个比较有疑问,因为起初AC的代码没写复制构造函数,用的编译器自动生成的,我就不理解为什么也可以AC,后来经博主回答编译器自动生成的复制构造函数都是形如A (const A&)这样的,而对于MyString这个类来说,由于它是子类,生成的默认复制构造函数也要自动调用父类的复制构造函数。

程序中创建MyString类的对象数组时,等号右边的赋值方式相当于调用了类型转换构造函数,这个地方有个点很容易出错,即当我们程序中的类可以用类型转换构造函数创建的时候,即我们可以直接在等号右边用相应的类型变量对该类进行初始化的时候,如果我们的复制构造函数的参数被我们写成了A&而非const A&,就会出现多个候选项的问题,即编译器会报错,不知道到底该调用哪个构造函数,因此在重写复制构造的时候一定要注意参数是const A&


#include <iostream>
#include <string>
#include <cstdlib>
using namespace std;
class MyString : public string
{
public:
MyString() {};
//1.0继承类继承父类所有的成员变量和成员函数,但不继承构造函数和析构函数
//1.1继承类的无参构造函数,会隐式调用父类的无参构造函数
MyString(const char * st) :string(st) {};
//1.2继承类的有参构造函数,如果父类也有有参构造函数,则必须显示调用它
//2.0这里的参数根据reference有两种选择,此处必须用const char*,"xxx"的类型是const char*
MyString(const MyString& s):string(s){}
//1.3继承类的复制构造函数必须要显示的调用父类的复制构造函数,不然就会默认调用父类的无参构造函数
MyString operator +(MyString & op2)
{
string s1 = *this;
string s2 = op2;
string s = s1 + s2;
return *new MyString(s.c_str());
}
MyString & operator +(const char * cs2)
{
string str1 = *this;
string s = str1 + cs2;
return *new MyString(s.c_str());
} MyString & operator()(int s, int l)
{
string str = substr(s, l);
return *new MyString(str.c_str());
}
}; MyString operator+(const char * op1, MyString & op2)
{
string st2 = op2;
string s = op1 + st2;
return *new MyString(s.c_str());
} int CompareString(const void * e1, const void * e2)
{
MyString * s1 = (MyString *)e1;
MyString * s2 = (MyString *)e2;
if (*s1 < *s2) return -;
else if (*s1 == *s2) return ;
else if (*s1 > *s2) return ;
}
int main()
{
MyString s1("abcd-"), s2, s3("efgh-");
MyString s4(s1);
MyString SArray[] = { "big","me","about","take" };
//这里等号右边的赋值操作相当于调用了MyString的转换构造函数,其实就是单一非const classname&参数的构造函数可以直接接受参数类型的变量
cout << "1. " << s1 << s2 << s3 << s4 << endl;
s4 = s3;
//3.0 operator=可以直接用string类里面的
s3 = s1 + s3;
s1+s3;
cout << "2. " << s1 << endl;
cout << "3. " << s2 << endl;
cout << "4. " << s3 << endl;
cout << "5. " << s4 << endl;
cout << "6. " << s1[] << endl;
s2 = s1;
s1 = "ijkl-";
s1[] = 'A';
cout << "7. " << s2 << endl;
cout << "8. " << s1 << endl;
s1 += "mnop";
cout << "9. " << s1 << endl;
s4 = "qrst-" + s2;
cout << "10. " << s4 << endl;
s1 = s2 + s4 + " uvw " + "xyz";
cout << "11. " << s1 << endl;
qsort(SArray, , sizeof(MyString), CompareString);
for (int i = ; i < ; ++i)
cout << SArray[i] << endl;
cout << s1(, ) << endl;
cout << s1(, ) << endl;
return ;
}

程序设计实习MOOC / 继承和派生——编程作业 第五周程序填空题1的更多相关文章

  1. 程序设计实习MOOC / 程序设计与算法(三)第一周测验

    作业题: 7. 填空(2分)简单的swap 通过码是 ( 请参考公告中的“关于编程作业的说明”完成编程作业(请注意,编程题都要求提交通过码,在openjudge上提交了程序并且通过以后,就可以下载到通 ...

  2. 程序设计实习MOOC / 程序设计与算法(一)第二周测验(2018春季)

    编程题: 1:对齐输出 总时间限制:  1000ms 内存限制:  65536kB 描述 读入三个整数,按每个整数占8个字符的宽度,右对齐输出它们. 输入 只有一行,包含三个整数,整数之间以一个空格分 ...

  3. 程序设计实习MOOC / 程序设计与算法(三)第二周测验

    6. 学生信息处理程序 总时间限制: 1000ms 内存限制: 1024kB 描述 实现一个学生信息处理程序,计算一个学生的四年平均成绩. 要求实现一个代表学生的类,并且类中所有成员变量都是[私有的] ...

  4. 程序设计实习MOOC / 程序设计与算法(二)第二周测验(2018春季)

    递归算法: 1:全排列 总时间限制:  1000ms 内存限制:  65536kB 描述 给定一个由不同的小写字母组成的字符串,输出这个字符串的所有全排列. 我们假设对于小写字母有'a' < ' ...

  5. 《程序设计入门——C语言》翁恺老师 第五周编程练习记录

    1 素数和(5分) 题目内容: 我们认为2是第一个素数,3是第二个素数,5是第三个素数,依次类推. 现在,给定两个整数n和m,0<n<=m<=200,你的程序要计算第n个素数到第m个 ...

  6. 【吴恩达课后编程作业】第二周作业 - Logistic回归-识别猫的图片

    1.问题描述 有209张图片作为训练集,50张图片作为测试集,图片中有的是猫的图片,有的不是.每张图片的像素大小为64*64 吴恩达并没有把原始的图片提供给我们 而是把这两个图片集转换成两个.h5文件 ...

  7. 团队作业第五周(HCL盐酸队)

    一.Alpha版本测试报告 1.测试计划 测试项目 上下移动   左右移动   发射子弹   与敌方坦克进行攻击 2.测试过程 测试截图 错误记录(提交issues到码云团队项目) 3.测试找出的bu ...

  8. Linux内核分析作业第五周

    系统调用的三个层次(下) 一.给MenuOS增加time和time-asm命令 1.克隆并自动编译 MenuOS rm menu -rf 强制删除原menu文件 git clone https://g ...

  9. 程序设计入门-C语言基础知识-翁恺-第五周:函数-详细笔记(五)

    目录 第五周:函数 5.1 函数 5-2 使用函数 5.3 课后习题 第五周:函数 5.1 函数 什么是函数? 函数是一块代码,接受零个或多个参数,做一件事情,并返回零个或一个值. 函数声明语法 返回 ...

随机推荐

  1. Java Numeric Formatting--reference

    I can think of numerous times when I have seen others write unnecessary Java code and I have written ...

  2. Java基础知识强化76:正则表达式之替换功能

    1. 替换功能: String类的replaceAll方法,如下: public String replaceAll(String regex, String replacement): 使用给定的r ...

  3. Android(java)学习笔记255:JNI之JNI概念

    1. JNI是什么? java native interface (java本机接口) 比如方法声明: public final native Class<?>  getClass(): ...

  4. nginx 配置轮询服务

    通常我们应用nginx做代理时,用到它的轮训服务 #设置轮询名称 upstream zyy{ server 127.0.0.1:8080  #本机的apache服务 } server { listen ...

  5. 一个好用的Python备份mysql的脚本

    前几天打算用Python写一个mysql脚本,上Google看了下老外写的,写的挺好的,原地址在http://tecadmin.net/python-script-for-mysql-database ...

  6. Fragment 创建 传递参数 跳转 典例

    抽取的控制Fragment的父Activity /**  * 抽象一个Activity托管我们的Single Fragment  */ public abstract class SingleFrag ...

  7. ruby 知识点

    $LOAD_PATH 执行 require 读取文件时搜索的目录名数组,也可以写作 $: 创建 URI 的时候可以直接这样 URI("http://www.dy2018.com/i/9751 ...

  8. 控制弹出div显示在鼠标附近的位置

    前一个页面: $("#txt_ocname").click(function () { art.dialog.open("/SelPosAll.aspx", { ...

  9. android WIFI的一些属性

    package com.example.wifitest; import java.util.List; import android.content.Context; import android. ...

  10. CClientDC

    CClientDC(客户区设备上下文)用于客户区的输出,它在构造函数中封装了GetDC(),在析构函数中封装了ReleaseDC()函数.一般在响应非窗口重画消息(如键盘输入时绘制文本.鼠标绘图)绘图 ...