本文和大家分享的主要是c++开发中,泛型使用导致的膨胀问题的解决,希望对大家有所帮助。
如下定义两个list,元素类型不同:
list<int> l1;listl2;
如果是用C语来做应该怎么办?它会对应list写一套代码,再对list写一套。每套都有相同的成员函数,只是变量类型各自不同罢了。
下面是list的C语言实现方式:
//! code-1struct list_int_item {
int value;
struct list_int_item *next;
};
struct list_int {
struct list_int_item *head;
size_tsize;
};
void list_int_insert(struct list_int *p, int value);int list_int_sort(struct list_int *p);bool list_int_empty(struct list_int *p);
...
下面是list的C语言实现方式:
//! code-2struct list_string_item {
string value;
struct list_string_item *next;
};
struct list_string {
struct list_string_item *head;
size_tsize;
};
void list_string_insert(struct list_int *p, string value);int list_string_sort(struct list_int *p);bool list_string_empty(struct list_int *p);
...
两者之间就是类型的差别。所以很多时间,在C语言中我们就用宏来替代它的类型,如下:
//! code-3#define LIST_DECLARE(TYPE)
struct list_##TYPE##_item {
TYPE## value;
struct list_##TYPE##_item *next;
};
struct list_##TYPE {
struct list_##TYPE##_item *head;
size_tsize;
};
void list_##TYPE##_insert(struct list_##TYPE *p, ##TYPE## value);
int list_##TYPE##_sort(struct list_##TYPE *p);
bool list_##TYPE##_empty(struct list_##TYPE *p);
...
然后在头文件中是这样定义list的:
//! code-4
LIST_DECLARE(double)
所以,泛型产生冗余代码是无法避免的,至少用C来做这样的泛型也是无法避免的。
既然无法避免的,那就看看怎么尽可能以避免上述的问题。在《Effective C++》中有一章节专门提到:不要在模板中使用不必要的参数。因为每一个不同的参数编译器都会为之生成一套相应的代码。
如果代码中只有一种数据类型,就算用该类型定义了多个变量,编译器是不是只会生成一套相关的代码?(应该是这样的)。
写个例子对比一下:(省略不必要的代码)
test1.cpp,里面只有map<INT, string="">,但定义了m1, m2, m3。
//! code-5
map<int, string> m1;
map<int, string> m2;
map<int, string> m3;
m1.insert(std::make_pair(1, "hello"));
m2.insert(std::make_pair(1, "hi"));
m3.insert(std::make_pair(1, "lichunjun"));
test2.cpp,与test1.cpp相比,里面有三个类型:
//! code-6
map<int, string> m1;
map<int, double> m2;
map<int, int> m3;
m1.insert(std::make_pair(1, "hello"));
m2.insert(std::make_pair(1, 1.2));
m3.insert(std::make_pair(1, 44));
结果,编译出来的可执行文件大小比较:
[[email protected]]$ lltest1test2
-rwxrwxr-x. 1 18784 Mar 19 22:01 test1
-rwxrwxr-x. 1 35184 Mar 19 22:03 test2
test2比test1大一倍,原因不用多说。
还有一个问题:指针是不是被认为是一个类型?
上面的list与list不能共用同一套代码,根据的原因是因为int与string这两种类型在空间大小与赋值的方式上都是不同的。所以,必须生成两套代码来实现。
而指针,不管是什么指针,它们都是一样的。我们可以用void*代表所有的指针类型。
于是我们将上面的代码改改,再测试一下:
//! code-7
map<int, string*> m1;
map<int, string*> m2;
map<int, string*> m3;
m1.insert(std::make_pair(1, new string("hello")));
m2.insert(std::make_pair(1, new string("hi")));
m3.insert(std::make_pair(1, new string("lichunjun")));
与
//! code-8
map<int, string*> m1;
map<int, double*> m2;
map<int, int*> m3;
m1.insert(std::make_pair(1, new string("hello")));
m2.insert(std::make_pair(1, new double(1.2)));
m3.insert(std::make_pair(1, new int(44)));
结果是这样的:
-rwxrwxr-x. 1 18736 Mar 19 23:05 test1-rwxrwxr-x. 1 35136 Mar 19 23:05 test2
预期的结果test1与test2相差不多,但从结果上看并没有什么优化,结果有点令人失望~
思考:C++有没有什么参数可以优化这个?
如果没有,为了节省空间,我们只能将所有的指针统一定义成void*类型了,在使用时再强制转换。
//! code-9
map<int, void*> m1;
map<int, void*> m2;
map<int, void*> m3;
m1.insert(std::make_pair(1, new string("hello")));
m2.insert(std::make_pair(1, new double(1.2)));
m3.insert(std::make_pair(1, new int(44)));
cout << *static_cast<STRING*>(m1[1]) << endl;
cout << *static_cast<double*>(m2[1]) << endl;
cout << *static_cast<int*>(m3[1]) << endl;
如上代码是将code-8的基础上,将所有的指定都定义成了void*,在使用的时候用static_cast进行强制转换成对应的指针类型。
如此得到的代码大小与code-7的比较,只多了16个字节。
但这种做法是很不可取的,必须用void*指针之后,编译器不再对类型进行检查,很容易把类型搞混淆。
来源:伯乐在线
C++中泛型使用导致的膨胀问题如何解决?
时间: 2024-11-08 01:42:11
C++中泛型使用导致的膨胀问题如何解决?的相关文章
ViewPager中嵌套HorizontalScrollView导致无法滑动换页冲突解决
在之前的开发过程中,遇到过ViewPager中嵌套HorizontalScrollView导致无法横向滑动换页,最终也是通过对onTouchEvent方法判断滑动的状态来进行控制左右的滑动,注意的是onTouchEvent的事件分发的方向正好与onInterceptTouchEvent相反,是从下往上分发的,所以也会先执行子View的onTouchEvent方法(如果想进一步了解View的分发机制可以去进一步查阅资料,比如http://blog.csdn.net/a553181867/artic
[转]:Delphi XE中泛型数组的使用范例
Delphi XE中泛型数组的使用范例,下面的范例简单的使用了泛型字符串数组,如用 TArray 代替 array of Word, 还可以使用 TArray 类提供的算法(就是少了点). uses Generics.Collections, Generics.Defaults; {测试 TArray 的 Sort 方法} procedure TForm1.Button1Click(Sender: TObject); var arr: TArray<string>; //同 array of
服务器中了病毒导致交换机无法正常运行
服务器:联想万全 问题:无法访问服务器上的网站,网断断断续续,最终导致服务器宕机. 原因:服务器中了病毒,导致IIS无法运行的同时,服务器也宕机. 解决方式: 首先,分析原因,IIS发布的所有网站无法运行,恢复初始IIS发布网站还是未成功.通过与团队技术人员沟通,最终确定操作系统中了病毒. 其次,恢复操作系统,恢复的过程中没有采用光驱安装,原因是光驱已损坏:用U盘恢复系统的时间失败,原因是从网上下了一个没有经过测试就使用的系统镜像,结果不兼容,导致安装失败: 再次,通过恢复先前的备份系统文件,文
Java中泛型在集合框架中的应用
泛型是Java中的一个重要概念,上一篇文章我们说过,当元素存入集合时,集合会将元素转换为Object类型存储,当取出时也是按照Object取出的,所以用get方法取出时,我们会进行强制类型转换,并且通过代码也可以看出来,我们放入其他类型时,如字符串,编译器不会报错,但是运行程序时会抛出类型错误异常,这样给开发带来很多不方便,用泛型就解决了这个麻烦 泛型规定了某个集合只能存放特定类型的属性,当添加类型与规定不一致时,编译器会直接报错,可以清楚的看到错误 当我们从List中取出元素时直接取出即可,不
Java 中 泛型的限定
泛型 一般 出现在集合中,迭代器中 也会出现! 泛型 是为了 提高代码的 安全性. 泛型 确保数据类型的唯一性. 在我们常用的容器中, 越是 单一 约好处理啊! 泛型的限定: ? 是通配符 指代 任意类型 泛型的限定上限: <? extends E> 接受 E 或者 E 的子类型. 泛型的限定下限: <? super E> 接收 E 或者 E 的父类. 泛型的限定上限 (定义父类 填装子类 类型!) 代码: package stu.love.v; import java
Java中泛型 使用
泛型: 1.5 之后出现 提高安全 1 泛型 确定 集合容器的类型. 2 <> 接收一种数据类型,(引用数据类型) ArrayList<String> lis = new ArrayList<String>() 目的: 将运行时期的 错误 转化到 编译时期,提高了安全性! 3 不需要 强制类型转换. 更加安全! 泛型的 擦除: 泛型在编译时期使用!使用完毕直接擦除. 编译完的时候 不存在 泛型. 好处: 使用了 泛型,不自需要强制类型
[转载]DllMain中不当操作导致死锁问题的分析--导致DllMain中死锁的关键隐藏因子2
(转载于breaksoftware的csdn博客) 本文介绍使用Windbg去验证<DllMain中不当操作导致死锁问题的分析--导致DllMain中死锁的关键隐藏因子>中的结论,调试对象是文中刚开始那个例子. 1 g 让程序运行起来 2 ctrl+break 中断程序 3 ~ 查看线程数 其实该程序自己运行起来的线程只有ID为0.TID为afc的线程.18c4线程是我们在windbg中输入ctrl+break,导致windbg在我们调试的进程中插入的一个中断线程.以后我们看到是这个线程的操作
[Android学习笔记]ListView中含有Button导致无法响应onItemClick回调的解决办法
转自:http://www.cnblogs.com/eyu8874521/archive/2012/10/17/2727882.html 问题描述: 当ListView的Item中的控件只是一些展示类控件时(比如TextView),注册ListView的监听setOnItemClickListener之后,当点击Item时候会触发onItemClick回调. 但是,当Item中存在Button(继承于Button)的控件时,onItemClick回调不会被触发. 解决方案: 在Item的布局文件
Java中使用FileputStream导致中文乱码问题的修改方案
1 package com.pocketdigi; 2 3 import java.io.File; 4 import java.io.FileInputStream; 5 import java.io.FileOutputStream; 6 import java.io.IOException; 7 import java.io.InputStreamReader; 8 import java.io.OutputStreamWriter; 9 10 public class Main { 11