今天是第一次听到C++还有个转换构造函数,之前经常见到默认构造函数、拷贝构造函数、析构函数,但是从没听说过转换构造函数,隐式转换函数也是一样,C++的确是够博大精深的,再次叹服!

         其实我们已经在C/C++中见到过多次标准类型数据间的转换方式了,这种形式用于在程序中将一种指定的数据转换成另一指定的类型,也即是强制转换,比如:int a = int(1.23),其作用是将1.23转换为整形1。然而对于用户自定义的类类型,编译系统并不知道如何进行转换,所以需要定义专门的函数来告诉编译系统改如何转换,这就是转换构造函数和类型转换函数!

一、转换构造函数

        转换构造函数(conversion constructor function) 的作用是将一个其他类型的数据转换成一个类的对象。

        当一个构造函数只有一个参数,而且该参数又不是本类的const引用时,这种构造函数称为转换构造函数。

        转换构造函数是对构造函数的重载。

        例如:

[cpp]

Complex(double r)

{

    real=r;

    imag=0;

}

        其作用是将double型的参数r转换成Complex类的对象,将r作为复数的实部,虚部为0。用户可以根据需要定义转换构造函数,在函数体中告诉编译系统怎样去进行转换。

        那么如何使用转换构造函数进行类型转换呢?我们看如下的例子:

[cpp]

// TypeSwitch.cpp : 定义控制台应用程序的入口点。

//

#include "stdafx.h"

#include <iostream>

using namespace std;

class Complex

{

public:

    Complex():real(0),imag(0){};

    Complex(double r, double i):real(r),imag(i){};

    Complex(double r):real(r),imag(0){};  // 定义转换构造函数

    void Print(){

        cout<<"real = " << real <<" image = "<<imag<<endl;

    }

    Complex& operator+(Complex c){

        return Complex(this->real + c.real, this->imag + c.imag);

    }

private:

    double real;

    double imag;

};

int main(int argc, _TCHAR* argv[])

{

    Complex c;

    c = 1.2;  // 调用转换构造函数将1.2转换为Complex类型

    c.Print();

    Complex c1(2.9, 4.2);

    Complex c2 = c1 + 3.1; // 调用转换构造函数将3.1转换为Complex类型

    c2.Print();

    return 0;

}

        不仅可以将一个标准类型数据转换成类对象,也可以将另一个类的对象转换成转换构造函数所在的类对象。如可以将一个学生类对象转换为教师类对象,可以在Teacher类中写出下面的转换构造函数:

[cpp]

Teacher(Student& s)

{

    num=s.num;

    strcpy(name,s.name);

    sex=s.sex;

}

        使用方法同上!

        注意:

        1.用转换构造函数可以将一个指定类型的数据转换为类的对象。但是不能反过来将一个类的对象转换为一个其他类型的数据(例如将一个Complex类对象转换成double类型数据)。

        2.如果不想让转换构造函数生效,也就是拒绝其它类型通过转换构造函数转换为本类型,可以在转换构造函数前面加上explicit!例如:

[cpp]

// TypeSwitch.cpp : 定义控制台应用程序的入口点。

//

#include "stdafx.h"

#include <iostream>

using namespace std;

class Complex

{

public:

    Complex():real(0),imag(0){};

    Complex(double r, double i):real(r),imag(i){};

    explicit Complex(double r):real(r),imag(0){};  // explicit禁止构造函数的转换功能

    void Print(){

        cout<<"real = " << real <<" image = "<<imag<<endl;

    }

private:

    double real;

    double imag;

};

int main(int argc, _TCHAR* argv[])

{

    Complex c1(1.2, 2.3), c2;

    double d;

    d = c1 + 1.1; // 调用类型转换函数将c1转换为double,编译出错!

    cout<<d<<endl;

    return 0;

}

二、类型转换函数

        用转换构造函数可以将一个指定类型的数据转换为类的对象。但是不能反过来将一个类的对象转换为一个其他类型的数据(例如将一个Complex类对象转换成double类型数据)。而类型转换函数就是专门用来解决这个问题的!

类型转换函数的作用是将一个类的对象转换成另一类型的数据。

        如果已声明了一个Complex类,可以在Complex类中这样定义类型转换函数:

[cpp]

operator double( )

{

     return real;

}

        类型转换函数的一般形式为:

        operator 类型名( )

        {

               实现转换的语句

        }

        注意事项:

        1.在函数名前面不能指定函数类型,函数没有参数。

        2.其返回值的类型是由函数名中指定的类型名来确定的。

        3.类型转换函数只能作为成员函数,因为转换的主体是本类的对象,不能作为友元函数或普通函数。

        4.从函数形式可以看到,它与运算符重载函数相似,都是用关键字operator开头,只是被重载的是类型名。double类型经过重载后,除了原有的含义外,还获得新的含义(将一个Complex类对象转换为double类型数据,并指定了转换方法)。这样,编译系统不仅能识别原有的double型数据,而且还会把Complex类对象作为double型数据处理。

[cpp]

// TypeSwitch.cpp : 定义控制台应用程序的入口点。

//

#include "stdafx.h"

#include <iostream>

using namespace std;

class Complex

{

public:

    Complex():real(0),imag(0){};

    Complex(double r, double i):real(r),imag(i){};

    Complex(double r):real(r),imag(0){};  // 定义转换构造函数

    void Print(){

        cout<<"real = " << real <<" image = "<<imag<<endl;

    }

    operator double(){ // 定义类型转换函数

        return real;

    }

private:

    double real;

    double imag;

};

int main(int argc, _TCHAR* argv[])

{

    Complex c1(1.2, 2.3);

    double d;

    d = c1 + 1.1; // 调用类型转换函数将c1转换为double

    cout<<d<<endl;

    return 0;

}

        本例中,对于d = c1 + 1.1;先调用类型转换函数将c1转为double类型,然后在与1.1相加!

        那么程序中的Complex类对具有双重身份,既是Complex类对象,又可作为double类型数据。Complex类对象只有在需要时才进行转换,要根据表达式的上下文来决定。转换构造函数和类型转换运算符有一个共同的功能: 当需要的时候,编译系统会自动调用这些函数,建立一个无名的临时对象(或临时变量)。

C++转换构造函数和隐式转换函数的更多相关文章

  1. C++转换构造函数和隐式转换函数 ~ 转载

    原文地址: C++转换构造函数和隐式转换函数 用转换构造函数可以将一个指定类型的数据转换为类的对象.但是不能反过来将一个类的对象转换为一个其他类型的数据(例如将一个Complex类对象转换成doubl ...

  2. 显示转换explicit和隐式转换implicit

    用户自定义的显示转换和隐式转换 显式转换implicit关键字告诉编译器,在源代码中不必做显示的转型就可以产生调用转换操作符方法的代码. 隐式转换implicit关键字告诉编译器只有当源代码中指定了显 ...

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

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

  4. explicit 只对构造函数起作用,用来抑制隐式转换。

    class A { private: int a; public: A(int x) :a(x){} void display(){ cout << a << endl; } ...

  5. 【C++自我精讲】基础系列五 隐式转换和显示转换

    [C++自我精讲]基础系列五 隐式转换和显示转换 0 前言 1)C++的类型转换分为两种,一种为隐式转换,另一种为显式转换. 2)C++中应该尽量不要使用转换,尽量使用显式转换来代替隐式转换. 1 隐 ...

  6. C++中的隐式转换和explicit

    隐式转换 c++中的数据类型转换分为隐式转换和显示转换: 显示转换即使用static_cast等方法进行转换,相关内容请参考 <C++数据类型转换>: 隐式转换则是编译器完成的,如,boo ...

  7. scala 隐式转换

    先参考这篇文章:http://www.jianshu.com/p/a344914de895 package com.test.scalaw.test /** * scala隐式转换 */ object ...

  8. C++隐式转换

    #include <iostream> using namespace std; class A { int a; public: A(int n):a(n) { cout << ...

  9. Scala入门系列(十二):隐式转换

    引言 Scala提供的隐式转换和隐式参数功能,是非常有特色的功能.是Java等编程语言所没有的功能.它可以允许你手动指定,将某种类型的对象转换成其他类型的对象.通过这些功能可以实现非常强大而且特殊的功 ...

随机推荐

  1. Hadoop报错:org.apache.hadoop.security.AccessControlException: Permission denied: user=xxxx

    问题出现原因: 因为远程提交hadoop的任务的情况下如果,没有hadoop 的系统环境变量,就会读取当前主机的用户名,所以Hadoop集群的节点中没有该用户名的权限,所以出现的异常. 解决方法: S ...

  2. [CodeIgniter4]讲解-加载静态页

    讲解 本教程旨在向您介绍CodeIgniter框架和MVC体系结构的基本原理.它将向您展示如何以逐步的方式构造基本的CodeIgniter应用程序. 在本教程中,您将创建一个基本的新闻应用程序.您将从 ...

  3. 【kuangbin带你飞】 MST专题

    唉,被班级合唱和复变考试搞得心力交瘁.新算法学不进去,更新下吧 A - Til the Cows Come Home The Head Elder of the tropical island of ...

  4. Dijkstra算法依据项目改进版,输出路径

    package dijkstra; import java.util.ArrayList; public class Dijkstra { ; /*private static int[][] Gra ...

  5. php文件上传 form表单形式

    1.php界面 <?php header( 'Content-Type:text/html;charset=utf-8 ');include_once("conn/conn.php&q ...

  6. Wannafly Camp 2020 Day 6I 你吓到我的马了.jpg - BFS

    暴力BFS即可 #include <bits/stdc++.h> using namespace std; int n,m,f[105][105]; char s[105][105]; s ...

  7. C++构造函数和重载函数运算符如何区分

    构造函数和重载函数运算符如何区分: class Distance { private: int feet; int inches; public: Distance(){ feet = ; inche ...

  8. "const char *" 类型的值不能用于初始化 "char *" 类型的实体

    用vs2019发现会出现这样的问题 最后发现只要设置一下就解决了,解决方案如下: 右击该项目,选择属性->c/c++->语言->将符合模式改为否就行了... 设置之后程序没有再报错, ...

  9. 浅析ReDoS

    ReDoS(Regular expression Denial of Service) 正则表达式拒绝服务攻击.开发人员使用了正则表达式来对用户输入的数据进行有效性校验, 当编写校验的正则表达式存在缺 ...

  10. 比较一下数据结构的链表和linux i2c驱动难度比较