设为首页 加入收藏

TOP

C++开发指导之使用编译期的契约:约束,在编译期发现程序的错误(一)
2014-11-24 11:02:52 】 浏览:1457
Tags:开发 指导 使用 编译 契约 约束 发现 程序 错误
利用 C/C++ 编译器对源程序进行编译的时候会检查语法错误和计算常量等特性,可以给我们的 C/C++源代码添加一些编译期的契约,要求源代码按一定的规则使用,这样的好处是可以减少很多使用错误,减少软件的 bug,因为这些 bug 在代码编译的时候就发现了,不用等测试或现场使用时才发现。
下面举一些编译期的契约,约束例子。
1. must_have_base()必须继承自
template
structmust_have_base
{
~must_have_base()
{
void (*p)(D*,B*) = constraints;
}
private:
static void constraints(D*pd, B* pb)
{
pb = pd;
}
};
模板原理:子类指针可以直接赋值给基类指针。
使用场景举例:
class Base
{
public:
Base();
};
template
class C : must_have_base
{ /* ... */ };
这个例子要求你使用模板类 C时,模板参数必须继承自类 Base。
2. must_be_subscriptable()必须可以按下标方式访问
template
structmust_be_subscriptable
{
static void constraints(Tconst &T_is_not_subscriptable)
{
sizeof(T_is_not_subscriptable[0]);
}
};
模板原理:就是按下标方式来访问进行检验。
使用场景举例:
template
void fun(T& t)
{
must_be_subscriptableconstraint;
// t[0]
// ...
}
要求输入参数要求可以用下标来访问
3. must_be_subscriptable_as_decayable_pointer()必须可以按下标方式访问,并且可以退化为原生指针
template
structmust_be_subscriptable_as_decayable_pointer
{
static void constraints(Tconst &T_is_not_decay_subscriptable)
{
sizeof(0[T_is_not_decay_subscriptable]);
}
};
模板原理:原生指针的特性,通过下标访问可以反过来,就是即可以这样 pointer[offset] 使用指针,也可以这样 offset[pointer]使用指针。
4. must_be_pod()必须为 POD类型
POD类型:POD意思是“plain-old-data”(C++-98:1.8;5),它是C++中的一个重要概念。POD类型必修满足以下条件:
将组成它的一个对象的各字节拷贝到一个字节数组中,然后再将它们重新拷贝回原先的对象所占的存储区中;此时对象应该仍具有它原来的值。
POD类型定义:标量类型、POD结构类型、POD联合类型,这些类型的数组,以及这些类型以 const/volatile修饰的版本。
POD结构:一个聚合体类,其任何非静态成员的类型都不能是如下任意一种:指向成员的指针、非 POD联合,以及以上这些类型的数组或引用,同时该聚合体类不允许包含用户自定义的拷贝赋值操作符和用户自定义的析够函数。
POD类型的重要作用:POD类型允许 C++与 C交互!
template
structmust_be_pod
{
static void constraints()
{
union{ T T_is_not_POD_type; };
}
};
模板原理:利用 POD类型可以放在 union中实现这个约束。
使用场景举例:
template
union must_be_pod
{
T t;
};
template
void SafeZeroMemory(T* p, size_t size)
{
must_be_pod();
memset(p, 0,size);
}
class A
{
public:
~A(){}
void Reset() { _value = 0; }
private:
int _value;
};
class B
{
public:
void Reset() { _value = 0; }
private:
int _value;
};
class C
{
private:
int _value;
};
void TestImpl()
{
A a;
B b;
C c;
int i = 0;
int* p = 0;
SafeZeroMemory(&a,sizeof(a)); // 编译失败
SafeZeroMemory(&b,sizeof(b)); // 编译通过
SafeZeroMemory(&c,sizeof(c)); // 编译通过
SafeZeroMemory(&i,sizeof(i)); // 编译通过
SafeZeroMemory(&p,sizeof(p)); // 编译通过
};
上面例子中,类 A有析构函数,因此它不是 POD结构。
一个类或结构如果有构造函数、析构函数、复制构造函数、赋值函数、或虚函数、它从非 POD的类或结构继承或者从多个类继承,都不是 POD结构。
5. must_be_same_size()大小必须相同
template
structsize_of
{
enum{ value = sizeof(T) };
};
template<>
structsize_of
{
enum{ value = 0 };
};
template
structmust_be_same_size
{
public:
~must_be_same_size()
{
void (*pfn)(void) =constraints;
(void)(pfn);
}
private:
static void constraints()
{
struct must_be_same_size_
{
int T1_must_be_same_size_as_T2 : size_of::value ==size_of::value;
};
}
};
模板原理:已命名位域不能有零宽度。
使用场景举例:
template
void ObjCopy(T1& l, const T2&r)
{
must_be_same_size();
must_be_pod();
must_be_pod();
memcpy(&l, &r,sizeof(l));
}www.2cto.com
structA { int fo
首页 上一页 1 2 下一页 尾页 1/2/2
】【打印繁体】【投稿】【收藏】 【推荐】【举报】【评论】 【关闭】 【返回顶部
上一篇POJ 2312 Battle City 优先多列+b.. 下一篇 poj 3581 Sequence

最新文章

热门文章

Hot 文章

Python

C 语言

C++基础

大数据基础

linux编程基础

C/C++面试题目