目标:

已知这个接口:

std::vector<double> add_vec(double *d1, double *d2)
{
.....
return result;
}

我们自定义了这个类:

class array
{
int size_;
double *data_;
....
};

要求实现:

array v1(10), v2(10);
add_vec(v1, v2);

留意到,当array 类型传入add_vec时,会遇到(double*) 类型的“试探性”强制转换, 所以,我们只要做一个从array到,(double*)的类型转换即可。

这样做的好处

  • 1.有了类型转换,我们不需要再写这样一个多态函数实现array 与array的相加
std::vector<double> add_vec(array d1, array d2)
{
.....
return result;
}
  • 2.以下的接口:
std::vector<double> add_vec(double* d1, double* d2)

可能重写起来非常麻烦,甚至add_vec(double*, double*) 是别人封装好的,我们看不到源代码,此时更谈不上重写

实现方法及测试代码:(直接在这个代码上增加,所以代码有些乱)

C++类型转换操作符(type conversion operator)

#include <iostream>

 class D {
public:
D(double d) : d_(d) {
data_=new double[10];
for(int i=0; i<10; i++)
data_[i]=d;
} ~D() { delete []data_; } /* “(int)D”类型转换 */
operator int() const {
std::cout << "(int)d called!" << std::endl;
return static_cast<int>(d_);
} // for: double me= *d1; //ok
double& operator*(void) {
return *data_;
} // for: double *me2= (double*)d1;
operator double*(void){
return data_;
} private:
double d_;
double *data_;
}; int add(int a, int b) {
return a + b;
} #include <vector>
std::vector<double> add_vec(double* d1, double* d2)
{
std::vector<double> r(10);
for(int i=0; i<10; i++){
r[i]=d1[i]+d2[i];
} return r;
} int main() {
D d1 = 1.1;
D d2 = 2.2;
std::cout << add(d1, d2) << std::endl; double me= *d1; //ok
std::cout<<"me:"<<me<<std::endl;
double *me2= (double*)d1; // ok std::vector<double> r;
r=add_vec(d1, d2);
for(int i=0; i<r.size(); i++){
std::cout<<r[i]<<" "<<std::endl;
} //std::cout << add_vec(d1, d2) << std::endl; return 0;
}

输出:

(int)d called!
(int)d called!
3
me:1.1
3.3 
3.3 
3.3 
3.3 
3.3 
3.3 
3.3 
3.3 
3.3 
3.3

补充:

注意到:

    double me=    *d1;
double *me= (double*)d1;

第二个的意义为把 D类型当成一个指针, 然后把把指针(即地址)赋值给me

第一个意义为:把D类型先假设指针,然后把指针指向的值转给me

二者等价,所以,可删除其中一个冗余定义,经过测试,代码仍能正常运行

    // for: double *me2= (double*)d1;
operator double*(void){
return data_;
}

但是如果:

    // for: double me=    *d1;      //ok
double& operator*(void) {
return *data_;
}

则会报错:

Press ENTER or type command to continue
main.cpp: In function ‘int main()’:
main.cpp:62:27: error: invalid cast from type ‘D’ to type ‘double*’
main.cpp:63:18: error: invalid conversion from ‘int’ to ‘double*’ [-fpermissive]
main.cpp:69:21: error: invalid conversion from ‘int’ to ‘double*’ [-fpermissive]
main.cpp:44:21: error:   initializing argument 1 of ‘std::vector<double> add_vec(double*, double*)’ [-fpermissive]
main.cpp:69:21: error: invalid conversion from ‘int’ to ‘double*’ [-fpermissive]
main.cpp:44:21: error:   initializing argument 2 of ‘std::vector<double> add_vec(double*, double*)’ [-fpermissive]

仔细理解,编译器这样做是有道理的,因为第一个重载只是第一个的特殊情况,不能包括第一个,而第二个重载却能包含第一个

更多参见

Can a cast operator be explicit?

How do conversion operators work in C++?

C++库研究笔记——操作符重载实现类型转换&这样做的意义的更多相关文章

  1. C#中如何利用操作符重载和转换操作符 (转载)

    操作符重载 有的编程语言允许一个类型定义操作符应该如何操作类型的实例,比如string类型和int类型都重载了(==)和(+)等操作符,当编译器发现两个int类型的实例使用+操作符的时候,编译器会生成 ...

  2. C#构造函数、操作符重载以及自定义类型转换

    构造器 构造器(构造函数)是将类型的实例初始化的特殊方法.构造器可分为实例构造器和类型构造器,本节将详细介绍有关内容. 实例构造器 顾名思义,实例构造器的作用就是对类型的实例进行初始化.如果类没有显示 ...

  3. 21.C++- "++"操作符重载、隐式转换之explicit关键字、类的类型转换函数

    ++操作符重载 ++操作符分为前置++和后置++,比如: ++a;  a++; ++操作符可以进行全局函数或成员函数重载 重载前置++操作符不需要参数 重载后置++操作符需要一个int类型的占位参数 ...

  4. 用ECMAScript4 ( ActionScript3) 实现Unity的热更新 -- 操作符重载和隐式类型转换

    C#中,某些类型会定义隐式类型转换和操作符重载.Unity中,有些对象也定义了隐式类型转换和操作符重载.典型情况有:UnityEngine.Object.UnityEngine.Object的销毁是调 ...

  5. C#中如何利用操作符重载和转换操作符

    操作符重载 有的编程语言允许一个类型定义操作符应该如何操作类型的实例,比如string类型和int类型都重载了(==)和(+)等操作符,当编译器发现两个int类型的实例使用+操作符的时候,编译器会生成 ...

  6. C++ operator overload -- 操作符重载

    C++ operator overload -- 操作符重载 2011-12-13 14:18:29 分类: C/C++ 操作符重载有两种方式,一是以成员函数方式重载,另一种是全局函数. 先看例子 # ...

  7. C++基础知识:操作符重载

    1.C++标准库: C++标准库并不是C++语言的一部分C++标准库是由C++语言编写而成的类库和函数的集合C++标准库中定义的类和对象都位于std命名空间中C++标准库的头文件都不带.h后缀C++标 ...

  8. C++学习笔记-操作符重载

    操作符重载(operator overloading)是一种形式的C++多态,C++将操作符重载扩展到用户自定义的类型,如允许使用+将两个自定义的对象相加,编译器将根据操作数的数目和类型决定使用那种加 ...

  9. 侯捷STL学习(四)--OOP-GP/操作符重载-泛化特化

    C++标准库第二讲 体系结构与内核分析 第1-7节为第一讲 读源代码前的准备 第八节:源代码分布 C++基本语法 模板的使用 数据结构和算法 本课程主要使用:Gnu C 2.9.1与Gun C 4.9 ...

随机推荐

  1. HDU 4946 共线凸包

    题目大意: 一些点在一张无穷图上面,每个点可以控制一些区域,这个区域满足这个点到达这个区域的时间严格小于其他点.求哪些点能够控制无穷面积的区域. 题目思路: 速度小的控制范围一定有限. 速度最大当且仅 ...

  2. 一位ACM过来人的心得(转)

    励志下! 刻苦的训练我打算最后稍微提一下.主要说后者:什么是有效地训练? 我想说下我的理解.很多ACMer入门的时候,都被告知:要多做题,做个500多道就变牛了.其实,这既不是充分条件.也不会是必要条 ...

  3. sharepoint Linq方式的增,删,查,改

    Site9527EntitiesDataContext (重要的类):连接实体与网站List操作 SPContext.Current.Web.Url:获取当前操作的页面 FirstOrDefault: ...

  4. cocos2d-x之 CCSpriteBatchNode 用法总结

    例子1: CCSpriteBatchNode* batch = [CCSpriteBatchNode batchNodeWithFile:@"table.png"];        ...

  5. chrome 远程调试(转)

    http://www.tuicool.com/articles/ZJfeAzi 由于 appspot.com被墙,一般调试不成功. 随着智能手机的普及,移动设备的浏览器功能越来越强大,我们用手机上网时 ...

  6. linux - 开机启动thunderbird、chromium

    cd ~/.config/autostart/(没有autostart,自己 mkdir autostart) vim thunderbird.desktop,输入以下: [Desktop Entry ...

  7. 2013.11.15 初学ant构建

    该做的事情都差不多做完了,今天开始用ant构建,所以学了下ant,其实要不是因为ubuntu时不时的抽风我应该早就可以开始构建了,但重写的时候也想清楚了一些逻辑,优化了一些地方.下面是我这辈子写的第一 ...

  8. 【转】Masonry介绍与使用实践(快速上手Autolayout)

    原文网址:http://adad184.com/2014/09/28/use-masonry-to-quick-solve-autolayout/ 前言 1 MagicNumber -> aut ...

  9. Oracle DBA 的常用Unix参考手册(一)

    作为一名Oracle DBA,在所难免要接触Unix,但是Unix本身又是极其复杂的,想要深刻掌握同样很不容易.那么到底我们该怎么入手呢?Donald K Burleson 的<Unix for ...

  10. Windows 编译器选项 Runtime Library

    https://msdn.microsoft.com/library/2kzt1wy3%28v=vs.100%29.aspx http://blog.csdn.net/ybxuwei/article/ ...