嵌入式技术
C语言常常让人觉得它所能表达的东西非常有限。它不具有类似第一级函数和模式匹配这样的高级功能。但是C非常简单,并且仍然有一些非常有用的语法技巧和功能,只是没有多少人知道罢了。
一、指定的初始化
很多人都知道像这样来静态地初始化数组:
int fibs[] = {1, 1, 2, 3, 5};
/* Entries may not correspond to actual numbers. Some entries omitted. */ #define EINVAL 1 #define ENOMEM 2 #define EFAULT 3 /* ... */ #define E2BIG 7 #define EBUSY 8 /* ... */ #define ECHILD 12 /* ... */现在,假设我们想为每个错误码提供一个错误描述的字符串。为了确保数组保持了最新的定义,无论头文件做了任何修改或增补,我们都可以用这个数组指定的语法。
char *err_strings[] = { [0] = "Success", [EINVAL] = "Invalid argument", [ENOMEM] = "Not enough memory", [EFAULT] = "Bad address", /* ... */ [E2BIG ] = "Argument list too long", [EBUSY ] = "Device or resource busy", /* ... */ [ECHILD] = "No child processes" /* ... */ };这样就可以静态分配足够的空间,且保证最大的索引是合法的,同时将特殊的索引初始化为指定的值,并将剩下的索引初始化为0。
struct point { int x; int y; int z; }
struct point p = {.x = 3, .y = 4, .z = 5};
#define FLAG_LIST(_) _(InWorklist) _(EmittedAtUses) _(LoopInvariant) _(Commutative) _(Movable) _(Lowered) _(Guard)它定义了一个FLAG_LIST宏,这个宏有一个参数称之为 _ ,这个参数本身是一个宏,它能够调用列表中的每个参数。举一个实际使用的例子可能更能直观地说明问题。假设我们定义了一个宏DEFINE_FLAG,比如:
#define DEFINE_FLAG(flag) flag, enum Flag { None = 0, FLAG_LIST(DEFINE_FLAG) Total }; #undef DEFINE_FLAG
enum Flag { None = 0, DEFINE_FLAG(InWorklist) DEFINE_FLAG(EmittedAtUses) DEFINE_FLAG(LoopInvariant) DEFINE_FLAG(Commutative) DEFINE_FLAG(Movable) DEFINE_FLAG(Lowered) DEFINE_FLAG(Guard) Total };接着,对每个参数都扩展DEFINE_FLAG宏,这样我们就得到了enum如下:
enum Flag { None = 0, InWorklist, EmittedAtUses, LoopInvariant, Commutative, Movable, Lowered, Guard, Total };然后,我们可能要定义一些访问函数,这样才能更好的使用flag列表:
#define FLAG_ACCESSOR(flag) bool is##flag() const { return hasFlags(1 << flag); } void set##flag() { JS_ASSERT(!hasFlags(1 << flag)); setFlags(1 << flag); } void setNot##flag() { JS_ASSERT(hasFlags(1 << flag)); removeFlags(1 << flag); } FLAG_LIST(FLAG_ACCESSOR) #undef FLAG_ACCESSOR
/* Force a compilation error if condition is false, but also produce a result * (of value 0 and type size_t), so it can be used e.g. in a structure * initializer (or wherever else comma expressions aren't permitted). */ /* Linux calls these BUILD_BUG_ON_ZERO/_NULL, which is rather misleading. */ #define STATIC_ZERO_ASSERT(condition) (sizeof(struct { int:-!(condition); }) ) #define STATIC_NULL_ASSERT(condition) ((void *)STATIC_ZERO_ASSERT(condition) ) /* Force a compilation error if condition is false */ #define STATIC_ASSERT(condition) ((void)STATIC_ZERO_ASSERT(condition))
STATIC_ASSERT(Total <= 32)它扩展为:
(void)sizeof(struct { int:-!(Total <= 32) })现在,假设Total<=32。那么-!(Total <= 32)等于0,所以这行代码相当于:
(void)sizeof(struct { int: 0 })这是一个合法的C代码。现在假设标志不止32个,那么-!(Total <= 32)等于-1,所以这时代码就相当于:
(void)sizeof(struct { int: -1 } )因为位宽为负,所以可以确定,如果标志的数量超过了我们指派的空间,那么编译将会失败。
审核编辑:汤梓红
全部0条评论
快来发表一下你的评论吧 !