前言
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]
類型的误窖。在尋找時候起點(diǎn)相同邁出的步子不同肯定結(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)境下運(yù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é)吧窃款。