【C语言】十六进制、二进制、字节、位、指针、数组

【C语言】十六进制、二进制、字节、位


文章目录

    • @[TOC](文章目录)
  • 前言
  • 一、十六进制、二进制、字节、位
  • 二、变量、指针、指针变量
  • 三、数组与指针
  • 四、指针自加运算
  • 五、二维数组与指针
  • 六、指向指针的指针
  • 七、指针变量作为函数形参
  • 八、函数指针
  • 九、函数指针数组
  • 十、参考文献
  • 总结

前言

使用工具:
1.控制器:STM32F103C8T6
2.仿真器:STLINK
3.C Primer Plus 第六版-1


提示:以下是本篇文章正文内容,下面案例可供参考

一、十六进制、二进制、字节、位

u16 A_Parameter[10],B_Parameter[10],Flash_Parameter[10];  //

在C语言或类似的环境中,u16 通常被定义为 unsigned short,这意味着它是一个无符号的16位整型数据。由于一个字节包含8位,因此 u16 类型的数据占用2个字节。

对于数组 u16 A_Parameter[10],它包含了10个 u16 类型的元素。每个元素占用2个字节,因此整个数组占用的字节数为:

10 元素×2 字节/元素=20 字节
所以,u16 A_Parameter[10] 占用了20个字节。

A_Parameter[0] 是数组 A_Parameter 的第一个元素,其类型为 u16,即 unsigned short。由于 u16 是一个16位的无符号整型,它占用2个字节。

因此,A_Parameter[0] 占用2个字节。
在这里插入图片描述

#define FLASH_SAVE_ADDR  0X0800E000 	

通过仿真器回读芯片FLASH中0X0800E000地址
十六进制表示

31 00 00 00 03 00 04 00 

A_Parameter[0] 占据两个字节,
一个字节占据8位,
十六进制表示0x31 00,
二进制表示0011 0001 0000 0000,
按位表示共有十六个0和1的位组成

在这里插入图片描述

二、变量、指针、指针变量

使用”&”来获取普通变量的地址
&变量名取得变量地址后就可以赋值给指针变量
这个代码里,我们定义了一个变量a, 定义了一个指针变量p。
我们通过运算符&把变量a的内存地址赋值给变量p,所以p指向了变量a的内存存储地址。
通过指针变量来改变变量a的值,因为指针变量p指向的是变量a的地址,所以改变指针变量p指向内存地址的数据就可以改变变量a的值。

unsigned char a ;//
unsigned char *p ;//
void Flash_Write(void)
{static unsigned int date=0;//¾²Ì¬Êý¾Ý£¬²»Ë溯Êý½øÈë°ÑÊý¾ÝÇå³ýif(++date==65535)date=0;//Êý¾Ý×Ô¼ÓB_Parameter[0]=M1.speed;//M1.speed//date	B_Parameter[1]=M2.speed;	B_Parameter[2]=3;	B_Parameter[3]=4;	printf("Êý¾ÝÀàÐ͵ĴóС\r\n");	printf("char=%d,int=%d,float=%d,double=%d\r\n",sizeof((unsigned char)B_Parameter[0])\,sizeof((unsigned int)B_Parameter[1])\,sizeof((float)B_Parameter[2])\,sizeof((double)B_Parameter[3]));printf("\r\n");	a=10;p=&a;//È¡µØÖ·printf("±äÁ¿aµÄµØÖ·\r\n");	printf("&a=0x%x\r\n",&a);	printf("p=0x%x\r\n",p);printf("&p=0x%x\r\n",&p);	printf("\r\n");		printf("±äÁ¿aµÄÖµ\r\n");	printf("a=%d\r\n",a);printf("*p=%d\r\n",*p);	printf("\r\n");		printf("ͨ¹ýÖ¸Õë±äÁ¿¸Ä±ä±äÁ¿aµÄÖµ\r\n");	*p=11;	printf("a=%d\r\n",a);	printf("*p=%d\r\n",*p);		printf("\r\n");		printf("ͨ¹ýÄÚ´æµØÖ·¸Ä±ä±äÁ¿aµÄÖµ\r\n");	*(unsigned int *)0x200005ac=12;	printf("a=%d\r\n",a);	printf("*p=%d\r\n",*p);		printf("\r\n");			STMFLASH_Write(FLASH_SAVE_ADDR,(u16*)B_Parameter,4);	
}

在这里插入图片描述

在这里插入图片描述

三、数组与指针

buff默认的是数组下标为0元素的存储地址,buff和&buff[0]是同一个内存地址,只是写法不一样
unsigned char buff[5]={1,2,3,4,5};
unsigned char 表示占用一个字节,即buff数组中所有成员各占一个字节,如buff0]=0x01,占用一个字节

unsigned char buff[5]={1,2,3,4,5};
unsigned char *p1;
unsigned char *p2;p1=buff;p2=&buff[0];printf("buff=0x%x\r\n",buff);	printf("&buff=0x%x\r\n",&buff[0]);	printf("p1_addr=0x%x\r\n",p1);	printf("p2_addr=0x%x\r\n",p2);	printf("\r\n");	printf("buff[0]=%d\r\n",buff[0]);	printf("*p1=%d\r\n",*p1);	printf("*p2=%d\r\n",*p2);	

在这里插入图片描述
在这里插入图片描述

四、指针自加运算

普通变量加减的是数值,而指针变量加减的是地址
首先,代码打印出数组buff每个元素的地址和值。
然后,指针p1被初始化为指向数组的第一个元素(&buff[0])。
接着,通过递增指针p1,代码遍历数组的每个元素,并打印出每个元素的地址和值。

#include <stdio.h>  int main() {  unsigned char buff[5] = {1, 2, 3, 4, 5};  unsigned char *p1;  // 打印数组每个元素的地址和值  printf("&buff[0]=0x%x buff[0]=%d\r\n", &buff[0], buff[0]);  printf("&buff[1]=0x%x buff[1]=%d\r\n", &buff[1], buff[1]);  printf("&buff[2]=0x%x buff[2]=%d\r\n", &buff[2], buff[2]);  printf("&buff[3]=0x%x buff[3]=%d\r\n", &buff[3], buff[3]);  printf("&buff[4]=0x%x buff[4]=%d\r\n", &buff[4], buff[4]);  p1 = &buff[0]; // 指针p1指向数组的第一个元素  // 使用指针遍历数组  printf("p1=0x%x *p1=%d\r\n", p1, *p1);  p1++;  printf("p1=0x%x *p1=%d\r\n", p1, *p1);  p1++;  printf("p1=0x%x *p1=%d\r\n", p1, *p1);  p1++;  printf("p1=0x%x *p1=%d\r\n", p1, *p1);  p1++;  printf("p1=0x%x *p1=%d\r\n", p1, *p1);  printf("\r\n");  return 0;  
}

在这里插入图片描述
在这里插入图片描述

五、二维数组与指针

首先,代码打印出二维数组buff2每个元素的地址和值。
然后,指针p1被初始化为指向数组的第一个元素(&buff2[0][0])。
接着,通过递增指针p1,代码遍历数组的每个元素,并打印出每个元素的地址和值。

#include <stdio.h>  int main() {  unsigned char buff2[2][3] = {{1, 2, 3}, {4, 5, 6}};  unsigned char *p1;  printf("五、二维数组与指针\r\n");  p1 = &buff2[0][0];  // 打印数组每个元素的地址和值  printf("&buff2[0][0]=0x%x buff2[0][0]=%d\r\n", &buff2[0][0], buff2[0][0]);  printf("&buff2[0][1]=0x%x buff2[0][1]=%d\r\n", &buff2[0][1], buff2[0][1]);  printf("&buff2[0][2]=0x%x buff2[0][2]=%d\r\n", &buff2[0][2], buff2[0][2]);  printf("&buff2[1][0]=0x%x buff2[1][0]=%d\r\n", &buff2[1][0], buff2[1][0]);  printf("&buff2[1][1]=0x%x buff2[1][1]=%d\r\n", &buff2[1][1], buff2[1][1]);  printf("&buff2[1][2]=0x%x buff2[1][2]=%d\r\n", &buff2[1][2], buff2[1][2]);  printf("\r\n");  // 使用指针遍历数组  printf("p1=0x%x *p1=%d\r\n", p1, *p1);  p1++;  printf("p1=0x%x *p1=%d\r\n", p1, *p1);  p1++;  printf("p1=0x%x *p1=%d\r\n", p1, *p1);  p1++;  printf("p1=0x%x *p1=%d\r\n", p1, *p1);  p1++;  printf("p1=0x%x *p1=%d\r\n", p1, *p1);  p1++;  printf("p1=0x%x *p1=%d\r\n", p1, *p1);  printf("\r\n");  return 0;  
}

在这里插入图片描述
在这里插入图片描述

在这里插入图片描述

六、指向指针的指针

指针变量分为存储地址和指向地址

a的初始值是*a,是10,存储地址是&a,是0x200005ac
p1的存储地址是&p1,是0x200005bc;指向地址是p1,是0x200005ac,是a的存储地址
p3的存储地址是&p3,是0x200005cc;指向地址是p3,是0x200005bc,是p1的存储地址
*p3是取p3的值,取0x200005bc这个地址的值,而这个地址的值是一个地址,是0x200005ac,是a的存储地址
**p3是取*p3的值,即取0x200005ac这个地址的值,是10

输出结果解释:

&a、&p1、&p3分别打印出变量a、指针p1、指针p3的内存地址。
a=10给变量a赋值。
p1=&a使指针p1指向变量a。
p3=&p1使指针p3指向指针p1。
接下来的几行打印出变量和指针的地址及它们所指向的值,展示了如何通过指针和指向指针的指针来访问和操作变量。

#include <stdio.h>  int main() {  unsigned char a;  unsigned char *p1;  unsigned char **p3;  printf("六、指向指针的指针变量\r\n");  printf("&a=%p\r\n", (void*)&a); // 打印变量a的地址  printf("&p1=%p\r\n", (void*)&p1); // 打印指针p1的地址  printf("&p3=%p\r\n", (void*)&p3); // 打印指针p3的地址  printf("\r\n");  a = 10;  p1 = &a;  p3 = &p1;  printf("&a=0x%x a=%d\r\n", &a, a); // 打印变量a的地址和值  printf("&p1=0x%x p1=0x%x\r\n", &p1, p1); // 打印指针p1的地址和值(即变量a的地址)  printf("&p3=0x%x p3=0x%x\r\n", &p3, p3); // 打印指针p3的地址和值(即指针p1的地址)  printf("*p1=%d\r\n", *p1); // 打印指针p1指向的值(即变量a的值)  printf("*p3=0x%x\r\n", *p3); // 打印指针p3指向的值(即指针p1的值,也就是变量a的地址)  printf("**p3=%d\r\n", **p3); // 打印指针p3指向的指针所指向的值(即变量a的值)  return 0;  
}

在这里插入图片描述
在这里插入图片描述

七、指针变量作为函数形参

代码解释:
#include <stdio.h>:包含标准输入输出库,以便使用printf函数。
void TEST7(unsigned char *p4):定义了一个函数TEST7,它接收一个指向unsigned char类型的指针p4作为参数。
在TEST7函数内部,*p4 = 5;语句将指针p4指向的值修改为5。
在main函数中,首先声明了一个unsigned char类型的变量a,并初始化为1。
TEST7(&a);调用TEST7函数,并将变量a的地址作为参数传递。这样,TEST7函数内的修改会影响到main函数中的a变量。
printf(“a=%d\r\n”, a);打印变量a的值,此时a的值已经被TEST7函数修改为5。
这段代码演示了如何通过指针参数在函数间共享和修改数据。

#include <stdio.h>  void TEST7(unsigned char *p4)  
{  *p4 = 5;  
}  int main() {  unsigned char a;  printf("七、指针变量作为函数参数 \r\n");  a = 1;  TEST7(&a);  printf("a=%d\r\n", a);  printf("\r\n");  return 0;  
}

在这里插入图片描述

八、函数指针

定义函数指针:

unsigned char (*func)(unsigned char,unsigned char);

这行代码定义了一个函数指针func,它指向一个接受两个unsigned char类型参数并返回一个unsigned char类型结果的函数。
定义函数:

unsigned char TEST8(unsigned char v1,unsigned char v2)  
{  return (v1+v2);  
}

这行代码定义了一个函数TEST8,它接受两个unsigned char类型的参数v1和v2,并返回它们的和。

使用函数指针调用函数:

func = TEST8;  
a = (*func)(1,2);

首先,将函数TEST8的地址赋给函数指针func。然后,通过函数指针func调用函数TEST8,并传递参数1和2。函数的结果被赋值给变量a。

打印结果:

printf("a=%d\r\n",a);

这行代码打印变量a的值,即函数TEST8的返回值。

#include <stdio.h>  unsigned char (*func)(unsigned char,unsigned char);  unsigned char TEST8(unsigned char v1,unsigned char v2)  
{  return (v1+v2);  
}  int main()  
{  unsigned char a;  printf("函数指针示例\r\n");  func = TEST8;  a = (*func)(1,2);  printf("a=%d\r\n",a);  printf("\r\n");  return 0;  
}

在这里插入图片描述

九、函数指针数组

定义函数指针数组:

void (*TEST9[3])()={func1,func2,func3};

这行代码定义了一个名为TEST9的函数指针数组,它包含3个元素,每个元素都是一个指向无参数、无返回值(void类型)函数的指针。数组在定义时就被初始化为包含func1、func2和func3这三个函数的地址。

定义函数:

void func1() { printf("func1 \r\n"); }  
void func2() { printf("func2 \r\n"); }  
void func3() { printf("func3 \r\n"); }

这三行代码定义了三个无参数、无返回值的函数func1、func2和func3,它们分别打印出各自的名称。

使用函数指针数组调用函数:

TEST9[0]();  
TEST9[1]();  
TEST9[2]();

这三行代码通过函数指针数组TEST9来调用func1、func2和func3这三个函数。数组的每个元素都是一个函数指针,通过在该指针后面加上()来调用对应的函数。

打印提示信息:

printf("¾Å¡¢º¯ÊýÖ¸ÕëÊý×é \r\n");

这行代码打印出一条提示信息,表明接下来将演示函数指针数组的使用。注意,这里的文本“¾Å¡¢”可能是某种特定编码下的字符,它在正常的ASCII或UTF-8编码下不会显示为预期的文本。在实际编程中,应使用可打印的ASCII或UTF-8字符来编写代码和注释。

#include <stdio.h>  void func1() { printf("func1 \r\n"); }  
void func2() { printf("func2 \r\n"); }  
void func3() { printf("func3 \r\n"); }  void (*TEST9[3])() = {func1, func2, func3};  int main() {  printf("函数指针数组示例 \r\n");  TEST9[0]();  TEST9[1]();  TEST9[2]();  return 0;  
}

在这里插入图片描述

十、参考文献

位、字节、16进制
c语言指针用法及实际应用详解,通俗易懂超详细!
VSCode中C语言程序输出时,控制台出现中文乱码的问题
VSCode编译C语言代码

总结

本文仅仅简单介绍了【C语言】十六进制、二进制、字节、位验证,评论区欢迎讨论。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/415034.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

高经费打造的史诗级视觉盛宴,惊叹于每一帧的奢华

8月29日&#xff0c;备受期待的《指环王&#xff1a;力量之戒》第二季终于上线了。这一季一上架就放出了三集&#xff0c;立刻引发了影迷们的热烈讨论。 自从2022年首季首播以来&#xff0c;《指环王&#xff1a;力量之戒》就一直备受瞩目。尽管首季受到了不少争议&#xff0c;…

【C++ Primer Plus习题】9.4

问题: 解答: main.cpp #include <iostream> #include "sales.h" using namespace std; using namespace SALES;int main() {Sales s1, s2;double de[QUARTERS] { 12.1,32.1,42.1,51.1 };setSales(s1, de, QUARTERS);showSales(s1);cout << endl;setSal…

springsecurity快速入门

Spring Security 是一个功能强大且高度可定制的安全框架&#xff0c;主要用于保护基于 Spring 的应用程序。它提供了一整套用于身份验证、授权、加密、会话管理等功能的工具和 API&#xff0c;从而帮助开发者快速、有效地保护应用程序。 Configuration EnableWebSecurity pu…

Hive 安装

目录 Hive 安装 Hive 安装地址 Hive 安装部署 安装 Hive 启动并使用 Hive Hive 安装 Hive 安装地址 1&#xff09;Hive 官网地址 Apache Hivehttp://hive.apache.org/ 2&#xff09;文档查看地址 GettingStarted - Apache Hive - Apache Software Foundationhttps://cwik…

“转移阻抗”?求你们不要再玩新梗了!

高速先生成员--黄刚 在SI这个行业待久了&#xff0c;Chris发现其实也蛮卷的&#xff0c;就好像前几周写的电容滤板半径这篇文章&#xff0c;最近一些和Chris很熟的网友也评论说&#xff1a;现在好好做设计&#xff0c;好好做仿真都不行啦&#xff1f;一定要发明一些听起来很高…

科研绘图系列:R语言多组极坐标图(grouped polar plot)

介绍 Polar plot(极坐标图)是一种二维图表,它使用极坐标系统来表示数据,而不是像笛卡尔坐标系(直角坐标系)那样使用x和y坐标。在极坐标图中,每个数据点由一个角度(极角)和一个半径(极径)来确定。角度通常从水平线(或图表的某个固定参考方向)开始测量,而半径则是…

VMS-WS-485使用简明教程(485通信类型变送器)

VMS-WS-485使用简明教程 该文章仅供参考&#xff0c;编写人不对任何实验设备、人员及测量结果负责&#xff01;&#xff01;&#xff01; 文章主要介绍485通信类型变送器的硬件连接、软件配置、数据读写以温湿度计算 1 硬件连接 2 软件配置 将变送器硬件部分正确连接后&…

物联网之云平台架构

一&#xff0c;一个典型的物联网云平台 一个典型的物联网&#xff08;IoT&#xff09;云平台需要实现多个功能&#xff0c;以支持物联网设备的接入、数据处理、设备管理、实时控制等需求。 &#xff08;一&#xff09;核心功能 1&#xff0c;设备接入与管理&#xff1a; - 设…

PE文件结构详解(非常详细)

最近在参考OpenShell为任务栏设置图片背景时&#xff0c;发现里面使用了IAT Hook&#xff0c;这一块没有接触过&#xff0c;去查资料的时候发现IAT Hook需要对PE文件结构有一定的了解&#xff0c;索性将PE文件结构的资料找出来&#xff0c;系统学习一下。 PE文件结构 Portable…

Java中类的成员介绍

我的后端学习大纲 我的Java学习大纲 4.类的成员&#xff1a; 3.1.类的成员 -> 属性介绍&#xff08;成员变量&#xff09;&#xff1a; a.语法格式&#xff1a; 1.修饰符 数据类型 属性名 初始化值 ;2.说明1: 修饰符 常用的权限修饰符有&#xff1a;private、缺省、prot…

【论文】A Collaborative Transfer Learning Framework for Cross-domain Recommendation

Intro 业界常见的跨域建模方案主要分为两种范式[22][32][5][36][17][14][20]&#xff1a;1) 将源样本和目标样本进行联合和混合&#xff0c;然后执行多任务学习技术&#xff0c;以提高在所有域中的性能&#xff1b;2) 使用混合或数据丰富的源域数据预先训练模型&#xff0c;然…

等保 2.0 Linux主机测评

以下结果以CentOS 7 为例&#xff0c;按照等保2.0标准&#xff0c;2021报告模板&#xff0c;三级系统要求进行测评。 一、身份鉴别 a)应对登录的用户进行身份标识和鉴别&#xff0c;身份标识具有唯一性&#xff0c;身份鉴别信息具有复杂度要求并定期更换。 输入 more /etc/s…

三元里等你!融合三个经典模型!Transformer-LSTM-SVM多变量时间序列预测(Matlab)

三元里等你&#xff01;融合三个经典模型&#xff01;Transformer-LSTM-SVM多变量时间序列预测&#xff08;Matlab&#xff09; 目录 三元里等你&#xff01;融合三个经典模型&#xff01;Transformer-LSTM-SVM多变量时间序列预测&#xff08;Matlab&#xff09;效果一览基本介…

4个方法快速恢复电脑删除文件

随着电脑的普及&#xff0c;我们越来越多地将重要数据存储在电脑中。然而&#xff0c;数据丢失的风险也随之增加。当您意外删除文件、格式化硬盘或遇到系统崩溃等情况时&#xff0c;如何恢复丢失的数据成为了一个待解决的问题。 一、回收站恢复 首先&#xff0c;最简单的恢复方…

深入探索批处理中的变量与命令:从基础到高级

更多内容前往&#xff1a;孔乙己大叔 在Windows环境中&#xff0c;批处理&#xff08;Batch&#xff09;文件是一种非常有用的脚本工具&#xff0c;允许用户自动化重复性任务。通过编写批处理脚本&#xff0c;用户可以执行一系列命令&#xff0c;而无需手动输入每个命令。变量是…

vue 动态替换父组件

替换父组件&#xff1f;&#xff1f; 什么鬼&#xff1f;&#xff1f;&#xff1f; 这个场景的确很少见&#xff01;&#xff01;不过我们要说的的确是要替换父组件&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&#xff01; 就是子组件内容不变但是父组件变…

node.js使用express框架实现api接口开发(从零开始,超简单可直接复制)

目录 一、效果图 二、实现 1、引入express框架依赖 2、 新建启动文件&#xff08;/server/index.js&#xff09; 3、新建接口函数文件&#xff08;/server/router.js&#xff09; 一、效果图 二、实现 1、引入express框架依赖 在项目文件夹根目录下&#xff0c;打开控制台…

算法题:单词接龙

单词接龙 字典 wordList 中从单词 beginWord 到 endWord 的 转换序列 是一个按下述规格形成的序列 beginWord -> s1 -> s2 -> … -> sk&#xff1a; 每一对相邻的单词只差一个字母。 对于 1 < i < k 时&#xff0c;每个 si 都在 wordList 中。注意&#xf…

RuoYi-Cloud 部署与配置 [CentOS7]

静态IP设置 # 修改网卡配置文件 vim /etc/sysconfig/network-scripts/ifcfg-ens33# 修改文件内容 TYPEEthernet PROXY_METHODnone BROWSER_ONLYno BOOTPROTOstatic IPADDR192.168.18.130 NETMASK255.255.255.0 GATEWAY192.168.18.2 DEFROUTEyes IPV4_FAILURE_FATALno IPV6INIT…

SpringBoot中基于MongoDB的findAndModify原子操作实现分布式锁原理详解

❃博主首页 &#xff1a; 「码到三十五」 &#xff0c;同名公众号 :「码到三十五」&#xff0c;wx号 : 「liwu0213」 ☠博主专栏 &#xff1a; <mysql高手> <elasticsearch高手> <源码解读> <java核心> <面试攻关> ♝博主的话 &#xff1a…