周星星 之 Blog

关注 ASM/C/C++

  VC知识库BLOG :: 首页 :: 新随笔 :: 联系 :: 聚合  :: 登录 ::
  133 随笔 :: 27 文章 :: 2676 评论 :: 11 Trackbacks
<2010年2月>
31123456
78910111213
14151617181920
21222324252627
28123456
78910111213

News

留言簿(52)

随笔分类

随笔档案

文章分类

文章档案

相册

相关链接

搜索

最新评论

阅读排行榜

评论排行榜

问:在c++里怎么能知道一个变量的具体类型,如:c#里的typeof.还有我怎么知道一个变量的类型是某个类型的子类,也就是实现关键字IS

答:
1。运行时获知变量类型名称,可以使用 typeid(变量).name,需要注意不是所有编译器都输出"int"、"float"等之类的名称,对于这类的编译器可以这样使用:float f = 1.1f; if( typeid(f) == typeid(0.0f) ) ……
2。对于多态类实例,想得到实际的类名称,需要使用到RTTI,这需要在编译的时候加上参数"/GR"。
3。对于普通变量,既然是自己写的,那当然也就应该知道它的类型,其实用不着运行时获知;对于多态类实例,既然需要运行时获知实际类型,那么就说明这里不具有多态性,既然没有多态性就不应该抽象它,这属于设计错误,总之,我认为RTTI是多余的。
4。对于多态类实例,使用 typeid(value) == typeid(value)来判断,不如使用 dynamic_cast 来判断,它们的原理是一样的。

事例代码:
#include <iostream>
using namespace std;
int main( void )
{
// sample 1
    cout << typeid(1.1f).name() << endl;
// sample 2
    class Base1
    {
    };
    class Derive1 : public Base1
    {
    };
    Derive1 d1;
    Base1& b1 = d1;
    cout << typeid(b1).name() << endl; // 输出"class Base1",因为Derive1和Base1之间没有多态性
// sample 3, 编译时需要加参数 /GR
    class Base2
    {
        virtual void fun( void ) {}
    };
    class Derive2 : public Base2
    {
    };
    Derive2 d2;
    Base2& b2 = d2;
    cout << typeid(b2).name() << endl; // 输出"class Derive2",因为Derive1和Base1之间有了多态性
// sample 4
    class Derive22 : public Base2
    {
    };
    // 指针强制转化失败后可以比较指针是否为零,而引用却没办法,所以引用制转化失败后抛出异常
    Derive2* pb1 = dynamic_cast<Derive2*>(&b2);
    cout << boolalpha << (0!=pb1) << endl; // 输出"true",因为b2本身就确实是Derive2
    Derive22* pb2 = dynamic_cast<Derive22*>(&b2);
    cout << boolalpha << (0!=pb2) << endl; // 输出"true",因为b2本身不是Derive2

    try {
        Derive2& rb1 = dynamic_cast<Derive2&>(b2);
        cout << "true" << endl;
    } catch( bad_cast )
    {
        cout << "false" << endl;
    }
    try {
        Derive22& rb2 = dynamic_cast<Derive22&>(b2);
        cout << "true" << endl;
    } catch( ... ) // 应该是 bad_cast,但不知道为什么在VC++6.0中却不行
    {
        cout << "false" << endl;
    }

    return 0;
}
posted on 2004-09-13 22:45 周星星 阅读(14642) 评论(14)  编辑 收藏

评论

# re: 关于typeid和RTTI的问答 2004-11-16 10:56 waterflier
>>3。对于普通变量,既然是自己写的,那当然也就应该知道它的类型,其实用不着运行时获知;对于多态类实例,既然需要运行时获知实际类型,那么就说明这里不具有多态性,既然没有多态性就不应该抽象它,这属于设计错误,总之,我认为RTTI是多余的。

不表赞同,RTTI是非常重要的特性.并不只是用来判断2个类型是否相等,或者是输出一个类名称而已.其意义在于可以获得class的元属性.不过对于C++来说,以他的设计理念,RTTI不可能完美的实现,所以使用的范围也非常的有限

# re: 关于typeid和RTTI的问答 2004-12-30 06:13 yesry
问题是:typeid,dynamic_cast要在dll,exe之间判断会出错。而RTTI是为扩展插件等作平台用的。
我的邮箱luodiping@21cn.com

# re: 关于typeid和RTTI的问答 2005-04-04 14:57 se
dynamic_cast 多态转换也要开起 RTTI

# re: 关于typeid和RTTI的问答 2005-06-12 16:59 still05017
似乎只要基类有一个虚函数,然后被派生,就可以使用RTTI来将一个多态指针转换其实际指向的对象类型。但是,一般情况下,基类肯定有一个虚析构函数,所以,RTTI,似乎总是可以使用成功。如果是这样,那还有什么意义?

# re: 关于typeid和RTTI的问答 2005-07-20 09:21 fatpenguin
星星兄,问个无关问题。我公司的系统是日文xp,看你有的正文(比如这篇)怎么很多半角大小的黑块呢?考到word里就没有问题了。改变编码没有反应,不知是否和字体有关。谢先!

# to fatpenguin: 2005-07-20 10:00 周星星
谢谢,但是我也不知道为什么?

# re: 关于typeid和RTTI的问答 2005-11-13 19:48 td
星星兄,问一下 参数 /GR
是怎么加得?

# re: 关于typeid和RTTI的问答 2007-07-27 16:54 强哥
3。对于普通变量,既然是自己写的,那当然也就应该知道它的类型,其实用不着运行时获知;对于多态类实例,既然需要运行时获知实际类型,那么就说明这里不具有多态性,既然没有多态性就不应该抽象它,这属于设计错误,总之,我认为RTTI是多余的。

你也太搞了吧,RTTI的用处绝不止这些,你当标准委员会的专家是吃白饭的啊

# re: 关于typeid和RTTI的问答 2007-11-29 08:29 ryf
dynamic_cast完成rtti功能的效率比较低吧!

# re: 关于typeid和RTTI的问答 2009-09-28 14:12 hljleo
这位老兄说话有点绝对哦。。。。。。

# re: 关于typeid和RTTI的问答 2010-01-29 09:50 andy
rtti在模板编程可是有很大作用

标题  
姓名  
主页
验证码 *
内容   
  登录  使用高级评论  Top
[使用Ctrl+Enter键可以直接提交]