【C语言】9.指针,字符串和数组的补充,指针函数&函数指针,结构体,枚举

  • 只有指针是可以运算(移动)的,数组名是不可以的。

     int x[10];
     x++;  //illegal
     int* p = x;
     p++; //legal
    
  • 两指针变量相减所得之差是两个指针所指数组元素之间相差的元素个数。

    • 实际上是两个指针值(地址)相减之差再除以该数组元素的长度(字节数)。
    • (pointer2地址值 - pointer地址值) / sizeof(所指向数据类型)
    • 指针之间可以相减,但不可以相加(相加无意义)。
  • 定义字符串:

    • 字符数组:

      char string[] = "hello";
      printf("%s\n",string);
      
    • 字符串指针指向字符串:

      char *str = "hello"
      
  • 使用字符数组来保存的字符串是存在”栈”里的,所以它是可读可写的,所以我们可以修改字符数组里的某个元素的值。

    但是,使用字符指针来保存字符串,它保存的是字符串常量地址,"常量区"是只读的,所以是不可改的。

    char *str = "hello";
    *(str+1) = 'w'; // 错误
    
  • 使用注意:

    char *str;
    scanf("%s", str); 
    
    /* str是一个野指针,他并没有指向某一块内存空间,所以不允许这样写。如果给str分配内存空间是可以这样用的 */
    
     /********* 数组的方法****************/  
    
     char name[20];  
     scanf("%s",name);    
    
    /************* 给字符针针分配内存空间的办法***********/  
    
     char *name;  
     name=(char*)malloc(50);      //此时name已经指向一个刚刚分配的地址空间。  
     scanf("%s",name);  
    

  • 指针函数(是函数,返回值是指针)注意:

    如果函数返回一个字符串,那么如果用一个数组以下面的形式来接的话,是会报错的:

    char *test() {
        return "hello";
    }
    
    int main(int argc, const char * argv[]) {
    
        char names[10];
    
        names = test();
    
        return 0;
    }
    

    这是因为,返回的字符串相当于一个这样的数组:{‘h’, ‘e’, ‘l’, ‘l’, ‘o’, ‘\0’},但是前面我们说过,数组如果在定义的时候没有用{}这种方式初始化,那么后面就不能再用这种方式初始化了,所以会出错。

    解决方法:将char names[10]改为char *names或者char names[10]直接等于test()

  • 函数指针(是指针,指向函数):

    • 格式:函数的返回值类型 (*指针变量名) (形参1, 形参2, ...);

      int sum(int a,int b)
      {
        return a + b;
      }
      
      int (*p)(int,int);
      p = sum;
      
    • 应用场景:

      • 调用函数

      • 将函数作为参数在函数间传递

      • 函数指针能更灵活:

        int minus(int a, int b) 
        {
          return (a - b);
        }
        
        int add(int a, int b)
        {
          return (a + b);
        }
        
        int myFunction(int a, int b, int (*funcP) (int, int))
        {
          return funcP(a, b);
        }
        
        int main()
        {
          int minusResult = myFunction(10, 20, minus);
            int addResult = myFunction(10, 20, add);
              ...
          return 0;
        }
        
        /*
          函数指针能让程序更灵活,比如后续有乘、除函数的时候,只需实现这两个函数然后在主函数调用myFunction函数即可。如果是多人协作,不同的人写不同的功能,如果我们来写myFunction那么基本就不用修改就可以一直使用,非常灵活。
        */
        

    • 技巧:

      1、把要指向函数头拷贝过来

      2、把函数名称使用小括号括起来

      3、在函数名称前面加上一个*

      4、修改函数名称

    • 使用注意:

      • 由于这类指针变量存储的是一个函数的入口地址,所以对它们作加减运算(比如p++)是无意义的。

      • 如上例,如果想使用p这个函数指针,可以直接向使用sum一样:

        int result = p(10, 10);
        

        也可以这样:

        int result = (*p)(10, 10);
        

  • 结构体是一种自定义数据类型,注意,它是数据类型

    struct Student {
         char *name;
         int age;
     };
    
     struct Student stu;
    

    注意,结构体的后面是有 ; 的。

  • 在使用结构体类型的时候,要加上struct关键字。

  • 定义结构体类型的同时定义变量:

    struct Student {
        char *name;
        int age;
    } stu;
    

    这种在定义的同时也定义了变量,就相当于:

    struct Student {
         char *name;
         int age;
     };
    
     struct Student stu;
    

    定义结构体类型的同时定义变量,以后如果想继续使用这个结构体类型,仍然可以使用常规的方式定义:

    struct Student newStu;
    
  • 匿名结构体定义结构体变量:

    struct {
        char *name;
        int age;
    } stu;
    

    这种匿名方式与上面的方式相比,虽然看起来更简洁(省去了结构名),但是要注意,这只能定义一个stu变量,而不能再定义新的变量,因为结构名没有了。

  • 结构体变量的初始化:

    1. 先定义结构体变量,然后再初始化

      struct Student {
           char *name;
           int age;
       };
      
      struct Student stu = {“QY", 21};
      
    2. 定义的同时初始化

      struct Student {
           char *name;
           int age;
       } stu = {“QY", 21};
      
    3. 可以使用另外一已经存在的结构体初始化新的结构体

      struct Student stu2 = stu;
      
    4. 先定义stu再初始化(与1对应):

      struct Student stu3;
      stu3 = (struct Student){"QY", 21};
      
      /*
      这种情况要与数组对应起来,因为前面说过数组是不可以这样子的,其实结构体是可以的,但是要加上强制类型转换,告诉编译器这不是数组而是一个结构体。
      */
      
    5. 下面的做法是错误的:

    6. struct Student {
           char *name;
           int age;
       };
      struct Student stu;
      stu = {“QY", 21}; // 错误,需要强制类型转换
      
    7. 或者以这种方式来给成员赋值:

      struct Student {
           char *name;
           int age;
       };
       struct Student stu;
      
       stu.age = 27;
      
  • 结构体变量占用存储空间大小

    1. 找到成员里占用字节最大的来分配空间,如int a, double b, char c,则一次分配8个字节。要注意对齐的问题来看总共多少个字节。注意声明变量的顺序,不同顺序分配的字节数也不一样。比如现在这个顺序要分配8 + 8 + 8个字节,但是如果int a, char c, double b则分配 8 + 8个字节。

    2. 可以通过#pragma pack(2)来设置对齐模数,这样一次分配2个字节。

      struct A
      {
          int my_test_a; // 分配2次,共4个字节
          char my_test_b; //分配1次,共2个字节
          double my_struct_a; //分配4次,共8个字节
          int my_struct_b; //分配2次,共4个字节
          char my_struct_c; //分配1,共2个字节
      }
      
      /* 结果是共有20个字节。
      */
      
  • 结构体数组:

    跟结构体变量一样,结构体数组也有3种定义方式 :

    • 先定义结构体类型,再定义结构体数组

      struct Student {
          char *name;
          int age;
      };
      struct Student stu[5];
      
    • 定义结构体类型的同时定义结构体数组

      struct Student {
          char *name;
          int age;
      } stu[5];
      
    • 匿名结构体定义结构体结构体数组

      struct {
          char *name;
          int age;
      } stu[5];
      

    结构体数组初始化:

    • 定义的同时进行初始化

      struct { 
        char *name; 
        int age; 
      } stu[2] = { {"jo", 27}, {"y", 30} };
      
    • 先定义,后初始化,整体赋值

      s[1] = (struct stu){23,"hello"};
      
    • 先定义,后初始化 分开赋值

      s[1].age=12; 
      strcpy(stu[1].name, "hello");
      
  • 结构指针变量中的值是所指向的结构变量的首地址

  • struct 结构名 *结构指针变量名

  • 示例:

    // 定义一个结构体类型
    struct Student {
        char *name;
        int age;
    };
    
    // 定义一个结构体变量
    struct Student stu = {“QY", 21};
    
    // 定义一个指向结构体的指针变量
    struct Student *p;
    
    // 指向结构体变量stu
    p = &stu;
    
    /*
    这时候可以用3种方式访问结构体的成员
    */
    // 方式1:结构体变量名.成员名
    printf("name=%s, age = %d \n", stu.name, stu.age);
    
    // 方式2:(*指针变量名).成员名
    printf("name=%s, age = %d \n", (*p).name, (*p).age);
    
    // 方式3:指针变量名->成员名
    printf("name=%s, age = %d \n", p->name, p->age);
    
  • 通过结构体指针访问结构体成员:

    • (*结构指针变量).成员名
    • 结构指针变量->成员名(用熟)
    • (pstu)两侧的括号不可少,因为成员符“.”的优先级高于“”。 如去掉括号写作pstu.num则等效于(pstu.num),这样,意义就完全不对了。
  • 在用结构指针的时候,就把它当作int *a来看,比如:

    struct Student stu = ... //相当于 int a = ...
    struct Student *p; //相当于 int *p
    p = &stu; //相当于 p = &a
    
  • 结构体变量作为参数传递是值传递,而结构指针则是址传递。

  • 枚举:

    枚举通常是一些固定的取值范围,如一年四季,一周七天这种。

    与结构体类似:

    enum 枚举名 {
        枚举元素1,
        枚举元素2,
        ……
    };
    
    enum Season {
        spring,
        summer,
        autumn,
        Winter
    };
    enum Season s;
    定义枚举类型的同时定义枚举变量
    enum Season {
        spring,
        summer,
        autumn,
         winter
    } s;
    省略枚举名称,直接定义枚举变量
    enum{
        spring,
         summer,
         autumn,
         winter
    } s;
    
    s = spring; //等价于s = 0;
    s = 3; //等价于s = winter;
    

    枚举类型本质上来说就是整形,默认为0,1,2,3…,但是也可以赋值,比如第三个赋值为10,那么第1个为0,第2个为1,第3个为10,第4个为11。

    通常来说,枚举的规范为:以k + 枚举名称 + 枚举元素名。比如:

    enum Season {
        kSeasonSpring,
        kSeasonSummer,
        kSeasonAutumn,
        kSeasonWinter
    };
    
    enum Season s = kSeasonSummer;
    
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,001评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,210评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,874评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,001评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,022评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,005评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,929评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,742评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,193评论 1 309
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,427评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,583评论 1 346
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,305评论 5 342
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,911评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,564评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,731评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,581评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,478评论 2 352

推荐阅读更多精彩内容