蓝桥杯之单片机学习(十九)——IIC协议(PCF8591、AT24C02)

一、EEPROM、AD/DA电路

AD/DA电路:光敏电阻、电压分压
EEOROM:保存数据——数据写进和读出
在这里插入图片描述
在这里插入图片描述

二、IIC协议介绍

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

三、PCF8591、AT24C02编程

3.1 PCF8591

在这里插入图片描述
先写在读

  1. 告诉ADC读哪个通道
  2. 然后接收回来
    在这里插入图片描述

3.2 AT24C02

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

四、练习

  1. 编程实现PCF8591读取通道1和通道3的电压值:
  2. 编程实现AT24C02的读写, 让0x55单元的数值每次上电自加1。

五、代码展示

#include "reg52.h"
#include "intrins.h"
#include "iic.h"
typedef unsigned char BYTE;
typedef unsigned int WORD;

unsigned char adc_table[3];
unsigned char volt_table[3];
unsigned char open_num_table[2];
//-----------------------------------------------

/* define constants */
#define FOSC 11059200L

#define T1MS (65536-FOSC/12/1000)   //1ms timer calculation method in 12T mode

/* define SFR */
sbit TEST_LED = P1^0;               //work LED, flash once per second

/* define variables */
WORD count;                         //1000 times counter

//-----------------------------------------------
unsigned char code T_display[]={0x3F,0x06,0x5B,0x4F,0x66,0x6D,0x7D,0x07,0x7F,0x6F};
unsigned char code T_COM[]={0x01,0x02,0x04,0x08,0x10,0x20,0x40,0x80}; 

/* Timer0 interrupt routine */
void tm0_isr() interrupt 1 using 1
{
		static char i;
    TL0 = T1MS;                     //reload timer0 low byte
    TH0 = T1MS >> 8;                //reload timer0 high byte
    if (count-- == 0)               //1ms * 1000 -> 1s
    {
					count = 1;               //reset counter
       		P2=0xe0;P0=~open_num_table[i];P2=0;
					P2=0xc0;P0=T_COM[i];P2=0;
					i++;
					if(i==2) i=0;
    }
}

//-----------------------------------------------



void Delay2ms()		//@11.0592MHz
{
	unsigned char i, j;

	_nop_();
	i = 4;
	j = 146;
	do
	{
		while (--j);
	} while (--i);
}

/* main program */
void main()
{
		unsigned char adc_val;
	  unsigned char open_num=0;
		float volt;
	
		open_num =read_24c02(0x55);
		write_24c02(0x55,++open_num);
		open_num_table[0]=T_display[open_num/10];
		open_num_table[1]=T_display[open_num%10];
    TMOD = 0x01;                    //set timer0 as mode1 (16-bit)
    TL0 = T1MS;                     //initial timer0 low byte
    TH0 = T1MS >> 8;                //initial timer0 high byte
    TR0 = 1;                        //timer0 start running
    ET0 = 1;                        //enable timer0 interrupt
    EA = 1;                         //open global interrupt switch
    count = 0;                      //initial counter
	
	write_adc(0x03); // ch3
	while(1)
	{
		EA=0;
		adc_val=read_adc(0x03);
		EA=1;
		volt=adc_val/255.0f*5;
		adc_table[0]=adc_val/100;//bai
		adc_table[1]=adc_val/10%10;//shi
		adc_table[2]=adc_val%10;
		
		volt_table[0]=T_display[(unsigned char)volt] | 0x80;
		volt_table[1]=T_display[(unsigned char)(volt*10)%10];
		volt_table[2]=T_display[(unsigned char)(volt*100)%10];
		Delay2ms();
	}
}

一些事项

PCF8591的通道1是光敏电阻,通道3是ADC转换
24C02与EEPROM可以掉电保存

  • 3
    点赞
  • 14
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
以下是基于STM32F103的IIC控制三个AT24C02的程序设计,其中每个AT24C02都有一个唯一的IIC地址。 ```c #include "stm32f10x.h" #include "stm32f10x_i2c.h" #define I2C1_SLAVE_ADDRESS_1 0xA0 #define I2C1_SLAVE_ADDRESS_2 0xA2 #define I2C1_SLAVE_ADDRESS_3 0xA4 void I2C1_Init(void); int main(void) { uint8_t data_write[2] = {0x00, 0x55}; uint8_t data_read[2]; I2C1_Init(); // 写入第一个AT24C02 I2C_GenerateSTART(I2C1, ENABLE); while (!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_MODE_SELECT)); I2C_Send7bitAddress(I2C1, I2C1_SLAVE_ADDRESS_1, I2C_Direction_Transmitter); while (!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_TRANSMITTER_MODE_SELECTED)); I2C_SendData(I2C1, data_write[0]); while (!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_BYTE_TRANSMITTED)); I2C_SendData(I2C1, data_write[1]); while (!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_BYTE_TRANSMITTED)); I2C_GenerateSTOP(I2C1, ENABLE); // 读取第二个AT24C02 I2C_GenerateSTART(I2C1, ENABLE); while (!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_MODE_SELECT)); I2C_Send7bitAddress(I2C1, I2C1_SLAVE_ADDRESS_2, I2C_Direction_Transmitter); while (!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_TRANSMITTER_MODE_SELECTED)); I2C_SendData(I2C1, data_write[0]); while (!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_BYTE_TRANSMITTED)); I2C_GenerateSTOP(I2C1, ENABLE); I2C_GenerateSTART(I2C1, ENABLE); while (!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_MODE_SELECT)); I2C_Send7bitAddress(I2C1, I2C1_SLAVE_ADDRESS_2, I2C_Direction_Receiver); while (!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_RECEIVER_MODE_SELECTED)); while (!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_BYTE_RECEIVED)); data_read[0] = I2C_ReceiveData(I2C1); I2C_GenerateSTOP(I2C1, ENABLE); // 写入第三个AT24C02 I2C_GenerateSTART(I2C1, ENABLE); while (!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_MODE_SELECT)); I2C_Send7bitAddress(I2C1, I2C1_SLAVE_ADDRESS_3, I2C_Direction_Transmitter); while (!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_TRANSMITTER_MODE_SELECTED)); I2C_SendData(I2C1, data_write[0]); while (!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_BYTE_TRANSMITTED)); I2C_SendData(I2C1, data_write[1]); while (!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_BYTE_TRANSMITTED)); I2C_GenerateSTOP(I2C1, ENABLE); while (1); } void I2C1_Init(void) { GPIO_InitTypeDef GPIO_InitStructure; I2C_InitTypeDef I2C_InitStructure; RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE); RCC_APB1PeriphClockCmd(RCC_APB1Periph_I2C1, ENABLE); GPIO_InitStructure.GPIO_Pin = GPIO_Pin_6 | GPIO_Pin_7; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_OD; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_Init(GPIOB, &GPIO_InitStructure); I2C_InitStructure.I2C_Mode = I2C_Mode_I2C; I2C_InitStructure.I2C_DutyCycle = I2C_DutyCycle_2; I2C_InitStructure.I2C_OwnAddress1 = 0x00; I2C_InitStructure.I2C_Ack = I2C_Ack_Enable; I2C_InitStructure.I2C_AcknowledgedAddress = I2C_AcknowledgedAddress_7bit; I2C_InitStructure.I2C_ClockSpeed = 400000; I2C_Init(I2C1, &I2C_InitStructure); I2C_Cmd(I2C1, ENABLE); } ``` 注意,这里的程序是演示如何写入和读取数据,实际应用中需要根据具体的需求进行修改。同时,需要在AT24C02上设置唯一的IIC地址,以便STM32F103可以正确地控制它们。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

周末不下雨

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值