【NodeMCU实时天气时钟温湿度项目 6】解析天气信息JSON数据并显示在 TFT 屏幕上(心知天气版)

        今天是第六专题,主要内容是:导入ArduinoJson功能库,借助该库解析从【心知天气】官网返回的JSON数据,并显示在 TFT 屏幕上。

        如您需要了解其它专题的内容,请点击下面的链接。
        第一专题内容,请参考:连接点亮SPI-TFT屏幕和UI布局设计
        第二专题内容,请参考:WIFI模式设置及连接
        第三专题内容,请参考:连接SHT30传感器,获取并显示当前环境温湿度数据(I2C)
        第四专题内容,请参考:通过NTPClient库获取实时网络时间并显示在TFT屏幕上
        第五专题内容,请参考:获取关于城市天气实况和天气预报的JSON信息(心知天气版)

一、【心知天气】官网JSON数据特点

        1、通过API返回的JSON数据,是未经压缩的明文JSON数据,可以直接使用ArduinoJson功能库解析需要的数值,JSON数据解析过程简单方便。
        2、天气实况数据。付费用户可获取全部数据;免费用户只返回天气现象文字、天气现象代码和气温 3 项数据,可供使用的数据比较少。数据更新频率,国内城市在 15 分钟左右,国际城市在 20 分钟左右。
        3、天气预报数据。可获取指定城市未来最多 15 天每天的白天和夜间预报。一般每天更新 3-4 次。付费用户可获取全部数据,免费用户只返回 3 天天气预报,可供使用的数据比较少。
        说明:如果仅作程序调试使用,可申请开通试用版,能通过API获取全部数据,试用期14天。

       

二、添加ArduinoJson库

        1、ArduinoJson库。是为 Arduino 和 IOT(Internet Of Things,物联网)开的 C++ JSON库,使用简单、高效,通用性强,在Github具备很高的活跃度。支持JSON数据的序列化、反序列化(或称序列化还原、解析)、MessagePack、流数据、过滤以及其他功能。官网链接,Github站点。
        2、添加库方法。打开 PlatformIO 界面,选择 Libraries 图标,在搜索栏内输入 ArduinoJson,在查询结果中选择ArduinoJson by Benoit Blanchon库,,添加到项目中。

        3、使用方法。解析JSON天气信息数据的程序代码,我们一般借助ArduinoJson官网提供的工具助手(Assistant),先生成基本代码,然后根据项目需要进行取舍,或进行格式转换后,再复制到项目中。本文第三部分我们进行示例说明。

        4、重要文档。关于安装库,建构和解析JSON,官方示例,API参考,FAQ等有关信息,请查阅官网 documentation 的内容。

三、解析JSON数据方法步骤 

        这是从【心知天气】官网通过API请求返回的原始JSON数据。

// 这是从【心知天气】官网通过API请求返回的原始JSON数据
{"results":[{"location":{"id":"WWE0TGW4PX6N","name":"济南","country":"CN","path":"济南,济南,山东,中国","timezone":"Asia/Shanghai","timezone_offset":"+08:00"},"now":{"text":"晴","code":"0","temperature":"28","feels_like":"27","pressure":"989","humidity":"31","visibility":"26.6","wind_direction":"南","wind_direction_degree":"201","wind_speed":"30.0","wind_scale":"5","clouds":"15","dew_point":""},"last_update":"2024-05-10T11:07:41+08:00"}]}

        在这部分,我们以【心知天气】返回的JSON数据为例,来说明生成解析代码的方法。
        1、打开ArduinoJson官网。点选页面上方菜单栏 [Assistant],显示以下界面。Board项,选择NodeMCU 1.0 (ESP-12E Module);Mode项,选择 Deserialize ;Input项,选择 String。点击右下方按钮 [Next:JSON]。


        2、复制JSON代码。将上述JSON原始数据复制到页面的【Input】输入框内,点击右下方按钮 【Next:Program】。

        3、生成解析程序代码。以下输入框内,就是使用助手自动生成解析代码。将代码复制到项目中,稍加调整修改后,就可以很方便地取出JSON内的数值了。

        4、测试。新建一个项目,把以下代码复制到 main.cpp 中,然后编译上传到开发板。

#include <Arduino.h>
#include <ArduinoJson.h>

void setup()
{
  // Initialize serial port
  Serial.begin(9600);
  while (!Serial)
    continue;

  // JSON input string.
  const char *json = "{\"results\":[{\"location\":{\"id\":\"WWE0TGW4PX6N\",\"name\":\"济南\",\"country\":\"CN\",\"path\":\"济南,济南,山东,中国\",\"timezone\":\"Asia/Shanghai\",\"timezone_offset\":\"+08:00\"},\"now\":{\"text\":\"晴\",\"code\":\"0\",\"temperature\":\"28\",\"feels_like\":\"27\",\"pressure\":\"989\",\"humidity\":\"31\",\"visibility\":\"26.6\",\"wind_direction\":\"南\",\"wind_direction_degree\":\"201\",\"wind_speed\":\"30.0\",\"wind_scale\":\"5\",\"clouds\":\"15\",\"dew_point\":\"\"},\"last_update\":\"2024-05-10T11:07:41+08:00\"}]}";

  // String input;
  JsonDocument doc;
  DeserializationError error = deserializeJson(doc, json);

  if (error)
  {
    Serial.print("deserializeJson() failed: ");
    Serial.println(error.c_str());
    return;
  }

  JsonObject results_0 = doc["results"][0];

  // Fetch the values
  Serial.println("");
  Serial.println("");

  Serial.print("地区名称: ");
  Serial.println(doc["results"][0]["location"]["name"].as<String>());

  Serial.print("当前天气: ");
  Serial.println(doc["results"][0]["now"]["text"].as<String>());

  Serial.print("当前温度: ");
  Serial.println(doc["results"][0]["now"]["temperature"].as<String>());

  Serial.print("空气湿度: ");
  Serial.println(doc["results"][0]["now"]["humidity"].as<String>());

  Serial.print("能见度: ");
  Serial.println(doc["results"][0]["now"]["visibility"].as<String>());

  Serial.print("风向: ");
  Serial.println(doc["results"][0]["now"]["wind_direction"].as<String>());

  Serial.print("风力: ");
  Serial.println(doc["results"][0]["now"]["wind_scale"].as<String>() + "级");

  Serial.print("风速: ");
  Serial.println(doc["results"][0]["now"]["wind_speed"].as<String>() + "公里/每小时");

  Serial.print("更新时间: ");
  Serial.println(doc["results"][0]["last_update"].as<String>().substring(0, 16));
}

void loop(){}

        如果您在串口监视器看到如下输出信息,那就说明成功了。

四、本项目获取和解析JSON数据结构和功能函数

        关于获取和解析天气信息JSON数据的数据结构和函数,都封装在weather_xinzhi.h文件中,其中主要包括 2 个数据结构 和 4 个功能 函数。
        1、实况天气数据结构:struct weather_now_data{},用于保存解析成功后的实况天气信息。

struct weather_now_data
{
  // 天气现象文字,包括阴晴雨雪等天气状态的描述
  String text = "";
  // 天气现象代码
  int code = -1;
  // 温度,单位为c摄氏度
  String temperature = "0";
  // 体感温度,默认单位:摄氏度
  int feels_like = 0;
  // 大气压强,默认单位:百帕
  int pressure = -1;
  // 相对湿度,0~100,单位为百分比
  int humidity = -1;
  // 能见度,默认单位:公里
  int visibility = -1;
  // 风向文字
  String wind_direction = "";
  // 风向角度,范围0~360,0为正北,90为正东,180为正南,270为正西
  String wind_direction_degree = "-1";
  // 风速,单位为km/h公里每小时或mph英里每小时
  int wind_speed = -1;
  // 风力等级
  String wind_scale = "-1";
  // 数据更新时间(该城市的本地时间)
  String last_update = "";
} wd;

        2、天气预报数据结构:struct weather_tmr_data{},用于保存获取到的天气预报信息。

struct weather_tmr_data
{
  String location_name;              // 预报地区名称
  String date = "";                  // 日期(该城市的本地时间)
  String text_day = "";              // 白天天气现象文字
  String code_day = "";              // 白天天气现象代码
  String text_night = "";            // 晚间天气现象文字
  String code_night = "";            // 晚间天气现象代码
  String high = "";                  // 当天最高温度
  String low = "";                   // 当天最低温度
  String precip = "";                // 降水概率,范围0~1,单位百分比(目前仅支持国外城市)
  String wind_direction = "";        // 风向文字
  String wind_direction_degree = ""; // 风向角度,范围0~360
  String wind_speed = "";            // 风速,单位km/h(当unit=c时)、mph(当unit=f时)
  String wind_scale = "";            // 风力等级
  String rainfall = "";              // 降水量,单位mm
  String humidity = "";              // 相对湿度,0~100,单位为百分比
  String last_update = "";           // 数据更新时间(该城市的本地时间)
};

weather_tmr_data wtd[3];

        3、获取实况天气函数:void get_now_Weather()

// 获取实况天气函数
void get_now_Weather()
{
  // 检查WI-FI是否已连接
  if (WiFi.status() == WL_CONNECTED)
  {
    // 准备发起请求
    std::unique_ptr<BearSSL::WiFiClientSecure> client(new BearSSL::WiFiClientSecure);
    client->setInsecure();
    HTTPClient https;

    // Serial.print("[HTTPS] begin...\n");

    // 请求接口
    if (https.begin(*client, "https://api.seniverse.com/v3/weather/now.json?key=" + key + "&location=" + city_name + "&language=zh-Hans&unit=c"))
    {
      // 获取HTTP的状态码,一般200=成功,出现其他的比如404、500 均为各种报错
      int httpCode = https.GET();
      Serial.println(httpCode);

      if (httpCode > 0)
      {
        if (httpCode == HTTP_CODE_OK || httpCode == HTTP_CODE_MOVED_PERMANENTLY)
        {
          // 从接口获取数据
          String payload = https.getString();
          Serial.println(payload);

          // 调用解析函数
          JsonDocument doc;
          DeserializationError err = deserializeJson(doc, payload);

          if (err.code() == DeserializationError::Ok)
          {
            get_now_weather_data(doc);
          }
          else
          {
            Serial.println("数据解析出错");
          }
        }
      }
      else
      {
        Serial.printf("[HTTP] GET... failed, error: %s\n", https.errorToString(httpCode).c_str());
      }
      https.end();
    }
  }
}

        4、获取未来几天的天气预报函数:void get_tmr_Weather(),可以在 https.begin() 函数中,根据需要修改起始日期和天数。

// 获取未来3天的天气预报
void get_tmr_Weather()
{
  // 检查WI-FI是否已连接
  if (WiFi.status() == WL_CONNECTED)
  {
    // 准备发起请求
    std::unique_ptr<BearSSL::WiFiClientSecure> client(new BearSSL::WiFiClientSecure);
    client->setInsecure();
    HTTPClient https;

    // 请求接口
    if (https.begin(*client, "https://api.seniverse.com/v3/weather/daily.json?key=" + key + "&location=" + city_name + "&language=zh-Hans&unit=c&start=1&days=3"))
    {
      // 获取HTTP的状态码,一般200=成功,出现其他的比如404、500 均为各种报错
      int httpCode = https.GET();
      Serial.println(httpCode);

      if (httpCode > 0)
      {
        if (httpCode == HTTP_CODE_OK || httpCode == HTTP_CODE_MOVED_PERMANENTLY)
        {
          // 从接口获取数据
          String payload = https.getString();
          Serial.println(payload);

          // 调用解析函数
          JsonDocument doc;
          DeserializationError err = deserializeJson(doc, payload);

          if (err.code() == DeserializationError::Ok)
          {
            get_tmr_weather_data(doc);
          }
          else
          {
            Serial.println("数据解析出错");
          }
        }
      }
      else
      {
        Serial.printf("[HTTP] GET... failed, error: %s\n", https.errorToString(httpCode).c_str());
      }
      https.end();
    }
  }
}

        5、提取实况天气具体数值的函数:void get_now_weather_data(JsonDocument &doc)

void get_now_weather_data(JsonDocument &doc)
{
  // 将数据保存到weahter_data 的结构体,方便后续调用
  wd.text = doc["results"][0]["now"]["text"].as<String>();
  wd.code = doc["results"][0]["now"]["code"];
  wd.temperature = doc["results"][0]["now"]["temperature"].as<String>();
  wd.feels_like = doc["results"][0]["now"]["feels_like"].as<int>();
  wd.pressure = doc["results"][0]["now"]["pressure"].as<int>();
  wd.humidity = doc["results"][0]["now"]["humidity"].as<int>();
  wd.visibility = doc["results"][0]["now"]["visibility"].as<int>();
  wd.wind_direction = doc["results"][0]["now"]["wind_direction"].as<String>();
  wd.wind_direction_degree = doc["results"][0]["now"]["wind_direction_degree"].as<String>();
  wd.wind_speed = doc["results"][0]["now"]["wind_speed"].as<int>();
  wd.wind_scale = doc["results"][0]["now"]["wind_scale"].as<String>();
  wd.last_update = doc["results"][0]["last_update"].as<String>().substring(0, 16);
  wd.last_update.replace("T", " ");
}

        6、提取未来天气预报具体数值的函数:void get_tmr_weather_data(JsonDocument &doc)

void get_tmr_weather_data(JsonDocument &doc)
{
  // 将数据保存到weahter_tmr_data 的结构体,方便后续调用
  wtd[0].location_name = doc["results"][0]["location"]["name"].as<String>();                          日期(该城市的本地时间)
  wtd[0].date = doc["results"][0]["daily"][0]["date"].as<String>();                                   日期(该城市的本地时间)
  wtd[0].text_day = doc["results"][0]["daily"][0]["text_day"].as<String>();                           // 白天天气现象文字
  wtd[0].code_day = doc["results"][0]["daily"][0]["code_day"].as<String>();                           // 白天天气现象代码
  wtd[0].text_night = doc["results"][0]["daily"][0]["text_night"].as<String>();                       // 晚间天气现象文字
  wtd[0].code_night = doc["results"][0]["daily"][0]["code_night"].as<String>();                       // 晚间天气现象代码
  wtd[0].high = doc["results"][0]["daily"][0]["high"].as<String>();                                   // 当天最高温度
  wtd[0].low = doc["results"][0]["daily"][0]["low"].as<String>();                                     // 当天最低温度
  wtd[0].precip = doc["results"][0]["daily"][0]["precip"].as<String>();                               // 降水概率,范围0~1,单位百分比(目前仅支持国外城市)
  wtd[0].wind_direction = doc["results"][0]["daily"][0]["wind_direction"].as<String>();               // 风向文字
  wtd[0].wind_direction_degree = doc["results"][0]["daily"][0]["wind_direction_degree"].as<String>(); // 风向角度,范围0~360
  wtd[0].wind_speed = doc["results"][0]["daily"][0]["wind_speed"].as<String>();                       // 风速,单位km/h(当unit=c时)、mph(当unit=f时)
  wtd[0].wind_scale = doc["results"][0]["daily"][0]["wind_scale"].as<String>();                       // 风力等级
  wtd[0].rainfall = doc["results"][0]["daily"][0]["rainfall"].as<String>();                           // 降水量,单位mm
  wtd[0].humidity = doc["results"][0]["daily"][0]["humidity"].as<String>();                           // 相对湿度,0~100,单位为百分比
  wtd[0].last_update = doc["results"][0]["last_update"].as<String>().substring(0, 16);                // 数据更新时间(该城市的本地时间)
  wtd[0].last_update.replace("T", " ");
}

        7、说明。关于 main.cpp 内容的修改,主要集中在调用既有功能函数和调整显示位置方面,请大家自行阅读代码。
        特别提醒修改两项内容:(1)WiFi连接的 ssid 和 password;(2)心知天气私钥。
        免费用户获的信息数据受到限制,可以申请开通试用版,测试体验全部信息数据。

// 项目 main.cpp 主要内容
#include <Arduino.h>
#include <TFT_eSPI.h>
#include <ArduinoJson.h>

// 连接wifi用的库
#include <ESP8266WiFi.h>
#include <ESP8266HTTPClient.h>

/**
   配置您所在环境的wifi 账号和密码
   注意:切勿连接 5G的频率、不要使用双频融合(路由器)
   注意:账号密码必须完全正确,包括字母大小写、空格、中划线、下划线
*/
const char *ssid = "xcb940";
const char *password = "87589940abc";

//const char *ssid = "LGCWZS";
//const char *password = "87129168";

// 构造函数,实例化 TFT 屏幕对象
TFT_eSPI tft = TFT_eSPI();

// WiFi 连接函数声明
void connectWiFi();

// 程序用到的字库文件,后面会详细说明
#include "hefeng-min-40px.h"
#include "weather_font20.h"
#include "weather_font16.h"

// 网络时钟的刷新频率
unsigned long last_ntp = 0;
const long interval_ntp = 1000; // 网络时钟的刷新频率(毫秒)

// 今日天气的刷新频率
unsigned long last_weather = 0;
const long interval_weather = 1000 * 60 * 5; // 今日天气的刷新频率(毫秒),每300毫秒更新一次

// 明日天气的刷新频率
unsigned long last_tmr_weather = 0;
const long interval_tmr_weather = 1000 * 60 * 60; // 明天天气的刷新频率(毫秒),每3600毫秒更新一次

// 温湿度传感器的刷新频率
unsigned long last_sht = 0;
const long interval_sht = 3000; // 温湿度传感器的刷新频率(毫秒),每3000毫秒更新一次

// 心知天气免费版密钥**************
// const String key = "**************";

// 心知天气试用版密钥**************,有效期14天,自2024-05-09始
const String key = "**************";

// 你的城市ID,获取方法参考课程
const String city_name = "jinan";

#include "sht30.h"
#include "ntptime.h"
#include "weather_xinzhi.h"

void setup()
{
  Serial.begin(115200);

  tft.init();
  tft.setSwapBytes(true);
  tft.setRotation(0);
  tft.fillScreen(TFT_BLACK);
  tft.setTextColor(TFT_WHITE, TFT_BLACK, true);

  tft.setTextSize(2);

  // 联网
  tft.println("Wi-Fi >> " + String(ssid));
  // initWiFi();
  connectWiFi();

  tft.println("");
  tft.setTextColor(TFT_WHITE, TFT_BLACK, true);

  tft.println("");

  // 今日天气
  tft.println("Get Today weather");
  get_now_Weather();
  get_tmr_Weather();

  tft.setTextSize(1);
  tft.println("");
  tft.setTextSize(2);

  // 对时
  tft.println("Get NTP Time");
  initNtp();

  // 温湿度传感器
  tft.println("");
  tft.println("load Sensor Data..");
  sht30_setup();

  tft.setTextColor(TFT_WHITE, TFT_BLACK, true);
  tft.println("");
  tft.println("start...");

  delay(500);
  tft.fillScreen(TFT_BLACK);
}

void loop()
{
  // 获取单片机启动至今的毫秒数
  unsigned long currentMillis = millis();

  // 显示当前日期,星期几,农历
  // update ntp 时间
  if (last_ntp == 0 || currentMillis - last_ntp >= interval_ntp)
  {
    last_ntp = currentMillis;
    loopNtp();

    tft.loadFont(weather_font16);
    tft.setTextColor(TFT_WHITE, TFT_BLACK, true);
    tft.drawString(dt.localDate + "  " + weekOfDate1(dt.year, dt.month, dt.day) + " " + outputLunarDate(dt.year, dt.month, dt.day), 0, 0);
    tft.unloadFont();

    tft.setTextSize(5);
    tft.setTextColor(TFT_GREEN, TFT_BLACK, true);
    tft.drawString(dt.localTime, 0, 30);
  }

  // 今日天气
  if (last_weather == 0 || currentMillis - last_weather >= interval_weather)
  {
    if (last_weather > 0)
    {
      get_now_Weather();
    }
    last_weather = currentMillis;

    // 擦除指定区域
    tft.fillRect(55, 90, 240, 40, TFT_BLACK);

    tft.setTextColor(TFT_YELLOW, TFT_BLACK, true);
    tft.loadFont(hefeng40);
    // tft.drawString(icon(wd.now_icon), 10, 90);
    tft.unloadFont();

    tft.loadFont(weather_font20);
    tft.setTextColor(TFT_WHITE, TFT_BLACK, true);
    tft.drawString("Now: " + String(wd.temperature) + "° " + wd.text, 55, 90);
    tft.drawString(wd.wind_direction + "风" + String(wd.wind_scale) + "级 " + wd.wind_speed + "KM/H", 55, 110);

    tft.drawLine(0, 140, 240, 140, TFT_WHITE);
  }

  // 明日天气
  if (last_tmr_weather == 0 || currentMillis - last_tmr_weather > interval_tmr_weather)
  {
    if (last_tmr_weather > 0){
      get_tmr_Weather();
    }
    last_tmr_weather = currentMillis;

    // 擦除指定区域
    tft.fillRect(55, 150, 240, 40, TFT_BLACK);

    tft.loadFont(hefeng40);
    tft.setTextColor(TFT_YELLOW, TFT_BLACK, true);
    // tft.drawString(icon(wtd[1].iconDay), 10, 150);
    tft.unloadFont();

    tft.loadFont(weather_font20);
    tft.setTextColor(TFT_WHITE, TFT_BLACK, true);
    tft.drawString("明天 " + String(wtd[0].low) + "° - " + String(wtd[0].high) + "°", 55, 150);
    tft.drawString(wtd[0].text_day+ ", " + "风力" + wtd[0].wind_scale + "级", 55, 170);

    // 这条线其实没必要重新绘制
    tft.drawLine(0, 200, 240, 200, TFT_WHITE);
  }

  // 温湿度传感器的数据
  if (last_sht == 0 || currentMillis - last_sht > interval_sht)
  {
    last_sht = currentMillis;

    sht30();

    tft.loadFont(weather_font20);
    tft.setTextColor(TFT_WHITE, TFT_BLACK, true);

    tft.drawString("室温:", 0, 210);
    tft.setTextColor(TFT_GREEN, TFT_BLACK, true);
    tft.drawString(String(sht_data.temperature) + "C ", 40, 210);

    tft.setTextColor(TFT_WHITE, TFT_BLACK, true);
    tft.drawString("湿度", 120, 210);

    tft.setTextColor(TFT_GREEN, TFT_BLACK, true);
    tft.drawString(String(sht_data.humidity) + "%  ", 170, 210); //+3(原为120)的x轴位置,因为视觉上似乎贴的优点近了
  }
}

// 连接wifi
void connectWiFi()
{

  Serial.print("Connecting to ");
  Serial.println(ssid);

  // 设置WiFi工作在终端模式,参数可选填WIFI_AP、WIFI_STA、WIFI_AP_STA、WIFI_OFF
  WiFi.mode(WIFI_STA);

  // 开始连接
  WiFi.begin(ssid, password);

  // 检查连接是否成功
  while (WiFi.status() != WL_CONNECTED)
  {
    delay(500);
    Serial.print(".");
    tft.print(".");
  }

  tft.println("");
  tft.println("");
  tft.setTextColor(TFT_GREEN, TFT_BLACK, true);
  tft.println(WiFi.localIP());

  // 设置:当路由器断开连接时,是否启动自动重新连接功能。true: 启用自动重新连接;false:不启用此功能
  WiFi.setAutoReconnect(true);

  // 设置:是否将WiFi参数保存于Flash中,默认为true,即在每次调用WiFi.begin()、WiFi.softAP()、WiFi.disconnect、WiFi.softAPdisconnect方法时都会将相关数据写入到Flash中;
  //       当设置为false时,以上动作将不会把数据写入Flash中,仅仅改变内存中的WiFi设置
  WiFi.persistent(true);

  // 连接成功后,在串口监视器显示自身IP地址,以下5行代码作调试用
  Serial.println("");
  Serial.println("WiFi connected");
  Serial.println("IP address: ");
  Serial.println(WiFi.localIP());
  Serial.println("");
}

五、项目运行展示

NodeMcu1.0_天气时钟温湿度展示

六、项目源代码下载

        百度网盘下载地址:
        https://pan.baidu.com/s/1-cENF6LaInjF3sw_qs40Vg?pwd=me32,
        提取码:me32

        部分代码来自于网络大神,如有异议,请联系。

参考文档
1. JSON 基本使用_json怎么用-CSDN博客
2. JSON——概述、JSON语法、序列化和反序列化_所有文档都可以通过json序列化吗-CSDN博客

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/623253.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

uniapp小程序:大盒子包裹小盒子但是都有点击事件该如何区分?

在开发过程中我们会遇到这种情况&#xff0c;一个大盒子中包裹这一个小盒子&#xff0c;两个盒子都有点击事件&#xff0c;例如&#xff1a; 这个时候如果点击评价有可能会点击到它所在的大盒子&#xff0c;如果使用css中的z-index设置层级的话如果页面的盒子多的话会混乱&…

AI工具的热门与卓越:揭示AI技术的实际应用和影响

文章目录 每日一句正能量前言常用AI工具创新AI应用个人体验分享后记 每日一句正能量 我们在我们的劳动过程中学习思考&#xff0c;劳动的结果&#xff0c;我们认识了世界的奥妙&#xff0c;于是我们就真正来改变生活了。 前言 随着人工智能&#xff08;AI&#xff09;技术的快…

极端天气对气膜建筑有什么影响吗—轻空间

气膜建筑在近年来的发展迅速&#xff0c;逐渐替代了一部分传统建筑&#xff0c;展现了良好的市场前景。然而&#xff0c;面对自然环境中的极端天气&#xff0c;如暴风、暴雨和暴雪&#xff0c;气膜建筑是否能够经受住考验是大家关注的焦点。轻空间带您探讨一下这些极端天气对气…

【漏洞复现】泛微OA E-Cology ResourceServlet文件读取漏洞

漏洞描述&#xff1a; 泛微OA E-Cology是一款面向中大型组织的数字化办公产品&#xff0c;它基于全新的设计理念和管理思想&#xff0c;旨在为中大型组织创建一个全新的高效协同办公环境。泛微OA E-Cology ResourceServlet存在任意文件读取漏洞&#xff0c;允许未经授权的用户…

Nurbs曲线

本文深入探讨了Nurbs曲线的概念、原理及应用&#xff0c;揭示了其在数字设计领域的独特价值和广泛影响。Nurbs曲线作为一种强大的数学工具&#xff0c;为设计师们提供了更加灵活、精确的曲线创建方式&#xff0c;从而极大地提升了设计作品的质感和表现力。文章首先介绍了Nurbs曲…

大数据之 Hadoop概述

用最简洁的语言跟大家表达我最想分享的知识 。 什么是Hadoop Hadoop框架核心模块 HDFS MapReduce Yarn Hive HBase Phoenix Zookeeper Impala Spark 分布式计算-Spark与Impala与Presto与Tez 今天主要跟大家简述一下hadoop&#xff0c;主要是图片的形式跟大家介绍&#xff0c;希…

Rpcx (二):传输

一、Transport 传输 rpcx 可以通过 TCP、HTTP、UnixDomain、QUIC和KCP通信。你也可以使用http客户端通过网关或者http调用来访问rpcx服务。 TCP 这是最常用的通信方式。高性能易上手。可以使用TLS加密TCP流量。 Example: 101basic 服务端使用 tcp 做为网络名并且在注册中心…

稚晖君独家撰文:具身智能即将为通用机器人补全最后一块拼图

具身智能新纪元。 *本文为稚晖君独家供稿,「甲子光年」经智元机器人授权发布。稚晖君本名彭志辉,先后任职OPPO、华为,现为智元机器人CTO、首席架构师。 在ChatGPT之后,又一个大模型概念火了——具身智能(Embodied AI)。 在学术界,图灵奖得主、上海期智研究院院长姚期…

IOS 苹果IAP(内购)之创建沙盒账号

IOS 苹果IAP&#xff08;内购&#xff09;之创建沙盒账号 沙盒账号是什么&#xff1f;沙盒账号创建的前提条件沙盒账号创建沙盒账号使用流程沙盒账号注意事项 沙盒账号是什么&#xff1f; 如果IOS应用里面用到了苹果应用内付费&#xff08;IAP&#xff09;功能&#xff0c;那么…

办公软件_EdrawMax 免安装版教程 (亿图图示综合图形图表设计软件)

前言 万兴亿图图示(Wondershare EdrawMax)是一款综合图形图表设计软件,Visio国产替代.亿图图示中文版(Edraw Max)是一款办公绘图软件的思维导图软件.无需任何绘图功底,即可轻松创建各类思维导图.亿图图示专家,提供大量事例和在线模板,用于创建流程图,信息图,组织结构图,科学教…

面试题:调整数字顺序,使奇数位于偶数前面

题目&#xff1a; 输入一个整数数组&#xff0c;实现一个函数&#xff0c;来调整该数组中数字的顺序 使得所有奇数位于数组的前半部分&#xff0c;所有偶数位于数组的后半部分 算法1&#xff1a; 利用快速排序的一次划分思想&#xff0c;从2端往中间遍历 时间复杂度&#x…

day5

利用迭代器&#xff01; #include <vector> #include <map>class Solution { public:std::vector<int> intersection(std::vector<int>& nums1, std::vector<int>& nums2) {std::map<int, int> Mymap;std::vector<int> qq…

金万维动态域名小助手怎么用?

金万维动态域名小助手是一个域名检测工具&#xff0c;使用此工具可以进行检测域名解析是否正确、清除DNS缓存、修改DNS服务器地址及寻找在线客服&#xff08;仅支持付费用户&#xff09;等操作。对不懂网络的用户是一个很好的检测域名的工具&#xff0c;下面我就讲解一下金万维…

面试中的算法(查找缺失的整数)

在一个无序数组里有99个不重复的正整数&#xff0c;范围是1~100&#xff0c;唯独缺少1个1~100中的整数。如何找出这个缺失的整数? 一个很简单也很高效的方法&#xff0c;先算出1~100之和&#xff0c;然后依次减去数组里的元素&#xff0c;最后得到的差值&#xff0c;就是那个缺…

ARM架构安全特性之通用平台安全服务

安全之安全(security)博客目录导读 目录 一、符合PSA认证标准 二、Arm平台安全规范 三、跨安全边界通信 四、FF-A 五、FF-M 六、开放和标准设备固件 七、Trustedfirmware.org 在一个需要高度信任设备的世界中&#xff0c;每个设备都必须是独一无二的可识别的、不可克隆…

网站服务器备案及域名购买配置教程

一、阿里云服务备案准备工作 1.什么是备案? 备案是指向相关部门提交网站信息,以便监管和管理互联网信息服务,未经备案的网站可能面临罚款甚至被关闭的风险。备案主要看您的网站或App等互联网信息服务解析到的服务器是否在中国内地(大陆),如果服务器在中国内地(大陆),…

携手鲲鹏昇腾 HashData展现云原生数仓创新力量

​5月9日-11日&#xff0c;鲲鹏昇腾开发者大会2024在北京中关村国际创新中心举行&#xff0c;众多行业领袖、专家学者及优秀开发们齐聚一堂&#xff0c;分享产业趋势、技术创新和应用实践。 酷克数据作为华为鲲鹏生态重要合作伙伴&#xff0c;受邀出席本次大会&#xff0c;展示…

springboot学习整理

视频&#xff1a;基础篇-01_springboot概述_哔哩哔哩_bilibili 介绍 spring boot 是spring提供的一个子项目&#xff0c;用于快速构建spring应用程序 spring构建&#xff1a; 1 导入依赖繁琐 &#xff1b; 2 项目配置繁琐 spring Framework: 核心 spring Boot :快速构建spring…

Spring的IOC和AOP机制?

我们是在使用Spring框架的过程中&#xff0c;其实就是为了使用IOC&#xff0c;依赖注入&#xff0c;和AOP&#xff0c;面向切面编程&#xff0c;这两个是Spring的灵魂。 主要用到的设计模式有工厂模式和代理模式。 IOC就是典型的工厂模式&#xff0c;通过sessionfactory去注入…

能效?性能?一个关于Windows下使用openssl speed进行速度测试的诡异问题

问题描述 最近的某个软件用到了openssl&#xff0c;所以就想着测试一下速度。我的电脑是惠普的&#xff0c;CPU是AMD Ryzen 7 PRO 6850HS&#xff0c;系统是Win11。我使用openssl自带的speed测试加密/解密的速度&#xff0c;命令大致如下&#xff1a; openssl speed -evp aes…