学习了RT-Thread的内核也有一段时间了,由于各种各样的琐事自己没有去做一个综合应用示例,刚最近有点时间,做了一个对接OneNET的历程,采用的是OneNET的EDP协议,关于OneNET的EDP协议可以点击跳转至OneNET EDP协议讲解与应用这篇博客看一下,这篇博客会比较详细的介绍对接EDP协议的思路。本篇博客会根据着重的将RT-Thread的应用思路,在本应用示例也只是用来软件定时器、线程管理、信号量而已,因为我用的是潘多拉开发板进行实验,所以AP6181驱动就直接用RT-Thread的,自己也只是在这个基础上做了应用。
使用AP6181 WiFi模组对接OneNET应用示例:STM32+AP6181(SDIO WiFi模组)对接OneNET,实现温湿度定时上报,控制上报周期、控制设备LED灯、蜂鸣器以及电机,掉线自动连接、WiFi自动连接。

整个工程代码点击跳转至GitHub获取。

一、移植需要修改的地方

在移植时,主要也就是修改几个地方,代码如下:

#define ONENET_EDP_PRODUCT_ID        "190254"  /* 产品id */
#define ONENET_EDP_PRODUCT_APIKEY   "cWPICK6PDU6cOHP=T0SqMcXWRc4="  /* api key */
#define ONENET_EDP_DEVICE_ID        "504890772"   /* 设备id */
#define ONENET_EDP_DEVICE_AUTHKEY   "edp20181122" /* 设备鉴权信息 */

需要修改的为产品ID、APIKEY、设备ID、设备鉴权信息,将这些信息修改为你自己在OneNET开发者中心的产品信息或设备信息。

二、连接OneNET思路

1、开机后,RT-Thread完成一系列的初始化后进入main线程,首先进入main函数会进行执行onenet_edp_sample();函数进行信号量创建、软件定时器创建以及线程创建:

rt_err_t onenet_edp_sample(void)
{/* 连接OneNET信号量 */link_onenet_sem = rt_sem_create("link_onenet_sem", 0, RT_IPC_FLAG_FIFO);if(link_onenet_sem == RT_NULL){rt_kprintf("[EDP]link_onenet_sem create failed\r\n");return RT_ERROR;}/* 发送心跳信号量 */send_heart_sem = rt_sem_create("send_heart_sem", 0, RT_IPC_FLAG_FIFO);if(send_heart_sem == RT_NULL){rt_kprintf("[EDP]send_heart_sem create failed\r\n");return RT_ERROR;}/* 用于获取温湿度和发送温湿度信号量 */humi_temp_sem = rt_sem_create("humi_temp_sem", 1, RT_IPC_FLAG_FIFO);if(humi_temp_sem == RT_NULL){rt_kprintf("[EDP]humi_temp_sem create failed\r\n");return RT_ERROR;}/* 用于定时触发上报温湿度数据信号量 */send_humi_temp_sem = rt_sem_create("send_humi_temp_sem", 0, RT_IPC_FLAG_FIFO);if(send_humi_temp_sem == RT_NULL){rt_kprintf("[EDP]send_humi_temp_sem create failed\r\n");return RT_ERROR;}/* 软件定时器,周期执行,120秒 */send_heart_timer = rt_timer_create("send_heart_timer", send_heart_timer_callback, RT_NULL, 120000, RT_TIMER_FLAG_SOFT_TIMER | RT_TIMER_FLAG_PERIODIC);if(send_heart_timer == RT_NULL){rt_kprintf("[EDP]send_heart_timer create failed\r\n");return RT_ERROR;}/* 软件定时器,周期执行,300秒 */send_humi_temp_timer = rt_timer_create("send_humi_temp_timer", send_humi_temp_timer_callback, RT_NULL, 30000, RT_TIMER_FLAG_SOFT_TIMER | RT_TIMER_FLAG_PERIODIC);if(send_humi_temp_timer == RT_NULL){rt_kprintf("[EDP]send_humi_temp_timer create failed\r\n");return RT_ERROR;}/* 连接设备线程 */link_dev_thread = rt_thread_create("link_dev_thread", onenet_edp_link_device_thread, RT_NULL, 2048, 4, 200);if(link_dev_thread == RT_NULL){rt_kprintf("[EDP]link_dev_thread create failed\r\n");return RT_ERROR;}rt_thread_startup(link_dev_thread);/* 发送心跳线程 */send_heart_thread = rt_thread_create("send_heart_thread", onenet_edp_send_heart_thread, RT_NULL, 1024, 4, 50);if(send_heart_thread == RT_NULL){rt_kprintf("[EDP]send_heart_thread create failed\r\n");return RT_ERROR;}rt_thread_startup(send_heart_thread);/* 平台下发命令或结果处理线程 */recv_data_pro_thread = rt_thread_create("recv_data_pro_thread",onenet_edp_receive_process_thread, RT_NULL, 4096, 4, 100);if(recv_data_pro_thread == RT_NULL){rt_kprintf("[EDP]recv_data_pro_thread create failed\r\n");return RT_ERROR;}rt_thread_startup(recv_data_pro_thread);/* 获取温湿度数据线程 */get_humi_temp_thread = rt_thread_create("get_humi_temp_thread", onenet_edp_get_humi_temp_thread, RT_NULL, 4096, 4, 50);if(get_humi_temp_thread == RT_NULL){rt_kprintf("[EDP]get_humi_temp_thread create failed\r\n");return RT_ERROR;}rt_thread_startup(get_humi_temp_thread);/* 上报温湿度数据到平台线程 */send_humi_temp_thread = rt_thread_create("send_humi_temp_thread", onenet_edp_send_humi_temp_thread, RT_NULL, 4096, 4, 100);if(send_humi_temp_thread == RT_NULL){rt_kprintf("[EDP]send_humi_temp_thread create failed\r\n");    return RT_ERROR;    }rt_thread_startup(send_humi_temp_thread);return RT_EOK;
}

2、完成信号量创建、软件定时器的线程创建后,会进行注册网络就绪事件回调、网络断开事件回调、以及连接网络、开启自动连接网络:

/* 注册网络就绪回调、 */
rt_wlan_register_event_handler(RT_WLAN_EVT_READY, wlan_ready_handler, RT_NULL);
/* 注册网络断开事件回调 */
rt_wlan_register_event_handler(RT_WLAN_EVT_STA_DISCONNECTED, wlan_station_disconnect_handler, RT_NULL);
/* 连接路由器 */
rt_wlan_connect(WLAN_SSID, WLAN_PASSWORD);
rt_thread_mdelay(1000);
/* 开启自动连接 */
wlan_autoconnect_init();
rt_wlan_config_autoreconnect(RT_TRUE);

3、网络就绪回调函数用于释放连接OneNET信号量,告诉系统已经有网络了,可以去连接OneNET了:

void wlan_ready_handler(int event, struct rt_wlan_buff *buff, void *parameter)
{rt_kprintf("wlan ready\r\n");if(link_onenet_sem != RT_NULL){rt_sem_release(link_onenet_sem);}
}

而另一方面,连接OneNET线程会移植挂起等待连接OneNET的信号量,指导获取到这个信号量才开始去连接OneNET:

/**************************************************************
函数名称:onenet_edp_link_device_thread
函数功能:连接OneNET平台设备线程
输入参数:parameter:线程入口参数
返 回 值:无
备    注:WiFi注册上网了后连接平台,连接成功则启动定时器发送心跳
**************************************************************/
void onenet_edp_link_device_thread(void *parameter)
{rt_err_t result = RT_ERROR;while(1){result = rt_sem_take(link_onenet_sem, RT_WAITING_FOREVER); /* 等待时间:一直等 */if(RT_EOK == result){#ifdef USING_DEVICEID_APIKEY_LINKonenet_edp_link_device(ONENET_EDP_SRV_ADDR, ONENET_EDP_SRV_PORT, ONENET_EDP_DEVICE_ID, ONENET_EDP_PRODUCT_APIKEY);#elseonenet_edp_link_device(ONENET_EDP_SRV_ADDR, ONENET_EDP_SRV_PORT, ONENET_EDP_PRODUCT_ID, ONENET_EDP_DEVICE_AUTHKEY);#endif}rt_thread_mdelay(1);rt_thread_yield();/* 放弃剩余时间片,进行一次线程切换 */}
}

三、数据上报和心跳发送思路

1、连接上OneNET之后,处理平台下发命令或结果的线程会得到连接成功返回结果并进行处理,启动软件定时器进行发送心跳和上报数据:

case 0:
{rt_kprintf("Tips: 连接成功\r\n");if(send_heart_timer != RT_NULL){rt_kprintf("[EDP]start send_heart_timer\r\n");rt_timer_start(send_heart_timer);/* 启动软件定时器定时发送心跳 */}if(send_humi_temp_timer != RT_NULL){rt_kprintf("[EDP]start send_humi_temp_timer\r\n");rt_timer_start(send_humi_temp_timer);/* 启动软件定时器定时上报数据 */}break;
}

2、软件定时器回调函数用于释放发送心跳信号量:

/**************************************************************
函数名称:send_heart_timer_callback
函数功能:软件定时器回调函数,用于定时发送信号量来触发发送心跳包
输入参数:parameter:入口参数
返 回 值:无
备    注:无
**************************************************************/
void send_heart_timer_callback(void *parameter)
{if(send_heart_timer != RT_NULL){rt_sem_release(send_heart_sem);}
}

3、发送心跳线程获取到信号量之后执行发送心跳操作:

/**************************************************************
函数名称:onenet_edp_send_heart_thread
函数功能:发送心跳线程
输入参数:parameter:线程入口参数
返 回 值:无
备    注:无
**************************************************************/
void onenet_edp_send_heart_thread(void *parameter)
{rt_err_t result = RT_ERROR;while(1){result = rt_sem_take(send_heart_sem, RT_WAITING_FOREVER); /* 等待时间:一直等 */if(RT_EOK == result){onenet_edp_send_heart();}rt_thread_mdelay(1);rt_thread_yield();/* 放弃剩余时间片,进行一次线程切换 */}
}

4、同理,上报数据也是等待信号量,一旦获取到就上报数据,为保证数据上报的同步,获取数据与上报数据用来二值信号量,实现代码:

/**************************************************************
函数名称:onenet_edp_get_humi_temp_thread
函数功能:获取温湿度数据线程
输入参数:parameter:线程入口参数
返 回 值:无
备     注:无
**************************************************************/
void onenet_edp_get_humi_temp_thread(void *parameter)
{rt_err_t result = RT_ERROR;while(1){result = rt_sem_take(send_humi_temp_sem, RT_WAITING_FOREVER); /* 等待时间:一直等 */if(RT_EOK == result){result = rt_sem_take(humi_temp_sem, RT_WAITING_FOREVER); /* 等待时间:一直等 */if(RT_EOK == result){aht10_read_data(&g_temperature, &g_humidity);}rt_sem_release(humi_temp_sem);  /* 释放信号量 */}rt_thread_mdelay(1);rt_thread_yield();/* 放弃剩余时间片,进行一次线程切换 */}
}/**************************************************************
函数名称:onenet_edp_send_humi_temp_thread
函数功能:发送温湿度数据线程
输入参数:parameter:线程入口参数
返 回 值:无
备     注:无
**************************************************************/
void onenet_edp_send_humi_temp_thread(void *parameter)
{rt_err_t result = RT_ERROR;while(1){result = rt_sem_take(send_humi_temp_sem, RT_WAITING_FOREVER); /* 等待时间:一直等 */if(RT_EOK == result){result = rt_sem_take(humi_temp_sem, RT_WAITING_FOREVER); /* 等待时间:一直等 */if(RT_EOK == result){/* 上传温湿度数据到平台 */if(RT_EOK == onenet_edp_send_data(FORMAT_TYPE3, ONENET_EDP_DEVICE_ID, ONENET_EDP_PRODUCT_APIKEY, humi_temp_data_stream, humi_temp_data_stream_cnt)){rt_kprintf("Humi_Temp Send success\n");}else{rt_kprintf("Humi_Temp Status Send failed\n");}}rt_sem_release(humi_temp_sem);  /* 释放信号量 */}rt_thread_mdelay(1);rt_thread_yield();/* 放弃剩余时间片,进行一次线程切换 */}
}

四、设备断开与网络断开处理

1、如果在有网络的情况下设备突然断开,那么就需要重新连接OneNET了,首先会停止所有定时器、接着重新发送连接OneNET信号量,处理如下:

case DISCONNECT:
{rt_kprintf("WARN:连接断开,准备重连\r\n");if(send_heart_timer != RT_NULL){rt_kprintf("[EDP]stop send_heart_timer\r\n");rt_timer_stop(send_heart_timer);/* 关闭软件定时器 */}if(send_humi_temp_timer != RT_NULL){rt_kprintf("[EDP]stop send_humi_temp_timer\r\n");rt_timer_stop(send_humi_temp_timer);/* 关闭软件定时器 */}if(link_onenet_sem != RT_NULL){rt_sem_release(link_onenet_sem);}break;
}

2、如果是网络断开,那么就需要停止软件定时器,处理如下:

void wlan_station_disconnect_handler(int event, struct rt_wlan_buff *buff, void *parameter)
{rt_kprintf("disconnect from the network!\n");if(send_heart_timer != RT_NULL){rt_kprintf("[EDP]stop send_heart_timer\r\n");rt_timer_stop(send_heart_timer);/* 关闭软件定时器 */}if(send_humi_temp_timer != RT_NULL){rt_kprintf("[EDP]stop send_humi_temp_timer\r\n");rt_timer_stop(send_humi_temp_timer);/* 关闭软件定时器 */}
}

五、命令处理思路

1、在命令处理里面,主要做了控制LED、BEEP、电机以及控制上报数据周期,其中控制数据上报周期用到了rt_timer_control函数,处理如下:

/**************************************************************
函数名称:onenet_edp_command_callback
函数功能:平台下发命令回调函数
输入参数:req:下发的命令
返 回 值:无
备    注:无
**************************************************************/
void onenet_edp_command_callback(char *req)
{char *time_ptr = NULL;unsigned int current_time = 0;rt_pin_mode(PIN_LED_R, PIN_MODE_OUTPUT);rt_pin_mode(PIN_MOTOR_A, PIN_MODE_OUTPUT);rt_pin_mode(PIN_MOTOR_B, PIN_MODE_OUTPUT);rt_pin_mode(PIN_BEEP, PIN_MODE_OUTPUT);led_ctrl(0);beep_ctrl(0);motor_ctrl(0);/* LED */if(strstr((const char*)req, "LED")){if(strcmp("LED:1", req) == 0){g_led_status = 1;led_ctrl(1);/* 上传LED状态数据到平台 */onenet_edp_send_data(FORMAT_TYPE3, ONENET_EDP_DEVICE_ID, ONENET_EDP_PRODUCT_APIKEY, led_status_data_stream, led_status_data_stream_cnt);}else if(strcmp("LED:0", req) == 0){g_led_status = 0;led_ctrl(0);/* 上传LED状态数据到平台 */onenet_edp_send_data(FORMAT_TYPE3, ONENET_EDP_DEVICE_ID, ONENET_EDP_PRODUCT_APIKEY, led_status_data_stream, led_status_data_stream_cnt);}else{}}/* 蜂鸣器 */else if(strstr((const char*)req, "BEEP")){if(strcmp("BEEP:1", req) == 0){g_beep_status = 1;beep_ctrl(1);/* 上传BEEP状态数据到平台 */onenet_edp_send_data(FORMAT_TYPE3, ONENET_EDP_DEVICE_ID, ONENET_EDP_PRODUCT_APIKEY, beep_status_data_stream, beep_status_data_stream_cnt);}else if(strcmp("BEEP:0", req) == 0){g_beep_status = 0;beep_ctrl(0);/* 上传BEEP状态数据到平台 */onenet_edp_send_data(FORMAT_TYPE3, ONENET_EDP_DEVICE_ID, ONENET_EDP_PRODUCT_APIKEY, beep_status_data_stream, beep_status_data_stream_cnt);}else{}}/* 电机 */else if(strstr((const char*)req, "MOTOR")){if(strcmp("MOTOR:1", req) == 0){g_motor_status = 1;motor_ctrl(1);/* 上传BEEP状态数据到平台 */onenet_edp_send_data(FORMAT_TYPE3, ONENET_EDP_DEVICE_ID, ONENET_EDP_PRODUCT_APIKEY, motor_status_data_stream, motor_status_data_stream_cnt);}else if(strcmp("MOTOR:0", req) == 0){g_motor_status = 0;motor_ctrl(0);/* 上传BEEP状态数据到平台 */onenet_edp_send_data(FORMAT_TYPE3, ONENET_EDP_DEVICE_ID, ONENET_EDP_PRODUCT_APIKEY, motor_status_data_stream, motor_status_data_stream_cnt);}else{}}else if(strstr((const char*)req, "TIME")){time_ptr = req + 5;g_data_time = (RT_TICK_PER_SECOND * atoi(time_ptr));/* 单位:秒 */if(g_data_time > 0){rt_kprintf("g_data_time:%d\r\n", g_data_time);if(RT_EOK == rt_timer_control(send_humi_temp_timer, RT_TIMER_CTRL_SET_TIME, &g_data_time)){rt_kprintf("修改定时上传温湿度周期成功,当前周期值为:%d\r\n", g_data_time);}else{rt_kprintf("修改定时上传温湿度周期失败\r\n");}}if(RT_EOK == rt_timer_control(send_humi_temp_timer, RT_TIMER_CTRL_GET_TIME, &current_time)){rt_kprintf("current_time:%d\r\n", current_time);}/* 上传获取数据周期值到平台 */onenet_edp_send_data(FORMAT_TYPE3, ONENET_EDP_DEVICE_ID, ONENET_EDP_PRODUCT_APIKEY, data_time_data_stream, data_time_data_stream_cnt);}else{rt_kprintf("receive other commond\r\n");}
}

六、在OneNET实现的应用示例

点击跳转至在OneNET实现的应用

七、图片示例

1、连接成功

2、发送心跳成功

3、上报数据成功

4、接收命令执行操作以及上报状态成功:

七、FinSH抓取连接OneNET、发送心跳、上报数据控制设备的信息

1、信息如下:
\ | /
- RT - Thread Operating System
/ | \ 4.0.0 build Mar 15 2019
2006 - 2018 Copyright by rt-thread team
lwIP-2.0.2 initialized!
[SFUD] Find a Winbond flash chip. Size is 16777216 bytes.
[SFUD] w25q128 flash device is initialize success.
msh />[I/FAL] RT-Thread Flash Abstraction Layer (V0.2.0) initialize success.
[I/OTA] RT-Thread OTA package(V0.1.3) initialize success.
[I/OTA] Verify ‘wifi_image’ partition(fw ver: 1.0, timestamp: 1529386280) success.
[I/WICED] wifi initialize done. wiced version 3.3.1
[I/WLAN.dev] wlan init success
[I/WLAN.lwip] eth device init ok name:w0
join ssid:MY_WiFi
[I/WLAN.mgnt] wifi connect success ssid:MY_WiFi
[Flash] EasyFlash V3.2.1 is initialize success.
[Flash] You can get the latest version on https://github.com/armink/EasyFlash .
wlan ready
socket create success!!!, socket_id:0
[I/WLAN.lwip] Got IP address : 192.168.43.5
socket connect success!!!
[EDP]EDP_PacketConnect send success
socket receive data
Tips: 连接成功
[EDP]start send_heart_timer
[EDP]start send_humi_temp_timer
[EDP]send heart
socket receive data
cmdid: 00a8d84c-d740-586c-94d1-4e6ac32dc883, req: LED:1, req_len: 5
Send 35 Bytes
socket receive data
Tips: HeartBeat OK
socket receive data
socket receive data
Tips: Send Ok
[EDP]send heart
socket receive data
Tips: HeartBeat OK
socket receive data
cmdid: 4b3ffb43-4f0c-50f3-9370-1ec9179658d8, req: LED:0, req_len: 5
Send 35 Bytes
socket receive data
socket receive data
Tips: Send Ok
socket receive data
cmdid: 3c039820-9487-5415-948d-f209550672e5, req: LED:1, req_len: 5
Send 35 Bytes
socket receive data
socket receive data
Tips: Send Ok
socket receive data
cmdid: e8f37c5b-d136-5ff5-95f2-40a530d9d553, req: LED:0, req_len: 5
Send 35 Bytes
socket receive data
socket receive data
Tips: Send Ok
socket receive data
cmdid: f38d85db-2b5b-51a9-94c7-6fda84de0ce1, req: LED:1, req_len: 5
Send 35 Bytes
socket receive data
socket receive data
Tips: Send Ok
socket receive data
cmdid: f2d9c7cc-0b02-594d-84b6-2b8e344a621f, req: LED:0, req_len: 5
Send 35 Bytes
socket receive data
socket receive data
Tips: Send Ok
[EDP]send heart
socket receive data
Tips: HeartBeat OK
[EDP]send heart
socket receive data
Tips: HeartBeat OK
[EDP]send heart
socket receive data
Tips: HeartBeat OK
socket receive data
cmdid: 4fc4d0de-078e-5ce5-bae1-cc4fdff23ec8, req: LED:1, req_len: 5
Send 35 Bytes
socket receive data
socket receive data
Tips: Send Ok
socket receive data
cmdid: 5f56d146-0313-5814-9c62-f5836221aebd, req: LED:0, req_len: 5
Send 35 Bytes
socket receive data
socket receive data
Tips: Send Ok
[EDP]send heart
Send 65 Bytes
Humi_Temp Send success
socket receive data
Tips: HeartBeat OK
socket receive data
socket receive data
Tips: Send Ok
[EDP]send heart
socket receive data
Tips: HeartBeat OK
[EDP]send heart
socket receive data
Tips: HeartBeat OK
[EDP]send heart
socket receive data
Tips: HeartBeat OK
[EDP]send heart
socket receive data
Tips: HeartBeat OK
[EDP]send heart
socket receive data
Tips: HeartBeat OK
[EDP]send heart
Send 65 Bytes
Humi_Temp Send success
socket receive data
Tips: HeartBeat OK
socket receive data
socket receive data
Tips: Send Ok
[EDP]send heart
socket receive data
Tips: HeartBeat OK
socket receive data
cmdid: 6235c513-bee0-5a46-a5ef-9282377f352a, req: MOTOR:1, req_len: 7
Send 37 Bytes
socket receive data
socket receive data
Tips: Send Ok
socket receive data
cmdid: fcbb4818-eeb7-5cac-b122-41b6329d2c99, req: MOTOR:0, req_len: 7
Send 37 Bytes
socket receive data
socket receive data
Tips: Send Ok

RT-Thread学习的综合应用——使用AP6181 WiFi模组对接OneNET应用示例相关推荐

  1. 乐鑫Esp32学习之旅 乐鑫 ESP-S2/S3 模组的实现 USB 无线网卡上网,为你的台式机装上无线WiFI上网吧。(附带源码)

    本系列博客学习由非官方人员 半颗心脏 潜心所力所写,仅仅做个人技术交流分享,不做任何商业用途.如有不对之处,请留言,本人及时更改. 系列一:ESP32系列模组基础学习系列笔记 1. 爬坑学习新旅程,虚 ...

  2. 嵌入式学习笔记7 WIFI相关1——WIFI模组选型

    一.选型指标: WIFI模组选型需要考虑的指标,列举如下: 发射功率 传输速率 传输距离 功耗 接收灵敏度 那么以上的各点主要取决于什么呢 取决于协议( IEEE 802.11xx )和频率( 2.4 ...

  3. Yeelink平台使用——远程控制 RT Thread + LwIP+ STM32

    1.前言     [2014年4月重写该博文]     经过若干时间的努力终于搞定了STM32+LwIP和yeelink平台的数据互通,在学习的过程中大部分时间花在以太网协议栈学习上,但是在RT Th ...

  4. RT Thread之 Uart2 操作

    官网连接:https://docs.rt-thread.org/#/rt-thread-version/rt-thread-standard/programming-manual/device/uar ...

  5. 使用RT Thread设备框架封装一个I2C设备——DS3231

    使用RT Thread设备框架封装一个I2C设备--DS3231 前言 ENV配置 I2C测试 将ds3231封装成一个字符设备 结语 前言 学习rt thread的I2C的时候,恰巧手上的板子留了d ...

  6. 基于GD32F103C8T6添加RT Thread nano设备框架并添加串口设备(以控制台console( uart0 )为例)

    最近没事琢磨了一下使用设备框架的问题.因为将串口注册到设备框架可以应用十分丰富的软件包. 于是就整理了一下手上的工程,重新将工程梳理了一遍. 像这样是十分清爽了,其中RTOS是操作系统源代码 并且学习 ...

  7. 上海大学计算机专硕与学硕,计算机学院2017届留沪研究生学习成绩综合评定工作通知...

    根据<上海大学毕业研究生学习成绩综合排名办法>(2017年)的原则,计算机学院2017届留沪研究生学习成绩综合评定工作,通知如下: 一.学院评定工作日程 1.5月17日前完成学院成绩综合排 ...

  8. 关于RT thread系统节拍时钟的配置

    关于RT thread系统节拍时钟的配置                  -----本文基于rt-thread-3.1.3版本编写 首先,使用RTthread OS时,要配置(或者明白)它的系统节拍 ...

  9. rt thread studio使用QBOOT和片外flash实现OTA升级

    我们这里要使用单片机外部flash作为OTA的下载分区,外部flash硬件连接关系 PB3-->SPI3_CLK PB4-->SPI3_MISO PB5-->SPI3_MOSI PE ...

最新文章

  1. ASP.NET DEMO 12 : CheckBoxList 实现单选
  2. html怎么让图片重叠_PS倒影效果怎么做,如何利用PS给产品制作倒影
  3. 《大道至简》第六章读后感及本次课后习题11.9
  4. CodeForces - 1030C Vasya and Golden Ticket(思维)
  5. 身体对腐朽灵魂的一次震撼:向京的雕塑与观众的行为
  6. 敏捷软件开发之结对编程
  7. 设计一个笔记本电脑类,属性随意,并且进行属性私有化,对外提供公开的set和get方法。 设计一个可插拔的接口:InsertDrawable,该接口有什么方法自行定义。
  8. 红黑树模拟软件_【程序员面试必备】红黑树详细图解
  9. 记一次服务器上架的总结和反思
  10. 图邻接表拓扑排序算法c语言完整,在用邻接表表示图时,拓扑排序算法时间复杂度为()...
  11. 2、Kmeans算法处理出租车数据
  12. QTP 10.0 破解版下载安装超详细教程
  13. ADB常用命令--测试人员必备
  14. linux给root权限命令,linux设置root权限命令
  15. charles 本地IP地址
  16. Fallback class must implement the interface annotated by @FeignClient
  17. 我们来了!多云架构时代,欢迎加入中国开源网络新势力
  18. 如何通过市场中性策略获利
  19. npm安装electron时报Error: EPERM: operation not permitted, scandir.....
  20. 王利芬对话蒲易 ——花店如何成为高端电商?_北京_歇会儿网

热门文章

  1. 简单rides和Memory切换缓存 Rides工具类主要方法
  2. 瓶盖扫码回收APP系统 废旧物品创造价值收益
  3. WPF 限制鼠标的移动范围
  4. RHCSA-A1.配置网络设置
  5. 如何区分生抽与老抽?
  6. 我的网页作品(div+css)
  7. 【瑞萨RA系列FSP库开发】RASC+Keil的环境搭建
  8. 在图片上添加水印的四个方法
  9. My 的第一篇博客!!!
  10. FGD针对目标检测的重点与全局知识蒸馏