C語言的基本數據類型
char、int、float、double
1> 字符charcharacter)
用單引號括起來的字符就是char;''括字符,""括 字符串。例如char c = 'A';
字符 是 用圖形來顯示的,但存儲時使用 圖形的編碼整數),因此字符 具備了字符和數據的兩重特性,最早的字符編碼就是ASCIIAmerican Standard Code for Information Interchange)編碼。'A' = 65 'a' = 97 '0' = 48。
字符 占 8位 二進制位bit), 1 個字節byte)。
#include <stdio.h>
int main()
{
char c1 = 'A';
char c2 = 97;
c1 = c1 + 1;
printf("c1=%c\n",c1);//B
printf("c1=%d\n",c1);//66
printf("c2=%c\n",c2);//a
printf("c2=%d\n",c2);//97
printf("sizeof(c1)=%d\n",sizeof(c1));
char c3 = '9';
int i = c3-'0';
printf("i=%d\n",i);
char c4 = -191;//取-191的後8位
printf("c4=%c,c4=%d\n",c4,c4);
return 0;
}/*練習:把數字字符'9'變成數字9*/附:sizeof() 用法:
sizeof(類型)、sizeof(變量)、sizeof(表達式)
都是計算 類型、變量的 內存大小以字節為單位)
sizeof(表達式)時,表達式 不進行運算,只估計結果,不推薦使用。
練習:輸入一個小寫字母字符,轉成對應的大寫字母字符後輸出:
#include <stdio.h>
int main()
{
printf("請輸入一個小寫字母\n");
char c1, c2;
scanf("%c", &c1);
c2 = c1 - 'a' + 'A';
printf("c1 = %c, c2 = %c\n", c1, c2);
return 0;
}在C語言中,有一些特殊的字符,叫轉義字符,比如:
\n 換行字符 '\n' 成立
\r 回車字符 回車回到本行行首,換行下一行的行首
\t 水平制表位 (n個空)
\\ \’ \? \" 4 個特殊字符的表示
2> 整型intinteger)
int 整型 代表 2個 或者 4個 字節,當前主流機器代表 4個字節(32位二進制)。整型還包括:
short int (簡寫short) 短整型 , 2個字節
long int (簡寫long) 長整型 , 4個字節
long long int(C99) 8個字節
unsigned int 無符號整數(非負數)
signed int 有符號整數(正負數),默認
整型 可以用 二進制、八進制、十進制、十六進制 表示。
整型的原碼表示法:
signed int 第一位看成符號位,0代表非負數,1代表負數,表述范圍: 負的2的31次方 到 正的2的31次方-1;
unsigned int 就是非負數,表述范圍: 0 到 正的2的32次方-1;
有符號和無符號在內存中 是一樣的,區別只是 如何處理第一位。有符號數 把第一位看成符號位,不參與運算,無符號數 把第一位看成 數字,參與運算。
#include <stdio.h>
int main()
{
int a = -1;
printf("unsigned a=%u,a=%d\n",a,a);
unsigned int b = -1;
printf("unsigned b=%u,b=%d\n",b,b);
return 0;
}進制轉換二進制-->十進制)
十六進制:0-9,10-15 用 a-fA-F)表示。
八進制、十進制、十六進制在C語言中的表示方法:
八進制 以0開頭,比如 011 = 8+1 = 9 int a = 011;
十六進制 以 0X/0x 開頭,比如 0xF1 = 15*16+1=241 int a = 0xF1;
二進制和十進制的轉換:
正數 2 -> 10
0111 1011 = 64+32+16 +8+2+1 =112 + 11 = 123
0101 1101 = 64 + 16 + 8 +4 +1 = 93
正數 10 -> 2
108 -> 108 -64 = 44 -32 = 12 - 8 =4 -4 =0 -> 0110 1100
0 1 1 0 1 1 0 0
87 -> 87 - 64 = 23 -16 = 7 -4-2-1 -> 0101 0111
練習: 0110 1011-> 107 125-> 0111 1101
負數 10 -> 2
對應的正數二進制 ,然後 按位取反 再 加1 (補碼)
-107 -> 0110 1011 -> 1001 0100+1=1001 0101
- 58 -> 0011 1010 -> 1100 0110
負數 2 ->10
先減1 再按位取反 得到對應的正數,前加 - 號即可
按位取反 再 加1 得到對應的正數,前加 - 號即可
1101 1011 -> 1101 1010 -> 0010 0101 -> -37
1101 1011 -> 0010 0100+1 -> 0010 0101 -> -37
1111 1111 -> -1
練習: -77 -> 0100 1101 -> 1011 0011
1110 0111 -> 0001 1001 -> -25
3> 浮點類型
float 單精度浮點
double 雙精度浮點
long double 擴展雙精度 (較少使用)
浮點數 就是小數,但 浮點 是近似值與計算機存儲有關)。
注:整數常量(字面值)的類型確定:
3 被認為是 int
3.5 被認為是 double
3.5f 被認為是 float
3.5L 被認為是 long double
35L 被認為是 long int
35LL 被認為是 long long int
35u 被認為是 unsigned int
35UL是unsigned long
但 類型是可以 進行類型的轉換的,類型轉換其實就是 內存的變化。
2. C語言的運算符
1> 運算符優先級:
括號的優先級最高 賦值運算符=)的優先級 極低。
2> 算數運算符:+ - * / %取余數)
int 相除,取整數部分,沒有小數捨棄小數部分);
% 只能用於整數,不能用於小數;
int 和 double 混合運算,把 int 轉換成 double 運算;
0 不能做 除數, 也不能取余, 否則引發 浮點數例外運行錯誤),並中斷程序;
0.0 可以做 除數,結果是 inf (無窮大);
#include <stdio.h>
int main()
{
int a = 5, b = 2, c = 0;
double d1 = 2.0, d2 = 0.0;
printf("a/b=%d\n", a / b);//2,3,2.5
printf("a%%b=%d\n", a % b);//1
printf("a/d1=%lf\n", a / d1);//2.0,2,5,2
//printf("a%%d1=%d\n", a % d1);//浮點不能取余,編譯出錯
//printf("a/c=%d\n", a / c);//浮點數例外,運行時出錯
printf("a/d2=%lf\n", a / d2);//a/d2=inf無限大)
printf("%d,%d,%d\n", -9 / 7, -9 % 7, 9 % -7);//-1,-2,2 求余數時候,余數的符號與被除數符號相同C99標准)
printf("main over\n");
return 0;
}注:在 C89 中,負數相關的 / % 都不確定,在C99中,除法向 0 取整,取余 i%j 的符號 由 i 決定。
練習:逆序打印2位數,比如 92-> 29整型的除法和取余)
#include <stdio.h>
int main()
{
int num, res = 0;
printf("請輸入一個兩位整數\n");
scanf("%d", &num);
res = num%10*10 + num/10;
printf("res=%d\n", res);
return 0;
}3> 賦值運算符=)
賦值運算符可以改變變量的值。在C語言中,除了 ++/-- 之外,只能用賦值運算符改變變量的值。 賦值運算符包括 = += -= *= /= ...
a += b; -> a = a + b; 賦值運算符 是 從 右向左 算,比如:
a=b=c= 0;// c=0->b=c->a=b; 但不推薦這種寫法。賦值運算符 是把右邊的值 賦給 左邊的變量,因此 左邊 不能是表達式,最好就是一個變量。左邊 又叫 左值。
i+j = 10; 錯 ,左邊是表達式 10 = i; 錯,左邊是 值,10是常數,不能被賦值。
4> 自增和自減++/--)
自增和自減 針對變量,5++ 錯。++/-- 可以用於 char 、int 、float、double 類型的變量,但 浮點型變量較少使用。
自增和自減 有兩種用法,前++ (++i)和後++ (i++) ,都會自增1。但 前++先自增後運算,用++之後的數 參與運算。 後++ 先運算後自增,用 ++之前的數參與運算。前++可以連用,後++可以連用)
經驗: i++/++i 單獨做為一個語句出現,不要和其他的語句混合。
#include <stdio.h>
int main()
{
int i=1,res;
double d = 1.0;
d++;
printf("d=%lf\n",d);//2.0
printf("i++=%d\n",i++);//1
printf("i=%d\n",i);//2
printf("++i=%d\n",++i);//3
printf("i=%d\n",i);//3
//res = i++ + ++i;//8 4+4
//res = i++ + i++;//6 3+3
res = ++i + ++i;//10 5+5
printf("res=%d,i=%d\n",res,i);
return 0;
} 5> 邏輯表達式
關系運算符:> >= < <=
== 是否相等 (= 賦值)
!= 是否不相等
邏輯表達式的值 為 1(邏輯真,成立) 或者 0(邏輯假,不成立)。但 在C語言中,非0 都是真,0 是假(針對變量、非邏輯表達式)。
關於真和假 (邏輯運算)可以使用邏輯運算符:
! 邏輯非 取相反)
&& 邏輯與 (並且) 都為真,結果 為真,否則假
|| 邏輯或 (或者) 都為假,結果 為假,否則真
&& 和 || 做了算法的優化,有短路效果。針對邏輯與,前面如果假,後面不進行運算,針對邏輯或,前面真,後面不進行運算。
#include <stdio.h>
/*
測試邏輯與 和 邏輯或 的短路效應
*/
int main()
{
int i=0,j=0;
if(i++&&j++){}//if(0&&j++) j++不運行
if(i--||j++){}//if(1||j++) j++不運行
printf("i=%d,j=%d\n",i,j);
return 0;
}6> 位運算符(只能是標准的int和char)
位與 & 按 二進制位進行與運算(2個整數) 兩個位都是1,結果1,否則 0
位或 | 按 二進制位進行或運算(2個整數) 兩個位都是0,結果0,否則 1
位反 ~ 按 二進制位進行取反運算(1個整數) 按位取反,~3+1=-3 -> ~3 = -4
位異或 ^ 按 二進制位進行異或運算(2個整數) 兩個位不同,結果為1,否則結果為0
注:
位與 可以 置 相應的二進制位 為0,也可以得到 對應的值。位與 111 0 111 置0.
位或 可以 置 相應的二進制位 為1,也可以得到 對應的值。位或 000 1 000 置1.
位異或 可以 反轉 二進制位。位異或 000 1 000 可以反轉。
#include <stdio.h>
int main()
{
int a=5,b=7;//0101 0111
printf("a&b=%d\n",a&b);//0101=5
printf("a|b=%d\n",a|b);//0111=7
printf("a^b=%d\n",a^b);//0010=2
printf("~a=%d\n",~a);//-6
printf("b<<2=%d,b>>2=%d\n",b<<2,b>>2);//11100=28,0001=1
b=b&0xFFFFFFFB;//111... 1011=11=B
printf("b=%d\n",b);
printf("%d\n",(b&0x00000004)!=0);//判斷b的倒數第三位是否為1
return 0;
}7> 左移和右移運算
左移位 << 按 二進制位進行向左移動運算。左移運算時,後面空位 補 0
右移位 >> 按 二進制位進行向右移動運算。右移運算時,前面空位時:有符號數補符號位,無符號數補0。
注:左移一位相當於 *2,右移一位相當於 /2 ,但效率更高。
8> &/*運算符
&取地址 * 根據地址取變量
printf中的%p用來輸出地址。
地址是按字節編號的,取地址取得的是一個變量的開始地址。
9> 三目運算符
(條件,邏輯值)? 表達式1 : 表達式2 ---> 如果 條件真,取 表達式1,否則取 表達式2
int a = 10; (a>10) ? 11 : 10 ; //如果a>10 取11,否則取10 (a>10) ?9.0: 8; //取值 8.0, 類型自動提升。 a>b ? a:b //取a,b的最大值
3. 類型轉換
1> 有浮點和整型運算,自動轉成 浮點後運算;
2> 小於intchar、short)的整型運算,自動轉成int後運算;
3> 強制類型轉換 格式:(目標類型) ()放 結果的類型;
強制類型轉換運算符 (目標類型) 可以省略。char c = (char) int;
本文出自 “雪狼” 博客,請務必保留此出處http://wolfzhxp.blog.51cto.com/2409990/1285495