typedef使用大全(转) 一、数组 typedef到处都是,但是能够真正懂得typedef使用的不算太多。对于初学者而言,看别人的源码时对到处充斥的typedef往往不知所错,而参考书又很少,所以在此给出一个源码,供大家参考 #include #
2011-05-13 23:45
761 查看
typedef使用大全(转)
一、数组
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
/*等价于 pn = &a[0]; */
*)a;
printf("/nThe a is :/n");
for
(int
i=0; i<sizeof
(a)/sizeof
(DataType); i++) printf("%4d ", pn[i]);
printf("/n");
pn = (int
/*等价于 pn = &b[0][0]; */
*)b;
printf("/nThe b is :/n");
for
(int
i=0; i<sizeof
(b)/sizeof
(DataType); i++) printf("%4d ", pn[i]);
printf("/n");
pn = (int
/*等价于 pn = &c[0][0][0]; */
*)c;
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;
}
二、结构体续 数组
#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);
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的使用中,最麻烦的是指向函数的指针,如果没有下面的函数,你知道下面这个表达式的定义以及如何使用它吗?
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
一、数组
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
/*等价于 pn = &a[0]; */
*)a;
printf("/nThe a is :/n");
for
(int
i=0; i<sizeof
(a)/sizeof
(DataType); i++) printf("%4d ", pn[i]);
printf("/n");
pn = (int
/*等价于 pn = &b[0][0]; */
*)b;
printf("/nThe b is :/n");
for
(int
i=0; i<sizeof
(b)/sizeof
(DataType); i++) printf("%4d ", pn[i]);
printf("/n");
pn = (int
/*等价于 pn = &c[0][0][0]; */
*)c;
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;
}
二、结构体续 数组
#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);
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的使用中,最麻烦的是指向函数的指针,如果没有下面的函数,你知道下面这个表达式的定义以及如何使用它吗?
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
相关文章推荐
- 使用 ext.js 3.0 和 .net 做的一个简单HR 系统,供大家参考
- 介绍一个工具给大家,做网站时,经常要上传文件到外网服务器,但是上传时往往需要很长时间,如果有一个文件对比工具……
- 用Gradle打包出jar文件 前面我们说过,在Android Studio里面使用Gradle来打包应用程序,一般都是build出来一个apk文件。但是有的同学是做实现层的开发,不直接做View层
- (2)int A[nSize],其中隐藏着若干0,其余非0整数,写一个函数int Func(int* A, int nSize),使A把0移至后面,非0整数移至数组前面并保持有序,返回值为原数据中第一个元素为0的下标。(尽可能不使用辅助空间且考虑效率及异常问题,注释规范且给出设计思路)
- 关于Fragment使用的那些坑,给出一个完美的填坑方式!源码相送
- 对于一个整数矩阵,存在一种运算,对矩阵中任意元素加一时,需要其相邻(上下左右),某一个元素也加一,现给出一正数矩阵,判断其是否能够由一个全零矩阵经过上述运算得到。
- 堆的数据结构能够使得堆顶总是维持最大(对于大根堆)或最小(对于小根堆),给定一个数组,对这个数组进行建堆,则平均复杂度是多少?如果只是用堆的 push 操作,则一个大根堆依次输入 3,7,2,4,1,5,8 后,得到的堆的结构示意图是下述图表中的哪个?
- MyEclipse下如何安装和使用ibatis插件(网上的资料对于myeclipse8.5根本就是没有用的,所以我还是自己选择了装了一个eclipse,然后将插件装在了eclipse中)
- 二者都是将 byte[] 转换成字符串,一个区别在于 Convert.ToBase64String 使用 base 64 数字编码,所以它生成的全部是 ASCII 字符。
- Java里面本质都是值传递——别人写的表达不清楚,所以我写了下,方便初学者福利 也方便C语言思维的人理解。
- 问题阐述: 本人使用mini6410开发了一个sqlite数据库的程序,在mini6410的linux系统下已经能够成功运行了。因为Android使用的也是linux内核,所以我想当然的认为按照同样
- 《c primer pius》第十章第6题,编写一个程序,初始化一个二维double数组,并利用练习2中的任一函数来把这个数组复制到另一个二维数组(因为二维数组是数组的数组,所以可以使用处理一维数组的
- typedef使用大全1(数组)
- 最近做了一个小东西,对于接口实现类。发现很多都是一个套路,所以备份下,以后可以经常用到! 这几个是单纯查询的实现类。
- Java里面本质都是值传递——别人写的表达不清楚,所以我写了下,方便初学者福利 也方便C语言思维的人理解。
- 用工厂模式写了一个数据库访问类库,但是对于数据库连接字符串和数据库类型有两种方法,不知道哪种好,大家看看了,附类关系图
- 一个数组里,数都是两两出现的,但是有三个数是唯一出现的,找出这三个数
- 只要你的答案让我明白了如和使用Microsoft VisualC++6 但是我一点基础也没有。所以向大家请教 我想学习C语言 加分不是问题 还望得到大家的帮助 我一定把分数加到你满意为止
- 罗马帝国开创了辉煌的人类文明,但他们的数字表示法的确有些繁琐,尤其在表示大数的时候,现在看起来简直不能忍受,所以在现代很少使用了。之所以这样,不是因为发明表示法的人的智力的问题,而是因为一个宗教的原因
- JavaScript基础 使用delete删除数组中一个元素的值 但是元素的所占的那个空间还在