再論C語言之typedef的用法

typedef使用大全1(數組)
typedef到處都是,但是能夠真正懂得typedef使用的不算太多。對於初學者而言,看別人的源碼時對到處充斥的typedef往往不知所錯,而參考書又很少,所以在此給出一個源碼,供大家參考
#include <stdio.h>
#include <iostream.h>
/* 避免Visual C的for與標準for的不同 */
#define for if (0);   else for
/* dim(a)是用於計算a的維數,不過只能計算數組的維數,不能計算指針的維數 */
#define dim(a) (sizeof(a)/sizeof(a[0])) 
/* N1到N4是幾個常量,以枚舉的形式定義 */
enum {N1 = 2, N2 = 3, N3 = 4, N4 = 5};
/* 這個C程序員都知道,就是將DataType定義爲int型,便於擴充 */
typedef int DataType;
/* 定義一個一維數組,數組的元素爲整型值 */
typedef DataType ARR1[N4];
/* 再定義一個一維數組,數組的元素維ARR1型,不過ARR1又是一個數組,所以
 * ARR2 實際上是一個矩陣
 */
typedef ARR1 ARR2[N3]; /* 此處完全等價爲typedef int ARR2[N3][N4];*/
/* 按照ARR2的解釋,ARR3也是一個一維數組,不過數組元素的類型是ARR2的類型
 * 所有ARR3是一個三維數組
 */
typedef ARR2 ARR3[N2]; /* 此處完全等價爲typedef int ARR3[N2][N3][N4];*/
/* 分別用定義好的ARR1,ARR2,ARR3定義三個變量a, b, c */
ARR1 a; /* 此處完全等價於:int a[N4]; */
ARR2 b; /* 此處完全等價於:int b[N3][N4]; */
ARR3 c; /* 此處完全等價於:int c[N2][N3][N4]; */
/* 下面函數給大家個示例看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");
    }
}
 
/* 下面函數給大家演示數組在內存中的排列 */
void exam_2()
{
    int *pn = NULL;
    pn = (int *)a; /* 等價於 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; /* 等價於 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; /* 等價於 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;
}
__________________________________________________________________________________________________________
 
typedef 使用大全2(結構體)
 
#define S(s) printf("%s/n", #s); s
typedef struct _TS1{
 int x, y;
} TS1, *PTS1, ***PPPTS1; // TS1是結構體的名稱,PTS1是結構體指針的名稱
// 也就是將結構體struct _TS1 命名爲TS1,
// 將struct _TS1 * 命名爲 PTS1
// 將struct _TS1 *** 命名爲 PPPTS1
 
typedef struct { // struct後面的結構體說明也可以去掉
 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;
//結構體內部的結構體也一樣可以定義
typedef TTS1::ITTS1 ITS1;
void test_struct()
{
 // 基本結構體重定義的使用
 TS1 ts1 = {100, 200};
 PTS1 pts1 = &ts1; // 完全等價於TS1* pts1 = &ts1;
 PPTS1 ppts1 = &pts1; // 完全等價於TS1** ppts1 = &pts1;
 PPPTS1 pppts1 = &ppts1; // 完全等價於 TS1*** pppts1 = &ppts1;
 TS2 ts2 = {99, 88};
 PTS2 pts2 = &ts2;   // 完全等價於 TS2* pts2 = &ts2;
 TTS1 itts1 = {{110, 220}, 10, 20};
 Its1* rits1 = &itts1.i;
 ITS1* &its1 = rits1; // 等價於 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);
 print
___________________________________________________________________________________________________
typedef使用大全3(指向函數的指針)
在typedef的使用中,最麻煩的是指向函數的指針,如果沒有下面的函數,你知道下面這個表達式的定義以及如何使用它嗎?
int (*s_calc_func(char op))(int, int);
如果不知道,請看下面的程序,裏面有比較詳細的說明
// 定義四個函數
int add(int, int);
int sub(int, int);
int mul(int, int);
int div(int, int);
// 定義指向這類函數的指針
typedef int (*FP_CALC)(int, int);
// 我先不介紹,大家能看懂下一行的內容嗎?
int (*s_calc_func(char op))(int, int);
// 下一行的內容與上一行完全相同,
// 定義一個函數calc_func,它根據操作字符 op 返回指向相應的計算函數的指針
FP_CALC calc_func(char op);
// 根據 op 返回相應的計算結果值
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;
}
 
// 這個函數的用途與下一個函數作業和調用方式的完全相同,
// 參數爲op,而不是最後的兩個整形
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); // 下面是類似的直接定義指向函數指針變量
 // 下面這行是不用typedef,來實現指向函數的指針的例子,麻煩!
 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, '/'));
}運行結果
 calc(100, 20, +) = 120
 calc(100, 20, -) = 80
 calc(100, 20, *) = 2000
 calc(100, 20, /) = 5
 
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章