0.目录

1.指针的判别

2.构造中的异常

3.令人迷惑的写法

4.小结

1.指针的判别

面试问题:

编写程序判断一个变量是不是指针。

指针的判别:

拾遗:

  • C++中仍然支持C语言中的可变参数函数
  • C++编译器的匹配调用优先级
    1. 重载函数
    2. 函数模板
    3. 变参函数

示例1——匹配调用优先级:

#include <iostream>

using namespace std;

void test(int i) // 1.重载函数
{
cout << "void test(int i)" << endl;
} template
<typename T>
void test(T i) // 2.函数模板
{
cout << "void test(T i)" << endl;
} void test(...) // 3.变参函数
{
cout << "void test(...)" << endl;
} int main(int argc, char *argv[])
{
int i = 0; test(i); return 0;
}

运行结果为:

[root@bogon Desktop]# g++ test.cpp
[root@bogon Desktop]# ./a.out
void test(int i)

示例2——匹配调用优先级:

#include <iostream>

using namespace std;

template
<typename T>
void test(T i) // 2.函数模板
{
cout << "void test(T i)" << endl;
} void test(...) // 3.变参函数
{
cout << "void test(...)" << endl;
} int main(int argc, char *argv[])
{
int i = 0; test(i); return 0;
}

运行结果为:

[root@bogon Desktop]# g++ test.cpp
[root@bogon Desktop]# ./a.out
void test(T i)

思路:

  • 将变量分为两类:指针 vs 非指针
  • 编写函数:
    1. 指针变量调用时返回true
    2. 非指针变量调用时返回false

函数模板与变参函数的化学反应:

示例——指针判断:

#include <iostream>

using namespace std;

class Test
{
public:
Test() { }
virtual ~Test() { }
}; template
<typename T>
bool IsPtr(T* v) // match pointer
{
return true;
} bool IsPtr(...) // match non-pointer
{
return false;
} int main(int argc, char *argv[])
{
int i = 0;
int* p = &i; cout << "p is a pointer: " << IsPtr(p) << endl; // true
cout << "i is a pointer: " << IsPtr(i) << endl; // false Test t;
Test* pt = &t; cout << "pt is a pointer: " << IsPtr(pt) << endl; // true
cout << "t is a pointer: " << IsPtr(t) << endl; // false return 0;
}

运行结果为:

[root@bogon Desktop]# g++ test.cpp
test.cpp: In function ‘int main(int, char**)’:
test.cpp:36: warning: cannot pass objects of non-POD type ‘class Test’ through ‘...’
[root@bogon Desktop]# ./a.out
p is a pointer: 1
i is a pointer: 0
pt is a pointer: 1
非法指令

(变参函数是C语言中的东西,根本不知道对象是什么,于是会报错。)

存在的缺陷:

  • 变参函数无法解析对象参数,可能造成程序崩溃!!

进一步的挑战:

  • 如何让编译器精确匹配函数,但不进行实际的调用?

示例——指针判断优化:

#include <iostream>

using namespace std;

class Test
{
public:
Test() { }
virtual ~Test() { }
}; template
<typename T>
char IsPtr(T* v) // match pointer
{
return 'd';
} int IsPtr(...) // match non-pointer
{
return 0;
} #define ISPTR(p) (sizeof(IsPtr(p)) == sizeof(char)) int main(int argc, char *argv[])
{
int i = 0;
int* p = &i; cout << "p is a pointer: " << ISPTR(p) << endl; // true
cout << "i is a pointer: " << ISPTR(i) << endl; // false Test t;
Test* pt = &t; cout << "pt is a pointer: " << ISPTR(pt) << endl; // true
cout << "t is a pointer: " << ISPTR(t) << endl; // false return 0;
}

运行结果为:

[root@bogon Desktop]# g++ test.cpp
[root@bogon Desktop]# ./a.out
p is a pointer: 1
i is a pointer: 0
pt is a pointer: 1
t is a pointer: 0

(只匹配,不运行,就不会报错。)

2.构造中的异常

2.1 如果构造函数中抛出异常会发生什么?

面试问题:

如果构造函数中抛出异常会发生什么情况?

构造函数中抛出异常:

  • 构造过程立即停止
  • 当前对象无法生成
  • 析构函数不会被调用
  • 对象所占用的空间立即收回

工程项目中的建议:

  • 不要在构造函数中抛出异常
  • 当构造函数可能产生异常时,使用二阶构造模式

示例——构造中的异常:

#include <iostream>

using namespace std;

class Test
{
public:
Test()
{
cout << "Test()" << endl; throw 0;
}
virtual ~Test()
{
cout << "~Test()" << endl;
}
}; int main(int argc, char *argv[])
{
Test* p = reinterpret_cast<Test*>(1); try
{
p = new Test();
}
catch(...)
{
cout << "Exception..." << endl;
} cout << "p = " << p << endl; return 0;
}

运行结果为:

[root@bogon Desktop]# g++ test.cpp
[root@bogon Desktop]# ./a.out
Test()
Exception...
p = 0x1

p指针并没有被赋值。(对象构造抛出异常,连空指针都不会返回,也就是说不会发生内存泄漏。)

2.2 如果析构函数中抛出异常会发生什么?

析构中的异常:

  • 避免在析构函数中抛出异常!!
  • 析构函数的异常将导致:
    1. 对象所使用的资源无法完全释放。

3.令人迷惑的写法

3.1 模板中的二义性

下面的程序想要表达什么意思?

历史上的原因:

  • 早期的C++直接复用class关键字来定义模板
  • 但是泛型编程针对的不只是类类型
  • class关键字的复用使得代码出现二义性

typename诞生的直接诱因:

  • 自定义类类型内部的嵌套类型
  • 不同类中的同一个标识符可能导致二义性
  • 编译器无法辨识标识符究竟是什么

示例1——能编译过的普通情况:

#include <iostream>

using namespace std;

int a = 0;

class Test_1
{
public:
static const int TS = 1;
}; class Test_2
{
public:
struct TS
{
int value;
};
}; template
< class T >
void test_class()
{
T::TS * a; // 1. 通过泛指类型 T 内部的数据类型 TS 定义指针变量 a (推荐的解读方式)
// 2. 使用泛指类型 T 内部的静态成员变量 TS 与全局变量 a 进行乘法操作
} int main(int argc, char *argv[])
{
test_class<Test_1>();
// test_class<Test_2>(); return 0;
}

示例2——模板中的二义性:

#include <iostream>

using namespace std;

int a = 0;

class Test_1
{
public:
static const int TS = 1;
}; class Test_2
{
public:
struct TS
{
int value;
};
}; template
< class T >
void test_class()
{
T::TS * a; // 1. 通过泛指类型 T 内部的数据类型 TS 定义指针变量 a (推荐的解读方式)
// 2. 使用泛指类型 T 内部的静态成员变量 TS 与全局变量 a 进行乘法操作
} int main(int argc, char *argv[])
{
// test_class<Test_1>();
test_class<Test_2>(); return 0;
}

运行结果为:

[root@bogon Desktop]# g++ test.cpp
test.cpp: In function ‘void test_class() [with T = Test_2]’:
test.cpp:33: instantiated from here
test.cpp:26: error: dependent-name ‘T::TS’ is parsed as a non-type, but instantiation yields a type
test.cpp:26: note: say ‘typename T::TS’ if a type is meant

示例3——使用typename解决模板中的二义性:

#include <iostream>

using namespace std;

int a = 0;

class Test_1
{
public:
static const int TS = 1;
}; class Test_2
{
public:
struct TS
{
int value;
};
}; template
< class T >
void test_class()
{
typename T::TS * a; // 1. 通过泛指类型 T 内部的数据类型 TS 定义指针变量 a (推荐的解读方式)
// 2. 使用泛指类型 T 内部的静态成员变量 TS 与全局变量 a 进行乘法操作
} int main(int argc, char *argv[])
{
// test_class<Test_1>();
test_class<Test_2>(); return 0;
}

typename的作用:

  1. 在模板定义中声明泛指类型
  2. 明确告诉编译器其后的标识符为类型

3.2 函数异常声明

下面的程序想要表达什么意思?

  • try ... catch用于分隔正常功能代码与异常处理代码
  • try ... catch可以直接将函数实现分隔为2部分
  • 函数声明和定义时可以直接指定可能抛出的异常类型
  • 异常声明成为函数的一部分可以提高代码可读性

函数异常声明的注意事项:

  • 函数异常声明是一种与编译器之间的契约
  • 函数声明异常后就只能抛出声明的异常
    1. 抛出其它异常将导致程序运行终止
    2. 可以直接通过异常声明定义无异常函数

示例——新的异常写法:

#include <iostream>

using namespace std;

int func(int i, int j) throw(int, char)
{
if( (0 < j) && (j < 10) )
{
return (i + j);
}
else
{
throw '0';
}
} void test(int i) try
{
cout << "func(i, i) = " << func(i, i) << endl;
}
catch(int i)
{
cout << "Exception: " << i << endl;
}
catch(...)
{
cout << "Exception..." << endl;
} int main(int argc, char *argv[])
{
test(5); test(10); return 0;
}

运行结果为:

[root@bogon Desktop]# g++ test.cpp
[root@bogon Desktop]# ./a.out
func(i, i) = 10
Exception...

4.小结

  • C++中依然支持变参函数
  • 变参函数无法很好的处理对象参数
  • 利用函数模板和变参函数能够判断指针变量
  • 构造函数和析构函数中不要抛出异常
  • class可以用来在模板中定义泛指类型(不推荐)
  • typename是可以消除模板中的二义性
  • try...catch 可以将函数体分成2部分
  • 异常声明能够提供程序的可读性

C++解析(30):关于指针判别、构造异常和模板二义性的疑问的更多相关文章

  1. 异常处理与MiniDump详解(2) 智能指针与C++异常

    write by 九天雁翎(JTianLing) -- blog.csdn.net/vagrxie 讨论新闻组及文件 一.   综述 <异常处理与MiniDump详解(1) C++异常>稍 ...

  2. 数据结构图文解析之:栈的简介及C++模板实现

    0. 数据结构图文解析系列 数据结构系列文章 数据结构图文解析之:数组.单链表.双链表介绍及C++模板实现 数据结构图文解析之:栈的简介及C++模板实现 数据结构图文解析之:队列详解与C++模板实现 ...

  3. 数据结构图文解析之:队列详解与C++模板实现

    0. 数据结构图文解析系列 数据结构系列文章 数据结构图文解析之:数组.单链表.双链表介绍及C++模板实现 数据结构图文解析之:栈的简介及C++模板实现 数据结构图文解析之:队列详解与C++模板实现 ...

  4. Flask框架(二)—— 反向解析、配置信息、路由系统、模板、请求响应、闪现、session

    Flask框架(二)—— 反向解析.配置信息.路由系统.模板.请求响应.闪现.session 目录 反向解析.配置信息.路由系统.模板.请求响应.闪现.session 一.反向解析 1.什么是反向解析 ...

  5. Delphi之通过代码示例学习XML解析、StringReplace的用法(异常控制 good)

    *Delphi之通过代码示例学习XML解析.StringReplace的用法 这个程序可以用于解析任何合法的XML字符串. 首先是看一下程序的运行效果: 以解析这样一个XML的字符串为例: <? ...

  6. HashMap 源码解析(一)之使用、构造以及计算容量

    目录 简介 集合和映射 HashMap 特点 使用 构造 相关属性 构造方法 tableSizeFor 函数 一般的算法(效率低, 不值得借鉴) tableSizeFor 函数算法 效率比较 tabl ...

  7. DRF框架(二)——解析模块(parsers)、异常模块(exception_handler)、响应模块(Response)、三大序列化组件介绍、Serializer组件(序列化与反序列化使用)

    解析模块 为什么要配置解析模块 1)drf给我们提供了多种解析数据包方式的解析类 form-data/urlencoded/json 2)我们可以通过配置来控制前台提交的哪些格式的数据后台在解析,哪些 ...

  8. C++解析(27):数组、智能指针与单例类模板

    0.目录 1.数组类模板 1.1 类模板高效率求和 1.2 数组类模板 1.3 堆数组类模板 2.智能指针类模板 2.1 使用智能指针 2.2 智能指针类模板 3.单例类模板 3.1 实现单例模式 3 ...

  9. C++学习笔记30,指针的引用(2)

    版权声明:本文为博主原创文章,未经博主同意不得转载. https://blog.csdn.net/guang_jing/article/details/32910093 能够创建不论什么类型的引用,包 ...

随机推荐

  1. itop4412学习-超级块操作

    1. 先看下超级块支持的函数列表,文件路径\4412_SCP精英版\Android源码\iTop4412_Kernel_3.0_20140521\iTop4412_Kernel_3.0\include ...

  2. javaweb(十)——HttpServletRequest对象(一)

    一.HttpServletRequest介绍 HttpServletRequest对象代表客户端的请求,当客户端通过HTTP协议访问服务器时,HTTP请求头中的所有信息都封装在这个对象中,通过这个对象 ...

  3. 如何下载YouTube 60fps视频

    YouTube上面不仅支持分辨率为4K和8K的视频,同时也开启了对60fps视频的支持.60帧的视频广泛用于游戏和体育视频中,使视频看起来更加流畅和细腻.对游戏玩家来说,YouTube对60fps支持 ...

  4. 003 -- Dubbo简单介绍

    1:Dubbo的基本概念 dubbo是阿里巴巴SOA服务治理 方案的核心框架,每天为20000+个服务次的数据量访问支持.dubbo是一个分布式的服务框架,致力于提供高性能和透明化的RPC远程服务调用 ...

  5. 3. IP地址转换函数

    一.字符串表示的IP地址需要被转化为整数(二进制数)方能使用 IPv4地址:点分十进制字符串 IPv6地址:十六进制字符串 有时(如记录日志),我们则要把整数(二进制数)表示的IP地址转化为可读的字符 ...

  6. 吴恩达机器学习笔记——正规方程(Normal Equation)

    问题描述:m examples : (x(1),y(1)), (x(2),y(2)),..., (x(m),y(m)) and n features; 计算方法:θ = (XTX)-1XTy; 计算过 ...

  7. Week4_Linux书本一二两章

    第一章的学习内容就是对Linux内核有一个基本的了解,同时知道一些关于Linux的知识. 学习Linux,可以自己有一台装有Linux操作系统的机器,源代码的作用无可替代: Linux发展历程简介:L ...

  8. 第四周 实验一 Java开发环境的熟悉 报告

    Java开发环境的熟悉 实验内容 1.IDEA的安装过程 2.使用IDEA代替虚拟机运行.编译.调试Java程序 实验要求 1.没有Linux基础的同学建议先学习<Linux基础入门(新版)&g ...

  9. TDGA-需求分析

    李青:绝对的技术控,团队中扮演“猪”的角色,勤干肯干,是整个团队的主心骨,课上紧跟老师的步伐,下课谨遵老师的指令,课堂效率高,他的编程格言“没有编不出来的程序,只有解决不了的bug”. 胡金辉:半两油 ...

  10. 团队Alpha冲刺(二)

    目录 组员情况 组员1(组长):胡绪佩 组员2:胡青元 组员3:庄卉 组员4:家灿 组员5:凯琳 组员6:丹丹 组员7:家伟 组员8:政演 组员9:黄鸿杰 组员10:刘一好 组员11:何宇恒 展示组内 ...