51单片机基础-IO扩展(并转串 74HC165)

第二十二章 IO扩展(并转串 74HC165)

1. 导入

上一章用 74HC595 实现"串→并"输出来扩展LED等。本章介绍与之互补的"并→串"输入扩展芯片 74HC165。它能把多路并行输入(如开关、按键、光耦量)用极少的MCU引脚串行读回,常与 74HC595 搭配实现"少线控多I/O"。

目标:

  • 理解 74HC165 引脚与装载/移位时序。
  • 连接单片机,读取 8/16/32 路输入。
  • 实现消抖与稳定扫描,打印或控制逻辑。
  • 了解多片级联、与 SPI 的配合技巧。

2. 硬件设计

2.1 74HC165 引脚速览(并转串)

  • D0~D7:8路并行输入(通常接按键/开关,建议上拉或下拉确定默认电平)
  • SH/LD(或 /PL):并行装载,低有效
  • CLK(CP):移位时钟,上升沿移位
  • CLK INH(CE):时钟禁止,高电平禁止移位,常接地(0)以使能
  • Q7:串行输出(接MCU输入)
  • Q7':串行输出的反相信号,同时用于多片级联的串行输入(接下一片的SER)

常用接法建议:

  • 多数输入为"按下=低",则每路输入加上拉电阻(到VCC,10kΩ左右),按键另一端接GND。
  • CLK INH 直接接GND(启用时钟);若要"冻结"移位可接MCU控制。
  • 多片级联:前一片 Q7' 接后一片 SER;CP、SH/LD 并联。

2.2 与 51 单片机连接示例

以 P1.0~P1.2 控制:

  • P1.0 ← Q7(数据输入,MCU读)
  • P1.1 → CLK(时钟输出)
  • P1.2 → SH_LD(并装,低有效)
  • CLK INH → GND(常使能)
  • D0~D7 → 8个按键(到GND),每路10k上拉至VCC

两片级联(16路):

  • 第1片 Q7' → 第2片 SER
  • 时钟、装载线并联到两片
  • 读取时得到16位数据,先出的是"最后一级"的最高位(注意位序,代码处理)

3. 时序与读取流程

  • 并行装载:保持 CLK=0,将 SH/LD 拉低≥t_w,芯片把 D0~D7 锁存到移位寄存器。
  • 移位输出:将 SH/LD 拉高,之后每个 CLK 上升沿把数据向 Q7 移一位。
  • 读法建议(稳妥顺序):
    1. CLK=0
    2. SH/LD=0(装载)→ 短延时 → SH/LD=1
    3. 循环8次:先读 Q7,再 CLK 上升沿→下降沿,进入下一位。

说明:首次读取的 Q7 对应 D7(常见版本),随后依次 D6...D0。


4. 软件实现(C51)

4.1 引脚与基础延时

c 复制代码
#include <reg52.h>
#include <intrins.h>

sbit HC165_DATA = P1^0;   // Q7 → MCU输入
sbit HC165_CLK  = P1^1;   // CP 时钟
sbit HC165_LD   = P1^2;   // SH/LD(低有效并装)

static void tiny_delay(void) { _nop_(); _nop_(); _nop_(); _nop_(); }

void delay_ms(unsigned int ms){
    unsigned int i,j;
    for(i=0;i<ms;i++) for(j=0;j<125;j++);
}

4.2 读8位(1片 74HC165)

c 复制代码
// 读取1片(8位),返回bit7..bit0(bit7是D7)
unsigned char hc165_read8(void)
{
    unsigned char i, val = 0;

    HC165_CLK = 0;         // 保证时钟低
    HC165_LD  = 0;         // 并装
    tiny_delay();
    HC165_LD  = 1;         // 转换为移位模式
    tiny_delay();

    for (i = 0; i < 8; i++) {
        // 先读当前Q7
        val <<= 1;
        if (HC165_DATA) val |= 0x01;

        // 再移位到下一位
        HC165_CLK = 1; tiny_delay();
        HC165_CLK = 0; tiny_delay();
    }
    return val;
}

4.3 读16位(2片级联)

c 复制代码
// 读取2片级联(16位):高字节为"后级"那片
unsigned int hc165_read16(void)
{
    unsigned char i;
    unsigned int val = 0;

    HC165_CLK = 0;
    HC165_LD  = 0; tiny_delay();  // 并装两片
    HC165_LD  = 1; tiny_delay();

    for (i = 0; i < 16; i++) {
        val <<= 1;
        if (HC165_DATA) val |= 0x0001;
        HC165_CLK = 1; tiny_delay();
        HC165_CLK = 0; tiny_delay();
    }
    return val; // bit15..bit0
}

说明:

  • 具体"哪片是高字节"取决于Q7'链路方向与布局,若与你的板卡相反,交换高低字节或在循环内倒序组装即可。
  • 若你的按键为"按下=低电平",读回位为0表示被按,后续逻辑可按需取反。

4.4 消抖与稳定检测

简易两次一致法:

c 复制代码
// 读取并简单消抖:两次一致才认定
unsigned char hc165_read8_debounced(void)
{
    unsigned char a = hc165_read8();
    delay_ms(5);
    unsigned char b = hc165_read8();
    return (a == b) ? a : 0xFF;  // 0xFF表示抖动/不稳定(按下=低时)
}

更稳妥可读N次做"多数票"或状态机消抖(按需扩展)。


5. 应用示例

5.1 示例A:8按键 → 串口打印键值(按下为低)

c 复制代码
// 串口初始化(9600bps)
void uart_init(void){
    TMOD |= 0x20; TH1=0xFD; TL1=0xFD; TR1=1;
    SCON = 0x50; EA=1; ES=0;
}
void uart_putc(char c){ SBUF=c; while(!TI); TI=0; }
void uart_puts(const char* s){ while(*s) uart_putc(*s++); }
void uart_put_hex8(unsigned char v){
    const char hx[]="0123456789ABCDEF";
    uart_putc(hx[(v>>4)&0xF]); uart_putc(hx[v&0xF]);
}

void main(){
    unsigned char last = 0xFF; // 默认全高(未按)
    uart_init();
    uart_puts("74HC165 Key Scan Start\r\n");

    while(1){
        unsigned char cur = hc165_read8_debounced(); // 低=按下
        if (cur != 0xFF && cur != last) {
            uart_puts("KEYS=0x"); uart_put_hex8(cur); uart_puts("\r\n");
            last = cur;
        }
    }
}
  • 若 D0 对应"最右键",当按下它时 KEYS 的最低位变为0。
  • 需要"哪个键被按下"的索引,可遍历每一位检测从1→0的边沿。

5.2 示例B:16按键(2片)→ 亮灭板载LED

假设某位被按下(读到0)则翻转 P1.7 指示灯:

c 复制代码
sbit LED = P1^7;

void main(){
    unsigned int last = 0xFFFF;
    LED = 1;
    while(1){
        unsigned int cur = hc165_read16();
        // 简单消抖
        delay_ms(5);
        if (cur == hc165_read16() && cur != last){
            // 任一位从1→0代表有新按下
            unsigned int changed = (last ^ cur);
            unsigned int pressed = changed & (~cur); // 由1变0
            if (pressed) {
                LED = !LED;
            }
            last = cur;
        }
    }
}

6. 与 74HC595 组合(节省I/O的键盘面板)

  • 用 74HC595 驱动行/列(或LED显示),用 74HC165 读入另一维的键值,MCU端只占 5~6 根线即可实现"显示+按键输入"面板。
  • 扫描流程:通过 595 逐列输出(1列为有效),经 165 读回行线状态;列→行的机械按键仍需消抖。
  • 注意时序:行切换与行稳定之间加短延时;避免显示切换引入的串扰影响读取。

7. SPI方式的小技巧

若MCU具备 SPI 外设:

  • Q7 → MISOCP → SCK(模式0),SH/LD 用GPIO控制,CLK INH 接GND。
  • 读流程:SH/LD=0→1 后,发出 8(或16/24/32)个SCK,并在 SPI 接收缓冲中取回数据,吞掉发送字节(一般发0x00)。
  • 这样可显著减轻软件位操作负担,提高采样速率与稳定性。

8. 常见问题与排查

  • 读值抖动/随机:
    • 输入悬空:务必上拉/下拉;线长用更小阻值或加RC滤波。
    • 时序不当:装载/移位时保证 CLK 在规范状态,读Q7与打时钟的先后顺序一致。
  • 位序与预期相反:
    • 74HC165 默认 Q7 先出 D7,若你希望D0在最低位,代码里倒序装配或硬件交换 D0...D7。
  • 级联顺序混乱:
    • 确保 Q7' → SER 串接方向正确;读多位时对应地高位/低位拼装。
  • 多片干扰:
    • 共地良好、去耦靠近芯片、CLK/SH/LD 线尽量短且加小电阻串联阻尼(如33~100Ω)以抑制过冲。

9. 小结

  • 74HC165 提供"并→串"输入扩展,极大节省 MCU 引脚,适合多按键、多开关回读。
  • 掌握了装载与移位时序、单片/级联读取、基础消抖与应用示例。
  • 与 74HC595 组合可形成"少线多I/O"的整机面板输入输出方案;有 SPI 更佳。

相关推荐
Tony小周3 小时前
使用QKeyEvent keyPress(QEvent::KeyPress, key模拟键盘发送事件,会导致主程序卡死
嵌入式硬件·qt
9527华安4 小时前
全国产化方案实现NVMe over 100G RDMA,解决智算超算中“存算”不匹配问题
fpga开发·nvme·rdma
碎碎思4 小时前
FPGA新闻速览-从漏洞到突破:FPGA技术在安全、架构与量子领域
安全·fpga开发
FPGA_ADDA4 小时前
100%全国产化4路125M FMC子卡
fpga开发·fmc子卡·全国产·4路ad采集·国产ad9653
普中科技5 小时前
【普中STM32F1xx开发攻略--标准库版】-- 第 13 章 STM32 位带操作
stm32·单片机·嵌入式硬件·arm·gpio·普中科技·位带操作
河南博为智能科技有限公司7 小时前
RS485转以太网串口服务器-串口设备联网的理想选择
大数据·服务器·人工智能·单片机·嵌入式硬件·物联网
国科安芯7 小时前
抗辐照MCU芯片在无人叉车领域的性能评估与选型建议
网络·人工智能·单片机·嵌入式硬件·安全
国科安芯8 小时前
抗辐照MCU芯片在激光雷达领域的适配性分析
网络·人工智能·单片机·嵌入式硬件·fpga开发
日更嵌入式的打工仔8 小时前
<RT1176系列14>CCM(Clock Controller Module)解读
单片机·嵌入式硬件