上篇文章我们学了关于定时器的三大组成部分及许多寄存器的概念问题,这篇文章我们就要开始讲解实操部分。
首先,我们先来看看本文最后写成的代码:


以上三张是代码的主函数,此外,代码中还需用到的独立按键检测代码在下面:


注意:头文件中#ifndef和#define后面必须跟一个空格!
看以上代码,我们来进行逐步讲解
配置TMOD
首先,我们先在主函数内部(循环外面)写一个定时器0的初始化函数_Timer0_Init(),我们先来配置TMOD定时器模式寄存器,TCON定时器控制寄存器以及定时器中断寄存器,如下图:
1.首先,我们先来配置TMOD,要想让定时器工作在定时器0模式且工作模式为模式1,那么,我们就需要把TMOD中的高四位全部置0,低四位中的最后一位置1,这就配置完成了。因为TMOD是不可位寻址,所以只能整体赋值,当然,我们可以简单粗暴的直接这样子写:TMOD=0X01,这样表示的就是0000 0001,很轻松的就达到了我们的目的,但是,这样子赋值的坏处就是无论高四位原本是什么值,最后都得被改变,为了不影响高四位原本的值,我们可以使用江科大老师教的另一种方法:或与是赋值法:原理就是,任何一个数,与上一个1都等于它本身,与上0都等于0,任何一个数,或上1都等于1,或上0都等于它本身
我们来看TMOD,要想实现配置完成而不影响高四位,我们可以先将TMOD&=0XF0,这个代码的意思是保持高四位不变,将低四位清零,接着我们再将TMOD|=0x01,这个代码的意思是,保持高四位不变,将最低为赋值为1,这样就很好的避免了在配置低四位的过程中,高四位被改变的情况。
配置TCON
2.TCON部分:接着,我们来讲计数系统,我们此型号(STC89C52)的单片机中,定时器0所使用的计数寄存器是两个8位寄存器连在一起的16位寄存器,最大值是65535,溢出后置0,无法自动重装(下一节串口所使用的寄存器是8位自动重装,和这个有区别),所以要想实现1s的定时器,需要给计数寄存器赋初值,那么,我们如何赋初值呢?
首先,我们想要给16位寄存器赋值,就需要将其分开,变为原来的两个8位寄存器分别赋值,其中TH0是高八位寄存器,TL0是低八位寄存器,我们要先给高8位赋初值,由于我们的目标是定一个1s的中断,所以,我们可以将计数系统溢出的时间设置为1ms一次,只要溢出1000次,便是1s,
因此我们只需要把计数寄存器的初值赋值为64535,就可以实现每1ms溢出一次,如何配置呢?
这里我们直接给答案:TH0=65535/256,TL0=65535%256,这样就可以将计数寄存器的初值赋值在64535,但是我们代码中给的确实TH0=0XFC,TL0=0X18,其实,这是系统软件算出来的代码,上面所写的是我们人为手算的吗,准确的来说,系统软件给出的代码比我们人为手算的精度高了1微秒,我们在TL0后面加上1,就可以实现和系统软件代码一样的精度。这里我们还是选择直接使用系统软件生成的代码。
3.当计数寄存器配置完成后,我们还需要看计数系统的下半部分,就是定时器0是否开启的开关TR0。由于我们在TMOD中直接给GATE赋值为0,所以定时器0的开关直接由TR0决定,所以TR0赋值1,并且将中断标志位TF0置0.至此,我们定时器中三大部分已经配置完成两部分。最后一部分便是中断系统。
配置中断系统
4.中断系统我们只需要找到T0,也就是定时器0所对应的中断,打开T0中断允许位ET0,再打开EA(总使能,相当于总开关),最后的中断优先级PT0系统默认为0,直接配置即可。
特别的,中断系统使用时还有特定的方式,如下图:

我们要用的是定时器0的中断,所以使用interrupt 1:即可
以上便是定时器0所需要配置的所有内容,接下来我们开始讲解独立按键控制led移位
1.首先,独立按键的头文件以及.c文件都在上面的图片中,独立按键控制led的原理如下:
独立按键原理:
独立按键的函数我们定义为keynum(),这个函数是有返回值的,因此我们需要再创建一个变量key来接收keynum()函数的返回值,接下来我们进入keynum()函数的内部,在内部,我们定义了一个变量password,这个用作返回值,是key接收的值。当按下P3_1按键时,我们先进行按键消抖和判断松手,在确认按下按键时,password赋值为1,剩下三个按键按下时以此类推,分别赋值为2,3,4。同时,password的初值要赋值为0,这样确保了在没按下按键时返回值也是0。写完代码后,将password return辉主函数由key接收。key接收了password后,要进行判断, 根据返回值的不同来判断按下了哪个按键,这些就是独立按键的原理。
主函数内部
独立按键的原理讲完后,我们就要来使用它,本实验只用到第一个独立按键,也就是P3_1这个按键,在主函数的while循环内,我们根据key的值来判断按键是否按下,所以直接用key当作判断条件if(key),如果按键按下,则key为1,进入条件内部,条件内部我们又定义了一个变量ledmode,按键每按下一次,ledmode++,它的初值是0,当ledmode>=2时,ledmode重新被赋值为0,这就确保了ledmode在0和1之间来回切换,两个数值对应两种状态,分别是led的左移和右移(接下来要讲的中断内部的内容),这样,我们主函数内部也配置好了,最后剩下一个中断。
2.中断内容,由前面我们讲过的定时器0的配置,我们给计数寄存器赋的初值是64535,每次溢出只需要1ms,所以每隔1ms,程序便会进入一次中断,这时如果我们想配置一个每隔0.5秒发生一次的程序,我们只需要进入500次中断即可,因此,我们定义了一个局部变量static unsigned int count(加static的原因是,count是一个局部变量,每次离开中断就会被销毁,下次进来时初值还是0,因此,static防止了count被销毁,保留了其初值,这也为后面累计500次中断做铺垫)。当中断满500次时,进入if(count>=500),进入后,一定要先将count置0,为下一次进入if做准备。当count置0以后,我们来判断ledmode的值,如果ledmode=0,(这里我们已经通过配置将P2_0点亮)我们就进行循环左移操作:用法如下:P2=_crol_(P2,1).如果ledmode=1,我们就进行循环右移操作,用法如下:P2=_cror_(P2,1),通过按下按键来改变ledmode的值,我们就实现了led流水灯左右变向的操作。
(_crol_ , _cror_的头文件是 #include<intrins.h>)
至此,我们独立按键控制led移位(定时器版)便完成了,实现了每隔0.5s led向左或向右移动一个位置,接下来,我们要实现定时器时钟。
定时器时钟
有了上个代码的基础,我们实现定时器时钟便是一件很轻松的事情,首先我们先看代码实现,如下图:

原理:我们要先定义sec(秒),min(分),hour(时)三个变量,再定一个1ms的中断,中断执行1000次,就是1s,sec++,当sec>=60时,先将sec清零后min++,当min>=60时,先将min清零,hour++,当hour>=24时,时钟达到计时上限,hour清零,全部重新开始计时。这便是定时器时钟的原理。另外,我们还要使用到lcd1602,时,分,秒在屏幕上如何输出使用lcd1602专门的代码即可,这里不再多说。
至此,本文两个项目的代码完成,感谢观看。
我在我的Rails项目中使用Pow和powifygem。现在我尝试升级我的ruby版本(从1.9.3到2.0.0,我使用RVM)当我切换ruby版本、安装所有gem依赖项时,我通过运行railss并访问localhost:3000确保该应用程序正常运行以前,我通过使用pow访问http://my_app.dev来浏览我的应用程序。升级后,由于错误Bundler::RubyVersionMismatch:YourRubyversionis1.9.3,butyourGemfilespecified2.0.0,此url不起作用我尝试过的:重新创建pow应用程序重启pow服务器更新战俘
我正在尝试修改当前依赖于定义为activeresource的gem:s.add_dependency"activeresource","~>3.0"为了让gem与Rails4一起工作,我需要扩展依赖关系以与activeresource的版本3或4一起工作。我不想简单地添加以下内容,因为它可能会在以后引起问题:s.add_dependency"activeresource",">=3.0"有没有办法指定可接受版本的列表?~>3.0还是~>4.0? 最佳答案 根据thedocumentation,如果你想要3到4之间的所有版本,你可以这
如果我使用ruby版本2.5.1和Rails版本2.3.18会怎样?我有基于rails2.3.18和ruby1.9.2p320构建的rails应用程序,我只想升级ruby的版本,而不是rails,这可能吗?我必须面对哪些挑战? 最佳答案 GitHub维护apublicfork它有针对旧Rails版本的分支,有各种变化,它们一直在运行。有一段时间,他们在较新的Ruby版本上运行较旧的Rails版本,而不是最初支持的版本,因此您可能会发现一些关于需要向后移植的有用提示。不过,他们现在已经有几年没有使用2.3了,所以充其量只能让更
我安装了ruby版本管理器,并将RVM安装的ruby实现设置为默认值,这样'哪个ruby'显示'~/.rvm/ruby-1.8.6-p383/bin/ruby'但是当我在emacs中打开inf-ruby缓冲区时,它使用安装在/usr/bin中的ruby。有没有办法让emacs像shell一样尊重ruby的路径?谢谢! 最佳答案 我创建了一个emacs扩展来将rvm集成到emacs中。如果您有兴趣,可以在这里获取:http://github.com/senny/rvm.el
有人知道在发布新版本的Ruby和Rails时收到电子邮件的方法吗?他们有邮件列表,RubyonRails有一个推特,但我不想听到那些随之而来的喧嚣,我只想知道什么时候发布新版本,尤其是那些有安全修复的版本。 最佳答案 从therailsblog获取提要.http://weblog.rubyonrails.org/feed/atom.xml 关于ruby-on-rails-如何在发布新的Ruby或Rails版本时收到通知?,我们在StackOverflow上找到一个类似的问题:
在应用开发中,有时候我们需要获取系统的设备信息,用于数据上报和行为分析。那在鸿蒙系统中,我们应该怎么去获取设备的系统信息呢,比如说获取手机的系统版本号、手机的制造商、手机型号等数据。1、获取方式这里分为两种情况,一种是设备信息的获取,一种是系统信息的获取。1.1、获取设备信息获取设备信息,鸿蒙的SDK包为我们提供了DeviceInfo类,通过该类的一些静态方法,可以获取设备信息,DeviceInfo类的包路径为:ohos.system.DeviceInfo.具体的方法如下:ModifierandTypeMethodDescriptionstatic StringgetAbiList()Obt
说在前面这部分我本来是合为一篇来写的,因为目的是一样的,都是通过独立按键来控制LED闪灭本质上是起到开关的作用,即调用函数和中断函数。但是写一篇太累了,我还是决定分为两篇写,这篇是调用函数篇。在本篇中你主要看到这些东西!!!1.调用函数的方法(主要讲语法和格式)2.独立按键如何控制LED亮灭3.程序中的一些细节(软件消抖等)1.调用函数的方法思路还是比较清晰地,就是通过按下按键来控制LED闪灭,即每按下一次,LED取反一次。重要的是,把按键与LED联系在一起。我打算用K1来作为开关,看了一下开发板原理图,K1连接的是单片机的P31口,当按下K1时,P31是与GND相连的,也就是说,当我按下去时
如果我一直输入geminstallrails使用不同版本的Rails会怎样?例如,我可以输入:geminstallrails--verson3.2.10或geminstallrails这给了我版本3.2.12。问题每次安装都会覆盖之前的吗?它会删除所有旧文件并添加我正在安装的新版本吗?或者如果我运行它两次,它会保留一些文件吗?我正在使用Ubuntu。 最佳答案 它将安装两个独立的gem。实际的可执行文件rails将调用最新版本。你可以覆盖它__例如,rails_3.2.10_将执行Rails3.2.10。bundler顺便说一下,如
我的Rails站点使用了一个确实不是很好的gem。每次我需要做一些新的事情时,我最终不得不花费与向实际Rails项目添加代码一样多的时间来为gem添加功能。但我不介意,我将我的Gemfile设置为指向我的gem的GitHub分支(我尝试提交PR,但维护者似乎已经下台)。问题是我真的没有找到一种合理的方法来测试我添加到gem的新东西。在railsc中测试它会特别好,但我能想到的唯一方法是a)更改~/.rvm/gems/.../foo。rb,这看起来不对或者b)升级版本,推送到Github,然后运行bundleup,这除了耗时之外显然是一场灾难,因为我不确定我所做的promise是否正
我正在寻找一个用ruby演示计时器的在线示例,并发现了下面的代码。它按预期工作,但这个简单的程序使用30Mo内存(如Windows任务管理器中所示)和太多CPU有意义吗?非常感谢deftime_blockstart_time=Time.nowThread.new{yield}Time.now-start_timeenddefrepeat_every(seconds)whiletruedotime_spent=time_block{yield}#Tohandle-vesleepinteravalsleep(seconds-time_spent)iftime_spent