概述
GCC C特性
在C语言的编程过程中,适当使用gcc的C语言特性,可以提高我们的编程效率,代码也更整洁,更清晰。
比如语句表达式,可以把一块语句当作一个表达式,返回最后的语句,个人常使用于宏的定义,感觉比用do{ }while(0)表示一块语句更直观,而且定义的宏还能当作表达式赋值给变量。
还有定义局部变量时,加上__cleanup__属性,当局部变量退出作用域时会自动调用指定cleanup函数,这样一来,我们可以对内存指针添加释放内存函数,让它自动调用free释放内存,当一个函数内,存在多处返回时不需要多处调用free,对多线程加锁场景,巧妙使用__cleanup__特性,也可以让锁自动进行解锁操作,不用担心忘记解锁而导致死锁。对__cleanup__特性的使用,glib库中定义的g_auto和g_autoptr宏就应用得很不错,值得学习。
GCC C扩展特性有很多,以下是个人觉得非常值得学习和使用的特性。
1、语句表达式
//表达式为一块语句,最后的为返回值,类似逗号表达式
({ int y = foo (); int z;
if (y > 0) z = y;
else z = - y;
z; })
#define foreach(v)
for(int i = 0; ({printf("i[%d] < v[%d]n", i, v); i < v;}); ({printf("i = %dn", i); i++;}))
2、本地声明的标签
#define SEARCH(value, array, target)
do {
__label__ found;
typeof (target) _SEARCH_target = (target);
typeof (*(array)) *_SEARCH_array = (array);
int i, j;
int value;
for (i = 0; i < max; i++)
for (j = 0; j < max; j++)
if (_SEARCH_array[i][j] == _SEARCH_target)
{ (value) = i; goto found; }
(value) = -1;
found:;
} while (0)
//普通标签不能在宏里使用,本地声明的标签可以
//使用语句表达式返回value
#define SEARCH(array, target)
({
__label__ found;
typeof (target) _SEARCH_target = (target);
typeof (*(array)) *_SEARCH_array = (array);
int i, j;
int value;
for (i = 0; i < max; i++)
for (j = 0; j < max; j++)
if (_SEARCH_array[i][j] == _SEARCH_target)
{ value = i; goto found; }
value = -1;
found:
value;
})
3、标签作为值
//不建议使用
void *ptr;
/* … */
ptr = &&foo;
goto *ptr;
static void *array[] = { &&foo, &&bar, &&hack };
goto *array[i];
4、嵌套函数
//只能用于 C ,不能用于 C++
hack (int *array, int size)
{
void store (int index, int value)
{ array[index] = value; }
intermediate (store, size);
}
//嵌套函数与本地标签一起使用
bar (int *array, int offset, int size)
{
__label__ failure;
int access (int *array, int index)
{
if (index > size)
goto failure;
return array[index + offset];
}
int i;
/* … */
for (i = 0; i < size; i++)
/* … */ access (array, i) /* … */
/* … */
return 0;
/* Control comes here from access
if it detects an error. */
failure:
return -1;
}
5、非本地跳转
6、__builtin_函数调用
extern int myprintf (FILE *f, const char *format, ...);
extern inline __attribute__ ((__gnu_inline__)) int
myprintf (FILE *f, const char *format, ...)
{
int r = fprintf (f, "myprintf: ");
if (r < 0)
return r;
int s = fprintf (f, format, __builtin_va_arg_pack ()); //可变参数调用,不使用宏
if (s < 0)
return s;
return r + s;
}
7、typeof or typeof
//可以和语句表达式一起使用
#define max(a,b)
({ typeof (a) _a = (a);
typeof (b) _b = (b);
_a > _b ? _a : _b; })
typeof (*x) y;
typeof (typeof (char *)[4]) y; //char *y[4];
#define pointer(T) typeof(T *)
#define array(T, N) typeof(T [N])
//__auto_type 类似于 C++ 中的 auto
#define max(a,b)
({ __auto_type _a = (a); //优先使用__auto_type,而不是typeof
__auto_type _b = (b);
_a > _b ? _a : _b; })
8、条件表达式(省略操作数)
x ? : y
9、128位整型
__int128
unsigned __int128
19、空结构体
struct empty {
}; //空结构体,占用字节数为 0
20、可变长度数组
FILE *
concat_fopen (char *s1, char *s2, char *mode)
{
char str[strlen (s1) + strlen (s2) + 1]; //数组长度不是常量表达式,长度可变
strcpy (str, s1);
strcat (str, s2);
return fopen (str, mode);
}
void
foo (int n)
{
struct S { int x[n]; }; //结构体中包含可变长度数组
}
27、非常量初始值
foo (float f, float g)
{
float beat_freqs[2] = { f-g, f+g };
/* … */
}
30、case 范围
switch(i)
{
case low ... high:
/* ... */
case 1 ... 5:
/* ... */
}
33、通用函数属性
constructor 构造函数属性
destructor 析构函数属性
34、变量__cleanup__属性
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
void free_str(char **str)
{
printf("free %pn", *str);
free(*str);
}
//设置变量退出作用域自定调用函数释放内存
#define FREE_STR __attribute__ ((__cleanup__(free_str)))
#define Str FREE_STR char
int main()
{
Str *p = malloc(100);
strcpy(p, "test for auto free.");
printf("%sn", p);
}
最后
以上就是活力蜗牛为你收集整理的GCC C语言特性的全部内容,希望文章能够帮你解决GCC C语言特性所遇到的程序开发问题。
如果觉得靠谱客网站的内容还不错,欢迎将靠谱客网站推荐给程序员好友。
本图文内容来源于网友提供,作为学习参考使用,或来自网络收集整理,版权属于原作者所有。
发表评论 取消回复