zoukankan      html  css  js  c++  java
  • C语言(三)- 结构体、结构体指针、位运算

    一、结构体

    1、一般形式

    不同类型数据组成的组合型数据结构,即结构体。

    结构体类型的一般形式:

    1 struct 结构体名{
    2 类型名  成员名1;
    3 类型名  成员名2;
    4 类型名  成员名3;
    5 ...... 6 };

    举个例子:

     1 #include<stdio.h>
     2 int main(void)
     3 {
     4 struct Date
     5     {
     6        int month;
     7        int day;
     8        int year;
     9     };
    10 struct Student
    11     {
    12        int num;
    13        char name[20];
    14        char sex;
    15        int age;
    16        struct Date birthday;    //birthday是一个struct Date结构体类型变量,即birthday使用了strcut Date结构体的数据结构形式
    17        char addr[30];
    18     };
    19 }

    2、定义结构体类型变量

     1 struct Date           //声明结构体类型,再定义是此类型的变量
     2     {
     3        int month;
     4        int day;
     5        int year;
     6     };
     7 struct Date birthday1,birthday2;
     8 
     9 或者
    10 
    11 struct Date            //在声明结构体类型的同时定义是此类型的变量
    12     {
    13        int month;
    14        int day;
    15        int year;
    16     } birthday1,birthday2;

    3、结构体变量初始化和引用

     使用上述  struct Date  结构体来进行说明。

    3.1初始化

     1 struct Date                        //声明结构体同时初始化       
     2     {
     3        int month;
     4        int day;
     5        int year;
     6     } birthday1={620,2015};       //char字符型变量使用" "
     7 
     8 
     9 或者
    10 
    11 struct Date birthday1={620,2015}; //先如上声明结构体,再初始化
    12 
    13 
    14 除此之外还可以对某一单一成员变量初始化:
    15 struct Date birthday1={.month=6}; 

    3.2成员变量操作

    结构体变量名.成员名

    如:birthday1.month

    结构体变量名.成员名.子成员名

    如:student1.birthday1.month

    注: 

    (1) .  是成员运算符,所有运算符中优先级最高,使用后相当于一个整体,即一个变量。

    (2)成员变量及之间可以进行算术运算。

    (3)同类的结构体变量可以相互赋值,如:birthday1=birthday2

    (4)可以引用结构体成员变量的地址,也可以引用结构体变量的地址。

    1 scanf("%d",&birthday1.month);          //输入&birthday1.month的值
    2 
    3 printf("%d",&birthday1);               //输出结构体变量&birthday1的首地址,即birthday1.month

    (5)在scanf函数中,如果结构体成员变量是数组,其本身就代表地址,前面不需要添加取地址符&。 

    二、结构体数组

     结构体数组一般形式:

     1 struct  机构体名
     2 {
     3 2 类型名  成员名1;
     4 3 类型名  成员名2;
     5 4 类型名  成员名3;
     6 5 ......
     7 }数组名[ 数组长度 ];
     8 
     9 或者
    10 
    11 结构体类型  数组名[ 数组长度 ];        //先声明好结构体类型

    举例子:

     1 #include<string.h>      //字符数组头文件
     2 #include<stdio.h>
     3 struct Person
     4     {
     5     char name[20];       //参选人名
     6     int count;                //合计票数
     7     }leader[3]={"ergouzi",0,"xiaobai",0,"goudaner",0};
     8 //定义结构体数组并初始化,三个人起始都是0票
     9 
    10 
    11 或者
    12 
    13 strcut Person leader[3]={"ergouzi",0,"xiaobai",0,"goudaner",0};

    注:结构体数组相对于结构体,简化了结构体变量定义,通过数组一次性定义很多结构体变量。

    三、结构体指针

    一个变量的地址称为该变量的指针,指针即内存地址。

    ①指针就是地址,地址就是指针;②地址是内存单元的编号;③指针变量就是存放内存地址的变量。

    &:取变量地址运算符,一般后接整型浮点型变量。*:取指针变量所指向的变量的内容的运算符,后接指针变量(即地址)。

    1 int x,y;
    2 int *p;     //表示p是指向整型量的指针变量,此时没有赋指针变量值,更不存在指向变量的内容。
    3 x=10;
    4 p=&x;       //将整型变量x的地址赋给了p,此时存在了指向变量的内容
    5 y=*p;       //将p所指向的变量的内容赋给了y,   即y=10

    结构指针是指向一种结构体类型的指针变量,它是结构体在内存中的首地址。

    地址:在内存中每8bit物理长度(即1字节)给的一个编号。在stm32f103zet6中存储器有4GB的地址空间:

    4GB的地址空间按照功能划分成8块,以Block 0为例:地址总数为0x2000 0000,换算成十进制为2*(16^7)= 536870912B = 512*1024*1024B =512*1024KB = 512MB

    stm32f103zet6为32处理器,寄存器也是32位。8位一个地址,单个寄存器的地址偏移量为4个地址。

    地址在C语言编程时,0x0000 0004在编译器来看只是一个十六进制数字,需要地址强制转换(unsigned int *)0x0000 0004。

     以stm32f103zet6中GPIOB寄存器的封装为例讲解结构体指针:

     1 /*
     2 GPIOx(A-G)是片上外设,为了系统化和结构化的封装寄存器,引入了结构体指针
     3  */
     4 /*外设基地址*/
     5 #define PERIPH_BASE                 ((unsign int)0x40000000)
     6 /*总线基地址*/
     7 #define APB1PERIPH_BASE          PERIPH_BASE
     8 #define APB2PERIPH_BASE          ((unsign int)0x00010000)
     9 #define AHBPERIPH_BASE           ((unsign int)0x00020000)
    10 /*GPIO外设基地址*/
    11 #define GPIOA_BASE                  (APB2PERIPH_BASE + 0x0800)
    12 #define GPIOB_BASE                  (APB2PERIPH_BASE + 0x0C00)
    13 #define GPIOC_BASE                  (APB2PERIPH_BASE + 0x1000)
    14 #define GPIOD_BASE                  (APB2PERIPH_BASE + 0x1400)
    15 #define GPIOE_BASE                  (APB2PERIPH_BASE + 0x1800)
    16 #define GPIOF_BASE                  (APB2PERIPH_BASE + 0x1C00)
    17 #define GPIOG_BASE                  (APB2PERIPH_BASE + 0x2000)
    18 /*寄存器基地址,GPIOB为例*/
    19 #define GPIOB_CRL                    (GPIOB_BASE + 0x00)
    20 #define GPIOB_CRH                    (GPIOB_BASE + 0x04)
    21 #define GPIOB_IDR                    (GPIOB_BASE + 0x08)
    22 #define GPIOB_ODR                    (GPIOB_BASE + 0x0C)
    23 #define GPIOB_BSRR                   (GPIOB_BASE + 0x10)
    24 #define GPIOB_BRR                    (GPIOB_BASE + 0x14)
    25 #define GPIOB_LCKR                   (GPIOB_BASE + 0x18)
    26 /*此时已知寄存器具体地址,可以通过指针&*赋值操作控制寄存器*/
    27 /*GPIOx都有一组功能相似的寄存器,只是地址不同却要为每个寄存器定义地址。为了方便访问寄存器,引入结构体封装寄存器*/
    28 
    29 
    30 /*使用结构体封装GPIO寄存器组*/
    31 typedef unsigned           int  uint32_t;     //无符号32位变量
    32 typedef unsigned short   int  uint16_t;    //无符号16位变量
    33 /*GPIO寄存器列表*/
    34 typedef struct{
    35       uint32_t CRL;          //端口配置低寄存器      地址偏移:0x00
    36       uint32_t CRH;         //端口配置高寄存器      地址偏移:0x04
    37       uint32_t IDR;          //数据输入寄存器         地址偏移:0x08
    38       uint32_t ODR;         //数据输出寄存器         地址偏移:0x0C
    39       uint32_t BSRR;       //位设置/清除寄存器      地址偏移:0x10
    40       uint32_t BRR;         //端口位清除寄存器       地址偏移:0x14
    41       uint16_t LCKR;       //端口配置锁定寄存器     地址偏移:0x18
    42 }GPIO_TypeDef;
    43 /*以上代码用typedef关键字声明了名为GPIO_TypeDef的结构体,结构体有七个成员变量。C语法,结构体内变量的存储空间是连续的,32位变量占4字节,16变量占2字节,连续排列。
    因此,给结构体设置好首地址,那么结构体成员的地址就确定下来,然后以结构体的形式访问寄存器。
    */ 44 45 /*结构体指针访问寄存器*/ 46 /*一般过程*/ 47 GPIO_TypeDef *GPIOx;//定义一个GPIO_TypeDef型结构体指针GPIOx 48 GPIOx = GPIOB_BASE;//GPIOx指针变量,把指针地址设为宏GPIOB_BASE地址 49 GPIOx->CRL = 0xFFFF; 50 uint32_t temp; 51 temp = GPIOx->CRL;//读取GPIOx_CRL寄存器的值到变量temp中 52 53 /*定义好GPIO端口首地址指针*/ 54 #define GPIOA ((GPIO_TypeDef *) GPIOA_BASE) 55 #define GPIOB ((GPIO_TypeDef *) GPIOB_BASE) 56 #define GPIOC ((GPIO_TypeDef *) GPIOC_BASE) 57 #define GPIOD ((GPIO_TypeDef *) GPIOD_BASE) 58 #define GPIOE ((GPIO_TypeDef *) GPIOE_BASE) 59 #define GPIOF ((GPIO_TypeDef *) GPIOF_BASE) 60 #define GPIOG ((GPIO_TypeDef *) GPIOG_BASE) 61 #define GPIOH ((GPIO_TypeDef *) GPIOH_BASE) 62 /*使用定义好的宏直接访问GPIOB端口的寄存器*/ 63 GPIOB->BSRR = 0xFFFF; 64 GPIOB->CRL = 0xFFFF; 65 GPIOB->ODR = 0xFFFF; 66 uint32_t temp; 67 temp = GPIOB->IDR;//读取GPIOB_IDR寄存器的值到变量temp中 68 69 70 /*(以下是访问结构体成员的说明)声明了两个变量,一个是指向结构GPIO_TypeDef的结构指针PGPIOB,GPIOB是一个GPIO_TypeDef结构变量*/ 71 struct GPIO_TypeDef *PGPIOB,GPIOB; 72 PGPIOB = &GPIOB; 73 /*结构体变量访问成员变量(以下三者是等价的)*/ 74 GPIOB.ODR = 0xFFFF; 75 /*结构体指针访问成员变量*/ 76 (*PGPIOB).ODR = 0xFFFF;//*优先级低于.,所以加() 77 PGPIOB->ODR = 0xFFFF;

    三、位运算(二进制运算)

    基本运算符:

    &按位与:均为1结果为1,否则为0。

    |按位或:有一个为1,结果为1。

    ^按位异或:二进位相异,结果为1。

    ~取反:0变1,1变0。

    <<左移:<<左边的运算数的各二进制左移若干位(<<右边的数为指定的位移数)

    >>右移:>>左边的运算数的各二进制右移若干位(>>右边的数为指定的位移数)

     1 /*某一位清零:  a &= ~(1<<x)*,二进制a右侧bitx清零/
     2 
     3 unsigned char a= 0x9f;       //a=1001 1111 b
     4 a &= ~(1<<2);                  //对bit2清零
     5 /*
     6      0000 0100 b                 //b表示二进制数
     7      1111 1011 b
     8      a = (1001 1111 b)&(1111 1011 b)
     9      a = 1001 1011 b
    10 */
    11 
    12 
    13 /*连续位清零:a &= ~(y<<x*k)*,y运算数,x一组数的位数,k组数*/
    14 a &= ~(3<<2*1); //运算数为3,每组位数为2位,清零第1组
    15                            //bit0、bit1为第0组,bit2、bit3为第1组
    16 /*
    17      0000 1100 b
    18      1111 0011 b
    19      a = (1001 1111 b)&(1111 0011 b)
    20      a = 1001 0011 b
    21 */
    22 
    23 
    24 /*连续位赋值*/
    25 unsigned char b= 1000 0011b;
    26 b |= (1<<2*2);
    27 /*
    28      0001 0000 b
    29      b = (1000 0011 b)|(0001 0000 b)
    30      b = 1001 0011 b
    31 */
    32 
    33 
    34 /*某一位取反*/
    35 //此时b = 1001 0011 b
    36 b ^=(1<<6)                      //bit6取反
    37 /*
    38      0100 0000 b
    39      b = (1001 0011 b)^(0100 0000 b)
    40      b = 1101 0011 b
    41 */

    7

  • 相关阅读:
    了解HDD或SDD磁盘的健康状态
    修复丢失的打开方式
    Invoke-WebRequest : 请求被中止: 未能创建 SSL/TLS 安全通道。
    绕过禁止未登陆用户访问
    debug
    更新已有数据
    编码格式(乱码)
    ajax
    Http
    科学的管理和规范标准
  • 原文地址:https://www.cnblogs.com/wuguangzong/p/8952367.html
Copyright © 2011-2022 走看看