C++/CLI——3继承与值类型、操作符重载与异常

2024-01-04 15:20

本文主要是介绍C++/CLI——3继承与值类型、操作符重载与异常,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

C++/CLI——3继承与值类型、操作符重载与异常

继承

C++/cli中的继承用法基本和C#中的用法相同,只不过要注意以下几点:

  1. 标准的C++在继承符号:之后,基类名称之前添加关键字public/protected/private,但是C++/CLI只支持public,所以可写可不写
  2. 标准C++声明抽象基类的方式是在内部至少将一个虚函数=0,设置为纯虚函数,而C++/CLI中则是在类声明后面加上abstract关键字
  3. 抽象函数在C++中叫做纯虚函数,也就是在虚函数后面=0,C++/CLI也支持这样做,另外第二种方式是可以在虚函数后面加上abstract关键字来表示为抽象函数
  4. C++/Cli支持密封类,也就是不能被继承,用sealed关键字ref class Myclass sealed
  5. C++/cli中抽象类和密封类可同时声明,sealed和abstact顺序任意
  6. C++/CLI不支持多继承,只能多继承接口,接口使用interface来声明interface clas IXmlWriter,且接口成员都是公共的和抽象的

案例演示:

interface class ICanShout //定义接口
{void Shout();
};ref class Animal abstract:ICanShout //定义抽象类
{
public:Animal(String^ n,int l);virtual void Eat() abstract;//抽象方法virtual void Shout() =0;//抽象方法String^ name;int legs;
};ref class Dog:Animal
{
public:Dog(String^ n,int l);void Eat() override;virtual void Shout() override;
};ref class Cat :Animal
{
public:Cat(String^ n, int l);void Eat() override;//写不写virtual都可以被重写了,因为基类定义了virtualvirtual void Shout() override;
};Animal::Animal(String^ n, int l)
{this->name = n;this->legs = l;
}Dog::Dog(String^ n, int l):Animal(n,l)
{
}void Dog::Eat()
{Console::WriteLine("我的名字是{0},我吃骨头", name);
}void Dog::Shout()
{Console::WriteLine("我会汪汪叫");
}Cat::Cat(String^ n, int l):Animal(n,l)
{}void Cat::Eat()
{Console::WriteLine("我的名字是{0},我吃鱼", name);
}void Cat::Shout()
{Console::WriteLine("我会喵喵叫");
}
//调用
int main(array<System::String^>^ args)
{Dog^ d = gcnew Dog("tom", 4);d->Eat();d->Shout();Cat^ c = gcnew Cat("jerry", 4);c->Eat();c->Shout();Animal^ a = d;a->Shout();Console::WriteLine("程序结束");
}

值类型

值类型的特点:

  1. 存在栈上
  2. 不进行垃圾回收
  3. 总是直接访问,不使用gcnew
  4. 拷贝类类型是直接拷贝
  5. 不能继承

结构

//使用value关键字
value struct  Point
{//默认publicint x, y;//构造函数不支持默认值
};

结构和类的基本区别:

  • 不能在结构定义时初始化成员,必须在构造器中初始化
  • 不能重写结构默认构造器,因为默认构造器要将所有成员设为默认值
  • 不能有析构和终结器
  • 不支持继承
  • 可实现接口

案例

value struct Line
{String^ name;Point p;
};//使用初始化器,初始化器只能是默认构造函数
Line l = { "线1",{3,4} };
Console::WriteLine(l.name);

枚举

//一定要用public 或者private来限定
//必须这样使用enum  class
public enum  class WeekDay
{Mondy,Tuesday
};
//必须使用类型名称限定
Console::WriteLine(WeekDay::Mondy);

枚举虽然是整数值,但是不能隐式转换,必须这样int day = static_cast<int>(WeekDay::Tuesday);

enum默认是int大小是32位,但是一般1个字节就可以容纳所有的值,为了节省内存,可以增加限定public enum class WeekDay:char

操作符重载

重载操作符的方法和规范基本和C++重载操作符一致,请看下面案例

value struct  IntVal
{
private:int value;
public:IntVal(int v) :value(v) {};int GetValue() { return value; }IntVal operator+(IntVal other){IntVal result(value + other.GetValue());return result;}IntVal operator+(int other){IntVal result(value + other);return result;}static IntVal operator+(int lhs, IntVal rhs){IntVal result(lhs + rhs.GetValue());return result;}
};int main(array<System::String^>^ args)
{IntVal one(1);IntVal two(2);IntVal three = one + two; //IntVal operator+(IntVal other)IntVal t = one + 2; //IntVal operator+(int other)IntVal t = 2 + one ; //static IntVal operator+(int lhs, IntVal rhs)Console::WriteLine(t.GetValue());//输出3Console::WriteLine("程序结束");
}

转换器

上面案例中仅仅实现了static IntVal operator+(int lhs, IntVal rhs),其实还需要实现(IntVal IntVal)\(IntVal,int)其实还有个更简单的办法,也就是重载转换操作符(),将int转换为IntVal。与标准C++不同,在C++中如果定义了int构造器,编译器就允许将int隐式转为IntVal,但是在C++/CLI中必须重载转换操作符

value struct  IntVal
{
private:int value;
public:IntVal(int v) :value(v) {};int GetValue() { return value; }static operator IntVal(int v) //重载转换操作符{return IntVal(v);}static IntVal operator+(IntVal lhs, IntVal rhs)//使用这一个配合重载操作符就可以替代三种签名函数{IntVal result(lhs.value + rhs.value);return result;}
};

另外,与C++不同的是,C++中+并不能自动得到+=,但是在C++/CLI中,重载+会自动得到+=,上面的重载函数完成后,可以实现

IntVal one(1);
one += 2;

递增和递减

标准C++要分别为前++和后++提供两个操作符重载,但是在C++/cli中只需要一个静态重载就可以,需注意要配合重载转换操作符

value struct  IntVal
{
private:int value;
public:IntVal(int v) :value(v) {};int GetValue() { return value; }static operator IntVal(int v) //重载转换操作符{return IntVal(v);}static IntVal operator++(IntVal i)//配合重载转换操作符{i.value++;return i;}
};int main(array<System::String^>^ args)
{IntVal one1(1);IntVal one2(1);Console::WriteLine((one1++).GetValue());Console::WriteLine(one1.GetValue());Console::WriteLine((++one2).GetValue());Console::WriteLine(one2.GetValue());Console::WriteLine("程序结束");
}

image-20240104103844420

引用类型重载操作符

为引用类型重载操作符和为值类型类似,就是要关注对象句柄

ref struct  IntVal
{
private:int value;
public:IntVal(int v) :value(v) {};int GetValue() { return value; }static operator IntVal^(int v) //重载转换操作符{return gcnew IntVal(v);}static IntVal^ operator+(IntVal^ lhs,IntVal^ rhs)//配合重载转换操作符{IntVal^ result = gcnew IntVal(lhs->value + rhs->value);return result;}
};int main(array<System::String^>^ args)
{IntVal^ one = gcnew IntVal(1);IntVal^ two = gcnew IntVal(2);IntVal^ three = one + two;IntVal^ four = two + 2;Console::WriteLine(three->GetValue());Console::WriteLine(four->GetValue());Console::WriteLine("程序结束");
}

image-20240104104435021

异常

C++/CLI中使用异常基本与C#中的使用方法相同。在动态类型转换上,safe_cast转型失败会抛出InvalidCastException,而dynamic_cast则会返回空指针。

这篇关于C++/CLI——3继承与值类型、操作符重载与异常的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

利用c++判断水仙花数并输出示例代码

《利用c++判断水仙花数并输出示例代码》水仙花数是指一个三位数,其各位数字的立方和恰好等于该数本身,:本文主要介绍利用c++判断水仙花数并输出的相关资料,文中通过代码介绍的非常详细,需要的朋友可以... 以下是使用C++实现的相同逻辑代码:#include <IOStream>#include <vec

基于C++的UDP网络通信系统设计与实现详解

《基于C++的UDP网络通信系统设计与实现详解》在网络编程领域,UDP作为一种无连接的传输层协议,以其高效、低延迟的特性在实时性要求高的应用场景中占据重要地位,下面我们就来看看如何从零开始构建一个完整... 目录前言一、UDP服务器UdpServer.hpp1.1 基本框架设计1.2 初始化函数Init详解

SpringBoot全局异常拦截与自定义错误页面实现过程解读

《SpringBoot全局异常拦截与自定义错误页面实现过程解读》本文介绍了SpringBoot中全局异常拦截与自定义错误页面的实现方法,包括异常的分类、SpringBoot默认异常处理机制、全局异常拦... 目录一、引言二、Spring Boot异常处理基础2.1 异常的分类2.2 Spring Boot默

SpringBoot的全局异常拦截实践过程

《SpringBoot的全局异常拦截实践过程》SpringBoot中使用@ControllerAdvice和@ExceptionHandler实现全局异常拦截,@RestControllerAdvic... 目录@RestControllerAdvice@ResponseStatus(...)@Except

C++ 右值引用(rvalue references)与移动语义(move semantics)深度解析

《C++右值引用(rvaluereferences)与移动语义(movesemantics)深度解析》文章主要介绍了C++右值引用和移动语义的设计动机、基本概念、实现方式以及在实际编程中的应用,... 目录一、右值引用(rvalue references)与移动语义(move semantics)设计动机1

C++ move 的作用详解及陷阱最佳实践

《C++move的作用详解及陷阱最佳实践》文章详细介绍了C++中的`std::move`函数的作用,包括为什么需要它、它的本质、典型使用场景、以及一些常见陷阱和最佳实践,感兴趣的朋友跟随小编一起看... 目录C++ move 的作用详解一、一句话总结二、为什么需要 move?C++98/03 的痛点⚡C++

Java方法重载与重写之同名方法的双面魔法(最新整理)

《Java方法重载与重写之同名方法的双面魔法(最新整理)》文章介绍了Java中的方法重载Overloading和方法重写Overriding的区别联系,方法重载是指在同一个类中,允许存在多个方法名相同... 目录Java方法重载与重写:同名方法的双面魔法方法重载(Overloading):同门师兄弟的不同绝

Go异常处理、泛型和文件操作实例代码

《Go异常处理、泛型和文件操作实例代码》Go语言的异常处理机制与传统的面向对象语言(如Java、C#)所使用的try-catch结构有所不同,它采用了自己独特的设计理念和方法,:本文主要介绍Go异... 目录一:异常处理常见的异常处理向上抛中断程序恢复程序二:泛型泛型函数泛型结构体泛型切片泛型 map三:文

详解C++ 存储二进制数据容器的几种方法

《详解C++存储二进制数据容器的几种方法》本文主要介绍了详解C++存储二进制数据容器,包括std::vector、std::array、std::string、std::bitset和std::ve... 目录1.std::vector<uint8_t>(最常用)特点:适用场景:示例:2.std::arra

C++构造函数中explicit详解

《C++构造函数中explicit详解》explicit关键字用于修饰单参数构造函数或可以看作单参数的构造函数,阻止编译器进行隐式类型转换或拷贝初始化,本文就来介绍explicit的使用,感兴趣的可以... 目录1. 什么是explicit2. 隐式转换的问题3.explicit的使用示例基本用法多参数构造