AVR单片机教程——随机点亮LED

本文隶属于AVR单片机教程系列。

之前我们做的闪烁LED和流水灯,灯效都是循环的。这次我们来尝试一些不一样的——每一次随机选择一个LED并点亮。

要实现随机的效果,我们要用C语言标准库中的相关设施:

1 #define RAND_MAX /*implementation defined*/
2 int rand();
3 void srand(unsigned seed);

以上设施都定义在 <stdlib.h> 中。其中,rand() 可以返回[0, RAND_MAX ]范围内的伪随机整数,srand() 用于给 rand() 提供种子,当种子相同时,多次调用 rand() 得到的序列是相同的,这就是为什么称 rand() 产生的数为“伪随机数”。如果使用 rand() 之前没有调用过 srand() ,则相当于调用过 srand(1) 。

利用这些工具,很容易就能写出一个随机LED的程序:

 1 #include <ee1/led.h>
 2 #include <ee1/delay.h>
 3 
 4 #include <stdlib.h>
 5 
 6 int main()
 7 {
 8     led_init();
 9     // srand(1);
10     while (1)
11     {
12         led_set(rand() % 4, LED_ON);
13         delay(200);
14         led_off();
15     }
16 }

rand() 返回[0, RAND_MAX ]范围内的整数,但 led_set 的第一个参数只有在 [0, 3] 范围内才有效,因此我们把 rand() 的返回值对4取模。

srand(1) 被打上注释,是因为这行调用没有必要。

把这段代码编译并烧写进单片机,你会发现LED闪烁的时间是不等长的,这是因为可能存在连续两次亮相同灯的情况。为了解决这个问题,我们引入一个变量,保存当前亮的LED,并让下一个亮的LED与当前的不同。代码如下:

 1 #include <ee1/led.h>
 2 #include <ee1/delay.h>
 3 
 4 #include <stdint.h>
 5 #include <stdlib.h>
 6 
 7 int main()
 8 {
 9     led_init();
10     // srand(0);
11     uint8_t cur = rand() % 4;
12     while (1)
13     {
14         led_set(cur, LED_ON);
15         delay(200);
16         uint8_t next = rand() % 3;
17         if (next >= cur)
18             next++;
19         led_set(cur, LED_OFF);
20         cur = next;
21     }
22 }

使连续两次不亮相同灯的核心代码是16~18行。程序生成一个[0, 2]范围内的随机值,3种取值概率相等,然后当此值大于或等于当前亮灯值时,让它自增。假设当前亮灯为1,则生成的随机数在值为0、1、2的情况下分别变成(映射为)0、2、3,因此下一次亮灯就是在当前没有亮的3个灯中等概率地选择一个。

按开发板上的RESET键可以让单片机复位。观察LED序列,你会发现对于每一次复位,LED序列都是一样的。这个问题我们暂时无法解决。

今天的作业:一个更复杂的随机效果,每次亮1~2个灯,连续两次不能有相同的灯亮,也不能都亮2个,总体来看亮2个的概率为1/3。

这里有一个hex文件,是作业的一个实现,以及一个.c源文件,把单片机程序的main函数复制到文件最后,用计算机的C编译器编译运行可以检查算法是否正确。一个正确的结果应该跟这个差不多:

原文地址:https://www.cnblogs.com/jerry-fuyi/p/11331642.html