动态网站制作指南 [  QQ表情  ]
[ 投票调查 ]
[ 企业邮箱 ]
[ 网站空间 ]
网络编程 | 站长之家 | 网页制作 | 图形图象 | 操作系统 | 冲浪宝典 | 软件教学 | 网络办公 | 邮件系统 | 网络安全 | 认证考试 | 系统进程
ASP源码 | .Net源码 | PHP源码 | JSP源码 | JAVA源码 | CGI源码 | VB源码 | C++源码 | Delphi源码 | PB源码 | VF源码 | 汇编 | 服务器
电脑书籍下载:程序设计书籍 | 数据库教程书籍 | 平面与多媒体书籍 | 网络通讯书籍 | 系统管理书籍 | 网络安全书籍 | 认证考试书籍
Firefox | IE | Maxthon | 迅雷 | 电驴 | BitComet | FlashGet | QQ | QQ空间 | Vista | 输入法 | Ghost | Word | Excel | wps | Powerpoint
asp | .net | php | jsp | Sql | c# | Ajax | xml | Dreamweaver | FrontPages | Javascript | css | photoshop | fireworks | Flash | Cad | Discuz!
当前位置 > 网站建设学院 > 网络编程 > C/C++教程
Tag:注入,存储过程,分页,安全,优化,xmlhttp,fso,jmail,application,session,防盗链,stream,无组件,组件,md5,乱码,缓存,加密,验证码,算法,cookies,ubb,正则表达式,水印,索引,日志,压缩,base64,url重写,上传,控件,Web.config,JDBC,函数,内存,PDF,迁移,结构,破解,编译,配置,进程,分词,IIS,Apache,Tomcat,phpmyadmin,Gzip,触发器,socket
网络编程:ASP教程,ASP.NET教程,PHP教程,JSP教程,C#教程,数据库,XML教程,Ajax,Java,Perl,Shell,VB教程,Delphi,C/C++教程,软件工程,J2EE/J2ME,移动开发
文章搜索服务
邮件订阅
输入你的邮件地址,
你将不会错过任何关于:
[ C/C++教程 ]的信息



本月文章推荐
.CPP程序--陀螺又称旋转矩阵.
.捕捉来自 Thread 的异常.
.网络应用三则.
.理论与实践相结合备战C上机考试.
.C++对象布局及多态探索之菱形结构.
.《c语言程序设计》第九章:预处理.
.C++面向对象编程入门:类(class).
.C语言初学者入门讲座 第十讲 函数.
.C++:谁动了我的指针.
.(02)数据结构题解-线性表.
.如何实现在一个Dbgrid中显示多数.
.C++头文件的结构.
.链表的c语言实现(十).
.新手入门:C/C++中数组和指针类型.
..
.C++基础:初始化类对象.
.开机密码改进and带解释版.
.探索C++的秘密之详解extern .
.RichEdit中实现查找功能.
.[收藏]ASP.Net生成静态HTML页 .

Bjarne:为什么不能为模板参数定义约束?

发表日期:2008-3-8 |



  可以的,而且方法非常简单和通用。

  看看这个:

template<class Container>
void draw_all(Container& c)
{
 for_each(c.begin(),c.end(),mem_fun(&Shape::draw));
}
  假如出现类型错误,可能是发生在相当复杂的for_each()调用时。例如,假如容器的元素类型是int,我们将得到一个和for_each()相关的含义模糊的错误(因为不能够对对一个int值调用Shape::draw的方法)。

  为了提前捕捉这个错误,我这样写:

template<class Container>
void draw_all(Container& c)
{
 Shape* p = c.front(); // accept only containers of Shape*s
 for_each(c.begin(),c.end(),mem_fun(&Shape::draw));
}
  对于现在的大多数编译器,中间变量p的初始化将会触发一个易于了解的错误。这个窍门在很多语言中都是通用的,而且在所有的标准创建中都必须这样做。在成品的代码中,我也许可以这样写:

template<class Container>

void draw_all(Container& c)
{
 typedef typename Container::value_type T;
 Can_copy<T,Shape*>(); // accept containers of only Shape*s
 for_each(c.begin(),c.end(),mem_fun(&Shape::draw));
}
  这样就很清楚了,我在建立一个断言(assertion)。Can_copy模板可以这样定义:

template<class T1, class T2> strUCt Can_copy {
 static void constraints(T1 a, T2 b) { T2 c = a; b = a; }
 Can_copy() { void(*p)(T1,T2) = constraints; }
};
  Can_copy(在运行时)检查T1是否可以被赋值给T2。Can_copy<T,Shape*>检查T是否是Shape*类型,或者是一个指向由Shape类公共继续而来的类的对象的指针,或者是被用户转换到Shape*类型的某个类型。注重这个定义被精简到了最小:

  一行命名要检查的约束,和要检查的类型

  一行列出指定的要检查的约束(constraints()函数)

  一行提供触发检查的方法(通过构造函数)

  注重这个定义有相当合理的性质:

  你可以表达一个约束,而不用声明或复制变量,因此约束的编写者可以用不着去设想变量如何被初始化,对象是否能够被复制,被销毁,以及诸如此类的事情。(当然,约束要检查这些属性的情况时例外。)

  使用现在的编译器,不需要为约束产生代码

  定义和使用约束,不需要使用宏

  当约束失败时,编译器会给出可接受的错误信息,包括“constraints”这个词(给用户一个线索),约束的名字,以及导致约束失败的具体错误(例如“无法用double*初始化Shape*”)。

  那么,在C++语言中,有没有类似于Can_copy——或者更好——的东西呢?在《C++语言的设计和演变》中,对于在C++中实现这种通用约束的困难进行了分析。从那以来,出现了很多方法,来让约束类变得更加轻易编写,同时仍然能触发良好的错误信息。例如,我信任我在Can_copy中使用的函数指针的方式,它源自Alex Stepanov和Jeremy Siek。我并不认为Can_copy()已经可以标准化了——它需要更多的使用。同样,在C++社区中,各种不同的约束方式被使用;到底是哪一种约束模板在广泛的使用中被证实是最有效的,还没有达成一致的意见。

  但是,这种方式非常普遍,比语言提供的专门用于约束检查的机制更加普遍。无论如何,当我们编写一个模板时,我们拥有了C++提供的最丰富的表达力量。看看这个:

template<class T, class B> struct Derived_from {
 static void constraints(T* p) { B* pb = p; }
 Derived_from() { void(*p)(T*) = constraints; }
};

template<class T1, class T2> struct Can_copy {
 static void constraints(T1 a, T2 b) { T2 c = a; b = a; }
 Can_copy() { void(*p)(T1,T2) = constraints; }
};

template<class T1, class T2 = T1> struct Can_compare {
 static void constraints(T1 a, T2 b) { a==b; a!=b; a<b; }
 Can_compare() { void(*p)(T1,T2) = constraints; }
};

template<class T1, class T2, class T3 = T1> struct Can_multiply {
 static void constraints(T1 a, T2 b, T3 c) { c = a*b; }
 Can_multiply() { void(*p)(T1,T2,T3) = constraints; }
};

struct B { };
struct D : B { };
struct DD : D { };
struct X { };

int main()
{
 Derived_from<D,B>();
 Derived_from<DD,B>();
 Derived_from<X,B>();
 Derived_from<int,B>();
 Derived_from<X,int>();

 Can_compare<int,float>();
 Can_compare<X,B>();
 Can_multiply<int,float>();
 Can_multiply<int,float,double>();
 Can_multiply<B,X>();
 Can_copy<D*,B*>();
 Can_copy<D,B*>();
 Can_copy<int,B*>();
}

// 典型的“元素必须继续自Mybase*”约束:

template<class T> class Container : Derived_from<T,Mybase> {

// ...

};
  事实上,Derived_from并不检查来源(derivation),而仅仅检查转换(conversion),不过这往往是一个更好的约束。为约束想一个好名字是很难的。

上一篇:C++箴言:确保公开继承模拟“is-a” 人气:173
下一篇:CB非可视组件的消息处理技巧 人气:288
浏览全部C/C++的内容 Dreamweaver插件下载 常用网页广告代码全集
  最新网站源码 最新软件下载
2008-7-25 WikyBlog v1.7.0.1 多国语言版
2008-7-25 乐彼网上开店系统(56770 Eshop)
2008-7-25 赛特网站管理系统sitecms v3.6.0
2008-7-25 Modoer多功能点评系统 v1.0.1 Bu
2008-7-25 Shangducms Teamsuit! v1.1.0 开
2008-7-25 幻影动漫网视频系统(Ppdong) v1.
2008-7-25 acteecompany企业网站建设系统 v
2008-7-25 恒浪整合管理系统 ims v4.1 ACCE
2008-7-25 艺术图库系统 v1.0 beta
2008-7-19 UltraEdit 简体中文增强版 14.10
2008-7-19 CentOS 5.2 i386 LiveCD
2008-7-19 Snapture多功能相机 v1.4
2008-7-19 iAcces中文输入法 v1.0Build016
2008-7-19 Cookbook烹饪秘籍 v2.5
2008-7-19 苹果专用DVD转换工具 v1.1.59汉化
2008-7-19 Modem修复软件ZiPhone修改版04.0
2008-7-19 AgileMessenger即时通讯工具美化
2008-7-19 Sketches画图软件 v0.7b6破解版


  发表评论
姓 名: 验证码:
内 容:
[ 汉字翻译拼音 ] [ 广告代码 ] [ 符号对照表 ] [ 进制转换 ] [ 经典小工具 ] [ 个税计算 ] [ 汉字简繁转换 ] [ 普通单位换算 ] [ 公制单位换算 ]
[ 生辰老黄历 ] [ 国内电话区号 ] [ 国家代码与域名缩写 ] [ 文字加密解密 ] [ 健康查询 ] [ 万年历 ] [ 手机号码查询 ] [ ip搜索 ] [ Google PR查询 ]
业务联系 | 广告刊登 | 频道合作 | 投稿荐稿 | 联系方式 | 加入收藏 | RSS订阅
Copyright © 2000-2008 www.knowsky.com All rights reserved | 网络实名:动态网站制作指南 | 沪ICP备05001343号