2025-03-03
103
在实现一个Excel异步导入功能需要异步解析excel,使用@Async注解启用了解析方法的异步功能。但在解析方法内有Feign远程调用,该接口涉及接口权限,而@Async标注的方法与web请求不在同一个线程中,获取不到request。因此需要从上传接口传递request至异步方法。了解到RequestContextHolder.setRequestAttributes(RequestAttributes attributes, boolean inheritable)方法该方法是 Spring 框架提供的一个工具方法,用于在当前线程中手动设置请求上下文(RequestAttributes)。它主要用于跨线程传递和管理请求的上下文。我们需要在开启新线程前通过该方法手动设置当前的请求,以在多线程中访问到当前的request请求对象。但是我在设置后,获取到的request为空。最后发现是excel解析完成时,之前的导入接口已经完成响应,request对象已经被销毁回收了。后又了解到AsyncContext,它用于控制异步请求的生命周期在HttpServletRequest中提供了一个startAsync()方法用于启动异步处理,它会返回 AsyncContext 对象。此时request在标记异步请求完成前不会被回收。在后台任务完成后,调用 asyncContext.complete() 来标记异步请求的完成。示例代码:private AsyncContext asyncContext = null;public void import(MultipartFile file){ ServletRequestAttributes sra = (ServletRequestAttributes) RequestContextHolder.getRequestAttributes(); RequestContextHolder.setRequestAttributes(sra, true); if (sra != null) { asyncContext = sra.getRequest().startAsync(); }// 调用异步解析方法parseFile(file)}@Asyncpublic void parseFile(MultipartFile file){// 一些耗时内容ServletRequestAttributes sra = (ServletRequestAttributes) RequestContextHolder.getRequestAttributes(); RequestContextHolder.setRequestAttributes(sra, true);// 此时能够正常完成Feign调用(已自定义RequestInterceptor拦截器,将所需信息传递)xxFeignClient.xxxxx();// 完成异步处理 asyncContext.complete();}参考:记一次springboot @Async处理导致后续request请求参数获取为空的坑
继续阅读»
2024-03-24
1371
之前将ESP8266接入巴法云,使用一段时间后,发现不够稳定,干脆直接在内网搭建一个HomeAssistant实现灯的控制.这里使用docker搭建:homeassistat(Home Assistant 2021.12.8):version: '3'services: homeassistant: privileged: true image: 'homeassistant/home-assistant:stable' container_name: 'homeassistant' environment: - TZ=Asia/Shanghai network_mode: host volumes: - /home/file/docker/homeassistant/config:/config restart: unless-stopped启动完成后访问服务器8123端口完成初始化.还需要安装一个mqtt服务端,这里同样使用docker搭建mosquitto:version: "3.0"services: mosquitto: container_name: 'mosquitto' image: 'eclipse-mosquitto:2.0.18' ports: - 1883:1883 - 9001:9001 volumes: - /home/file/docker/mosquitto/config:/mosquitto/config - /home/file/docker/mosquitto/data:/mosquitto/data - /home/file/docker/mosquitto/log:/mosquitto/logmosquitto.confpersistence true#persistence_location /mosquitto/data#log_dest file /mosquitto/log/mosquitto.loglistener 9001port 1883allow_anonymous true启动mqtt服务器,然后来到HomeAssistant管理页面,左侧配置-设备与服务,右下角添加集成,找到MQTT,输入刚刚搭建的mqtt服务器.烧录开发板,跟前一篇文章的巴法云代码一样,这里再贴一下,设备接线参考ESP8266通过巴法云接入米家我这边使用的WiFiManager版本为0.16.0,发现使用最新版2.0.17有无法连接MQTT服务器的问题#include ESP8266WiFi.h#include ESP8266WebServer.h#include WiFiManager.h#include Ticker.h //#include PubSubClient.h#include "PubSubClient.h" //默认,加载MQTT库文件 WiFiClient espClient; //wifi客户端模式PubSubClient client(espClient);Ticker ticker; //定时器对象#define ID_MQTT "esp8266-01" //客户端id#define D1 5 //D1#define D2 4 //D2const char* topic = "light001"; //主题名字,与巴法云控制台的主题一致const int B_led = D1; //引脚值,此引脚用于连接继电器的信号端const int led_switch = D2; //引脚值,此引脚用于连接继电器的信号端//**************************************************//const char* mqtt_server = "192.168.4.200"; //默认,MQTT服务器const int mqtt_server_port = 1883; //默认,MQTT服务器端口int count; // Ticker计数用的变量int state = 0; //灯的状态// 建立WiFiManager对象WiFiManager wifiManager; //灯光函数及引脚定义void turnOnLed();void turnOffLed();//计数void tickerCount(){ count++;}// 连接MQTT服务器void reconnect() { Serial.print("Attempting MQTT connection..."); // Attempt to connect if (client.connect(ID_MQTT)) { Serial.println("connected"); Serial.print("subscribe:"); Serial.println(topic); //订阅主题,如果需要订阅多个主题,可发送多条订阅指令client.subscribe(topic2);client.subscribe(topic3); client.subscribe(topic); } else { Serial.print("failed, rc="); Serial.println(client.state()); Serial.println(" try again in 1 seconds"); // Wait 5 seconds before retrying delay(1000); }//}}// 收到信息后的回调函数void receiveCallback(char* topic, byte* payload, unsigned int length) { Serial.print("来了Topic:"); Serial.println(topic); String msg = ""; for (int i = 0; i length; i++) { msg += (char)payload[i]; } Serial.print("Msg:"); Serial.println(msg); if (msg == "on") {//如果接收字符on,亮灯 turnOnLed();//开灯函数 } else if (msg == "off") {//如果接收字符off,关灯 turnOffLed();//关灯函数 } msg = ""; }//开灯void turnOnLed() { Serial.println("turn on light"); digitalWrite(B_led, HIGH); state = 1;}//关灯void turnOffLed() { Serial.println("turn off light"); digitalWrite(B_led, LOW); state = 0;}// 发布信息void pubMQTTmsg(){ // 建立发布主题 //巴法云个性设置,推送消息时:主题名后加/set推送消息,表示向所有订阅这个主题的设备们推送消息, //假如推送者自己也订阅了这个主题,消息不会被推送给它自己,以防止自己推送的消息被自己接收。 String topicString = "light001/set" ; char publishTopic[topicString.length() + 1]; //转换成字符数组 strcpy(publishTopic, topicString.c_str()); // 建立发布信息,当前D1引脚状态 String messageString; if(digitalRead(B_led)){ messageString = "on"; } else { messageString = "off"; } char publishMsg[messageString.length() + 1]; //转换成字符数组 strcpy(publishMsg, messageString.c_str()); // 实现ESP8266向主题发布信息,并在串口监视器显示出来 if(client.publish(publishTopic, publishMsg)){ Serial.println("Publish Topic:"); Serial.println(publishTopic); Serial.println("Publish message:"); Serial.println(publishMsg); } else { Serial.println("Message Publish Failed."); }}//程序入口void setup() { pinMode(B_led, OUTPUT); //设置引脚为输出模式 pinMode(led_switch, OUTPUT); // 开关引脚为输入模式 digitalWrite(B_led, LOW);//默认引脚上电低电平 Serial.begin(9600); //设置波特率9600 //********************自动配置网络************************ // wifiManager.setTimeout(20); wifiManager.setConfigPortalTimeout(120); // 自动连接WiFi。以下语句的参数是连接ESP8266时的WiFi名称// wifiManager.autoConnect("AutoConnectAP"); // 如果您希望该WiFi添加密码,可以使用以下语句: wifiManager.autoConnect("8266","xxxxx"); //重置wifi设置 // wifiManager.resetSettings(); // WiFi连接成功后将通过串口监视器输出连接成功信息 Serial.println(""); Serial.print("ESP8266 Connected to "); Serial.println(WiFi.SSID()); // WiFi名称 Serial.print("IP address:\t"); Serial.println(WiFi.localIP()); // IP //******************************************************* client.setServer(mqtt_server, mqtt_server_port);//设置mqtt服务器 client.setCallback(receiveCallback); //mqtt消息处理 ticker.attach(1, tickerCount); // Ticker定时对象 Serial.println("setup结束");}//循环执行 void loop() {// Serial.println("循环"); int val = digitalRead(B_led); int ctl_val = digitalRead(led_switch); if(ctl_val == 1){//按键按下 delay(400); state = !state; digitalWrite(B_led, state);// digitalWrite(led_switch, LOW); digitalWrite(led_switch, LOW); int ctl_val = digitalRead(led_switch); Serial.print("开2关值为 :\t"); Serial.println(ctl_val); } if (client.connected()) { client.loop(); if (count = 10){ // 每隔10秒钟发布一次信息 Serial.println("心跳报"); pubMQTTmsg(); count = 0; } }else{ reconnect(); }}烧录完成后,现在为HomeAssistant添加该设备在homeassistant的configuration.yaml中最后添加一行light: !include light01.yaml随后在configuration.yaml同级目录添加light01.yaml,内容为- platform: mqtt name: "卧室灯" state_topic: "light001/set" command_topic: "light001" payload_on: "on" payload_off: "off" optimistic: false qos: 0 retain: true完成后重启homeassistant即可连接HomeKit配置-设备与服务中添加HomeKit,然后开放端口在通知中可以看到绑定二维码,使用iPhone的家庭APP扫码绑定用户名-高级模式参考文章ESP8266通过MQTT接入Home Assistant实践ESP8266通过MQTT接入Home Assistant
继续阅读»
2024-03-06
1520
效果演示视频(apple设备无法播放):用到的硬件:ESP8266-NodeMCU开发板、继电器模块、常开门禁开关、杜邦线若干。在巴法云建立帐号并添加MQTT设备云,将私钥复制到下方代码ID_MQTT变量中米家添加设备:我的 - 其他平台设备 - 巴法,按提示绑定巴法云帐号使用Arduino将以下程序烧录至esp8266开发板:/** 智能语言控制控制,支持同时天猫、小爱、小度、google Assistent控制* 也同时支持web控制、小程序控制、app控制,定时控制等* 项目示例:通过发送on或off控制开关* 官网:bemfa.com*/#include ESP8266WiFi.h //默认,加载WIFI头文件#include DNSServer.h#include ESP8266WebServer.h#include WiFiManager.h#include Ticker.h //#include PubSubClient.h#include "PubSubClient.h" //默认,加载MQTT库文件 WiFiClient espClient; //wifi客户端模式PubSubClient client(espClient);Ticker ticker; //定时器对象//********************需要修改的部分*******************//#define ID_MQTT "xxx" //用户私钥,控制台获取#define D1 5 //D1#define D2 4 //D2const char* topic = "light002"; //主题名字,与巴法云控制台的主题一致const int B_led = D1; //引脚值,此引脚用于连接继电器的信号端const int led_switch = D2; //引脚值,此引脚用于连接继电器的信号端//**************************************************//const char* mqtt_server = "bemfa.com"; //默认,MQTT服务器const int mqtt_server_port = 9501; //默认,MQTT服务器端口int count; // Ticker计数用的变量int state = 0; //灯的状态// 建立WiFiManager对象WiFiManager wifiManager; //灯光函数及引脚定义void turnOnLed();void turnOffLed();//计数void tickerCount(){ count++;}// 连接MQTT服务器void reconnect() { Serial.print("Attempting MQTT connection..."); // Attempt to connect if (client.connect(ID_MQTT)) { Serial.println("connected"); Serial.print("subscribe:"); Serial.println(topic); //订阅主题,如果需要订阅多个主题,可发送多条订阅指令client.subscribe(topic2);client.subscribe(topic3); client.subscribe(topic); } else { Serial.print("failed, rc="); Serial.println(client.state()); Serial.println(" try again in 1 seconds"); // Wait 5 seconds before retrying delay(1000); }}// 收到信息后的回调函数void receiveCallback(char* topic, byte* payload, unsigned int length) { Serial.print("Topic:"); Serial.println(topic); String msg = ""; for (int i = 0; i length; i++) { msg += (char)payload[i]; } Serial.print("Msg:"); Serial.println(msg); if (msg == "on") {//如果接收字符on,亮灯 turnOnLed();//开灯函数 } else if (msg == "off") {//如果接收字符off,关灯 turnOffLed();//关灯函数 } msg = ""; }//开灯void turnOnLed() { Serial.println("turn on light"); digitalWrite(B_led, HIGH); state = 1;}//关灯void turnOffLed() { Serial.println("turn off light"); digitalWrite(B_led, LOW); state = 0;}// 发布信息void pubMQTTmsg(){ // 建立发布主题 //巴法云个性设置,推送消息时:主题名后加/set推送消息,表示向所有订阅这个主题的设备们推送消息, //假如推送者自己也订阅了这个主题,消息不会被推送给它自己,以防止自己推送的消息被自己接收。 String topicString = "light002/set" ; char publishTopic[topicString.length() + 1]; //转换成字符数组 strcpy(publishTopic, topicString.c_str()); // 建立发布信息,当前D1引脚状态 String messageString; if(digitalRead(B_led)){ messageString = "on"; } else { messageString = "off"; } char publishMsg[messageString.length() + 1]; //转换成字符数组 strcpy(publishMsg, messageString.c_str()); // 实现ESP8266向主题发布信息,并在串口监视器显示出来 if(client.publish(publishTopic, publishMsg)){ Serial.println("Publish Topic:"); Serial.println(publishTopic); Serial.println("Publish message:"); Serial.println(publishMsg); } else { Serial.println("Message Publish Failed."); }}//程序入口void setup() { pinMode(B_led, OUTPUT); //设置引脚为输出模式 pinMode(led_switch, OUTPUT); // 开关引脚为输入模式 digitalWrite(B_led, LOW);//默认引脚上电低电平 Serial.begin(9600); //设置波特率9600 //********************自动配置网络************************ // wifiManager.setTimeout(20); wifiManager.setConfigPortalTimeout(120); // 自动连接WiFi。以下语句的参数是连接ESP8266时的WiFi名称// wifiManager.autoConnect("AutoConnectAP"); // 如果您希望该WiFi添加密码,可以使用以下语句: wifiManager.autoConnect("8266"); //重置wifi设置 // wifiManager.resetSettings(); // WiFi连接成功后将通过串口监视器输出连接成功信息 Serial.println(""); Serial.print("ESP8266 Connected to "); Serial.println(WiFi.SSID()); // WiFi名称 Serial.print("IP address:\t"); Serial.println(WiFi.localIP()); // IP //******************************************************* client.setServer(mqtt_server, mqtt_server_port);//设置mqtt服务器 client.setCallback(receiveCallback); //mqtt消息处理 ticker.attach(1, tickerCount); // Ticker定时对象 Serial.println("setup结束");}//循环执行 void loop() { int val = digitalRead(B_led); int ctl_val = digitalRead(led_switch); if(ctl_val == 1){//按键按下 delay(400); state = !state; digitalWrite(B_led, state); digitalWrite(led_switch, LOW); int ctl_val = digitalRead(led_switch); Serial.print("开2关值为 :\t"); Serial.println(ctl_val); } if (client.connected()) { client.loop(); if (count = 10){ // 每隔10秒钟发布一次信息 pubMQTTmsg(); count = 0; } }else{// mqtt链接断开,并且已连接上AP,则重连mqtt if(wifiManager.getWLStatusString().equals("WL_CONNECTED")){ reconnect(); } }}继电器及门禁开关分别使用D1、D2引脚作为信号触发,D1在收到开灯指令时输出高电平触发继电器动作,从而点亮灯泡D2连接门禁开关,门禁开关另一端连接开发板3.3V引脚,在按下开关时,将D2引脚拉高,作为手动控制灯开关:按下并松开开关,灯将点亮或熄灭开发板断网时仍可通过门禁开关控制灯光烧录完成后开发板将自动重启,首次未配网时,开发板将启用ssid为8266的接入点,使用无线设备连接该热点,进入网络认证页面完成网络配置.接线图如下:[图片]两条橙色线分别接入3.3v及D2引脚绿色线接5v引脚给继电器供电蓝色线接GND引脚给继电器供电灰色线接D1引脚作为继电器信号线继电器设置为高电平触发模式继电器COM接灯开关内的220V火线继电器NO接灯的控制线开发板上电或重启时,继电器为断开状态,COM口与NO为开路状态,灯不点亮发送开灯指令时,继电器吸合,COM口与NO为短路状态,灯会被点亮使用小爱语音(提前在米家绑定巴法云)控制开关或者手动发送mqtt指令on、off可以完成控制附上硬件购买截图[图片][图片][图片]更新:通过siri控制这里使用php编写一段发布mqtt消息的代码:?php/** * Created by PhpStorm. * User: ZXQ * Date: 2024/3/9 * Time: 23:43 */require('./phpMQTT.php');function console_log($data){ echo("scriptconsole.log(".json_encode($data).")/script");}$server = 'bemfa.com';$port = 9501;$clientId = 'xx';$username = null;$password = null;$clean_session = false;if (($_SERVER['REQUEST_METHOD']==='GET')){ $action = $_GET['action']; if ($action=='on'){ $mqtt = new Bluerhinos\phpMQTT($server, $port, $clientId); if ($mqtt-connect(true, NULL, $username, $password)) { //public用法:publish($topic主题, $content内容, $qos = 0, $retain = false): void $mqtt-publish('light002/set', 'on', 0, false); $mqtt-close(); } else { echo "Time out!\n"; } } if ($action=='off'){ $mqtt = new Bluerhinos\phpMQTT($server, $port, $clientId); if ($mqtt-connect(true, NULL, $username, $password)) { //public用法:publish($topic主题, $content内容, $qos = 0, $retain = false): void $mqtt-publish('light002/set', 'off', 0, false); $mqtt-close(); } else { echo "Time out!\n"; } } echo json_encode("success");}phpMQTT.php来自phpMQTT.php部署完成后,在IOS中添加两条快捷指令,分别为开灯和关灯,操作选择"从URL获取内容",输入开关灯对应URL,然后直接对Siri说开灯或者关灯就可以了,虽说这种方式有点low,但也算实现了吧...[图片]
继续阅读»
2024-01-19
1318
淘了一台单网口J4125设备,安装了ESXI虚拟服务器系统,在上面运行了Windows、linux等系统实现家庭服务搭建。ESXI上的虚拟机都会分配一个局域网地址供外部访问,由于某个服务需要使用第二条网络(第二条宽带),单网口无法实现,于是购买Netgear GS108T通过vlan技术扩展设备的网口,该设备有8个网口。网口规划:1口:原网络入口2口:第二条网络入口3口:汇聚口,接入ESXI主机,同时承载两条网络数据,后续在ESXI中使用vlan分离两条网络的数据4-5口:原网络的lan口,相当于在实现ESXI服务器功能的前提下,扩展出了原来网络两个lan口备用6-7口:第二条网络的lan口,类似4-5口作用。8口:交换机管理口,为了4567口能正常访问到汇聚到3口中的虚拟机设备,需要将他们的pvid设为对应的vlanid交换机配置:创建了11、12两个vlan,分别对应原有网络及第二条网络[图片]各vlan的标记设置[图片][图片][图片]这样设置完成后,插在第4-5口的设备(及原网络的扩展lan口)无法访问ESXI中的虚拟机,但能访问这条线路来之前的所有设备(例如光猫管理页面)因此需要修改4-5口的pvid为11,这样从4-5口设备进来的数据,会被打上vlan11标签,可以从1口去除vlan标签后流出,也可带上vlan标签进入3口进行后续流转6-7口则将pvid设置为12(这里是关键点,因为默认所有口的pvid都为1,所有数据从端口进来都会被打上vlan1的标签,无法流转到3口中,造成无法访问虚拟机)在网上看到过类似问题:查看,我之前使用光猫加openwrt路由器实现iptv单线复用也遇到过类似情况,当时理解不够深刻[图片]在此交换机就设置完成了,只有通过8口才能进入交换机管理页面,猜测也可通过修改交换机vlanid为11或12实现在局域网内管理交换机,没有尝试过。ESXI设置:由于ESXI主机只有一个网口,因此我之前用的都是默认的VMNetwork端口组,现在需要新建一个端口组用于分离第二条网络的数据修改VM Network的vlanid为11,新建vlanid为12的端口组同时需要修改ManagementNetwork的vlanid为你常用的id,因为现在入口数据包含了多vlanid的数据,我这边后续只能通过在vlanid11的设备访问ESXI管理页面(交换机的3-4口以及上级光猫下的设备)[图片]这样就完成了配置,在需要第二条网络的虚拟机配置中添加对应网络vlanid的端口组即可
继续阅读»