Arduino U8glib 数字闹钟及温度计(不需时钟模块)

前言

最近想用Arduino制作一个数字闹钟,但因为没有独立的时钟模块造成时间误差太大让我困扰,但还是想了一个办法,让程序自动校正时间,于是不需要时钟模块,时间也能相当准确,效果如下图:

材料

名称 数量
Arduino UNO 一个
LCD 12864显示屏 一个
LM35 热敏电阻 一个
3W小喇叭 一个
面包版 一个
跳线 数根

校正时间的方法

以准确时间为标准,测量一段时间内的误差(比如测量3小时后的误差),计算每秒钟的误差毫秒数,取整后然后每秒进行校正,由于取整,仍然会有误差,再将误差扩大60倍,也就是每分钟的误差,取整后每分钟再进行校正,以此类推,直到你认为精确度足够。

Arduino代码

#include <U8glib.h>

#define BUZZER 10//喇叭针脚

U8GLIB_ST7920_128X64_4X u8g(13, 12, 11);//配置屏幕针脚

int base = 0, ms = 0, s = 0, mi = 48, hr = 19, week = 6;//进行初始化时间
int nMi = 0, nHr = 8;//闹钟时间
double tem = 0;//温度

void setup() {
  pinMode(BUZZER, OUTPUT);
  u8g.setRot180();//旋转屏幕180度
  base = millis();//初始化基准时间
}

void loop() {
  calc();//更新时间
  func();//处理函数
  u8g.firstPage(); do draw(); while (u8g.nextPage());//刷新屏幕
}

void func() {
  tem = (double)analogRead(A0) * (5 / 10.23);//读取并计算温度
  if (hr == nHr && mi == nMi) {//播放闹钟1分钟
    tone(BUZZER, 200 + ms);//随毫秒数改变闹钟音调,产生铃声
  } else {
    noTone(BUZZER);//结束播放铃声
    digitalWrite(BUZZER, LOW);//设置为低电平
  }
}

void calc() {
  ms = millis() - base;
  if (ms > 999) {
    ms = 0;
    base = millis() - 61; //用于校正误差,每过1s提前61ms
    if (++s > 59) {
      s = 0;
      base = millis() + 8; //用于校正误差,每过1min延迟8ms
      if (++mi > 59) {
        mi = 0;
        base = millis() - 38; //用于校正误差,每过1hour提前38ms
        if (++hr > 23) {
          hr = 0;
          base = millis() - 8; //用于校正误差,每过1day提前8ms
          if (++week > 6) week = 0;
        }
      }
    }
  }
}

void draw() {
  for (int x = 0; x < 2 * s; x++)//按照秒数绘制宽4个像素的进度条
    for (int y = 0; y < 4; y++)
      u8g.drawPixel(x + 5, y + 3);

  u8g.setFont(u8g_font_gdb30n);//设置gdb30像素数字类字体

  if (hr > 9) {//如果数字只有一位就添加0在前面
    u8g.setPrintPos( 8, 41);
  } else {
    u8g.drawStr(10, 41, "0");
    u8g.setPrintPos( 32, 41);
  }
  u8g.print(hr);

  if (s % 2 == 0)//冒号在秒数为偶数时才绘制,达到冒号闪烁的效果
    u8g.drawStr( 56, 37, ":");

  if (mi > 9) {//如果数字只有一位就添加0在前面
    u8g.setPrintPos( 70, 41);
  } else {
    u8g.drawStr(70, 41, "0");
    u8g.setPrintPos( 94, 41);
  }
  u8g.print(mi);

  u8g.setFont(u8g_font_gdb14r);//设置gdb14像素常用类字体
  switch (week) {//绘制星期数到屏幕
    case 0: u8g.drawStr(6, 61, "Sun"); break;
    case 1: u8g.drawStr(6, 61, "Mon"); break;
    case 2: u8g.drawStr(6, 61, "Tue"); break;
    case 3: u8g.drawStr(6, 61, "Wed"); break;
    case 4: u8g.drawStr(6, 61, "Thu"); break;
    case 5: u8g.drawStr(6, 61, "Fri"); break;
    default: u8g.drawStr(6, 61, "Sat");
  }

  u8g.setPrintPos( 56, 61);//绘制温度示数到屏幕
  u8g.print(tem);
  u8g.drawStr(111, 61, "C");
}

可动态调整时间的Arduino代码

使用一个电位器和一个按钮进行调整时间,电位器选择模式或调整数字,按钮进行选定,具体请从代码中理解。

#include <U8glib.h>

#define BUZZER 10//喇叭针脚
#define YJ 7//额外的控制硬件

U8GLIB_ST7920_128X64_4X u8g(13, 12, 11);//配置屏幕针脚

int base = 0, ms = 0, s = 55 , mi = 59, hr = 5, week = 6;//进行初始化时间
int nMi = 0, nHr =  6;//闹钟时间
double tem = 0;//温度
char mode = 0;//模式

void setup() {
  pinMode(BUZZER, OUTPUT);
  pinMode(YJ, OUTPUT);
  u8g.setRot180();//旋转屏幕180度
  base = millis();//初始化基准时间
}

void loop() {
  func();//处理函数
  u8g.firstPage(); do draw(); while (u8g.nextPage());//刷新屏幕
}

void func() {
  if (map(analogRead(A2), 0, 1010, 0, 1) == 1) {
    switch (mode) {
      case 0: calc(); break; //更新时间
      case 1: hr = map(analogRead(A1), 0, 1020, 0, 23); break;//设置小时
      case 2: mi = map(analogRead(A1), 0, 1020, 0, 59); break;//设置分钟
      case 3: s = map(analogRead(A1), 0, 1020, 0, 59); break;//设置秒
      case 4: week = map(analogRead(A1), 0, 1020, 0, 6); break;//设置星期
      case 5: nHr = map(analogRead(A1), 0, 1020, 0, 23); break;//设置闹钟小时
      case 6: nMi = map(analogRead(A1), 0, 1020, 0, 59); break;//设置闹钟分钟
    }
  } else {
    mode = map(analogRead(A1), 0, 1010, 0, 6);//读取模式
    calc();//更新时间
  }

  tem = (double)analogRead(A0) * (5 / 10.23);//读取并计算温度
  if (hr == nHr && mi == nMi) {//播放闹钟1分钟
    digitalWrite(YJ, LOW);
    tone(BUZZER, 200 + ms);//随毫秒数改变闹钟音调,产生铃声
  } else {
    digitalWrite(YJ, HIGH);
    noTone(BUZZER);//结束播放铃声
    digitalWrite(BUZZER, LOW);//设置为低电平
  }
}

void calc() {
  ms = millis() - base;
  if (ms > 999) {
    ms = 0;
    base = millis() - 61; //用于校正误差,每过1s提前61ms
    if (++s > 59) {
      s = 0;
      base = millis() + 8; //用于校正误差,每过1min延迟8ms
      if (++mi > 59) {
        mi = 0;
        base = millis() - 38; //用于校正误差,每过1hour提前38ms
        if (++hr > 23) {
          hr = 0;
          base = millis() - 8; //用于校正误差,每过1day提前8ms
          if (++week > 6) week = 0;
        }
      }
    }
  }
}

void draw() {
  switch (mode) {
    case 0: {
        for (int x = 0; x < 2 * s; x++)//按照秒数绘制宽4个像素的进度条
          for (int y = 0; y < 4; y++)
            u8g.drawPixel(x + 5, y + 3);

        u8g.setFont(u8g_font_gdb30n);//设置gdb30像素数字类字体

        if (hr > 9) {//如果数字只有一位就添加0在前面
          u8g.setPrintPos( 8, 41);
        } else {
          u8g.drawStr(10, 41, "0");
          u8g.setPrintPos( 32, 41);
        }
        u8g.print(hr);

        if (s % 2 == 0)//冒号在秒数为偶数时才绘制,达到冒号闪烁的效果
          u8g.drawStr( 56, 37, ":");

        if (mi > 9) {//如果数字只有一位就添加0在前面
          u8g.setPrintPos( 70, 41);
        } else {
          u8g.drawStr(70, 41, "0");
          u8g.setPrintPos( 94, 41);
        }
        u8g.print(mi);

        u8g.setFont(u8g_font_gdb14r);//设置gdb14像素常用类字体
        switch (week) {//绘制星期数到屏幕
          case 0: u8g.drawStr(6, 61, "Sun"); break;
          case 1: u8g.drawStr(6, 61, "Mon"); break;
          case 2: u8g.drawStr(6, 61, "Tue"); break;
          case 3: u8g.drawStr(6, 61, "Wed"); break;
          case 4: u8g.drawStr(6, 61, "Thu"); break;
          case 5: u8g.drawStr(6, 61, "Fri"); break;
          default: u8g.drawStr(6, 61, "Sat");
        }

        u8g.setPrintPos( 56, 61);//绘制温度示数到屏幕
        u8g.print(tem);
        u8g.drawStr(111, 61, "C");
        break;
      }
    case 1: {
        u8g.setFont(u8g_font_gdb30n);//设置gdb30像素数字类字体
        if (hr > 9) {//如果数字只有一位就添加0在前面
          u8g.setPrintPos( 8, 41);
        } else {
          u8g.drawStr(10, 41, "0");
          u8g.setPrintPos( 32, 41);
        }
        u8g.print(hr);
        break;
      }
    case 2: {
        u8g.setFont(u8g_font_gdb30n);//设置gdb30像素数字类字体
        if (mi > 9) {//如果数字只有一位就添加0在前面
          u8g.setPrintPos( 70, 41);
        } else {
          u8g.drawStr(70, 41, "0");
          u8g.setPrintPos( 94, 41);
        }
        u8g.print(mi);
        break;
      }
    case 3: {
        for (int x = 0; x < 2 * s; x++)//按照秒数绘制宽4个像素的进度条
          for (int y = 0; y < 4; y++)
            u8g.drawPixel(x + 5, y + 3);

        u8g.setFont(u8g_font_gdb30n);//设置gdb30像素数字类字体
        if (s % 2 == 0)//冒号在秒数为偶数时才绘制,达到冒号闪烁的效果
          u8g.drawStr( 56, 37, ":");
        break;
      }
    case 4: {
        u8g.setFont(u8g_font_gdb14r);//设置gdb14像素常用类字体
        switch (week) {//绘制星期数到屏幕
          case 0: u8g.drawStr(6, 61, "Sun"); break;
          case 1: u8g.drawStr(6, 61, "Mon"); break;
          case 2: u8g.drawStr(6, 61, "Tue"); break;
          case 3: u8g.drawStr(6, 61, "Wed"); break;
          case 4: u8g.drawStr(6, 61, "Thu"); break;
          case 5: u8g.drawStr(6, 61, "Fri"); break;
          default: u8g.drawStr(6, 61, "Sat");
        }
        break;
      }
    case 5: {
        u8g.setFont(u8g_font_gdb30n);//设置gdb30像素数字类字体
        if (nHr > 9) {//如果数字只有一位就添加0在前面
          u8g.setPrintPos( 8, 41);
        } else {
          u8g.drawStr(10, 41, "0");
          u8g.setPrintPos( 32, 41);
        }
        u8g.print(nHr);
        break;
      }
    case 6: {
        u8g.setFont(u8g_font_gdb30n);//设置gdb30像素数字类字体
        if (nMi > 9) {//如果数字只有一位就添加0在前面
          u8g.setPrintPos( 70, 41);
        } else {
          u8g.drawStr(70, 41, "0");
          u8g.setPrintPos( 94, 41);
        }
        u8g.print(nMi);
        break;
      }
  }
}

注意

不同的机器和环境可能有不同的误差,建议自己进行校正后进行使用。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,744评论 6 502
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,505评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 163,105评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,242评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,269评论 6 389
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,215评论 1 299
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,096评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,939评论 0 274
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,354评论 1 311
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,573评论 2 333
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,745评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,448评论 5 344
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,048评论 3 327
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,683评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,838评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,776评论 2 369
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,652评论 2 354

推荐阅读更多精彩内容