您好,登錄后才能下訂單哦!
梓益C語言學習總結
一、靜態變量與普通變量的區別
1.1、普通的全局變量
概念:
在函數外部定義的變量
作用范圍:
全局變量的作用范圍,是程序的所有地方。
只不過用之前需要聲明。聲明方法 extern int num; 聲明的時候,不要賦值。
生命周期:
程序運行的整個過程,一直存在,直到程序結束。
注意:定義普通的全局變量的時候,如果不賦初值,它的值默認為0
1.2、靜態全局變量 static
概念:
定義全局變量的時候,前面用static 修飾。
作用范圍:
static 限定了靜態全局變量的,作用范圍
只能在它定義的.c(源文件)中有效
生命周期:
在程序的整個運行過程中,一直存在。
注意:定義靜態全局變量的時候,如果不賦初值,它的值默認為0
2.1、普通的局部變量
概念:
在函數內部定義的,或者復合語句中定義的變量
作用范圍:
在函數中定義的變量,在函數中有效
在復合語句中定義的,在復合語句中有效。
生命周期:
在函數調用之前,局部變量不占用空間,調用函數的時候,才為局部變量開辟空間,函數結束了,局部變量就釋放了。
在復合語句中定義的亦如此。
注意:定義局部變量的時候,如果不賦初值,它的值為隨機值
2.2、靜態的局部變量
概念:
定義局部變量的時候,前面加static 修飾
作用范圍:
在它定義的函數或復合語句中有效。
生命周期:
第一次調用函數的時候,開辟空間賦值,函數結束后,不釋放,以后再調用函數的時候,就不再為其開辟空間,也不賦初值,
用的是以前的那個變量。
注意:
1:定義普通局部變量,如果不賦初值,它的值是隨機的。
2:定義靜態局部變量,如果不賦初值,它的值是0
3:普通全局變量和靜態全局變量如果不賦初值,它的值為0
二、頭文件路徑
#include<> 在系統路徑下查找頭文件
#include“ ” 在當前目錄下查找頭文件
三、a++ 先調用a再把它的值加一,++a 先把a的值加一再調用a
四、宏定義 #define
1、宏定義后不加分號,宏定義的值不能改變
2、宏是在預編譯的時候進行替換,無變量,不占內存
3、宏定義可以防止不知含義的數字出現,方便修改
4、可以定義任何類型
不帶參宏
#define PI 3.14
在預編譯的時候如果代碼中出現了PI 就用 3.14去替換。
宏的好處:只要修改宏定義,其他地方在預編譯的時候就會重新替換。
終止宏:#undef PI //終止PI的作用
帶參宏(宏函數)
#define S(a,b) (a)*(b)
注意帶參宏的形參 a和b沒有類型名,
S(2,4) 將來在預處理的時候替換成 實參替代字符串的形參,其他字符保留,(2) * (4)
五、條件編譯(選擇性編譯)
#ifdef AAA
代碼段一
#else
代碼段二
#endif
如果在當前.c ifdef 上邊定義過AAA ,就編譯代碼段一,否則編譯代碼段二
常用在.h文件頭部,如:文件fun.h開頭寫上
#ifndef FUN_H
# define FUN_H
.......代碼段.......
# endif
防止fun.h文件被重復引用
六、動態庫和靜態庫
一:動態編譯
動態編譯使用的是動態庫文件進行編譯
gcc hello.c -o hello
咱們使用的是動態編譯方法
二:靜態編譯
靜態編譯使用的靜態庫文件進行編譯
gcc -static hello.c -o hello
三:靜態編譯和動態編譯區別
1:使用的庫文件的格式不一樣,動態編譯使用動態庫,靜態編譯使用靜態庫
注意:
1:靜態編譯要把靜態庫文件打包編譯到可執行程序中。
2:動態編譯不會把動態庫文件打包編譯到可執行程序中,它只是編譯鏈接關系
制作靜態庫:.c文件mylib.c
gcc -c mylib.c -o mylib.o生成mylib.o
ar rc libtestlib.a mylib.o生成靜態庫libtestlib.a
注意:靜態庫起名的時候必須以lib開頭以.a結尾
編譯程序:gcc -static mytest.c libtestlib.a -o mytest
制作動態鏈接庫:gcc -shared mylib.c -o libtestlib.so //使用gcc編譯、制作動態鏈接庫
動態鏈接庫的使用:gcc mytest.c libtestlib.so -o mytest
七、
出現段錯誤,一定是指針用錯了
定義字符數組時,記得初始化為空 char str[20]=" ",將空間定義大點,后期可以進行優化
八、創建鏈表的兩種方法
頭插法-有返回值
(STU *)link_creat_head(STU *p_head,STU *p_new)
{
if(p_head==NULL) //當第一次加入鏈表為空時,head執行p_new
{
p_head=p_new;
p_new->next=NULL;
}
else //第二次及以后加入鏈表
{
p_new->next=p_head; //將新申請的節點加入鏈表
p_head=p_new;
}
return p_head;
}
尾插法-無返回值
void link_creat_head(STU **p_head,STU *p_new)
{
STU *p_mov=*p_head;
if(*p_head==NULL) //當第一次加入鏈表為空時,head執行p_new
{
*p_head=p_new;
p_new->next=NULL;
}
else //第二次及以后加入鏈表
{
while(p_mov->next!=NULL)
{
p_mov=p_mov->next; //找到原有鏈表的最后一個節點
}
p_mov->next=p_new; //將新申請的節點加入鏈表
p_new->next=NULL;
}
}
鏈表逆序:每個節點的next指針的方向調頭反向
STU *reverse(STU *head)
{
STU *pf,*pb,*r;
pf=head;
pb=pf->next;
while(pb!=NULL)
{
r=pb->next;
pb->next=pf;
pf=pb;
pb=r;
}
head->next=NULL;
head=pf;
return head;
}
九、fgets
stdout、stdin 標準輸入輸出緩沖區指針
用fgets代替scanf進行輸入讀取,避免緩沖區溢出
buf1[20]=" ";
fgets(buf1, sizeof(buf1)-1, stdin);
buf1(strlen(buf1)-1)='\0';
printf("%s\n", buf1);
如果和scanf一起用,要先讀出緩沖區中的\n, char ch=0; scanf("%c", &ch);
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。