前言
C語言學(xué)過了, 認(rèn)真思考過嗎?幾個栗子, 或許讓你得到一些裝逼技能。
example
int a[5] = {1, 2, 3, 4, 5};
printf("%d\n", 0[a]);
printf("%d\n", *a);
說明
a[0] = *a = *(a + 0) = *(0 + a) = 0[a]
然而這種寫法很少見, 但是可以用。比如: 1[a]用來訪問數(shù)值中下標(biāo)為1的元素
example
int a[5] = {1, 2, 3, 4, 5};
printf("%p\n", a);
printf("%p\n", &a);
int (*p)[5] = &a;
p++;
int *q = (int *)p;
q--;
printf("%d\n", *q);
輸出
0x7fff5fbff7f0
0x7fff5fbff7f0
5
說明
a是數(shù)組首個元素的地址和&a[0]等價。 雖然他們的值都和&a一樣, 但是意義是不一樣的, &a是整個數(shù)組的首字節(jié)地址。感覺沒區(qū)別, 但是上面的例子會讓你理解, 因為他們類型, 不一樣+1的時候步長也不一樣。a是int *類型的, 而&a是 int(*)[5]類型的。在尋找時候起點相同邁出的步子不同肯定結(jié)果就不同了。結(jié)果就是先跑到了數(shù)組的最后, 然后回來一個步長, 自然也就是5了。
example
char *str = "Hello, world";
printf("%s\n", str);
printf("%s\n", "Hello");
printf("%c\n", *"world");
printf("%c\n", "Hello"[0]);
printf("%c\n", 0["Hello"]);
printf("%c\n", 1["Hello"]);
printf("%c\n", *("Hello" + 1));
輸出
Hello, world
Hello
w
H
H
e
e
說明
這個初學(xué)者可能很難理解吧, 因為對字符串的存儲不理解。字符串在只讀數(shù)據(jù)段, 在棧段只保存著它的首個字符地址, 這樣就很方便的找到它了。 難道第一行中char *str = "Hello, world", 你不覺得str是一個指針嗎?(指針是什么?指針就是一個變量, 只是用來保存地址而已)那么"Hello, world"就是一個地址, 一個字符一個字節(jié)保存哪一個字節(jié)的地址呢?當(dāng)然保存首個元素的地址。還值得一提的是字符數(shù)組保存在棧段, 所以可以被修改, 但是只讀數(shù)據(jù)段, 是不能改的。如果改了就.....下圖

example
你認(rèn)識下面的p是什么嗎?
int p[10]int *p[10]int (*p[10])(int, int)int (*p)(int, int)int (*p[10])(int, int);
int p[10]
這是一個數(shù)組
int *p[10]
這是一個指針, 指向什么呢? 去掉*就是指向的東西, int [10], int [10]是一個什么呢? int a[10]這是一個數(shù)組。所以p是一個數(shù)組, 數(shù)組的每一個元素都是指針。
int (*p[10])(int, int)
有了上面的基礎(chǔ)那就很好解釋了。p是一個數(shù)組, 每一個元素都是一個指針, 每個指針指向一個函數(shù)(int a(int, int)), 函數(shù)是輸入為兩個int, 返回值是int類型。
int (*p)(int, int)
p是一個指針, 指向一個返回值是int的函數(shù), 這個函數(shù)就是int (int, int)
example
終端原始環(huán)境下運行
#include <stdio.h>
#include <unistd.h>
int main(){
while(1){
sleep(1);
printf("Hello, world");
}
return 0;
}
現(xiàn)象
什么都沒有出現(xiàn), 但是把sleep改為usleep就會打印。
原因是數(shù)據(jù)緩沖區(qū)滿了, 或者 \n 沖刷函數(shù)就會輸出到屏幕上。
如果你還不知道, 再學(xué)學(xué)吧。