找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 1372|回复: 7
收起左侧

C语言指针的相关问题 程序分析

[复制链接]
回帖奖励 20 黑币 回复本帖可获得 20 黑币奖励! 每人限 1 次
ID:1079270 发表于 2023-8-7 10:14 | 显示全部楼层 |阅读模式


#include <reg52.h>


#define LCD1602_DB  P0
sbit LCD1602_RS = P1^0;
sbit LCD1602_RW = P1^1;
sbit LCD1602_E  = P1^5;


bit flag500ms = 0;   //500ms定时标志
unsigned char T0RH = 0;  //T0重载值的高字节
unsigned char T0RL = 0;  //T0重载值的低字节
//待显示的第一行字符串
unsigned char code str1[] = "Kingst Studio";
//待显示的第二行字符串,需保持与第一行字符串等长,较短的行可用空格补齐
unsigned char code str2[] = "Let's move...";


void ConfigTimer0(unsigned int ms);
void InitLcd1602();
void LcdShowStr(unsigned char x, unsigned char y,
                unsigned char *str, unsigned char len);


void main()
{
    unsigned char i;
    unsigned char index = 0;  //移动索引
    unsigned char pdata bufMove1[16+sizeof(str1)+16]; //移动显示缓冲区1
    unsigned char pdata bufMove2[16+sizeof(str2)+16]; //移动显示缓冲区2


    EA = 1;            //开总中断
    ConfigTimer0(10);  //配置T0定时10ms
    InitLcd1602();     //初始化液晶
    //缓冲区开头一段填充为空格
    for (i=0; i<16; i++)
    {
        bufMove1[ i] = ' ';
        bufMove2[ i] = ' ';
    }
    //待显示字符串拷贝到缓冲区中间位置
    for (i=0; i<(sizeof(str1)-1); i++)
    {
        bufMove1[16+i] = str1[ i];
        bufMove2[16+i] = str2[ i];
    }
    //缓冲区结尾一段也填充为空格
    for (i=(16+sizeof(str1)-1); i<sizeof(bufMove1); i++)
    {
        bufMove1[ i] = ' ';
        bufMove2[ i] = ' ';
    }
   
    while (1)
    {
        if (flag500ms)  //每500ms移动一次屏幕
        {
            flag500ms = 0;
            //从缓冲区抽出需显示的一段字符显示到液晶上
            LcdShowStr(0, 0, bufMove1+index, 16);
            LcdShowStr(0, 1, bufMove2+index, 16);
            //移动索引递增,实现左移
            index++;
            if (index >= (16+sizeof(str1)-1))
            {   //起始位置达到字符串尾部后即返回从头开始
                index = 0;
            }
        }
    }
}
/* 配置并启动T0,ms-T0定时时间 */
void ConfigTimer0(unsigned int ms)
{
    unsigned long tmp;  //临时变量
   
    tmp = 11059200 / 12;      //定时器计数频率
    tmp = (tmp * ms) / 1000;  //计算所需的计数值
    tmp = 65536 - tmp;        //计算定时器重载值
    tmp = tmp + 12;           //补偿中断响应延时造成的误差
    T0RH = (unsigned char)(tmp>>8);  //定时器重载值拆分为高低字节
    T0RL = (unsigned char)tmp;
    TMOD &= 0xF0;   //清零T0的控制位
    TMOD |= 0x01;   //配置T0为模式1
    TH0 = T0RH;     //加载T0重载值
    TL0 = T0RL;
    ET0 = 1;        //使能T0中断
    TR0 = 1;        //启动T0
}
/* 等待液晶准备好 */
void LcdWaitReady()
{
    unsigned char sta;
   
    LCD1602_DB = 0xFF;
    LCD1602_RS = 0;
    LCD1602_RW = 1;
    do {
        LCD1602_E = 1;
        sta = LCD1602_DB; //读取状态字
        LCD1602_E = 0;
    } while (sta & 0x80); //bit7等于1表示液晶正忙,重复检测直到其等于0为止
}
/* 向LCD1602液晶写入一字节命令,cmd-待写入命令值 */
void LcdWriteCmd(unsigned char cmd)
{
    LcdWaitReady();
    LCD1602_RS = 0;
    LCD1602_RW = 0;
    LCD1602_DB = cmd;
    LCD1602_E  = 1;
    LCD1602_E  = 0;
}
/* 向LCD1602液晶写入一字节数据,dat-待写入数据值 */
void LcdWriteDat(unsigned char dat)
{
    LcdWaitReady();
    LCD1602_RS = 1;
    LCD1602_RW = 0;
    LCD1602_DB = dat;
    LCD1602_E  = 1;
    LCD1602_E  = 0;
}
/* 设置显示RAM起始地址,亦即光标位置,(x,y)-对应屏幕上的字符坐标 */
void LcdSetCursor(unsigned char x, unsigned char y)
{
    unsigned char addr;
   
    if (y == 0)  //由输入的屏幕坐标计算显示RAM的地址
        addr = 0x00 + x;  //第一行字符地址从0x00起始
    else
        addr = 0x40 + x;  //第二行字符地址从0x40起始
    LcdWriteCmd(addr | 0x80);  //设置RAM地址
}
/* 在液晶上显示字符串,(x,y)-对应屏幕上的起始坐标,
   str-字符串指针,len-需显示的字符长度 */
void LcdShowStr(unsigned char x, unsigned char y,
                unsigned char *str, unsigned char len)
{
    LcdSetCursor(x, y);   //设置起始地址
    while (len--)         //连续写入len个字符数据
    {
        LcdWriteDat(*str++);  //先取str指向的数据,然后str自加1
    }
}
/* 初始化1602液晶 */
void InitLcd1602()
{
    LcdWriteCmd(0x38);  //16*2显示,5*7点阵,8位数据接口
    LcdWriteCmd(0x0C);  //显示器开,光标关闭
    LcdWriteCmd(0x06);  //文字不动,地址自动+1
    LcdWriteCmd(0x01);  //清屏
}
/* T0中断服务函数,定时500ms */
void InterruptTimer0() interrupt 1
{
    static unsigned char tmr500ms = 0;
   
    TH0 = T0RH;  //重新加载重载值
    TL0 = T0RL;
    tmr500ms++;
    if (tmr500ms >= 50)
    {
        tmr500ms = 0;
        flag500ms = 1;
    }
}
问:bufMove1+index代入void LcdShowStr(unsigned char x, unsigned char y,
                unsigned char *str, unsigned char len)的 LcdWriteDat(*str++)的流程是什么?

回复

使用道具 举报

ID:908826 发表于 2023-8-7 18:41 | 显示全部楼层
先执行间接运算操作,然后指针值加一指向下一个内存地址
回复

使用道具 举报

ID:1079566 发表于 2023-8-7 19:07 | 显示全部楼层
bufMove1+index代入void LcdShowStr(unsigned char x, unsigned char y,
                unsigned char *str, unsigned char len)的 LcdWriteDat(*str++)的流程是什么?
流程: 类似如下
LcdWriteDat( bufMove1[index])
index++ (这里只是展示效果,实际不影响函数外的index)

或:
*str = & bufMove1[index]  // 或 *str= bufMove1 + index
LcdWriteDat(*str)
str++
回复

使用道具 举报

ID:883242 发表于 2023-8-7 21:38 | 显示全部楼层
原来你是不知道下面两句话是完全等效的。
  1. bufMove1+index
复制代码
  1. &bufMove1[index]
复制代码

bufMove1作为一个数组名,本身就是指针常量,+index运算后得到一个指针变量,当然可以当成unsigned char *类型的参数传递进去。
而bufMove1[index],这就不是一个指针了,而是一个unsigned char类型的数据,必须在前面加&才能变成unsigned char *类型的参数传递进去。

评分

参与人数 1黑币 +12 收起 理由
xnysfh + 12 很给力!

查看全部评分

回复

使用道具 举报

ID:844772 发表于 2023-8-8 08:36 | 显示全部楼层
当bufMove1+index代入void LcdShowStr(unsigned char x, unsigned char y, unsigned char *str, unsigned char len)的LcdWriteDat(*str++)时,流程如下:

LcdSetCursor(x, y):设置LCD光标起始地址为(x, y)。
进入while循环,len--:len是要显示的字符数量,每循环一次,len减1。
LcdWriteDat(*str++):从str指向的地址取出一个字符数据,然后str自增1,指向下一个字符。将取出的字符数据写入LCD的数据寄存器,显示在LCD屏幕上。
重复步骤2和步骤3,直到len减到0,即所有字符都显示完毕。
实际上,bufMove1+index代入*LcdShowStr(unsigned char x, unsigned char y, unsigned char *str, unsigned char len)的LcdWriteDat(*str++)时,bufMove1+index的内容会被当作字符串的起始地址,而不是单个字符的地址。因此,函数会显示从bufMove1+index开始的len个字符。

评分

参与人数 1黑币 +12 收起 理由
xnysfh + 12 很给力!

查看全部评分

回复

使用道具 举报

ID:1079270 发表于 2023-8-8 11:25 | 显示全部楼层
Hephaestus 发表于 2023-8-7 21:38
原来你是不知道下面两句话是完全等效的。

非常感谢,没想到我的问题这么抽象还能明白我想表达的意思,不愧是大佬
回复

使用道具 举报

ID:372579 发表于 2023-8-12 09:11 | 显示全部楼层
51大神好多,逛逛就可以学到很多知识
回复

使用道具 举报

ID:385774 发表于 2023-8-14 11:01 | 显示全部楼层
Hephaestus 发表于 2023-8-7 21:38
原来你是不知道下面两句话是完全等效的。

受教了。。。。谢谢
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

手机版|小黑屋|51黑电子论坛 |51黑电子论坛6群 QQ 管理员QQ:125739409;技术交流QQ群281945664

Powered by 单片机教程网

快速回复 返回顶部 返回列表