原碼:將最高位作為符號位(0代表正,1代表負),其余各位代表數值本身的絕對值。
反碼:一個數如果值為正,那么反碼和原碼相同。如果值為負,那么符號位為1不變,其它各位與原碼相反。
補碼:正數的原碼、反碼、補碼都相同。負數,最高位符號位為1不變,其余各位取反,最后+1。計算機存放數據都是按照補碼的形式放,然后進行計算的。
1.數組
定義:內存連續,并且是用一種數據類型的變量
類型 數組名[];
在內存當中是連續的空間地址
數組初始化:類型 數組名[N]={0};表示N個元素都初始化為0。
二維數組定義:int array[3][4];
二維數組初始化:int array[3][4]={{1,2,3,4},{5,6,7,8},{9,10,11,12}};
字符串數組定義:char array[100];
對C語言字符串就是最后一個元素以'\0'結尾的數組。
字符串初數組始化:char array[]="hello";
scanf函數不安全提醒:主要是scanf函數沒有解決緩沖區溢出的問題和不能成功獲取完整的含空格的字符串,我們可以添加一個宏定義#define _CRT_SECURE_NO_WARNINGS或#pragma warning(disable:4996)
我們也可以用gets函數來獲取一個含空格的字符串。
我們也可以用fgets函數來獲取一個含空格的字符串并解決緩沖區溢出的問題,用法fgets(字符串數組名,sizeof(字符串數組名),stdin);stdin表示標準輸入。fgets函數最后會自動補上'\n'
puts函數:打印字符串,與printf不同,puts函數會在最后自動添加一個‘\n‘。
fputs函數用法:fputs(字符串數組名,stdout);與puts函數區別是它不會在最后加上\n
strlen函數求字符串長度。不會將最后的\0也計算在內。
strcat(a,b);把b追加到a的后面。a必須要有足夠的空間。
當a的緩沖區空間較小,b很大的時候,strcat也面臨安全問題,所以要用strncat函數,用法:strncat(a,b,sizeof(a)-strlen(a)-1);這樣就不會有溢出了。
memset(a,0,sizeof(a));將a內存初始化為0;
strcmp(a,"str");表示如果兩個參數所指的字符串內容相同,函數返回0。
strncmp(a,"exit",4);4表示只比較前4位。
strcpy(b,a);表示將a拷貝到b。
strncpy:有限拷貝。
sprintf(a,"ni shi shui%d",3)表示將ni shi shui3輸入到字符串數組a中。
strchr(a,'c'):表示在字符數組a查找字符'c',返回一個指針且指向字符‘c’的位置
strstr(a,"345"):表示在字符數組a查找字符串“345”。返回一個指針且指向字符串“345”的首地址。
strtok(a,"¥"):表示以¥為結束符,返回指向前面的字符串或字符的首地址或地址的指針。
atoi(a)函數表示將字符串a轉化為整數。atof、atol都是一樣的。
3.函數
在定義函數時,函數名后面括弧中的變量名稱為形式參數,簡稱形參。在調用函數時,函數后面括號中的變量或表達式稱為實際參數,簡稱實參。
形參在未出現函數調用時,它們并不占用內存單元,只有發生函數調用的時候形參才被分配內存,函數調用完成后,形參所占的內存被釋放。
實參可以是變量、常量、表達式。
在定義函數時,一定要指定形參和實參的數據類型。
形參和實參的數據類型一定要可兼容。
在C語言中,實參和形參的數據傳遞是單向值傳遞,只由實參傳遞給形參,而不能由形參傳遞給實參。
return語句后面的語句不會被執行。
exit(0)函數表示退出程序。
頭文件的使用:函數原型放在頭文件里,就不用在使用函數的時候去聲明該函數了,直接將放置函數原型的頭文件預編譯即可。使用頭文件需要#include,定義頭文件需要#ifndef與#endif。
函數的封裝:如果放在.c文件就是代碼級的封裝,如果放在.dll就是二進制的封裝。
動態庫.dll的特點:程序運行會自動加載dll文件,且dll文件可以被多個程序調用。
靜態庫的特點:程序直接編譯靜態庫,編譯完成后就不需要靜態庫的文件了,因為已經包含在程序內部了,這樣一來程序所占的內存會變大,且如果靜態庫的功能發生變化,就需要重新編譯程序了。
linux下的動態庫so文件封裝示例:
新建一個a.c文件:
輸入gcc -c a.c -fPIC -o a.o表示編譯a.c文件,與位置無關,輸出a.o文件
輸入gcc -shared -o liba.so a.o表示編譯一個有關a.o二進制文件的共享庫
-fPIC的意思是生成一個與位置無關的代碼。主要是因為每個程序都有一個入口地址(系統根據實際情況動態分配的),程序內部調用的函數距離該入口地址有一段距離叫做偏移地址,但是動態庫編譯的時候不知道偏移地址是多少,主要取決與哪個程序調用它,所以要指定-fPIC生成一個與位置無關的代碼。
gcc鏈接的時候要加-shared選項,意思是生成一個so共享庫。
對于linux或Unix,一個so文件,文件擴展名必須是so,文件名的前三個字母必須是lib。
要用到這個動態庫文件so,需要新建一個頭文件來聲明動態庫文件so里面的函數。輸入vi a.h
#ifndef __AH表示如果沒有定義一個__AH,就定義一個__AH,#define __AH表示定義一個__AH。
輸入vi main.c表示編寫一個可執行文件,來調用庫文件.so。
輸入gcc -c main.c -o main.o表示只編譯不鏈接,指定輸出文件名稱為main.o
輸入gcc -o main.out main.o -L. -la表示鏈接main.o文件,指定輸出main.out可執行文件,并且在當前目錄下尋找-la文件(代表liba,so文件)。
linux不再當前目錄下尋找可執行程序,同時也不在當前目錄下找so庫文件,解決辦法——修改用戶配置文件:第一:cd(返回主目錄);第二vi .bash_profile;第三:添加export LD_LIBRARY_PATH=$LD_LINRARY_PATH:.;第四:wq;第五:生效. .bash_profile?
然后因為linux不再當前目錄下尋找可執行文件,所以我們要進入main.out,如,我在1文件中存放main.out可執行文件,所以我要進入cd 1,然后輸入./main.out表示在當前目錄下執行main.out文件
VS2013下的代碼封裝
打開VS2013,新建項目,選擇Win32應用程序導向
下一步
選擇dll,點完成。
添加新建項.cpp文件,在里面寫代碼,實現判斷兩個數的大小的功能,編寫完編譯一下
尋找自己存儲VS項目文件處,找到編譯好的mydll.dll文件,該文件就是我們封裝好的,供其它程序加載調用的動態庫文件
那么dll文件怎么用呢?我們知道封裝的代碼功能是判斷兩個數的大小,這時候,我們就要新建一個項目,并且添加新建項,編寫代碼,一個程序的入口main。編譯一下。
并將封裝好的.dll文件復制到我們新建項目usedll中和可執行文件一起,并將.lib庫文件放到usedll項目文件中和.c文件一起
這時候,需要一個頭文件,來聲明我們封裝的函數和我們所需要用到的庫文件,代碼:#pragma comment(lib,"mydll.lib")。
成功實現封裝代碼的功能
dll文件封裝起來,主要是給第三方使用的,這樣別人就不會看到里面的代碼,而且dll可以被多個程序調用,可以被其它高級語言調用。
注意C++寫的函數,C語言是不能調用的,但是C++是可以調用的C語言寫的函數,