C语言 数组相关乱七八糟 和 一些宏定义

322 阅读1分钟

本文已参与「新人创作礼」活动,一起开启掘金创作之路。

数组

char b[] = "1234";
int main()
{
    char a[] = "1234";
    std::cout << "Hello World!\n";
}

a是指针,指向局部变量“1234”,存在栈中
b指向全局变量“1234”。

   int a[3][3] = { 1,2,3,4,5,6,7,8,9 };
    int*  p= &a[0];

错误写法。

int a[3][3] = { 1,2,3,4,5,6,7,8,9 };
    int*  p=(int*) &a[0];

p指向了a的第一个元素。

int a[3][3] = { 1,2,3,4,5,6,7,8,9 };
    int*  p=(int*) (&a[0]+1);
    p++;

p先指向了4,后指向了5。

Int a[10]={1,2,3,4,5,6,7,8,9,10};
Int *p=&a[3];
Int b=p[3];
Printf(“%d”,b);
Return 0;

输出7。

char h[] = "hello";
    std::cout << sizeof(h)<<sizeof(*h);

6 1

Char **a[5][6];
cout<<sizeof(a);

64位系统下面输出240;表示字符二重指针数组。

数组等价表示

数组int a[3][4]中的a[2][1]用其他形式表示

int a1=*(*(a+2)+1);
int a2=*(a[2]+1);
int* p=&a[0][0];
int a3=p[9];

宏定义

除法取整

x对a向下取整数倍ALIGN_DOWN(x, a) eg(65,3)->63
x对a向上取整数倍ALIGN_UP(x, a) eg(65,3)->66

#define ALIGN_DOWN(x, a) ((x)-(x)%a)
#define ALIGN_UP(x, a) ((x)+(a)-(x)%a)

向幂的倍数取整

内核源码有涉及

#define ALIGN_2N_DOWN(x, a) ((x)&~(a - 1))
#define ALIGN_2N_UP(x, a) ((x + a - 1)&~(a - 1))

一年多少秒

#define SECONDE_PER_YEAR (60*60*24*365)UL

编译器会计算常量表达式,注意加括号,UL表示长整型,16位芯片可能溢出。

返回结构体首地址

内核中有如下写法,很妙!

/*
 * 选自 linux-2.6.7 内核源码
 * filename: linux-2.6.7/include/linux/stddef.h
 */
#define offsetof(TYPE, MEMBER) ((size_t) &((TYPE *)0)->MEMBER)

已知一个结构体类型和成员变量,返回结构体的首地址。

/*
 * 选自 linux-2.6.7 内核源码
 * filename: linux-2.6.7/include/linux/stddef.h
 */
#define container_of(ptr, type, member) ({			\
        const typeof( ((type *)0)->member ) *__mptr = (ptr);	\
        (type *)( (char *)__mptr - offsetof(type,member) );})

typeof 关键字是GNU C 标准中的,其作用在于其能够自动推导表达式类型,然后用({})表达式

宏判断大小,不用<>?

#include <stdio.h>
#define max(a,b) ((((int)(a)-(int)(b))&0x80000000)>>31)
int main()
{
   /* 我的第一个 C 程序 */
   printf("Hello, World! \n");
   int a=4,b=5;
	printf("%x",max(a,b));
   return 0;
}