关键词:
C语言常常让人觉得它所能表达的东西非常有限。它不具有类似第一级函数和模式匹配这样的高级功能。但是C非常简单,并且仍然有一些非常有用的语法技巧和功能,只是没有多少人知道罢了。
一、指定的初始化
很多人都知道像这样来静态地初始化数组:
int fibs[] = 1, 1, 2, 3, 5;
C99标准实际上支持一种更为直观简单的方式来初始化各种不同的集合类数据(如:结构体,联合体和数组)。
二、数组
我们可以指定数组的元素来进行初始化。这非常有用,特别是当我们需要根据一组#define来保持某种映射关系的同步更新时。来看看一组错误码的定义,如:
/* 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:
struct point p = .x = 3, .y = 4, .z = 5;
当我们不想将所有字段都初始化为0时,这种作法可以很容易的在编译时就生成结构体,而不需要专门调用一个初始化函数。
对联合体来说,我们可以使用相同的办法,只是我们只用初始化一个字段。
四、宏列表
C中的一个惯用方法,是说有一个已命名的实体列表,需要为它们中的每一个建立函数,将它们中的每一个初始化,并在不同的代码模块中扩展它们的名字。这在Mozilla的源码中经常用到,我就是在那时学到这个技巧的。例如,在我去年夏天工作的那个项目中,我们有一个针对每个命令进行标记的宏列表。其工 作方式如下:
#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
对FLAG_LIST(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
一步步的展示其过程是非常有启发性的,如果对它的使用还有不解,可以花一些时间在gcc –E上。
五、编译时断言
这其实是使用C语言的宏来实现的非常有“创意”的一个功能。有些时候,特别是在进行内核编程时,在编译时就能够进行条件检查的断言,而不是在运行时进行,这非常有用。不幸的是,C99标准还不支持任何编译时的断言。
但是,我们可以利用预处理来生成代码,这些代码只有在某些条件成立时才会通过编译(最好是那种不做实际功能的命令)。有各种各样不同的方式都可以做到这一点,通常都是建立一个大小为负的数组或结构体。最常用的方式如下:
/* 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))
如果(condition)计算结果为一个非零值(即C中的真值),即! (condition)为零值,那么代码将能顺利地编译,并生成一个大小为零的结构体。如果(condition)结果为0(在C真为假),那么在试图生成一个负大小的结构体时,就会产生编译错误。
它的使用非常简单,如果任何某假设条件能够静态地检查,那么它就可以在编译时断言。例如,在上面提到的标志列表中,标志集合的类型为uint32_t,所以,我们可以做以下断言:
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 )
因为位宽为负,所以可以确定,如果标志的数量超过了我们指派的空间,那么编译将会失败。
--- EOF ---
————————————————
不管你是转行也好,初学也罢,进阶也可,如果你想学编程~
【值得关注】我的 编程学习交流俱乐部【点击进入】
分享几个超级实用的c语言小技巧!
在嵌入式开发中,C语言的一些小技巧可以帮我们提高开发效率,事半功倍。C语言常常让人觉得它所能表达的东西非常有限。它不具有类似第一级函数和模式匹配这样的高级功能。但是C非常简单,并且仍然有一些非常... 查看详情
c语言进阶学习笔记七程序执行+调试技巧(实用技巧篇)(代码片段)
文章目录一、程序执行篇①预处理详解②宏定义③define标识符字符串④define宏名(参数表)字符串⑤宏和函数对比二、调试技巧篇①什么是bug?②调试是什么?有多重要?③debug和release的介绍④windows环境调试介绍⑥如何写出... 查看详情
c语言炫酷的文件操作(代码片段)
文件操作什么是文件程序文件数据文件(本文重点)文件名文件的打开和关闭文件指针文件函数相对路径与绝对路径相对路径绝对路径输入输出流二进制读写fwirtefread什么是文件磁盘上的文件是文件但是在程序设计中... 查看详情
vs环境——c语言实用调试技巧(代码片段)
目录什么是bug?调试的目的调试的基本步骤Debug和Release的介绍windows环境调试介绍 1.选择Debug调试版本 2.学会快捷键 3.调试窗口的使用 1.局部变量窗口 2.监视窗口 3.内存查看窗口 4.调用堆栈窗口什么是bug?bug是计算... 查看详情
c语言初阶笔记程序员必须要会的实用调试技巧(含库函数strcpy和strlen的模拟实现)!!(代码片段)
目录 为什么要学习调试?什么是bug?调试是什么,有多重要?调试的概念及重要性调试的步骤Debug和Release的介绍windows环境调试介绍1.调试环境的准备2.学会快捷键F5F9F10F11CTRL+F53.调试的时候查看程序当前信息查看临... 查看详情
c语言初阶笔记程序员必须要会的实用调试技巧(含库函数strcpy和strlen的模拟实现)!!(代码片段)
目录 为什么要学习调试?什么是bug?调试是什么,有多重要?调试的概念及重要性调试的步骤Debug和Release的介绍windows环境调试介绍1.调试环境的准备2.学会快捷键F5F9F10F11CTRL+F53.调试的时候查看程序当前信息查看临... 查看详情
分享几个实用的c语言小技巧!
C语言常常让人觉得它所能表达的东西非常有限。它不具有类似第一级函数和模式匹配这样的高级功能。但是C非常简单,并且仍然有一些非常有用的语法技巧和功能,只是没有多少人知道罢了。一、指定的初始化很多人都... 查看详情
c语言有这个就够了七.实用调试技巧
(一)什么是BUG历史上第一个bug导致程序运行错误的对象(二)调试是什么调试就是破案的过程,因为有人写代码是这样的:1.调试又称除错,是发现和减少计算机程序或电子仪器设备中程序错误的一个过程。2.调试的基本步骤... 查看详情
c语言高级实用技巧(代码片段)
字符串相关1、字符串拼接字符串拼接的时候,由于长度问题,可能导致内存溢出。以下这种方法比较安全。chartest_str[256]=0;snprintf(test_str+strlen(test_str),sizeof(test_str)-strlen(test_str),"%s",...);内存相关1、free直接free空... 查看详情
最炫酷的h5怎么实现的
一般炫酷的H5都是专业的开发或工具制作出来的,依赖模板只能得到模块化的H5作品。推荐使用意派Epub360H5工具,专业的组件,灵活的触发行为,专为有想法的设计师打造的交互制作利器,这里简单介绍四种炫酷的H5刷屏技巧。刷... 查看详情
c语言学习技巧
学习C语言的方法 首先大家要明白学习无捷径!但是学习是有方法的。 那么学习C语言有什么好的方法呢?根据我自己多年的总结,以及很多编程前辈的经验,主要有以下几个方面: 分清主次... 查看详情
apicloud开发者进阶之路|超级实用技巧
...的!本周《30天,App开发从0到1》将要给大家分享第十章超级实用技巧,我们精选了其中的一个章节--地图模块的实用扩展:在地图上添加自定义按钮,一起来学习吧。主要内容本章将向读者介绍一些开发中常用的功能和技巧,这... 查看详情
实用c语言管道小程序
编写使用管道的程序】我们在文件夹下,c语言做的每个程序都有一个独立的功能,我们可以将多个小程序使用管道连接到一起。我们现在写一个程序aver.c,求任意个数的平均值:#include<stdio.h>intmain(){ints,n;scanf("%d,%d",&s,&... 查看详情
unity之shadergraph实现超级炫酷的溶解效果入门级教程(代码片段)
Unity之ShaderGraph实现超级炫酷的溶解效果入门级教程前言一,效果展示二,原理介绍三,实现效果四,使用示例五,完整图示前言若你的工程还没有进行基础配置,请查看我的博文Unity之ShaderGraph入门使用详... 查看详情
超级实用的vue小技巧,用了之后直呼好棒~(代码片段)
实用的vue小技巧,来源于真实企业实战——记得收藏~1.原型注入2.对象冻结3.img加载失败4.递归组件5.解决繁乱的template——render函数6.延迟更新——Vue.nextTick()7.自定义指令1.原型注入全局属性和方法可以这么定义方便调用和获... 查看详情
超级实用的vue小技巧,用了之后直呼好棒~(代码片段)
实用的vue小技巧,来源于真实企业实战——记得收藏~1.原型注入2.对象冻结3.img加载失败4.递归组件5.解决繁乱的template——render函数6.延迟更新——Vue.nextTick()7.自定义指令1.原型注入全局属性和方法可以这么定义方便调用和获... 查看详情
几点实用的c语言技巧
...注+星标公众号,不错过精彩内容来源 |百问科技C语言常常让人觉得它所能表达的东西非常有限。它不具有类似第一级函数和模式匹配这样的高级功能。但是C非常简单,并且仍然有一些非常有用的语法技巧和功能,... 查看详情
炫酷的花式滑块滑动无缝切换特效(代码片段)
💂个人网站:【海拥】【小霸王游戏机】【大转盘】🤟风趣幽默的前端学习课程:👉28个案例趣学前端💅想寻找共同学习交流、摸鱼划水的小伙伴,请点击【摸鱼学习群】【学习文档】💬免费且实用... 查看详情