typedef到處都是,但是能夠真正懂得typedef使用的不算太多。對于初學(xué)者而言,看別人的源碼時(shí)對到處充斥的typedef往往不知所錯(cuò),而參考書又很少,所以在此給出一個(gè)源碼,供大家參考
#include <stdio.h>
#include <iostream.h>
/* 避免Visual C的for與標(biāo)準(zhǔn)for的不同 */
#define for if (0); else for
/* dim(a)是用于計(jì)算a的維數(shù),不過只能計(jì)算數(shù)組的維數(shù),不能計(jì)算指針的維數(shù) */
#define dim(a) (sizeof(a)/sizeof(a[0]))
/* N1到N4是幾個(gè)常量,以枚舉的形式定義 */
enum {N1 = 2, N2 = 3, N3 = 4, N4 = 5};
/* 這個(gè)C程序員都知道,就是將DataType定義為int型,便于擴(kuò)充 */
typedef int DataType;
/* 定義一個(gè)一維數(shù)組,數(shù)組的元素維整型值 */
typedef DataType ARR1[N4];
/* 再定義一個(gè)一維數(shù)組,數(shù)組的元素維ARR1型,不過ARR1又是一個(gè)數(shù)組,所以
* ARR2 實(shí)際上是一個(gè)矩陣
*/
typedef ARR1 ARR2[N3]; /* 此處完全等價(jià)為typedef int ARR2[N3][N4];*/
/* 按照ARR2的解釋,ARR3也是一個(gè)一維數(shù)組,不過數(shù)組元素的類型是ARR2的類型
* 所有ARR3是一個(gè)三維數(shù)組
*/
typedef ARR2 ARR3[N2]; /* 此處完全等價(jià)為typedef int ARR3[N2][N3][N4];*/
/* 分別用定義好的ARR1,ARR2,ARR3定義三個(gè)變量a, b, c */
ARR1 a; /* 此處完全等價(jià)于:int a[N4]; */
ARR2 b; /* 此處完全等價(jià)于:int b[N3][N4]; */
ARR3 c; /* 此處完全等價(jià)于:int c[N2][N3][N4]; */
/* 下面函數(shù)給大家個(gè)示例看a,b,c如何使用 */
void exam_1()
{
for (int i=0; i<dim(a); i++) a[i] = i+1;
for (int i=0; i<dim(b); i++) for (int j=0; j<dim(b[0]); j++)
b[i][j] = (i+1)*10 + (j+1);
for (int i=0; i<dim(c); i++) for (int j=0; j<dim(c[0]); j++)
for (int k=0; k<dim(c[0][0]); k++) c[i][j][k] = (i+1)*100 + (j+1)*10 + (k+1);
printf("\nThe a is :\n");
for (int i=0; i<dim(a); i++) printf("%4d ", a[i]);
printf("\n");
printf("\nThe b is :\n");
for (int i=0; i<dim(b); i++)
{
for (int j=0; j<dim(b[0]); j++) printf("%4d ", b[i][j]);
printf("\n");
}
printf("\nthe c is:\n");
for (int i=0; i<dim(c); i++)
{
for (int j=0; j<dim(c[0]); j++)
{
for (int k=0; k<dim(c[0][0]); k++) printf("%4d ", c[i][j][k]);
printf("\n");
}
printf("\n");
}
}
/* 下面函數(shù)給大家演示數(shù)組在內(nèi)存中的排列 */
void exam_2()
{
int *pn = NULL;
pn = (int *)a; /* 等價(jià)于 pn = &a[0]; */
printf("\nThe a is :\n");
for (int i=0; i<sizeof(a)/sizeof(DataType); i++) printf("%4d ", pn[i]);
printf("\n");
pn = (int *)b; /* 等價(jià)于 pn = &b[0][0]; */
printf("\nThe b is :\n");
for (int i=0; i<sizeof(b)/sizeof(DataType); i++) printf("%4d ", pn[i]);
printf("\n");
pn = (int *)c; /* 等價(jià)于 pn = &c[0][0][0]; */
printf("\nThe c is :\n");
for (int i=0; i<sizeof(c)/sizeof(DataType); i++) printf("%4d ", pn[i]);
printf("\n");
}
int main(int argc, char* argv[])
{
exam_1();
exam_2();
return 0;
}
續(xù) 數(shù)組
#define S(s) printf("%s\n", #s); s
typedef struct _TS1{
int x, y;
} TS1, *PTS1, ***PPPTS1; // TS1是結(jié)構(gòu)體的名稱,PTS1是結(jié)構(gòu)體指針的名稱
// 也就是將結(jié)構(gòu)體struct _TS1 命名為TS1,
// 將struct _TS1 * 命名為 PTS1
// 將struct _TS1 *** 命名為 PPPTS1
typedef struct { // struct后面的結(jié)構(gòu)體說明也可以去掉
int x, y;
} TS2, *PTS2;
typedef PTS1 *PPTS1; // 定義PPTS1是指向PTS1的指針
typedef struct _TTS1{
typedef struct ITTS1 {
int x, y;
} iner;
iner i;
int x, y;
} TTS1;
//結(jié)構(gòu)體內(nèi)部的結(jié)構(gòu)體也一樣可以定義
typedef TTS1::ITTS1 ITS1;
void test_struct()
{
// 基本結(jié)構(gòu)體重定義的使用
TS1 ts1 = {100, 200};
PTS1 pts1 = &ts1; // 完全等價(jià)于TS1* pts1 = &ts1;
PPTS1 ppts1 = &pts1; // 完全等價(jià)于TS1** ppts1 = &pts1;
PPPTS1 pppts1 = &ppts1; // 完全等價(jià)于 TS1*** pppts1 = &ppts1;
TS2 ts2 = {99, 88};
PTS2 pts2 = &ts2; // 完全等價(jià)于 TS2* pts2 = &ts2;
TTS1 itts1 = {{110, 220}, 10, 20};
Its1* rits1 = &itts1.i;
ITS1* &its1 = rits1; // 等價(jià)于 TTS1::ITTS1 *its1 = &(itts1.i);
printf("ts1\t = (%d, %d)\n*pts1\t = (%d, %d)\n"
"**ppts1\t = (%d, %d)\n***pppts1= (%d, %d)\n\n",
ts1.x, ts1.y, pts1->x, pts1->y,
(**ppts1).x, (**ppts1).y, (***pppts1).x, (***pppts1).y);
printf("ts2\t = (%d, %d)\n*pts2\t = (%d, %d)\n\n",
ts2.x, ts2.y, pts2->x, pts2->y);
printf("itts1\t = [(%d, %d), %d, %d]\n*its1\t = (%d, %d)\n\n",
itts1.i.x, itts1.i.y, itts1.x, itts1.y, its1->x, its1->y);
S(pts1->x = 119);
S(pts2->y = 911);
S(its1->x = 999);
printf("ts1\t = (%d, %d)\n*pts1\t = (%d, %d)\n"
"**ppts1\t = (%d, %d)\n***pppts1= (%d, %d)\n\n",
ts1.x, ts1.y, pts1->x, pts1->y,
(**ppts1).x, (**ppts1).y, (***pppts1).x, (***pppts1).y);
printf("ts2\t = (%d, %d)\n*pts2\t = (%d, %d)\n\n",
ts2.x, ts2.y, pts2->x, pts2->y);
printf("itts1\t = [(%d, %d), %d, %d]\n*its1\t = (%d, %d)\n\n",
itts1.i.x, itts1.i.y, itts1.x, itts1.y, its1->x, its1->y);
S((*ppts1)->y = -9999);
printf("ts1\t = (%d, %d)\n**ppts1\t = (%d, %d)\n\n",
ts1.x, ts1.y, (*ppts1)->x, (*ppts1)->y);
S((**pppts1)->x = -12345);
S((***pppts1).y = -67890);
printf("ts1\t = (%d, %d)\n*pts1\t = (%d, %d)\n"
"**ppts1\t = (%d, %d)\n***pppts1= (%d, %d)\n\n",
ts1.x, ts1.y, pts1->x, pts1->y,
(**ppts1).x, (**ppts1).y, (***pppts1).x, (***pppts1).y);
}
在typedef的使用中,最麻煩的是指向函數(shù)的指針,如果沒有下面的函數(shù),你知道下面這個(gè)表達(dá)式的定義以及如何使用它嗎?
int (*s_calc_func(char op))(int, int);
如果不知道,請看下面的程序,里面有比較詳細(xì)的說明
// 定義四個(gè)函數(shù)
int add(int, int);
int sub(int, int);
int mul(int, int);
int div(int, int);
// 定義指向這類函數(shù)的指針
typedef int (*FP_CALC)(int, int);
// 我先不介紹,大家能看懂下一行的內(nèi)容嗎?
int (*s_calc_func(char op))(int, int);
// 下一行的內(nèi)容與上一行完全相同,
// 定義一個(gè)函數(shù)calc_func,它根據(jù)操作字符 op 返回指向相應(yīng)的計(jì)算函數(shù)的指針
FP_CALC calc_func(char op);
// 根據(jù) op 返回相應(yīng)的計(jì)算結(jié)果值
int calc(int a, int b, char op);
int add(int a, int b)
{
return a + b;
}
int sub(int a, int b)
{
return a - b;
}
int mul(int a, int b)
{
return a * b;
}
int div(int a, int b)
{
return b? a/b : -1;
}
// 這個(gè)函數(shù)的用途與下一個(gè)函數(shù)作業(yè)和調(diào)用方式的完全相同,
// 參數(shù)為op,而不是最后的兩個(gè)整形
int (*s_calc_func(char op)) (int, int)
{
return calc_func(op);
}
FP_CALC calc_func(char op)
{
switch (op)
{
case '+': return add;
case '-': return sub;
case '*': return mul;
case '/': return div;
default:
return NULL;
}
return NULL;
}
int calc(int a, int b, char op)
{
FP_CALC fp = calc_func(op); // 下面是類似的直接定義指向函數(shù)指針變量
// 下面這行是不用typedef,來實(shí)現(xiàn)指向函數(shù)的指針的例子,麻煩!
int (*s_fp)(int, int) = s_calc_func(op);
// ASSERT(fp == s_fp); // 可以斷言這倆是相等的
if (fp) return fp(a, b);
else return -1;
}
void test_fun()
{
int a = 100, b = 20;
printf("calc(%d, %d, %c) = %d\n", a, b, '+', calc(a, b, '+'));
printf("calc(%d, %d, %c) = %d\n", a, b, '-', calc(a, b, '-'));
printf("calc(%d, %d, %c) = %d\n", a, b, '*', calc(a, b, '*'));
printf("calc(%d, %d, %c) = %d\n", a, b, '/', calc(a, b, '/'));
}
運(yùn)行結(jié)果
calc(100, 20, +) = 120
calc(100, 20, -) = 80
calc(100, 20, *) = 2000
calc(100, 20, /) = 5