开发者社区> shy丶gril> 正文
阿里云
为了无法计算的价值
打开APP
阿里云APP内打开

写KeilC遇到的坑

简介:
+关注继续查看

1、跟vs中不同,keilC中的的变量必须声明在最前,不然就出错。

2、KeilC中的错误提示功能实在是很弱。没有办法精确到具体是哪个具体的变量或标点符号出问题。

3、KeilC中的unsigned int能表示的数太小,只能有五位数,导致烧到单片机上的时候,数码管显示有问题。

unsigned char,8位,最大值255
unsigned int,16位,最大值65535

4、8针的引脚只要用unsigned char就可以控制,因为unsigned char 是8位的。

5、在LCD1602输出了脏数据。

复制代码
 1 #include<reg51.h>
 2 #define uchar unsigned char
 3 #define uint unsigned int
 4 uchar code table[]="Hello World!";
 5 uchar code table1[]="I am a boy!";
 6 ///此处省略了一系列的函数定义
 7 void main()
 8 {
 9         int num;
10     init();
11     write_command(0x80+0x10);
12     for(num=0;num<20;num++)
13     {
14         write_data(table[num]);
15         delay(5);
16     }
17     //此处省略了一些无关内容
18 }
复制代码

在以上代码段中,我们期望在LCD1602的第一行输出"Hello World!       ",注意这个输出后面是空白的,但是输出了"Hello World!       **** I am a"(***代表乱码)。这就有点无奈了。调了好久,一开始以为是ram里面有暂存的数据,单片机断电后,再试还是不行。后面联想到编译原理中,两个字符串table和table1都是用code定义的数组,他们存储位置是程序存储器。又因为table1定义的位置正好在table的下面,所以当以上代码行12对应处遍历table数组超出table本身的索引时,自然就输出了table1的内容,从而造成脏数据。

 

6、头文件的定义。

①当我们定义头文件的时候,最后不要在头文件里面定义define uchar unsigned char 这类的东西,因为当引用了两次头文件或者不同的头文件里面都有同样的define的时候,程序会出问题。

②当定义头文件的时候,为了避免二次引用造成重复定义,最好在头文件上#ifndef __I2C_H__  #define __I2C_H__ 、#endif(其中__I2C_H__是自己定义的一个名称如以下代码段所示参考链接

 

复制代码
 1 #ifndef __I2C_H__
 2 #define __I2C_H__
 3 
 4 //========函数区============================================
 5 extern void I2C_start();  //开始信号
 6 extern void I2C_stop();   //停止
 7 extern void I2C_respons();  //应答
 8 extern void I2C_write_byte(unsigned char date);
 9 extern unsigned char I2C_read_byte();
10 extern void I2C_write_address(unsigned char address,unsigned char date);
11 extern unsigned char I2C_read_address(unsigned char address);
12 //========函数区结束============================================
13 #endif
复制代码

 

 7、按键反应很慢。

下面的程序是一个按下独立按键,然后在LED屏幕上显示当前计数值的程序。程序要实现:当按下独立按键1的时候,等待。当再次按下,程序往下运行。我发现当我按下,程序可以很快地反应;但是再次按下想让他运行的时候,却要等待大概3~5秒的时间。细看以下代码段加灰的地方和中断函数,终于发现了原因。原来,当忙等待的时候,中断函数int0()中的Count还在不断计数,也就是Count可能大于20,那么就不满足Count==20这个条件。而要等再次满足Count==20,则要等待Count计满重新归零(注意Count是无符号数,无符号数计满的时候自动归零)。

复制代码
#include<reg51.h>
unsigned char Count=0;
/***********************************************************
    重新计数
***********************************************************/
void ReCount(){
    Count=0;  //开始下一轮1s的计时
    TH0=T_High_50ms;// 计算初值,并将其写入TH0、TL0或TH1、TL1。
    TL0=T_Low_50ms;
}
void main()
{
//去掉不必要的初始化函数
while(1){
Waiting();
//一个忙等待函数,当按下独立按键1的时候,等待。当再次按下, //程序往下运行 if(Count==20){//定时器0 1s时间到 ReCount(); //……省略做其他事情的函数 } } } void int0() interrupt 1//定时/计数器T0中断 { Count++; }
复制代码

 8、强制转化的问题

除了强制转化成bit型是取高位外(代码段1),其他从位数少到位数多类型的强制类型转换都是取低位。

复制代码
 1 unsigned char aChar=0x00;//除了强制转化成bit型是取高位外
 2 unsigned int aInt=0xaa55;
 3 aChar=(unsigned char)aInt;
 4 //结果:aChar=0x55
 5 
 6 ///////////////////////////////////////////////////////////////
 7 bit aBit;
 8 unsigned char aChar=0x01; //其他从位数少到位数多类型的强制类型转换都是取低位。
 9 aBit=(bit)aChar;
10 //结果:aBit=0;
复制代码

9、全局变量定义在头文件(*.h)或源文件(*.c)的思考。

看了一些代码,似乎在头文件的定义中都尽量地避免了对全局变量的定义。我想头文件是给人引用的,也就是这部分的内容必须暴露,为了函数的封装性更好,更好的方法就是将全局变量尽可能封装在源文件 (*.c)。这样,使用者不会因为头文件中含有全局变量而感到困惑,也不用因为全局变量而去了解整个C语言函数库的实现细节。

10、C#和Java都是支持函数的重载的,但是C语言中不支持函数的重载。

1 int add(int a,int b);             //函数1
2 int add(int a,int b,int c);       //函数2

C语言中,如果你定义了函数1,那么就不能再定义函数2.不然编译就不通过了。究其原因,可能是因为C语言中有函数指针这东西(见一下代码段,重载会让C编译器不知道将哪个函数地址赋给函数指针)。

复制代码
#include <stdio.h>
int add(int a,int b);
//int add(int a , int b , int c);
int sub(int a,int b);
void result(int (* pf)() , int a , int b);
main(){
   int i,j=1;
   int (* pf)();
   pf=add;        //这里如果重载了add函数,那么这里编译器就不知道要把哪个add
   result(pf,i,j);//赋值给pf
   pf=sub;
   result(pf,i,j);
}

int add(int a , int b){
     return a+b;
}
int sub(int a,int b){
     return a-b;
}
void result(int (*pf)() , int a , int b){
   int value;
   value = (*pf)(a , b);
   printf("%d\t",value);
}
复制代码

 

 


作者:kissazi2 
出处:http://www.cnblogs.com/kissazi2/ 
本文版权归作者所有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,否则保留追究法律责任的权利。

转载:http://www.cnblogs.com/kissazi2/archive/2013/05/25/3098325.html

版权声明:本文内容由阿里云实名注册用户自发贡献,版权归原作者所有,阿里云开发者社区不拥有其著作权,亦不承担相应法律责任。具体规则请查看《阿里云开发者社区用户服务协议》和《阿里云开发者社区知识产权保护指引》。如果您发现本社区中有涉嫌抄袭的内容,填写侵权投诉表单进行举报,一经查实,本社区将立刻删除涉嫌侵权内容。

相关文章
应用 Serverless 化,让业务开发心无旁骛
本文是在 2022 杭州云栖大会上分享的内容。伴随着云产品的全面 Serverless 化,应用架构也发生了很大变化,我们认为应用 Serverless 化是未来的趋势,可以让业务开发更加心无旁骛的聚焦业务创新。
0 0
重磅嘉宾畅聊大数据&AI开源话题,零距离感受激荡开源江湖
「开源人说」第四期——大数据& AI专场在今年云栖大会举办,阿里巴巴开源委员会大数据AI领域副主席王峰和阿里云AI开源项目EasyRec负责人施兴现场分享热门开源项目背后的故事。开源中国创始人&CTO红薯,白鲸开源联合创始人代立冬,浙大博导赵俊博,InfoQ总编辑王一鹏、Apache软件基金会成员李钰等嘉宾圆桌共话,对开源热点及痛点问题展开激烈讨论。
0 0
消灭空指针,Java 8 给我们更好的解决方案
在平时的业务开发中,空指针是我们经常遇到的问题,
0 0
拥抱开源,云原生时代下的开源牧码人的初心与坚守
王峰 阿里巴巴开源委员会大数据AI领域副主席 阿里云开源大数据平台负责人 Flink中文社区发起人
0 0
『简单·易用』 一站式数据管理与服务
市场报告显示,2025年将有75%数据库迁移到云上,将有30%数据是实时数据,到2024年末数据保护法将会保护到 75% 人口。整体趋势可以总结为上云、实时以及安全。 我们认为,库仓一体是未来趋势。因为核心数据从数据库产生,而实现库仓一体化可以降低用户使用成本。
0 0
我们是否对现代前端开发框架过于崇拜了?
前端界有两个“教派”,一个叫 Vue,一个叫 React。Vue 的成员看不起 React,React 成员鄙视 Vue,他们认为手中的“教义”就是真理,可以消灭世界一切苦难。
0 0
使用vue3结合高德实现定位功能
使用vue3结合高德地图实现定位功能
0 0
「开源人说」|大咖齐聚首,大数据&AI开源话题对碰
「开源人说」第四期——大数据& AI专场在今年云栖大会举办,阿里巴巴开源委员会大数据AI领域副主席王峰和阿里云AI开源项目EasyRec负责人施兴现场分享热门开源项目背后的故事。开源中国创始人&CTO红薯,白鲸开源联合创始人代立冬,浙大博导赵俊博,InfoQ总编辑王一鹏、Apache软件基金会成员李钰等嘉宾圆桌共话,对开源热点及痛点问题展开激烈讨论。
0 0
干货!6个方面,32条总结教你提升职场经验
初入职场总是会感觉到迷茫,一身力气没有地方发挥,希望成长过程不要走弯路,本文从成长的捷径、功夫在日常、学会被管理、思维转换、要栓住情绪、成为 Leader 六个方面,总结了 32 条职场经验
0 0
+关注
文章
问答
文章排行榜
最热
最新
相关电子书
更多
低代码开发师(初级)实战教程
立即下载
阿里巴巴DevOps 最佳实践手册
立即下载
冬季实战营第三期:MySQL数据库进阶实战
立即下载