資料介紹
描述
如果你有一只物聯網寵物,它會吃什么?WiFi SSID,當然!
Nerd 是一種無線電子寵物,它通過收集 WiFi SSID 以及一些休息和陽光來生存。
為了讓它茁壯成長,你必須平衡離線和在線模式與光明和黑暗,以確保它有一個適當的日常吃-睡-書呆子程序。
如果沒有 WiFi 太久,它將使用其內置的壓電揚聲器以摩爾斯電碼發送 SOS。離線時間越長,蜂鳴聲越多。
簡而言之
書呆子將每半小時醒來一次以掃描周圍的網絡。如果它檢測到新網絡,它將存儲它們并在低功耗模式下重新進入睡眠狀態(以節省電池壽命)。否則它會用蜂鳴器發出噪音來抱怨,直到你喂它或把它放在黑暗中。
它還會通過連接到您的 wifi 網絡了解何時在家。在家時,Nerd 將能夠連接到互聯網并獲取當前時間和日期。
如果超過兩天不喂,它會急劇死亡,發出很大的噪音。
成分
- RGB LED
- 蜂鳴器
- 電池
- 220歐姆電阻
學習目標
- 管理完整的 WiFi 功能
- 在閃存中存儲數據
- 管理時間和實時時鐘
- 管理低功耗模式
想知道更多?
本教程是讓您熟悉 MKR1000 和 IoT 的一系列實驗的一部分。所有實驗都可以使用 MKR IoT Bundle 中包含的組件構建。
- 書呆子
設立董事會
為了實現所有功能,我們將使用以下庫:
- WiFi101 //連接到互聯網并掃描網絡
- FlashStorage // 保存值,以便它們不會在每次重新啟動時被擦除
- RTCZero // 管理時間觸發事件
- ArduinoLowPower //節省電池電量
- WiFiUdp // 從互聯網上獲取時間和日期
掃描 WiFi 網絡
書呆子渴望網絡!
掃描網絡非常簡單, 只需上傳此示例草圖或轉到> 示例 > WiFi101 > ScanNetworksAdvanced以獲得更多擴展版本。
#include
#include
void setup() {
//Initialize serial and wait for port to open:
Serial.begin(9600);
while (!Serial) {
; // wait for serial port to connect. Needed for native USB port only
}
// scan for existing networks:
Serial.println();
Serial.println("Scanning available networks...");
listNetworks();
}
void loop() {
delay(10000);
// scan for existing networks:
Serial.println("Scanning available networks...");
listNetworks();
}
void listNetworks() {
// scan for nearby networks:
Serial.println("** Scan Networks **");
int numSsid = WiFi.scanNetworks();
if (numSsid == -1)
{
Serial.println("Couldn't get a WiFi connection");
while (true);
}
// print the list of networks seen:
Serial.print("number of available networks: ");
Serial.println(numSsid);
// print the network number and name for each network found:
for (int thisNet = 0; thisNet < numSsid; thisNet++) {
Serial.print(thisNet + 1);
Serial.print(" SSID: ");
Serial.println(WiFi.SSID(thisNet));
Serial.flush();
}
Serial.println();
}
在閃存中存儲值
當然,您不希望書呆子每次沒電時就死掉!
為了避免這種行為,我們將在閃存中保存一些變量(作為食物量),以便即使在電路板關閉并再次打開后也可以檢索它們。
您可以使用以下方法了解基本功能:
示例 > FlashStorage > FlashStoreAndRetrieve
我們將保存網絡的名稱,以便書呆子只吃一次,我們還將使用保存這些 SSID 的數組來計算它在白天吃的食物量。
保存在閃存中的值將在電路板重置后保留,但不會在新草圖的上傳后保留。每次上傳新草圖時,閃存也會被清空。
此草圖將掃描網絡并將 SSID 保存在閃存中。
#include
#include
#include
#define MAGIC_NUMBER 0x7423 // arbitrary number to double check the saved SSID
#define MaxNet 30 // max amount of network to be saved
int PosToBeSaved = 0; // Variable used to navigate the array of networks
int daily_amount_of_food = 12; // The amount of food per day needed to survive
// Struct of variable to be saved in flash memory
typedef struct {
int magic;
boolean valid[MaxNet];
char SSIDs[MaxNet][100];
} Networks;
FlashStorage(my_flash_store, Networks);
Networks values;
void setup() {
Serial.begin(115200);
while(!Serial); // wait until the Serial montior has be opened
delay(2000);
values = my_flash_store.read(); // Read values from flash memory
if (values.magic == MAGIC_NUMBER) { // If token is correct print saved networks
Serial.println("saved data:");
Serial.println("");
for (int a = 0; a < MaxNet; a++) {
if (values.valid[a]) {
Serial.println(values.SSIDs[a]);
} else {
PosToBeSaved = a;
}
}
}
}
void loop() {
// Temporarly save the number of networks
int networks_already_saved = PosToBeSaved;
getNetwork();
if (PosToBeSaved >= daily_amount_of_food) {
Serial.println("Enough food for today");
}
delay(5000);
}
// Feed the Nerd with networks's SSID
void getNetwork() {
// scan for nearby networks:
Serial.println("\n*Scan Networks*\n");
int numSsid = WiFi.scanNetworks();
delay(1000);
if (numSsid == -1)
{
Serial.println("There are no WiFi networks here..");
} else {
Serial.print("number of available networks: ");
Serial.println(numSsid);
// print the network number and name for each network found:
for (int thisNet = 0; thisNet < numSsid; thisNet++) {
Serial.print("SSID: ");
Serial.println(WiFi.SSID(thisNet));
delay(500);
char* net = WiFi.SSID(thisNet);
bool canBeSaved = true;
// check if the network has already been saved
for (int a = 0; a < PosToBeSaved ; a++) {
if (values.valid[a]) {
if (strncmp(net, values.SSIDs[a], 100) == 0 || strnlen(net, 100) == 0) {
Serial.println("Not saved");
canBeSaved = false;
}
}
}
// Store ssid name
if (canBeSaved && PosToBeSaved < MaxNet) {
if (strlen(net) + 1 < 100 && strlen(net) > 0) { // check if the SSID name fits 100 bytes
memset(values.SSIDs[PosToBeSaved], 0, sizeof(values.SSIDs[PosToBeSaved])); // set all characters to zero
memcpy(values.SSIDs[PosToBeSaved], net, strlen(net) + 1); // copy "net" to values.SSDs[thisNet]
values.valid[PosToBeSaved] = true;
values.magic = MAGIC_NUMBER;
my_flash_store.write(values);
Serial.println(String(values.SSIDs[PosToBeSaved]) + " saved in position " + String(PosToBeSaved));
PosToBeSaved ++;
}
else {
Serial.println(" network skipped");
}
}
}
}
}
請注意我們如何定義可以保存的最大網絡數量,以避免內存空間問題:
#define MaxNet 30
int PosToBeSaved = 0;
已用于導航保存網絡名稱的數組并測量已吃的食物量。
管理時間
我們可以將實時時鐘 (RTC) 的功能與 WiFi 相結合,以獲取當前時間和日期,然后設置板的內部時鐘。通過這種方式,我們可以在很長一段時間內觸發基于時間的事件,而無需使用millis()
當您必須將毫秒轉換為天時可能會很棘手的函數。
我們將從網絡時間協議(NTP) 時間服務器獲取時間,然后使用它設置 RTC。請注意,收到的時間采用紀元格式,即自 1970 年 1 月 1 日以來的秒數。
您可以從example > WiFi101 > WiFiUdpNtpClient運行基本草圖
在下面的草圖中,我們將事物放在一起:掃描網絡功能和與時間相關的功能。
-
bool atHome = false;
用于觸發 WiFi 連接,因此請求服務器獲取時間。
-
check_home()
用于掃描所有可用網絡并查看其中一個是否為家庭 WiFi 網絡。
-
connect_WiFi()
然后調用,它將板連接到 WiFi,觸發對服務器的請求并打印當前時間。
-
rtc.setEpoch(epoch + GMT);
用于以紀元格式以當前時間啟動 RTC,修改 GMT 變量以將時間調整為您當前的時區。
#include
#include
#include
#include
#include
WiFiUDP udp;
WiFiUDP Udp;
RTCZero rtc;
#define MAGIC_NUMBER 0x7423 // arbitrary number to double check the saved SSID
#define MaxNet 30 // max amount of network to be saved
const char* home_ssid = SECRET_SSID; // your network SSID (name)
const char* password = SECRET_PSWD; // your network password
int PosToBeSaved = 0; // Variable used to navigate the array of networks
int daily_amount_of_food = 12; // The amount of food per day needed to survive
bool atHome = false;
// Struct of variable to be saved in flash memory
typedef struct {
int magic;
boolean valid[MaxNet];
char SSIDs[MaxNet][100];
int alive_days;
int last_time_feeded;
} Networks;
FlashStorage(my_flash_store, Networks);
Networks values;
void setup() {
Serial.begin(115200);
delay(2000);
rtc.begin(); // enable real time clock functionalities
values = my_flash_store.read(); // Read values from flash memory
if (values.magic == MAGIC_NUMBER) { // If token is correct print saved networks
Serial.println("saved data:");
Serial.println("");
for (int a = 0; a < MaxNet; a++) {
if (values.valid[a]) {
Serial.println(values.SSIDs[a]);
} else {
PosToBeSaved = a;
}
}
}
}
void loop() {
if(!atHome) check_home();
// Temporarly save the number of networks
int networks_already_saved = PosToBeSaved;
getNetwork();
if (PosToBeSaved >= daily_amount_of_food) {
Serial.println("Enough food for today");
}
}
void check_home() {
int numSsid = WiFi.scanNetworks();
if (numSsid != -1) {
for (int thisNet = 0; thisNet < numSsid; thisNet++) {
delay(100);
if (strncmp(WiFi.SSID(thisNet), home_ssid, 100) == 0) {
Serial.println("Yay, I'm home \n");
atHome = true;
connect_WiFi();
}
}
}
}
void connect_WiFi() {
if (WiFi.status() != WL_CONNECTED) {
while (WiFi.begin(home_ssid, password) != WL_CONNECTED) {
delay(500);
}
Serial.println("WiFi connected \n");
GetCurrentTime();
printTime();
}
}
// Feed the Nerd with networks's SSID
void getNetwork() {
// scan for nearby networks:
Serial.println("*Scan Networks*");
int numSsid = WiFi.scanNetworks();
delay(1000);
if (numSsid == -1)
{
Serial.println("There are no WiFi networks here..");
} else {
Serial.print("number of available networks: ");
Serial.println(numSsid);
// print the network number and name for each network found:
for (int thisNet = 0; thisNet < numSsid; thisNet++) {
Serial.print("SSID: ");
Serial.println(WiFi.SSID(thisNet));
delay(500);
char* net = WiFi.SSID(thisNet);
bool canBeSaved = true;
// check if the network has already been saved
for (int a = 0; a < PosToBeSaved ; a++) {
if (values.valid[a]) {
if (strncmp(net, values.SSIDs[a], 100) == 0 || strnlen(net, 100) == 0) {
Serial.println("Not saved");
canBeSaved = false;
}
}
}
// Store ssid name
if (canBeSaved && PosToBeSaved < MaxNet) {
if (strlen(net) + 1 < 100 && strlen(net) > 0) { // check if the SSID name fits 100 bytes
memset(values.SSIDs[PosToBeSaved], 0, sizeof(values.SSIDs[PosToBeSaved])); // set all characters to zero
memcpy(values.SSIDs[PosToBeSaved], net, strlen(net) + 1); // copy "net" to values.SSDs[thisNet]
values.valid[PosToBeSaved] = true;
values.last_time_feeded = rtc.getEpoch();
values.magic = MAGIC_NUMBER;
my_flash_store.write(values);
Serial.println(String(values.SSIDs[PosToBeSaved]) + " saved in position " + String(PosToBeSaved));
PosToBeSaved ++;
}
else {
Serial.println(" network skipped");
}
}
}
}
}
/*************************************************
Start an UDP connection to get the time in unix,
then set the real time clock (rtc)
************************************************/
unsigned int localPort = 2390; // local port to listen for UDP packets
IPAddress timeServer(129, 6, 15, 28); // time.nist.gov NTP server
const int NTP_PACKET_SIZE = 48; // NTP time stamp is in the first 48 bytes of the message
byte packetBuffer[ NTP_PACKET_SIZE]; //buffer to hold incoming and outgoing packets
const int GMT = 1 * 60 * 60; //change this to adapt it to your time zone hours*minutes*seconds
unsigned long epoch;
void GetCurrentTime() {
int numberOfTries = 0, maxTries = 6;
do {
epoch = readLinuxEpochUsingNTP();
numberOfTries++;
}
while ((epoch == 0) || (numberOfTries > maxTries));
if (numberOfTries > maxTries) {
Serial.print("NTP unreachable!!");
while (1);
}
else {
Serial.print("Epoch received: ");
Serial.println(epoch);
rtc.setEpoch(epoch + GMT);
Serial.println();
}
}
unsigned long readLinuxEpochUsingNTP()
{
Udp.begin(localPort);
sendNTPpacket(timeServer); // send an NTP packet to a time server
// wait to see if a reply is available
delay(1000);
if ( Udp.parsePacket() ) {
Serial.println("NTP time received");
// We've received a packet, read the data from it
Udp.read(packetBuffer, NTP_PACKET_SIZE); // read the packet into the buffer
//the timestamp starts at byte 40 of the received packet and is four bytes,
// or two words, long. First, esxtract the two words:
unsigned long highWord = word(packetBuffer[40], packetBuffer[41]);
unsigned long lowWord = word(packetBuffer[42], packetBuffer[43]);
// combine the four bytes (two words) into a long integer
// this is NTP time (seconds since Jan 1 1900):
unsigned long secsSince1900 = highWord << 16 | lowWord;
// now convert NTP time into everyday time:
// Unix time starts on Jan 1 1970. In seconds, that's 2208988800:
const unsigned long seventyYears = 2208988800UL;
// subtract seventy years:
Udp.stop();
return (secsSince1900 - seventyYears);
}
else {
Udp.stop();
return 0;
}
}
// send an NTP request to the time server at the given address
unsigned long sendNTPpacket(IPAddress & address)
{
// set all bytes in the buffer to 0
memset(packetBuffer, 0, NTP_PACKET_SIZE);
// Initialize values needed to form NTP request
// (see URL above for details on the packets)
packetBuffer[0] = 0b11100011; // LI, Version, Mode
packetBuffer[1] = 0; // Stratum, or type of clock
packetBuffer[2] = 6; // Polling Interval
packetBuffer[3] = 0xEC; // Peer Clock Precision
// 8 bytes of zero for Root Delay & Root Dispersion
packetBuffer[12] = 49;
packetBuffer[13] = 0x4E;
packetBuffer[14] = 49;
packetBuffer[15] = 52;
// all NTP fields have been given values, now
// you can send a packet requesting a timestamp:
Udp.beginPacket(address, 123); //NTP requests are to port 123
Udp.write(packetBuffer, NTP_PACKET_SIZE);
Udp.endPacket();
}
void printTime() {
// Print date...
Serial.print(rtc.getDay());
Serial.print(" / ");
Serial.print(rtc.getMonth());
Serial.print(" / ");
Serial.print(rtc.getYear());
Serial.print("\t");
// ...and time
print2digits(rtc.getHours());
Serial.print(": ");
print2digits(rtc.getMinutes());
Serial.print(": ");
print2digits(rtc.getSeconds());
Serial.println("");
}
void print2digits(int number) {
if (number < 10) {
Serial.print("0");
}
Serial.print(number);
實現當前時間允許我們使用像這樣的簡單函數來管理 Nerd 的生死:
if(rtc.getEpoch() - values.last_time_fed >= 86400*2){
// complain and eventually die :(
}
其中86400
是一天中的秒數, values.last_time_fed
是以紀元格式存儲的時間值,其中 Nerd 上次被喂食并以紀元格式rtc.getEpoch()
返回當前時間。
低功耗模式
我們可以使用低功耗模式讓我們的電路板進入睡眠狀態。這意味著它將禁用其大部分功能(包括 WiFi)以節省電池電量。
由于我們希望 Nerd 定期醒來,我們可以輕松設置一個計時器:
#include "ArduinoLowPower.h"
int sleeping_time = 5000; // 5 seconds
bool awake = true;
void setup() {
Serial.begin(9600);
LowPower.attachInterruptWakeup(RTC_ALARM_WAKEUP, WakeUp, CHANGE);
pinMode(LED_BUILTIN, OUTPUT);
}
void loop() {
if (awake) {
digitalWrite(LED_BUILTIN, HIGH);
delay(500);
digitalWrite(LED_BUILTIN, LOW);
delay(500);
}
awake = false;
Serial.println("going to sleep");
LowPower.sleep(sleeping_time);
}
void WakeUp() {
Serial.println("awake");
awake = true;
}
請注意,該WakeUp()
函數附加到中斷,這意味著它不能包含任何包含延遲的代碼。但是我們可以設置布爾變量來觸發循環中的事件。
- 寵物食品分配器開源設計
- 寵物喂食器開源硬件
- 自動寵物喂食器開源硬件
- 寵物食品消費追蹤器開源分享
- 赫伯特機器人虛擬寵物開源分享
- HugWear可穿戴寵物開源分享
- 無線電廣播TA2003開源分享
- 自制無線電通信開源項目
- 帶TEF6686的RDS無線電開源
- 無線電控制椰子收割機開源分享
- 對于減量AGC無線電開源項目
- 無線電控制車開源項目
- 解析無線電偽碼測目標距離系統的設計 0次下載
- 軟件無線電基礎 51次下載
- 軟件無線電RFID測試平臺課件下載 32次下載
- 無線電時鐘的dcf信號是什么意思 818次閱讀
- 廚房內的無線電源解決方案 640次閱讀
- 軟件無線電安全之GNU Radio基礎知識 4946次閱讀
- 什么是無線電頻譜 最全最新無線通信頻率分配 9647次閱讀
- 兩種常見無線電架構對比 939次閱讀
- X和Ku-頻段小型無線電設計 1453次閱讀
- 無線電波的傳播方式和頻率分配 2.1w次閱讀
- 無線電信號的發送和接收 1.5w次閱讀
- 帶你了解無線電遙控技術 1w次閱讀
- 軟件無線電架構的詳細概述 1.6w次閱讀
- 基于Zedboard的開源軟件定義無線電設備——Panoradio 6662次閱讀
- 基于FPGA的軟件無線電平臺設計詳細教程 3694次閱讀
- 基于SDR技術的無線電設計方法 1642次閱讀
- 軟件定義無線電的創新應用 2071次閱讀
- 簡易無線電接收器電路 1.2w次閱讀
下載排行
本周
- 1山景DSP芯片AP8248A2數據手冊
- 1.06 MB | 532次下載 | 免費
- 2RK3399完整板原理圖(支持平板,盒子VR)
- 3.28 MB | 339次下載 | 免費
- 3TC358743XBG評估板參考手冊
- 1.36 MB | 330次下載 | 免費
- 4DFM軟件使用教程
- 0.84 MB | 295次下載 | 免費
- 5元宇宙深度解析—未來的未來-風口還是泡沫
- 6.40 MB | 227次下載 | 免費
- 6迪文DGUS開發指南
- 31.67 MB | 194次下載 | 免費
- 7元宇宙底層硬件系列報告
- 13.42 MB | 182次下載 | 免費
- 8FP5207XR-G1中文應用手冊
- 1.09 MB | 178次下載 | 免費
本月
- 1OrCAD10.5下載OrCAD10.5中文版軟件
- 0.00 MB | 234315次下載 | 免費
- 2555集成電路應用800例(新編版)
- 0.00 MB | 33566次下載 | 免費
- 3接口電路圖大全
- 未知 | 30323次下載 | 免費
- 4開關電源設計實例指南
- 未知 | 21549次下載 | 免費
- 5電氣工程師手冊免費下載(新編第二版pdf電子書)
- 0.00 MB | 15349次下載 | 免費
- 6數字電路基礎pdf(下載)
- 未知 | 13750次下載 | 免費
- 7電子制作實例集錦 下載
- 未知 | 8113次下載 | 免費
- 8《LED驅動電路設計》 溫德爾著
- 0.00 MB | 6656次下載 | 免費
總榜
- 1matlab軟件下載入口
- 未知 | 935054次下載 | 免費
- 2protel99se軟件下載(可英文版轉中文版)
- 78.1 MB | 537798次下載 | 免費
- 3MATLAB 7.1 下載 (含軟件介紹)
- 未知 | 420027次下載 | 免費
- 4OrCAD10.5下載OrCAD10.5中文版軟件
- 0.00 MB | 234315次下載 | 免費
- 5Altium DXP2002下載入口
- 未知 | 233046次下載 | 免費
- 6電路仿真軟件multisim 10.0免費下載
- 340992 | 191187次下載 | 免費
- 7十天學會AVR單片機與C語言視頻教程 下載
- 158M | 183279次下載 | 免費
- 8proe5.0野火版下載(中文版免費下載)
- 未知 | 138040次下載 | 免費
評論
查看更多