51单片机与C语言入门全攻略

本文还有配套的精品资源,点击获取 menu-r.4af5f7ec.gif

简介:51单片机是用于教学的经典微控制器,本文详细介绍了51单片机的基础架构、内部组件及其与C语言结合的编程方法。文章将帮助读者从零开始,系统学习单片机编程的各个方面,包括C语言基础、存储器模型、输入输出操作、中断系统、定时器/计数器、串行通信和开发流程等。此外,通过实例应用巩固理论知识,提升实践技能。 51单片机

1. 51单片机基础架构介绍

51单片机作为一款经典的微控制器,广泛应用于嵌入式系统开发中,是学习微控制器原理和应用的良好起点。本章旨在为读者提供51单片机的架构概述,为后续深入学习奠定基础。

1.1 单片机硬件组成概述

51单片机硬件架构包括处理器核心、存储器、输入输出接口、定时器/计数器和中断系统等。这些组件共同协作,使得51单片机能够完成各种控制任务。

1.2 核心处理器特性

核心处理器基于8051架构,包含一个8位的CPU,具有一定的指令集支持,可以进行基本的算术和逻辑操作。其处理器时钟速率、寄存器组和特殊功能寄存器(SFR)是实现各种操作的关键。

1.3 存储器结构简析

51单片机内置ROM和RAM,用于存储程序和数据。其中,程序存储器可存储用于执行的机器码,数据存储器则用于存放程序运行时的临时数据。理解其结构对于编程和系统优化至关重要。

本章通过对51单片机硬件组成和核心特性进行概述,帮助读者建立对单片机硬件基础的认识。接下来的章节将逐步深入到软件编程和具体应用的探讨中。

2. C语言编程基础

2.1 C语言的基本语法

C语言是单片机编程中广泛使用的语言,因为它具有直接控制硬件的能力。要有效地使用C语言为51单片机编程,首先必须掌握它的基本语法。

2.1.1 C语言的语句和表达式

在C语言中,语句是构建程序的基石,而表达式则用于实现计算和赋值等操作。语句以分号结束,可以是声明、赋值、函数调用等。表达式则由运算符和操作数组成,用于产生值。

int a = 5; // 声明并初始化变量a
a = a + 1; // 表达式赋值,a的值变为6

在编写C语言代码时,需要注意运算符的优先级。例如,算术运算符的优先级高于赋值运算符。

int result = 3 + 2 * 5; // result 为 13,先计算乘法,后加法

编写C语言程序时,还应该合理地使用括号来明确运算的先后顺序。

2.1.2 C语言的控制结构

控制结构使程序能够执行条件判断和循环。条件判断中最常用的是if-else结构,而循环结构包括for、while和do-while。

if (condition) {
    // 条件为真时执行的代码
} else {
    // 条件为假时执行的代码
}

控制结构的选择取决于特定情况的需求。if-else适合简单的条件判断,而循环结构则适用于需要重复执行的操作。

2.2 C语言的函数和模块化编程

函数是C语言的核心概念之一,它是完成特定任务的代码块。函数的使用使得代码更加模块化,易于理解和维护。

2.2.1 函数的定义和声明

函数的定义包括返回类型、函数名、参数列表和函数体。

int add(int a, int b) { // 函数定义
    return a + b;       // 返回计算结果
}

int main() {
    int sum = add(3, 4); // 函数声明后调用
    return 0;
}

函数声明是在调用函数之前,对函数的返回类型、名称和参数列表的声明。它告诉编译器该函数的接口信息,使得编译器能够在调用点处检查参数匹配。

2.2.2 模块化编程的优势和实现方式

模块化编程的优势在于将复杂程序分解成较小的、易于管理的模块。每个模块负责一块特定的逻辑,这样可以提高代码的重用性和可维护性。

模块化编程可以通过定义和使用函数来实现。例如,可以定义一个函数用于初始化单片机的某个硬件模块,另一个函数用于处理输入输出操作。

// 初始化单片机硬件模块的函数
void init_hardware() {
    // 初始化代码
}

// 处理输入输出操作的函数
void process_io() {
    // 输入输出处理代码
}

每个函数都应该有一个清晰定义的功能,并且尽量避免在多个函数中重复相同的代码。这样,当需要修改程序的一部分时,开发者只需要关注一个模块,而不会影响到其他部分。

以上内容是第二章的核心部分,提供了C语言编程基础的概述,包括了语法、函数和模块化编程的基本知识。接下来的章节将继续深入探讨C语言的高级特性以及与51单片机编程更紧密相关的内容。

3. 存储器模型的理解

3.1 51单片机的内存结构

3.1.1 内存的分类和特点

在51单片机中,内存主要分为内部RAM、外部RAM、程序存储器(ROM)和特殊功能寄存器。内部RAM用于存储变量和临时数据,其大小和地址范围由51单片机的型号决定。内部RAM通常分为两个部分:工作寄存器区和用户RAM区。

工作寄存器区提供了快速的存储访问,被用于频繁使用的变量和寄存器。它位于0x00到0x1F的地址范围,共有32个字节,分为4组,每组8个寄存器(R0到R7)。用户RAM区则提供了通用的数据存储空间,其大小因不同的51单片机型号而异,典型的如8051系列拥有128字节的用户RAM区。

外部RAM则是用来扩展单片机的存储容量,它位于地址空间的高端。外部RAM的访问速度相比内部RAM要慢,但可以提供更多的存储空间。

程序存储器(ROM)用于存放单片机的程序代码,通常是只读的。51单片机的ROM可以是掩膜ROM、EPROM或EEPROM,其容量同样因型号而异。

特殊功能寄存器(SFR)是专门用于控制单片机内部各种功能的寄存器。它们有着固定的地址,并且可以影响单片机的工作模式和功能。例如,控制串行通信的SBUF寄存器和控制定时器的TMOD寄存器。

3.1.2 内存的寻址方式

51单片机提供了多种内存寻址方式,以满足不同情况下的内存访问需求:

  • 立即寻址(Immediate Addressing):操作数是常数,直接编码在指令中,例如 MOV A, #0x20
  • 寄存器寻址(Register Addressing):操作数是寄存器中的内容,例如 MOV A, R0
  • 直接寻址(Direct Addressing):通过指定内存地址来访问数据,例如 MOV A, 0x20
  • 间接寻址(Indirect Addressing):通过寄存器或寄存器对中的内容作为地址来访问数据,例如 MOV A, @R0
  • 变址寻址(Indexed Addressing):寄存器的内容加上一个常数作为内存地址,常用于访问数组数据,例如 MOV A, 0x20[R0]
  • 位寻址(Bit Addressing):针对特殊功能寄存器中的特定位进行操作,用于位操作指令,例如 SETB P1.0

这些寻址方式使得51单片机能够灵活地访问和操作内存中的数据,为编程提供了便利。

3.2 存储器的操作和管理

3.2.1 常见的存储器操作函数

在使用51单片机进行编程时,操作存储器往往需要通过特定的函数或指令。常见的内存操作函数包括:

  • MOV 指令:用于数据传输,可以将立即数、寄存器内容、直接寻址或间接寻址的内容传送到目标位置。
  • XCH 指令:用于交换两个寄存器或寄存器与直接地址内容。
  • CLR 指令:清除(将内容置为0)一个位操作的寄存器或位。
  • SETB CLR 指令:用于对位寄存器进行设置和清除操作。

对于更高级的存储器管理操作,可以使用C语言的库函数,如 memset memcpy

#include <string.h>

char buffer[10];
memset(buffer, 0, sizeof(buffer)); // 清零buffer数组

3.2.2 存储器管理的策略和实现

为了有效地管理存储器资源,需要考虑以下几个策略:

  • 内存分配策略:通常使用静态分配和动态分配两种方式。静态分配在编译时确定,而动态分配允许程序在运行时申请和释放内存。
  • 内存访问优化:合理利用内存访问指令,如减少间接寻址的使用,以提高程序的执行效率。
  • 内存保护机制:使用堆栈保护、内存界限检查等技术来避免内存溢出和越界访问的问题。
  • 内存碎片管理:由于动态内存分配可能产生碎片,需要设计合理的内存管理算法来减少内存碎片,例如使用伙伴系统。

实际的存储器管理实现中,开发者应根据应用场景选择合适的策略,并通过编程实现这些策略。例如,嵌入式系统中经常采用静态分配策略来保证系统的实时性和稳定性。

在继续深入分析存储器模型的理解之前,了解51单片机内存结构及其操作方法是基础,对于设计和实现嵌入式应用至关重要。接下来,我们将探讨如何通过代码块、表格和mermaid格式流程图等元素,对存储器的操作和管理策略进行更直观的展示和分析。

4. 输入输出操作实践

4.1 输入输出端口的配置

4.1.1 输入输出端口的特性

在51单片机中,输入输出端口(I/O端口)是单片机与外部设备进行数据交换的重要接口。输入端口用于读取外部设备或传感器的数据,而输出端口则用于控制外围设备,如LED灯、继电器等。输入输出端口具有以下特点:

  1. 双向性 :大多数单片机的I/O端口具有双向操作的能力,既可以作为输入端口读取数据,也可以作为输出端口发送数据。
  2. 高阻态 :某些I/O端口可被配置为高阻态(High-Impedance),这种模式下,端口对电路的影响可以忽略不计,用于实现三态逻辑。
  3. 电气特性 :I/O端口的电气特性要与外部电路相匹配,需要考虑电流、电压等参数。
4.1.2 配置输入输出端口的步骤和示例

配置51单片机的I/O端口需要遵循以下步骤:

  1. 确定I/O端口的引脚 :首先明确需要配置的I/O端口的引脚号。
  2. 设置端口模式 :根据需要将I/O端口配置为输入模式或输出模式。
  3. 编写控制代码 :编写相应的控制代码,根据硬件连接情况和功能需求配置I/O端口。

以Keil C51编译器为例,配置P1.0作为输出端口的示例代码如下:

#include <REGX51.H>

void main() {
  P1 = 0xFF; // 将P1端口的所有引脚设置为高电平
  P1_0 = 0;  // 将P1.0设置为低电平,用作输出

  while(1) {
    // 在此处编写代码,控制P1.0的状态
  }
}

在上述代码中,我们首先包含了51单片机的寄存器定义文件 REGX51.H 。然后在 main 函数中,我们将P1端口的所有位设置为高电平,接着将P1.0设置为低电平,配置为输出状态。在 while 循环中可以添加更多的控制逻辑来实现具体功能。

4.2 输入输出操作的程序实现

4.2.1 如何编写输入输出程序

编写输入输出程序需要深入了解I/O端口的硬件连接和功能需求。以下是一个输入输出程序编写的基本步骤:

  1. 初始化端口 :在程序开始时初始化需要使用的I/O端口。
  2. 读取输入端口数据 :根据需求编写读取外部信号的代码。
  3. 处理数据 :根据读取的数据进行必要的处理。
  4. 输出处理结果 :将处理后的数据输出到外部设备。

以下是一个简单的读取P1端口输入并根据输入控制P2端口输出的示例代码:

#include <REGX51.H>

void main() {
  // 初始化P1为输入端口,P2为输出端口
  P1 = 0xFF; // 设置P1端口为高阻态
  P2 = 0x00; // 设置P2端口为输出0

  while(1) {
    if (P1 == 0xFE) { // 如果P1.0为低电平(外部设备输入)
      P2 = 0xFF;     // 则P2端口输出高电平
    } else {
      P2 = 0x00;     // 否则P2端口输出低电平
    }
  }
}

在这个例子中,我们首先将P1端口初始化为高阻态输入模式,并将P2端口设置为输出模式。在 while 循环中,我们不断检查P1.0的状态,根据该状态控制P2端口输出高或低电平。

4.2.2 输入输出操作的调试和优化

调试和优化输入输出程序需要遵循以下步骤:

  1. 单步执行 :使用调试工具单步执行程序,观察每个I/O端口的状态变化。
  2. 逻辑分析 :分析程序中逻辑判断的正确性,确保逻辑流程符合预期。
  3. 硬件测试 :连接实际硬件设备,检查I/O操作是否按预期工作。
  4. 优化代码 :根据测试结果对代码进行优化,比如减少不必要的操作,提高程序的运行效率。

优化时可以考虑以下几点:

  • 减少延时 :使用硬件定时器代替软件延时,提高程序的响应速度。
  • 代码去抖动 :对于输入信号,可以采用软件或硬件方法去除抖动,避免误判。
  • 模块化编程 :将输入输出操作封装成模块,提高代码的复用性和可维护性。

通过这些步骤,我们能够确保输入输出操作的准确性和程序的高效性。在实际应用中,合理的调试和优化策略对保证系统的稳定运行至关重要。

5. 中断系统的学习

5.1 中断系统的原理和分类

5.1.1 中断系统的概念和功能

在计算机系统中,中断是一种机制,它允许处理器对外部或内部事件做出快速响应,并暂时中止当前运行的程序,转而处理更为紧急的任务。中断可以看作是一种特殊的输入,它来自于CPU外部或内部的异步事件,当这些事件发生时,处理器会暂停当前执行的程序,转而执行一个特定的中断服务程序(ISR)。

中断系统的主要功能有:

  • 实时性处理 :中断响应可以暂停当前程序的执行,允许立即处理紧急事件,如外部设备请求、错误条件等,提高了系统的实时性。
  • 资源共享 :中断机制允许CPU和I/O设备等共享资源,通过中断服务程序协调它们的活动。
  • 任务调度 :在多任务操作系统中,中断可以用于任务调度,根据系统需求和任务优先级切换执行任务。

5.1.2 中断的分类和特性

中断可以根据其来源和功能被分为几种不同类型,不同类型的中断在单片机中具有不同的处理方式和优先级。中断主要分为以下几类:

  • 硬件中断 :来自外部设备的中断信号,如按键输入、串行通信等。
  • 软件中断 :通过执行特定的中断指令(如在x86架构中的INT指令)由软件程序触发的中断。
  • 定时器中断 :由单片机内置的定时器/计数器在计数达到预设值时产生的中断。
  • 异常中断 :指由程序执行中的错误或异常情况触发的中断,如除零错误、内存访问违规等。

中断还具有以下特性:

  • 非可屏蔽中断(NMI) :不能被任何指令或软件操作关闭的中断,如电源故障。
  • 可屏蔽中断 :可以被程序中的特定指令或硬件条件屏蔽的中断。
  • 优先级 :中断请求的处理顺序,通常由硬件或软件指定中断优先级,高优先级的中断可以打断低优先级中断的处理。

5.2 中断的编程实现

5.2.1 中断服务程序的设计和编写

为了处理中断,需要编写中断服务程序。在51单片机中,中断向量地址表规定了每个中断入口的地址。中断服务程序的编写步骤通常包括:

  1. 初始化中断 :设置中断源、开启中断允许寄存器IE中的相应位,以及设置中断优先级。
  2. 编写中断服务程序 :将中断服务程序链接到相应的中断向量地址。
  3. 编写中断返回指令 :在中断服务程序末尾添加 RETI 指令,使CPU能够返回到被中断的程序继续执行。

下面是一个简单的中断服务程序编写示例:

// 中断服务程序示例
void timer0_isr (void) interrupt 1 // 中断号为1,对应定时器0中断
{
    // 处理定时器中断
    // ...
    RETI(); // 中断返回
}

5.2.2 中断的管理策略和实现方法

中断管理策略包括中断嵌套、中断优先级的分配以及中断屏蔽。实现方法涉及硬件配置和软件编写两个方面。

硬件配置 : - 配置中断触发方式(上升沿触发或下降沿触发)。 - 设置中断允许寄存器,控制中断使能。

软件编写 : - 在程序中合理设置中断优先级,确保关键任务的及时响应。 - 使用中断嵌套,允许高优先级中断打断低优先级中断的处理。 - 当需要暂时关闭某个中断时,可以通过修改中断允许寄存器来实现。

中断嵌套的示例代码:

void main()
{
    EA = 1; // 允许全局中断
    EX0 = 1; // 允许外部中断0
    IT0 = 1; // 设置INT0为下降沿触发
    // ...
}

void ext0_isr() interrupt 0
{
    // 处理外部中断0
    // ...
    EX0 = 1; // 在中断服务程序内重新允许外部中断0,实现嵌套
    // ...
}

在以上代码中,首先在主函数中通过设置中断允许位和中断触发方式来配置中断。在外部中断0的中断服务程序中,重新允许了该中断的触发,这样即使在中断服务程序内部也能够响应新的中断请求,实现中断嵌套。

6. 定时器/计数器的应用

6.1 定时器/计数器的工作原理

6.1.1 定时器/计数器的功能和特性

定时器/计数器是单片机中重要的功能模块,它可以用来计时、计量事件发生的次数,或者生成精确的时间延迟。通常51单片机内置有两个定时器/计数器,它们可以工作在不同的模式下,实现不同的功能。

在定时功能下,定时器以一定的速率递增计数,当计数值达到预设值时,产生定时器溢出,通过溢出中断来执行特定任务。在计数功能下,定时器通过对外部事件进行计数,可用于测量外部脉冲的频率或周期。

6.1.2 定时器/计数器的配置和使用

定时器/计数器的配置和使用通常包括以下几个步骤:

  1. 设置定时器模式:根据需要配置定时器工作在定时模式还是计数模式。
  2. 设置定时器初值:根据定时器溢出时间或计数次数,设置定时器的初值。
  3. 启动定时器:通过设置控制寄存器,启动定时器进行计数。
  4. 中断处理:配置中断,实现当定时器溢出时触发中断服务程序,执行相关操作。

具体到51单片机的寄存器,我们需要操作TMOD(定时器模式寄存器)以及TCON(定时器控制寄存器)来配置定时器的工作模式和控制启停。

6.2 定时器/计数器的编程实践

6.2.1 编写定时器/计数器的程序

以下是一个使用定时器0产生定时中断的简单示例代码:

#include <REGX51.H>

// 定时器初始化函数
void Timer0_Init() {
    TMOD &= 0xF0; // 清除定时器0的模式位
    TMOD |= 0x01; // 设置定时器0为模式1(16位定时器)
    TH0 = 0xFC;   // 装载定时初值,这里设置为1ms左右
    TL0 = 0x18;
    ET0 = 1;      // 开启定时器0中断
    TR0 = 1;      // 启动定时器0
}

// 定时器0中断服务程序
void Timer0_ISR() interrupt 1 {
    // 重新装载初值
    TH0 = 0xFC;
    TL0 = 0x18;
    // 用户代码区,定时执行的代码放在这里
}

void main() {
    EA = 1;       // 开启全局中断
    Timer0_Init(); // 初始化定时器0
    while(1) {
        // 主循环中其他代码
    }
}

6.2.2 定时器/计数器的调试和优化

编写完定时器程序后,需要进行调试以确保定时器能够准确地工作。调试过程中,可以利用串口打印或其他调试工具来观察定时器溢出的时间点,确保其符合预期。

在程序运行过程中,如果发现定时器的精确度不够或者效率不高,可能需要对定时器的初值进行调整。此外,如果中断服务程序执行时间较长,可能需要考虑优化中断服务程序,以减少对定时精度的影响。

定时器/计数器的优化往往需要根据实际应用场景进行细致的调整。例如,在需要高精度定时的应用中,可能需要校准晶振频率来提高定时的精确度;在多任务系统中,可能需要使用优先级管理来确保定时器中断的及时响应。

通过以上章节的介绍,我们对定时器/计数器的工作原理和编程应用有了较为全面的了解。下一章节我们将探讨串行通信协议的实践,这在单片机应用中同样重要。

本文还有配套的精品资源,点击获取 menu-r.4af5f7ec.gif

简介:51单片机是用于教学的经典微控制器,本文详细介绍了51单片机的基础架构、内部组件及其与C语言结合的编程方法。文章将帮助读者从零开始,系统学习单片机编程的各个方面,包括C语言基础、存储器模型、输入输出操作、中断系统、定时器/计数器、串行通信和开发流程等。此外,通过实例应用巩固理论知识,提升实践技能。

本文还有配套的精品资源,点击获取 menu-r.4af5f7ec.gif

单片机c语言编程100个实例目录1 函数的使用和熟悉 实例3:用单片机控制第一个灯亮 实例4:用单片机控制一个灯闪烁:认识单片机的工作频率 实例5:将 P1口状态分别送入P0、P2、P3口:认识I/O口的引脚功能 实例6:使用P3口流水点亮8位LED 实例7:通过对P3口地址的操作流水点亮8位LED 实例8:用不同数据类型控制灯闪烁时间 实例9:用P0口、P1 口分别显示加法和减法运算结果 实例10:用P0、P1口显示乘法运算结果 实例11:用P1、P0口显示除法运算结果 实例12:用自增运算控制P0口8位LED流水花样 实例13:用P0口显示逻辑""运算结果 实例14:用P0口显示条件运算结果 实例15:用P0口显示按位"异或"运算结果 实例16:用P0显示左移运算结果 实例17:"万能逻辑电路"实验 实例18:用右移运算流水点亮P1口8位LED 实例19:用if语句控制P0口8位LED的流水方向 实例20:用swtich语句的控制P0口8位LED的点亮状态 实例21:用for语句控制蜂鸣器鸣笛次数 实例22:用while语句控制LED 实例23:用do-while语句控制P0口8位LED流水点亮 实例24:用字符型数组控制P0口8位LED流水点亮 实例25: 用P0口显示字符串常量 实例26:用P0 口显示指针运算结果 实例27:用指针数组控制P0口8位LED流水点亮 实例28:用数组的指针控制P0 口8 位LED流水点亮 实例29:用P0 、P1口显示整型函数返回值 实例30:用有参函数控制P0口8位LED流水速度 实例31:用数组作函数参数控制流水花样 实例32:用指针作函数参数控制P0口8位LED流水点亮 实例33:用函数型指针控制P1口灯花样 实例34:用指针数组作为函数的参数显示多个字符串 单片机c语言编程100个实例目录2 实例35:字符函数ctype.h应用举例 实例36:内部函数intrins.h应用举例 实例37:标准函数stdlib.h应用举例 实例38:字符串函数string.h应用举例 实例39:宏定义应用举例2 实例40:宏定义应用举例2 实例41:宏定义应用举例3 * 中断、定时器中断、定时器 *中断、定时器*中断、定时器 / 实例42:用定时器T0查询方式P2口8位控制LED闪烁 实例43:用定时器T1查询方式控制单片机发出1KHz音频 实例44:将计数器T0计数的结果送P1口8位LED显示 实例45:用定时器T0的中断控制1位LED闪烁 实例46:用定时器T0的中断实现长时间定时 实例47:用定时器T1中断控制两个LED以不同周期闪烁 实例48:用计数器T1的中断控制蜂鸣器发出1KHz音频 实例49:用定时器T0的中断实现"渴望"主题曲的播放 实例50-1:输出50个矩形脉冲 实例50-2:计数器T0统计外部脉冲数 实例51-2:定时器T0的模式2测量正脉冲宽度 实例52:用定时器T0控制输出高低宽度不同的矩形波 实例53:用外中断0的中断方式进行数据采集 实例54-1:输出负脉宽为200微秒的方波 实例54-2:测量负脉冲宽度 实例55:方式0控制流水灯循环点亮 实例56-1:数据发送程序 实例56-2:数据接收程序 实例57-1:数据发送程序 实例57-2:数据接收程序 实例58:单片机向PC发送数据 实例59:单片机接收PC发出的数据 *数码管显示*数码管显示 数码管显示数码管显示*/ 实例60:用LED数码显示数字5 实例61:用LED数码显示器循环显示数字0~9 实例62:用数码管慢速动态扫描显示数字"1234" 实例63:用LED数码显示器伪静态显示数字1234 实例64:用数码管显示动态检测结果 实例65:数码秒表设计 实例66:数码时钟设计 实例67:用LED数码管显示计数器T0的计数值 实例68:静态显示数字“59” 单片机c语言编程100个实例目录3 键盘控制*键盘控制* *键盘控制 *键盘控制 */ 实例69:无软件消抖的独立式键盘输入实验 实例70:软件消抖的独立式键盘输入实验 实例71:CPU控制的独立式键盘扫描实验 实例72:定时器中断控制的独立式键盘扫描实验 实例73:独立式键盘控制的4级变速流水灯 实例74:独立式键盘的按键功能扩展:"以一当四" 实例75:独立式键盘调时的数码时钟实验 实例76:独立式键盘控制步进电机实验 实例77:矩阵式键盘按键值的数码管显示实验 //实例78:矩阵式键盘按键音 实例79:简易电子琴 实例80:矩阵式键盘实现的电子密码锁 液晶显示LCD*液晶显示LCD *液晶显示LCD * *液晶显示LCD*液晶显示LCD *液晶显示LCD */ 实例81:用LCD显示字符'A' 实例82:用LCD循环右移显示"Welcome to China" 实例83:用LCD显示适时检测结果 实例84:液晶时钟设计 *一些芯片的使用*24c02 DS18B20 X5045 ADC0832 DAC0832 DS1302 红外遥控/ 实例85:将数据"0x0f"写入AT24C02再读出送P1口显示 实例86:将按键次数写入AT24C02,再读出并用1602LCD显示 实例87:对I2C总线上挂接多个AT24C02的读写操作 实例88:基于AT24C02的多机通信 读取程序 实例89:基于AT24C02的多机通信 写入程序 实例90:DS18B20温度检测及其液晶显示 实例91:将数据"0xaa"写入X5045再读出送P1口显示 实例92:将流水灯控制码写入X5045并读出送P1口显示 实例93:对SPI总线上挂接多个X5045的读写操作 实例94:基于ADC0832的数字电压表 实例95:用DAC0832产生锯齿波电压 实例96:用P1口显示红外遥控器的按键值 实例97:用红外遥控器控制继电器 实例98:基于DS1302的日历时钟 实例99:单片机数据发送程序 实例100:电机转速表设计 模拟霍尔脉冲
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值