认识C中的结构体

  C中结构体是另外一种表示数据形式的方式,结构体中可以表示C中的基本数据形式,如int,double....结构体可以让我们更好的表示数据。下面来看看结构体。

  说到结构体首先要了解的是它的申明形式,要申明一个结构体形式如下:

 1 #include<stdio.h>
 2 #define LEN 20
 3 
 4 //申明一个结构体
 5 struct name {
 6     char firstname[LEN];
 7     char lastname[LEN];
 8 };
 9 
10 //申明一个结构体
11 struct guy {
12     struct name handle;        //结构体嵌套
13     char favfood[LEN];
14     char job[LEN];
15     int age;
16 };
17 
18 int main(int argc, char* argv[])
19 {
20   ........
21 }

  在结构体中可以有基本的数据类型例如上面的int,char数组,也可以嵌套有其他结构体,例如上面结构体guy中就存在另外一个结构体name。值得注意的是,我们在申明一个结构体的时候,计算机并没有为数据分配空间,而是在我们创建结构变量的时候才会进行分配。创建结构变量的形式如下:

 1 struct guy new_guy; 

  在计算机执行到上面这句时,计算机会为变量分配内存。如上guy中包含一个name结构体,name结构体中是由两个20个char所占字节组成,共计2*20=40字节,而favfood和job也是两个长度为20的char数组,age是一个int类型,所以一个guy类型的变量所占字节数共计为40+20+20+4=84。

  现在我们知道如何申明一个结构体和定义一个结构体变量了。先看看下面的代码:

 1 struct guy new_guy1 = { //结构体的初始化
 2         {"zhou","xuanyu"},
 3         "tomato",
 4         "student",
 5          22
 6     };
 7 struct guy new_guy2 = { //结构体的初始化
 8         .handle = {"zhou","xuanyu"},
 9         .job = "student",
10     };

  上面一段代码是对一个结构体进行初始化,对于new_guy1,我们对它的每一个属性都初始化了,而对于new_guy2我们仅仅初始化了他的handle和job。对于一个结构体我们应该怎么访问结构体中的每一个项呢?不错,是使用 . 运算符。例如:

 1 printf("%s",new_guy2.handle); ,就可以打印出"zhouxuanyu"。再看看下面这段代码(其中的结构体guy在上面已定义):

 1 int main(int argc, char* argv[])
 2 {
 3     struct guy new_guy[2] = {    //定义一个结构体数组
 4         {
 5             {"zhou","xuanyu"},"tomato","student",22    //初始化数组第一项
 6         },
 7         {
 8             {"hu","jiannan"},"fruit","student",22    //初始化数组第二项
 9         }
10     };
11     
12     struct guy * him; //指向结构体guy的指针
13     him = &new_guy[0]; //将指针指向guy数组
14     
15     printf("address #1:%p, #2:%p\n",&new_guy[0],&new_guy[1]);
16     printf("pointer #1:%p, #2:%p\n",him,him+1);
17     
18     printf("him->handle.firstname is %s,(*him).age is %d\n",him->handle.firstname,(*him).age);
19     him++;
20     printf("him->job is %s,(*him).handle.lastname is %s\n",him->job,(*him).handle.lastname);
21 }

  在上面这段代码中,我们定义了一个guy类型的数组,定义一个结构体数组的形式和定义一般数组一样。在上面代码第12行中我们申明了一个指向结构体的指针him,利用指针我们可以对数据的操作会更加灵活。在13行,我们初始化了这个指针,这里要注意的是结构体的名称和数组名不同,数组名就是数组其实元素的地址,所以可以这样做int *p = a,假设a是一个int类型的数组,但是结构体不行,必须使用&取地址。所用上面15,16行打印出来的地址是一样的。上面的代码中又出现了一个新的运算符->。它的作用和刚刚说的 . 运算符一样,只是作用的对象不一样,简单的记法就是:指针使用->访问结构体中的项,而结构体变量使用.访问它的项。到现在我们已经大概了解了结构体的基本用法。下面来看看结构体在函数中使用的几种方式:

 1 #include<stdio.h>
 2 #define LEN 20
 3 
 4 struct Book {                 //定义结构体
 5     char bookname[LEN];
 6     double price;
 7     char author[LEN];
 8 };
 9 
10 //三个函数都用于改变书的价格
11 double changebook1(double);        //因为只改变书的价格,直接将价格作为参数传入函数
12 double changebook2(struct Book *);    //传入一个指向Book类型的指针
13 double changebook3(struct Book);    //传入一个结构体
14 
15 
16 int main(int argc, char* argv[])
17 {
18     struct Book book = {
19         "C Primer Plus",75.0,"John"
20     };
21     
22     struct Book * p_book = &book;
23     
24     printf("book's price is: %f\n",book.price);
25     printf("after changebook1(),return value is: %.2f, book's price is: %.2f\n",changebook1(book.price),book.price);
26     double new_price2 = changebook2(p_book);
27     printf("after changebook2(),return value is: %.2f, book's price is: %.2f\n",new_price2,book.price);
28     double new_price3 = changebook3(book);
29     printf("after changebook3(),return value is: %.2f, book's price is: %.2f\n",new_price3,book.price);
30 }
31 
32 double changebook1(double price){
33     return price * 2;
34 }
35 
36 double changebook2(struct Book * book){
37     book->price *= 2;
38     return book->price;
39 }
40 
41 double changebook3(struct Book book){
42     book.price *= 4;
43     return book.price;
44 }

  上面的代码中,我们定义了三个函数用来改变book的价格,第一个是直接将double类型的pirce传入,第二个是将一个指向结构体的指针传入,第三个是将整个结构体传入。最后打印出函数运行之后的结构,如下:

  

 

 

 

 

  从结果可以看出,只有changebook2()函数才真正的将书的价格改变了。这是为什么呢?原因是这样的,方法二是将一个指针作为参数传入函数,指针指向的是原始数据,所以我们所做的所有操作都是在原始数据上进行的。而方法三传入的是一个结构体,在调用函数changebook3()的时候,会根据模板Book创建一个自动变量,然后这个变量的成员会被初始化与原始数据一样副本。而我们所进行的操作都是在这个副本上进行的,所以不会对原始数据有任何改变。但是这样的做法会对内存有很大的浪费,所以我们一般使用的第二种方式,但是第二种方式会有破坏数据的可能,该怎么办呢?在C中const关键字可以解决这个问题。如果我们对原始数据不需要进行修改,这时候可以将传入函数的指针用const修饰。

posted @ 2015-05-18 20:00  轩宇一页  阅读(921)  评论(0编辑  收藏  举报