作者 : 万境绝尘
转载请注明出处 : http://www.hanshuliang.com/?post=30
.
结构体概述 : 结构体是 多个 变量的集合, 变量的类型可以不同;
-- 可进行的操作 : 结构体可以进行 拷贝 赋值操作, 可以作为 函数参数 和 函数返回值;
1. 结构体的基本使用
结构体声明 : struct 结构标记 {结构成员} 普通变量;
-- 结构体示例 :
代码语言:javascript复制 struct student
{
char *name;
int age;
};
-- 结构标记 : struct 后面的 student 是结构标记, 这个标记 可写 可不写, 其作用是 为结构命名, 结构标记可以代表 {} 中的声明的所有的成员变量;
-- 结构成员 : 在 {} 中定义的变量就是结构成员;
-- 普通变量 : 在声明结构体的时候后面可以加上若干普通变量, 相当于定义结构体变量;
结构体变量声明 : 可以在定义的时候声明变量, 也可以在定义完结构体使用 结构标记 声明变量;
-- 定义结构体时声明变量 : 这种声明变量的方式可以不用 结构标记, 变量名写在 花括号 后面, 用头号隔开;
代码语言:javascript复制 struct student
{
char *name;
int age;
} s1, s2, s3;
-- 使用结构标记声明 : 结构标记 student 代表了花括号的声明, 是 结构体的简写, 可以使用结构标记代替花括号中的内容;
代码语言:javascript复制struct student s4, s5, s6;
结构体内存分配 : 结构体内存是在声明变量的时候分配的, 如果只声明了结构体, 没有声明对应变量, 那么不会分配内存;
结构体变量初始化 :
-- 声明结构体的时候初始化 : struct student s1 = {"Tom", 12} ; 注意 初值表中必须时结构体对应类型的常量表达式;
-- 声明之后初始化 : 结构体变量名.成员名 可以访问结构体中的成员变量, s1.name = "Tom"; s2.age = 12;
结构体嵌套 : 结构体中的成员变量可以是 结构体变量;
代码语言:javascript复制 struct student
{
char *name;
int age;
} s1;
struct class
{
struct student s1;
struct student s2;
} c1;
结构体代码示例 :
代码语言:javascript复制/*************************************************************************
> File Name: base_struct.c
> Author: octopus
> Mail: octopus_work.163.com
> Created Time: 2014年03月24日 星期一 10时49分46秒
************************************************************************/
#include<stdio.h>
int main(int argc, char **argv)
{
/*
* 声明结构体 同时声明变量s1
*/
struct student
{
char *name;
int age;
} s1;
/*
* 结构体嵌套
*/
struct class
{
struct student s1;
struct student s2;
} c1;
struct student s2 = {"Tom", 12};/*只有声明的时候才能对结构体初始化才能使用花括号赋值*/
struct class c2 = {{"Jack", 13}, {"Pig", 15}};
s1.name = "Hack"; /*变量声明后对结构体赋值只能一个一个赋值*/
s1.age = 14;
//s1 = {"fuck", 1}; /*只有在初始化的时候才能使用 花括号初始化结构体变量*/
c1.s1.name = "CJ";
c1.s1.age = 21;
c1.s2.name = "KW";
c1.s2.age = 22;
/*访问结构体中的变量, 使用 . 进行访问*/
printf("s1 : name = %s, age = %d n", s1.name, s1.age);
printf("s2 : name = %s, age = %d n", s2.name, s2.age);
printf("c1 : s1 : name = %s, age = %d ; s2 : name = %s, age = %d n", c1.s1.name, c1.s1.age, c1.s1.name, c1.s2.age);
printf("c2 : s1 : name = %s, age = %d ; s2 : name = %s, age = %d n", c2.s1.name, c2.s1.age, c2.s1.name, c2.s2.age);
return 0;
}
执行结果 :
代码语言:javascript复制octopus@octopus-Vostro-270s:~/code/c/struct$ gcc base_struct.c
octopus@octopus-Vostro-270s:~/code/c/struct$ ./a.out
s1 : name = Hack, age = 14
s2 : name = Tom, age = 12
c1 : s1 : name = CJ, age = 21 ; s2 : name = CJ, age = 22
c2 : s1 : name = Jack, age = 13 ; s2 : name = Jack, age = 15
2. 结构体与函数
结构体的合法操作 :
-- 整体复制 : 结构体可以复制;
-- 整体赋值 : 声明结构体的时候可以整体赋值, 在其它情况下不可以;
-- & 取地址 : 使用 & 运算符获取 结构体地址;
-- 访问成员 : 使用 结构体变量名.成员变量名 可以访问成员变量;
函数传递结构体方法 :
-- 传递结构体成员 : 将结构体拆开, 将其中的成员变量分别传入;
代码语言:javascript复制struct class create_class(struct student s2, struct student s3)
{
struct class c1 = {s2, s3};
return c1;
}
-- 传递结构体 : 将整个结构体当做参数传入, 这种情况和传递其它类型参数一样, 都是通过值传递的;
代码语言:javascript复制struct class create_class(struct student s2, struct student s3)
{
struct class c1 = {s2, s3};
return c1;
}
-- 传递结构体指针 : 传递结构体的指针, 访问形式如下;
代码语言:javascript复制/*
* 传入一个结构体指针
* 通过指针访问结构体的方法 :
* (*结构体指针变量).成员变量 访问;
* 结构体指针变量 -> 成员变量 访问;
*/
void printf_struct_pointer(struct student *s)
{
printf("student : (*s).name = %s, (*s).age = %d n", (*s).name, (*s).age);
printf("student : s->name = %s, s->age = %d n", s->name, s->age);
}
结构体指针访问 :
-- 示例 : 定义一个结构体指针;
代码语言:javascript复制 struct student
{
char *name;
int age;
}*p;
-- "." 和 "->"优先级 : 这两个运算符都是从左到右运算, 都是右结合; "." 和 "->" 优先级比 "*" , " " 优先级要高; 这两个运算符 与 () [] 是四个优先级最高的运算符;
-- p->age 分析 : 是对结构体中的 age 进行自增操作;
-- *p->name 分析 : 获取 结构体中的 name 字符串的值(注意不是指针|地址);
-- *p ->name 分析 : 先获取 name 字符串的值, 再将p自增;
结构体函数示例 :
代码语言:javascript复制/*************************************************************************
> File Name: method_struct.c
> Author: octopus
> Mail: octopus_work.163.com
> Created Time: 2014年03月24日 星期一 14时46分16秒
************************************************************************/
#include<stdio.h>
/*声明一个结构体类型, 其成员变量是普通变量*/
struct student
{
char *name;
int age;
};
/*声明一个结构体类型, 其成员变量是结构体*/
struct class
{
struct student s1;
struct student s2;
};
/*
* 传递 2 个结构体的成员变量
* 在函数中创建结构体并返回
*/
struct student create_student(char *name, int age)
{
struct student s1 = {name, age};
return s1;
}
/*
* 传递 2 个结构体变量
*/
struct class create_class(struct student s2, struct student s3)
{
struct class c1 = {s2, s3};
return c1;
}
/*
* 传入一个结构体指针
* 通过指针访问结构体的方法 :
* (*结构体指针变量).成员变量 访问;
* 结构体指针变量 -> 成员变量 访问;
*/
void printf_struct_pointer(struct student *s)
{
printf("student : (*s).name = %s, (*s).age = %d n", (*s).name, (*s).age);
printf("student : s->name = %s, s->age = %d n", s->name, s->age);
}
int main(int argc, char **argv)
{
/*使用函数获取一个结构体, 传入结构体的值*/
struct student s1 = create_student("Tom", 11);
printf("student : name = %s, age = %d n", s1.name, s1.age);
/*创建一个成员变量是结构体的结构体, 并打印结构体数据*/
struct class c1 = create_class(create_student("Jack", 12), create_student("CJ", 13));
printf("c1 : s1 : name = %s, age = %d ; s2 : name = %s, age = %d n", c1.s1.name, c1.s1.age, c1.s2.name, c1.s2.age);
/*声明结构体指针*/
struct student *p = &s1;
printf_struct_pointer(p);
return 0;
}
执行结果 :
代码语言:javascript复制octopus@octopus-Vostro-270s:~/code/c/struct$ gcc method_struct.c
octopus@octopus-Vostro-270s:~/code/c/struct$ ./a.out
student : name = Tom, age = 11
c1 : s1 : name = Jack, age = 12 ; s2 : name = CJ, age = 13
student : (*s).name = Tom, (*s).age = 11
student : s->name = Tom, s->age = 11
3. 结构体数组
(1) 结构体数组声明初始化
声明结构体数组 :
-- 声明结构体的时候声明结构体数组 : 格式为 : struct 结构标记 {} 数组名[];
-- 使用结构标记声明结构体数组 : 格式为 : struct 结构标记 数组名[];
结构体数组声明初始化 :
-- 逐个元素初始化 : 数组名[] = {{结构体1}, {结构体2}};
-- 总体初始化 : 数组名[] = {常量1, 常量2 ...};
结构体初始化 :
代码语言:javascript复制/*************************************************************************
> File Name: array_struct.c
> Author: octopus
> Mail: octopus_work.163.com
> Created Time: 2014年03月24日 星期一 16时40分15秒
************************************************************************/
#include<stdio.h>
/*
* 声明结构体
* 同时也声明结构体类型数组
* 为数组初始化
* 直接将每个结构体成员的值依次列出即可
*/
struct student
{
char *name;
int age;
} team1[] =
{
"Tom", 12,
"Jack", 13
};
int main(int argc, char **argv)
{
int i;
/*将每个结构体初始化, 赋值, 每个结构体初始化内容使用 花括号括起来*/
struct student team2[] = {{"CJ", 34}, {"KI", 32}};
for(i = 0; i < 2; i )
{
printf("team1 : team1[i].name = %s, team1[i].age = %d n", team1[i].name, team1[i].age);
}
for(i = 0; i < 2; i )
{
printf("team2 : team2[i].name = %s, team2[i].age = %d n", team2[i].name, team2[i].age);
}
return 0;
}
执行结果 :
代码语言:javascript复制octopus@octopus-Vostro-270s:~/code/c/struct$ gcc array_struct.c
octopus@octopus-Vostro-270s:~/code/c/struct$ ./a.out
team1 : team1[i].name = Tom, team1[i].age = 12
team1 : team1[i].name = Jack, team1[i].age = 13
team2 : team2[i].name = CJ, team2[i].age = 34
team2 : team2[i].name = KI, team2[i].age = 32
(2) 结构体数组示例程序
需求 : 实现一个统计 C 语言关键字出现次数;
代码 :
代码语言:javascript复制/*************************************************************************
> File Name: word_count.c
> Author: octopus
> Mail: octopus_work.163.com
> Created Time: 2014年03月24日 星期一 17时12分32秒
************************************************************************/
#include<stdio.h>
#include<ctype.h>
#include<string.h>
#define MAXWORD 10
/*
* 定义结构体类型 key
* 该类型结构体中存储一个 字符串(关键字) 和 int 数据(关键字出现次数)
* 同时声明一个结构体数组
* 对结构体数组进行初始化
*
*
*/
struct key
{
char *word;
int count;
}key_count[] =
{
"auto", 0,
"break", 0,
"case", 0,
"char", 0,
"const", 0,
"continue", 0,
"default", 0,
"void", 0,
"volatitle", 0,
"while", 0
};
int main(int argc, char **argv)
{
int n;
char word[MAXWORD];
/*循环接收字符串, 如果字符串与结构体数组中匹配, 结构体的count */
while(getword(word, MAXWORD) != EOF)
if(isalpha(word[0]))
if((n = binsearch(word, key_count, 10)) >= 0)
key_count[n].count ;
/*打印大于0的关键字 及 个数*/
for (n = 0; n < 10; n )
if(key_count[n].count > 0)
printf("- %sn", key_count[n].count, key_count[n].word);
return 0;
}
/*
* 重要api解析 :
* int getc(FILE *stream) 从标准输入流中读取字符
* int ungetc(int c, FILE *stream) 将字符退回到标准输入流中
* int isspace(int c) 判断字符是否是 空格 'f' 'r' 'n' 't' 'v'
* int isalpha(int c) 判断是否是字母
*/
int getword(char *word, int lim)
{
int c, getc(FILE*), ungetc(int, FILE*);
char *wp = word;
/*过滤空格, 如果输入的不是 空, 就继续向下执行*/
while(isspace(c = getc(stdin)));
/*如果输入的不是结束符, 那么 wp指针, 先取值, 之后地址自增*/
if(c != EOF)
*wp = c;
/*如果输入的不是字母, 直接返回, 关键字里面没有数字开头的*/
if(!isalpha(c))
{
*wp = '