C语言中一些不被熟知的特性

designated initializers(c99)

C99允许你对结构体中指定的变量初始化,如

struct Foo {

int x;

int y;

int z;

};

Foo foo = {.z = 3, .x = 5};

这其中x会默认初始化为0

指定初始化也可适用于数组,如下面三个等价的数组定义

int a[5] = {[1] = 2, [4] = 5};

int a[] = {[1] = 2, [4] = 5};

int a[5] = {0, 2, 0, 0, 5};

restricted pointers(c99)

限定词restricted用于限定一个指针(如名,告诉编译器该指针的内存访问在任何情况下都只能通过该指针进行,其余指向无效.如

int f(const int* restrict x, int* y) {

(*y)++;

int z = *x;

(*y)--;

return z;

}

会被编译器优化为

int f(const int* restrict x, int* y) {

return *x;

}

静态和限制性数组参数(c99)

void f(int a[static 10]) {

/* ... */

}

传递给f()的指针如果为null或者小于10编译器会warning!

又如void f(int a[const]) {

/* ... */

}

中a不可被修改。

泛型表达式(c11)

使用X Maroc(http://en.wikipedia.org/wiki/X_Macro),编译器会在compile time选择适当的形式

#define cbrt(X) _Generic((X), \

long double: cbrtl, \

default: cbrt, \

float: cbrtf \

)(X)

默认形式为cbrt(),如果实参为float就会被替换为cbrtf(),etc.

Switch条件判断case(c11)

case语句可以出现在if-else结构和循环结构中

switch (a)

{

case 1:;

// ...

if (b==2)

{

case 2:;

// ...

}

else case 3:

{

// ...

for (b=0;b<10;b++)

{

case 5:;

// ...

}

}

break;

case 4:

// ...

break;

}

不使用”+”号实现加法操作

#include <stdio.h>

int add(int a,int b){

if(a!=0&&b!=0)

return printf("%*c%*c",a,‘\r‘,b,‘\r‘);

else return a!=0?a:b;

}

int main(){

int A = 0, B = 0;

printf("Enter the two numbers to add\n");

scanf("%d %d",&A,&B);

printf("Required sum is %d",add(A,B));

return 0;

}

用printf()返回值实现,也可以使用bitwise实现

void返回值的函数体中使用return

static void foo (void) { }

static void bar (void) {

return foo(); // Note this return statement.

}

int main (void) {

bar();

return 0;

}

#include in strange places

#include <stdio.h>

void main()

{

printf

#include "fragment.c"

}

("dayum!\n");#text in Fragment.c

这我也无力吐槽..

有范围的switch-case(gcc extension)

switch(c) {

case ‘A‘ ... ‘Z‘:

Doa();

break;

case 1 ... 5 :

Dob();

}

C属于a-z字符时执行Doa();属于1-5时执行Dob();

编译器通过的最短C语言程序

main;

作者没有提到编译环境,虽然我的编译器会报错..

四个等价

a [ i ] == * (a + i) == * (i + a) == i [ a ];

古董级键盘福利

下表左右都是等价的

于是就出现这样的(:乱码code

!E() ??!??! C();

default替换(C++乱入)

#include<iostream>

int main(){

int x = 2;

switch(x){

case 1:

cout<<"1";

break;

abc1:

cout<<"2";

}

}

c++ switch-case结构中的default可以被任何标签替换,如上面的abc1

奇怪的等价

# 1 "1.c"

# 1 "<command-line>"

# 1 "/usr/include/stdc-predef.h" 1 3 4

# 1 "<command-line>" 2

# 1 "1.c"

int main(){ return 0;}

等价于

#ifdef SOME_MACRO

void foo (){}

#endif

int main(){ return 0;}

且都能编译通过(其实就是预处理器- -)

时间: 2024-10-12 12:34:26

C语言中一些不被熟知的特性的相关文章

C语言中关于时间的函数

C语言中关于时间的函数 本文从介绍基础概念入手,探讨了在C/C++中对日期和时间操作所用到的数据结构和函数,南平私家侦探(http://user.qzone.qq.com/778607337)并对计时.时间的获取.时间的计算和显示格式等方面进行了阐述.本文还通过大量的实例向你展示了time.h头文件中声明的各种函数和数据结构的详细使用方法. 关键字:UTC(世界标准时间),Calendar Time(日历时间),epoch(时间点),clock tick(时钟计时单元) 注:linux系统时间如

从头开始-04.C语言中流程控制

分支结构: if语句:当条表达式满足的时候就执行if后面大括号中语句 三种格式: if,if else , if else if else 特点:1.只有一个代码块会被执行 2.若有else那么必有一个代码会被执行 3.一个if语句中只能以if开头,最多只能有一个else,可以有任意个else if switch(表达式){ case 常量: ... break; case 常量1: ... break; default: ... break; } : 使用场景:当一个变量只有几个固定的取值的时

sqlite学习笔记8:C语言中使用sqlite之创建表

前面已经说了如何打开和关闭数据库,这次要说得是如何执行SQL语句,来创建一张表. 要用的的函数: sqlite3_exec(sqlite3* db, const char *sql, sqlite_callback callback, void *data, char **errmsg) 参数: db:已经打开的数据库实例 sql:SQL语句,是一个字符串 callback:是一个回调函数 data:做为回调函数的第一个参数 errmsg:用于带回错误信息 该回调函数有两种返回值类型. 1.返回

C语言中内存的管理

一  Handler作用和概念 包含线程队列和消息队列,实现异步的消息处理机制,跟web开发的ajax有异曲同工之妙. 1.运行在某个线程上,共享线程的消息队列: 2.接收消息.调度消息,派发消息和处理消息: 3.实现消息的异步处理: Handler能够让你发送和处理消息,以及Runnable对象:每个Handler对象对应一个Thread和Thread的消息队列.当你创建一个Handler时,它就和Thread的消息队列绑定在一起,然后就可以传递消息和runnable对象到消息队列中,执行消息

C语言中的函数、数组与指针

1.函数:当程序很小的时候,我们可以使用一个main函数就能搞定,但当程序变大的时候,就超出了人的大脑承受范围,逻辑不清了,这时候就需要把一个大程序分成许多小的模块来组织,于是就出现了函数概念:  函数是C语言代码的基本组成部分,它是一个小的模块,整个程序由很多个功能独立的模块(函数)组成.这就是程序设计的基本分化方法: (1) 写一个函数的关键: 函数定义:函数的定义是这个函数的实现,函数定义中包含了函数体,函数体中的代码段决定了这个函数的功能: 函数声明:函数声明也称函数原型声明,函数的原型

c语言中的副作用!!千万小心!

今天刚看完书上的副作用,博主觉得呢,副作用其实就在改变变量的值,也就是一个赋值操作!不过刚刚在知道上还是犯了错!!尴尬啊!! 大家都知道,c语言中的赋值操作符是自右向左结合的!! 下面有一个关于赋值中的副作用的例子: #include <stdio.h> int main() { int a = 3; a += a -= a*a; printf("%d\n",a); return 0; } 想到答案了吗?有没有人觉得答案是-3的呢? 小伙伴犯了和博主一样的错咯,正解应该是这

想精度高,可以考虑用c语言中的函数gettimeofday

大家好: 在 win32 + bcb 时, 有个 GetTickCount() 返回第统启动到现在的 tick, 单位 ms.请问在 Linux + qt5 怎样实现呢? 如果用 QDateTime , 精度只能到秒,而且运行过程中,如果用户改充了系统时间,就会出错了. 用QElapsedTimer. 想精度高,可以考虑用c语言中的函数gettimeofday,微秒级精度 #include <sys/time.h>int gettimeofday(struct timeval*tv, stru

Go语言中使用MySql数据库

Go语言中使用MySql数据库 1.MySQL驱动 Go中支持MySQL的驱动目前比较多,有如下几种,有些是支持database/sql标准,而有些是采用了自己的实现接口,常用的有如下几种: https://github.com/Go-SQL-Driver/MySQL 支持database/sql,全部采用go写. https://github.com/ziutek/mymysql 支持database/sql,也支持自定义的接口,全部采用go写. https://github.com/Phil

Go语言中使用SQLite数据库

Go语言中使用SQLite数据库 1.驱动 Go支持sqlite的驱动也比较多,但是好多都是不支持database/sql接口的 https://github.com/mattn/go-sqlite3 支持database/sql接口,基于cgo(关于cgo的知识请参看官方文档或者本书后面的章节)写的 https://github.com/feyeleanor/gosqlite3 不支持database/sql接口,基于cgo写的 https://github.com/phf/go-sqlite