[嵌入式系统-69]:RT-Thread-组件:网络组件“组”,RT-Thread系统通向外部网络世界的入口

目录

RT-Thread 提供的网络世界入口 - 网络组件

1. 总概

2. AT

3. Lwip: 轻量级IP协议栈

4. W5500

5. Netdev

6. RT-Thread SAL(Socket Abstraction Layer)套接字和BSD套接字区别

RT-Thread SAL 套接字接口示例

BSD 套接字接口示例


RT-Thread 提供的网络世界入口 - 网络组件

作为一款在 RTOS 领域对网络支持很丰富的 RT-Thread,对设备联网功能的支持的工具就是接下来要讨论的 net 组件。它位于 ./rt-thread/components/net 路径下;作为一款基础组件, env 与 Studio 的工程配置界面的配置项都依赖该目录下的 Kconfig 文件。我们对网络功能的选择,或者其配置都属于该 Kconfig 的管理范畴。

从图片中的信息也可以了解到,AT,Lwip,Netdev,SAL 这些我们相对熟悉的功能也位于该目录中。这四个基础组件,可以驱动的网络设备的种类多到各个方面。

从有线网络的各种 ETH 的方式,带 PHY 的 enc28j60,纯接口的 RJ45,还有自带网络协议栈的 W5500都可以直接经过配置,通过我们的 net 组件实现设备联网的需求;除了有线网络的接入方式,无线网络的接入方式则更多,2G模块,3G模块,4G模块,Cat-1模块,Cat-4模块,NB-IOT模块,乃至 5G 模块这些依赖基站运营商的入网方式,例如 SIM800,EC20,AIR720,L610,N58,M5311 等,这些不同厂家,不同工作频率的模组均可以通过 NET 组件入网;WIFI 这种无需运营商直接提供的网络的入网方式,例如 ESP8266,W60x,rw007 等。

通过上面的简单介绍,大家应该可以基本了解 Net 组件的作用,即为使用 RT-Thread 的设备提供上网能力。上面的联网设备,品类繁多,方式各异,但其实追踪到依赖的基础 net 能力其实也没有多复杂。实际上大部分的功能依赖的基础能力,只有两组而已,即 AT ,Lwip;而到用户,也就开发者层面,大部分就只用到 SAL 这一个概念。

相信大家看过这份文档,主要描述了 SAL 提供的编程接口,也就是大家常用的 socket 接口;通过 SAL 接口大家已经可以实现 MQTT,NTP,TFTP,TCP Client,TCP Server,webclient,UDP Client,UDP Server,webnet 等这些大家经常会使用的工具。像浏览器一样访问网站,webclient;模拟一个网站,webnet;连接各种云平台,Mqtt;获取准确授时,NTP ;通过网络传输文件 ,TFTP ;诸如此类功能, SAL 提供了轻松访问网络的一把钥匙。你可以理解这把钥匙是向上的,但是这篇文章现在要讲的,是向下的,看看 SAL 这样组件,又是谁在支撑着它。

Net 组件的主要内容有 4 个方面,分别是 sal ,AT,lwip 与 netdev,我们先从 AT 和 lwIP 讲起,在不同的方式中介绍 sal 的用途;然后讲 netdev 的作用。

1. 总概

对于一个给定的模组,他的联网方式有时并不是单一的;以常见的 Cat-4 (4G) 模块为例,一方面它可以使用 AT 命令这种经典的方式来实现网络连接(并不一定是通过串口传送AT命令,也可以通过线程间通信传送AT命令);另一方面又可以通过 PPP 拨号,使用 lwip 的方式来实现网络连接。或者在以 W60x 为例,一方面可以使用 AT 的方式来实现联网功能;另一方面又可以使用裸 IP 包的方式来实现网络连接。在选取上网模块时,应当考虑片上资源的情况,以及网络质量的问题,选择比较适合的联网方式。条条大路通罗马,短途长途,步行开车还是有些小区别的。

2. AT

AT 命令一开始并没有那么丰富,一开始是用作拨号的功能,也就是电话业务方面的内容。后来随着时间发展,AT 命令的范围也一步步进行了扩展,而逐渐支持了网络的一些功能,例如 TCP ,UDP 的功能;而到现在,有些模块厂商甚至直接使用 AT 命令扩展的方式,实现了诸如 MQTT,HTTP,HTTPS 之类的功能。这些使用扩展 AT 命令方式实现的上层应用对模块依赖度高,在 RT Thread 的 IOT Package 的栏目中,可以留意到这样的软件包,例如:bc28_mqtt。RT-Thread 的 AT socket 功能,则是更接近底层,实现 TCP UDP 这样的接口,通过 SAL 来实现 MQTT 这种上层应用。

RT-Thread 的 AT 组件的设计目的就是使设备能发送和解析 AT 命令。为了到达这个目的,契合 AT 命令的常见逻辑,提供了AT client 功能和 AT Server 功能。

  • 对于 AT Client 来说,是由 MCU 去向 Modem 发送 AT 命令,由 Modem 响应命令,并给出回复。这种方式我们常用的,也是我们通过支持 AT 的模组上网的常用方式。
  • 对于 AT Server 来说,是由 MCU 作为类似 Modem 的功能,由其他 MCU 向这个 MCU 发送 AT 命令,由这个 MCU 响应 AT 命令并做出回复。

具体的 AT 的使用,可以参考 AT 文档;更加详细,底层的实操的内容在专门的文档中有描述;在查阅文档时,可以同时浏览代码,通过文章的说明,我们应该能了解到 AT 组件的使用方法。AT 组件的适用范围也进而清晰起来:

  • AT 命令用作模块控制
  • AT 命令用作数据的解析

对 AT Client 功能做大规模应用的软件包为 at_device ,作为一款使用频次很高的软件包,可以看到他充分利用了 AT 组件的能力,一方面针对模块控制的能力设置了 at_device_xxx.c 来提供模块的基础控制操作;一方面也提供了 at_socket_xxx.c 来提供数据解析能力。对于整体的 net 组件而言,at_socket_xxx.c 实际上沟通了 AT 组件与 SAL,使得上层应用能依赖 at_socket_xxx.c 的函数,实现对应的网络功能 MQTT,HTTP....

static const struct sal_socket_ops at_socket_ops =
{at_socket,at_closesocket,at_bind,NULL,at_connect,NULL,at_sendto,at_recvfrom,at_getsockopt,at_setsockopt,at_shutdown,NULL,NULL,NULL,
#ifdef SAL_USING_POSIXat_poll,
#endif /* SAL_USING_POSIX */
};static const struct sal_netdb_ops at_netdb_ops =
{at_gethostbyname,NULL,at_getaddrinfo,at_freeaddrinfo,
};复制错误复制成功

从 AT 组件的 at/at_socket/at_socket.h 中可以看到 AT 组件实现的 sal 的一部分函数,从这些函数命令也能看出来 at_socket 是不支持 server 模式的。

对于 at_device 软件包来说,at_socket_xxx.c 的内容,也主要是对 at_socket.h 内容的实现,例如:

/* AT socket operations function */
struct at_socket_ops
{int (*at_connect)(struct at_socket *socket, char *ip, int32_t port, enum at_socket_type type, rt_bool_t is_client);int (*at_closesocket)(struct at_socket *socket);int (*at_send)(struct at_socket *socket, const char *buff, size_t bfsz, enum at_socket_type type);int (*at_domain_resolve)(const char *name, char ip[16]);void (*at_set_event_cb)(at_socket_evt_t event, at_evt_cb_t cb);int (*at_socket)(struct at_device *device, enum at_socket_type type);
};复制错误复制成功

如果你能浏览对应的 at_device 中 at_socket_xxx.c 的代码,你可以发现几乎都是上述函数的实现,都是根据不同的模块的特性来处理对应逻辑。


在论坛中也能看到 at_device 的影响:

  1. at_device使用bc28报错
  2. 4.0.3版本串口接收buff设置大于128时,使用at_device有问题
  3. air720 at-device回复命令冲突
  4. AT组件接收不到数据
  5. .....

除了这些使用 at_device 能搜索到的问题,还有一些因为使用了 at_device 而产生的问题;这些问题的出现,对于在使用 AT 组件时的使用体验影响是很大的。

在遇到问题时,希望能小心地验证问题,并尝试解决问题;在问题难以解决时,请详细描述问题复现的场景,按照格式在论坛中提问。问题描述清晰,复现手段准确,更能吸引小伙伴们的参与,共同解决问题。

  • AT RAW 能准确显示 AT 交互中的 modem 返回的数据,再数据异常需要调试时,建议打开。

市面上的各种模块,at_device 已经囊括了最常使用的一些部分,不过还有更多的模块等待着小伙伴们一起帮忙完善。也许是因为在小伙伴提交代码时并没有那么多的业务量或者功能,在一次次的功能增加迭代中,导致了 at_device 有时不能做到随取随用而产生了一些问题,也希望使用的小伙伴能发现问题,做出修改并给予 at_device 软件包自己的 PR,一起推动 at_device 更加易用。

3. Lwip: 轻量级IP协议栈

大家基本都是看过这个 TCP/IP 模型的,在嵌入式的网络协议栈入口中,lwip 就是很重要的一个入口;

lwip 是开源的一个协议栈,已经经过很多厂家和开发者使用后,得到验证和肯定的嵌入式 TCP/IP 网络协议栈,lwip 的全称为 A Lightweight TCP/IP stack,一个轻量级的 TCP/IP 协议栈。

所以可以知道:lwip 并不是嵌入式上唯一的 TCP/IP 协议栈,但 lwip 是经过验证的,能胜任工作的 TCP/IP 协议栈。

在 RT-Thread 中,提供了三个 lwip 的版本供用户使用: lwip1.4.1,lwip2.0.2,lwip2.1.2 ;

通常大家在使用有线连接的方式使用网络时,一部分是使用的 lwip 协议栈,而另一部分则是使用了 w5500 这种方式来实现。

难道 lwip 只有在有线网络中才能使用吗?其实,也不尽然。对于使用裸 IP 包方式实现的网络数据的交互,而不是简单使用 AT 命令的话;rw007 这种 WIFI 设备,Air720 这种 Cat-4 设备,都是可以使用 Lwip 来使用网络功能的。

AT 方式lwip 方式
rw007AT 命令实现数据交互(与模块版本有关)裸 IP 包形式实现数据交互
Air720AT 命令实现数据交互PPP 包形式实现数据交互

通过表格可以发现 net 组件的意图:即通过数据交互的方式来实现网络数据传输。对于 AT 方式来说,网络数据在 AT 命令中,需要从 AT 数据中解析出想要使用的网络数据;这种方式不需要 Lwip 庞大的协议栈,但很明显需要比较强大的解析能力,而且频繁转换解析对网络效率也是有影响的。

对于 lwip 方式来说,网络数据直接以 IP/PPP 包的方式传输,这是网络世界通用的格式;无论是我们的交换机,路由器,它们都能识别 IP 报文,更加轻便而且无需进行复杂的相互转换,但这种方法需要有 lwip 协议栈的参与,需要根据项目对 lwip 做一些细微得调整。

在 RT-Thread 中,我们对 lwip 做了一些小修改,也使用 Kconfig 的方式提供了一些简单的优化配置项供用户进行简单的配置。在上面提到,我们对 lwip 有一些修改,主要的修改内容包括 netif 与 mem 方面;在 netif 中,我们在上层抽象了 netdev 的结构,对于 lwip 而言是对 netif 的继承和常用的一些接口函数的函数,而对于不是使用 lwip 实现的网络通信功能来说,netdev 可以使开发者更轻松的使用;在 mem 方面,我们使用 RT-Thread 自身的内存分配算法取代了 lwip 的内存堆分配算法,不包括内存池部分。查看 lwipx.x.x 下的j脚本文件 Sconscript 文件,可以留意到.\src\core\mem.c文件是没有参与编译的;也就是说在使用 pbuf_alloc(PBUF_RAW, len, PBUF_RAM)语句来申请 pbuf 空间时,我们实际上使用得是 RT-Thread 通用的内存堆算法。当然无论是 lwip 还是 RT-Thread ,内存堆与内存池的申请确实是各有优劣的,还是需要针对使用场景才能确定孰优孰劣。在本篇文章中,主要讲解 lwip 在 net 中的作用,具体的 lwip 配置方法和内容讲解会在另一篇文章中仔细描述。

4. W5500

我们经常使用的 net 组件中的两项已经介绍了基本情况;除了这两种方式,w5500 作为一个拥有 TCP/IP 协议栈的一个独立模块,也是一种通用的方式。在 RT-Thread 中使用 w5500 的方式也很简单。只需在工程中使用 wiznet 软件包就可以使用标准的 BSD socket 接口来实现网络编程了。

通过网络框架图可以看到:wiznet 是独立与 AT socket 与 lwip 的,但还是需要 netdev 与 SAL 的参与才能轻松的使用。

从sal,netdev,w5500 的对接过程来看,在 BSD socket 的接口方面,只是实现了常用的一些接口;这些接口已经囊括了 IOT 设备中会使用的常用能力。

static const struct sal_socket_ops wiz_socket_ops =
{wiz_socket,wiz_closesocket,wiz_bind,wiz_listen,wiz_connect,wiz_accept,wiz_sendto,wiz_recvfrom,wiz_getsockopt,wiz_setsockopt,wiz_shutdown,NULL,NULL,NULL,
#ifdef SAL_USING_POSIXwiz_poll,
#endif /* SAL_USING_POSIX */
};static const struct sal_netdb_ops wiz_netdb_ops =
{wiz_gethostbyname,NULL,wiz_getaddrinfo,wiz_freeaddrinfo,
};复制错误复制成功

在 w5500 的使用中,也有一部分小伙伴会遇到问题。比如,一些 Linux 上的网络应用在迁移到 w5500 的平台上时会出现一些参数的不兼容问题。比如设置某些 socket 的属性,设置一个 TCP 的属性等等;有些设置在 Linux 平台上可能是可以使用的,有些则可能会产生编译或者执行过程的逻辑异常,不符合函数预期。

为了在嵌入式平台上实现网络功能,AT命令,lwip 协议栈,wiznet 这些方法都是实现了 TCP/IP 协议中的一大部分功能;并没有完全实现整个协议栈的内容。可能有些函数特性在嵌入式的平台上是不太符合预期的。这时候就需要发现问题的小伙伴及时在论坛中提出问题,更多的小伙伴参与讨论,才能使我们的 net 组件功能更丰富。

5. Netdev

在 lwip 的介绍中,我们提到 netdev 是对 lwip 中 netif 的继承。因为 netdev 从 netif 中取出了一些字节用于关键信息的填充。我们在 cmd 控制台上,经常会用到 ifconfig命令,然后得到下面的回复:

在控制台中可以显示每个注册为 netdev 的信息,IP地址,网关地址,子网掩码,以及 DNS 服务器地址;还有 MTU 长度,MAC 地址,以及该 netdev 的属性(是否启用,链路是否正常,网络是否正常,DHCP 是否打开,ARP 功能是否开启,广播是否支持,IGMP组播能力是否支持)。这些内容中经常使用到的一般有三个属性:

重要属性意义
UP / DOWN网卡是否启用
LINK_UP / LINK_DOWN链路是否正常
INTERNET_UP / INTERNET_DOWN网络是否正常
  • LINK_UP 的意义为链路是否正常,在 netdev 启用后,LINK_UP 是作为 PING 命令是否可用的关键。如果 为 LINK_DOWN ,PING 命令将禁用。毕竟,网线都没有连接,PING 命令不可以用也是合情合理的喽。LINK_DOWN 状态下禁用 PING 命令很合理,但是如果 netdev 的状态被异常置为 LINK_DOWN 就很苦恼了。如果被代码错误执行到设置为 LINK_DOWN,PING 命令将不可用。netdev 的使用,可以参考文档介绍。
  • INTERNET_UP 的意义为网络是否正常。实际上,RT-Thread 的代码逻辑中,把这个位作为标志位,但很少利用这个标志位做逻辑判断。这样做的原因,是因为判断 INTERNET_UP 的方式为访问 link.rt-thread.org,如果能有回复,就把标志位置为 INTERNET_UP,否则置为 INTERNET_DOWN。这个逻辑在一般情况下算是正常,但是在一些网络拓扑中,很有可能被网络标记位不可达(被墙了,PS :也有可能是 RT-Thread 的网络主机重启了),所以哪怕实际上是可以连接网络的,却显示为 INTERNET_DOWN。

除此以外,netdev 还有 PING 功能的抽象,就是上面提到的 LINK_DOWN 状态不可用的那个。说它是抽象,是因为 netdev 中并没有直接实现全部的 ping 逻辑,而是依赖底层的提供提供的 PING 能力。

也许会有小伙伴想问,netdev 没有提供 ping 功能是因为代码不够高大上,能力不足吗?都是连接到一个 IP 地址上面,使用 socket 的接口不可以吗?为什么一定要抽象出一个 ping 接口,让底层去实现这个接口,而不是 netdev 层直接实现呢?其实这并不是这个原因,因为 ping 功能发送的数据,和通过 socket 发出的数据,格式并不是完全相同的。虽然统称为 IP 数据报文,但实际上 netdev 这个层级是感受不到这个数据的。具体的内容可以在下一篇 lwip 的讲解中来一起解释。

6. RT-Thread SAL(Socket Abstraction Layer)套接字和BSD套接字区别

RT-Thread SAL(Socket Abstraction Layer)套接字和BSD套接字在功能和使用上存在一些区别,尽管它们都用于网络通信,但各有其特定的设计和应用场景。

RT-Thread SAL套接字:

  1. 定义与功能:SAL是RT-Thread操作系统中的一个组件,用于抽象和统一多种网络协议栈接口。它提供了一个Socket层面的TLS加密传输特性,并支持标准BSD Socket API。SAL套接字抽象层位于网络硬件层和应用层之间,通过它,RT-Thread系统能够适配下层不同的网络协议栈,并提供给上层统一的网络编程接口,方便不同协议栈的接入
  2. 特点与优势:SAL套接字的主要特点包括抽象和统一多种网络协议栈接口,提供统一的FD(文件描述符)管理,以及支持标准BSD Socket API等。这使得开发者可以更方便地使用read/write、poll/select等函数来操作网络功能。此外,SAL的孕育而出,使得RT-Thread可以无缝接入各式各样的网络芯片或模块,极大地提升了RT-Thread在IoT领域对于不同网络硬件的兼容性。

BSD套接字:

  1. 定义与功能:BSD套接字最早是由BSD(伯克利软件套件)在1982年引入的通信机制,目前已被广泛移植到主流的操作系统中。对于应用开发人员来说,套接字(socket)是一个抽象层,是一种特殊的I/O接口,独立于具体协议的网络编程接口,也是一种文件描述符。应用程序可以通过它发送或接收数据,可对其进行像对文件一样的打开、读写和关闭等操作。套接字允许应用程序将I/O插入到网络中,并与网络中的其他应用程序进行通信。
  2. 特点与优势:BSD套接字具有跨平台、兼容性好、易于理解和使用等优点。它支持各种形式的网络应用,并且还是一种进程间通信的机制。套接字可以看成是一种特殊的管道,但与管道不同的是,套接字所能容纳的数据不受限制。

总结来说,RT-Thread SAL套接字和BSD套接字在功能和使用上存在一些差异。RT-Thread SAL套接字是RT-Thread操作系统中的一个组件,主要用于抽象和统一多种网络协议栈接口,提升系统的兼容性和易用性;而BSD套接字则是一种广泛使用的网络通信机制,具有跨平台、兼容性好等优点,适用于各种网络应用。在实际应用中,可以根据具体需求和场景选择合适的套接字类型。

RT-Thread 的 SAL (Socket Abstraction Layer) 套接字与 BSD 套接字在接口层面上是非常相似的,因为它们都遵循 BSD Socket API 规范。下面我会分别给出一个 RT-Thread SAL 套接字和 BSD 套接字的简单接口示例,以便进行比较。

RT-Thread SAL 套接字接口示例

在 RT-Thread 中使用 SAL 套接字进行网络通信时,通常会遵循以下步骤:

  1. 创建套接字

 

c复制代码

int sockfd = socket(AF_INET, SOCK_STREAM, 0);
if (sockfd < 0) {
/* 错误处理 */
}
  1. 绑定套接字

 

c复制代码

struct sockaddr_in server_addr;
memset(&server_addr, 0, sizeof(server_addr));
server_addr.sin_family = AF_INET;
server_addr.sin_port = htons(8080);
server_addr.sin_addr.s_addr = INADDR_ANY;
if (bind(sockfd, (struct sockaddr *)&server_addr, sizeof(server_addr)) < 0) {
/* 错误处理 */
}
  1. 监听套接字

 

c复制代码

if (listen(sockfd, 5) < 0) {
/* 错误处理 */
}
  1. 接受连接

 

c复制代码

struct sockaddr_in client_addr;
socklen_t client_len = sizeof(client_addr);
int connfd = accept(sockfd, (struct sockaddr *)&client_addr, &client_len);
if (connfd < 0) {
/* 错误处理 */
}
  1. 发送和接收数据

 

c复制代码

char buf[1024];
int len = recv(connfd, buf, sizeof(buf), 0);
if (len > 0) {
/* 处理接收到的数据 */
}
/* 发送数据 */
send(connfd, "Hello, Client!", strlen("Hello, Client!"), 0);
  1. 关闭套接字

 

c复制代码

close(sockfd);
close(connfd);
BSD 套接字接口示例

BSD 套接字的接口与 RT-Thread SAL 套接字的接口非常相似,因为 RT-Thread SAL 本身就是基于 BSD Socket API 设计的。所以上面的示例代码在 BSD 环境下也是适用的。

不过,需要注意的是,虽然接口相似,但在具体实现和性能上可能会有所不同。

RT-Thread SAL 针对嵌入式系统和物联网设备进行了优化,可能具有更小的内存占用和更高的效率。而 BSD 套接字则更广泛地应用于各种操作系统和环境中

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

如若内容造成侵权/违法违规/事实不符,请联系瞎胡闹网进行投诉反馈,一经查实,立即删除!

相关文章

css: hover 划过显示/隐藏 div 样式

1. 图例: 划过用display: block;和 display: none; 显示div和隐藏div div: <div class="sectorBox"> <div v-for="(item, index) in sectorList" :key="index" class="sill"> <div class="si…

大数据Scala教程从入门到精通第五篇:Scala环境搭建

一&#xff1a;安装步骤 1&#xff1a;scala安装 1&#xff1a;首先确保 JDK1.8 安装成功: 2&#xff1a;下载对应的 Scala 安装文件 scala-2.12.11.zip 3&#xff1a;解压 scala-2.12.11.zip 4&#xff1a;配置 Scala 的环境变量 在Windows上安装Scala_windows安装scala…

通过红黑树封装 map 和 set 容器(1):红黑树的迭代器

一、红黑树的迭代器 红黑树的遍历默认为中序遍历 —— key 从小到大&#xff0c;因此 begin() 应该获取到红黑树的最左节点 —— 最小&#xff0c;end() 获取到红黑树最右节点的下一个位置&#xff0c; operator() 也应保证红黑树的遍历为中序的状态。 首先对红黑树节点进行改造…

FMEA助力智能电网升级:构建安全、高效、可靠的电力网络

随着科技的不断进步&#xff0c;智能电网已成为现代电力行业的重要发展方向。而在这个过程中&#xff0c;FMEA&#xff08;失效模式和影响分析&#xff09;作为一种重要的质量管理工具&#xff0c;正日益发挥着其在智能电网建设中的赋能作用。本文将从FMEA的基本概念出发&#…

Springboot 集成 Consul 实现服务注册中心-05

因为后续很多模块都要用到注册中心&#xff0c;所以此处先实现此模块。 Consul简介 Consul是一个开源的服务发现和配置管理工具&#xff0c;具有跨平台、运行高效等特点。它由HashiCorp公司开发&#xff0c;并使用Go语言编写。Consul主要用于实现分布式系统中的服务发现、健康…

python中一些莫名其妙的异常

目录 一、字符串中空格\xa0二、文件写入为空问题三、Counter对NAN空值的统计问题 一、字符串中空格\xa0 对于文本中的一些空格&#xff0c;原始状态时显示为普通“空格”&#xff08;其实是latin1编码字符&#xff09;&#xff0c;但是经过split()操作后&#xff0c;这些latin…

MOSFET场效应管栅极驱动电流的计算

MOSFET驱动 MOSFET场效应管是电压驱动器件&#xff0c;输入有电容&#xff0c;因此为可靠驱动MOSFET&#xff0c;栅极需要施加较大的驱动电流。 功率MOSFET开关模型 该模型显示了影响开关性能的最重要的寄生器件。 图1 MOSFET开通过程 MOSFET场效应管的开通动作可分为如下…

《起风了》观后感

我想宫崎骏的电影是很多人心目中美好的回忆&#xff0c;每当听到有他的新电影要上映&#xff0c;总是迫不及待想去捧场&#xff0c;一刷二刷三刷却还是依然看得津津有味&#xff0c;这就是宫崎骏电影独特的魅力。《起风了》跟他的其他电影有很明显的不同&#xff0c;他的大部分…

I forgot my Plex Account PIN; how can I reset it? How can I change my PIN?

If you’ve set a PIN on your Plex account, it’s possible to reset or remove that PIN. Related Page: Plex Home Regular Plex Account If you know the current PIN If the current PIN is known, then simply edit the current PIN on the Settings > Users &…

ESP8266固件烧写

概述 因为手上有块闲置的ESP8266开发板&#xff0c;想着拿来倒腾一下WIFI探针&#xff0c;倒腾了一阵测试成功&#xff0c;博文记录用以备忘 硬件 ESP8266 NodeMCU 环境 Windows 11 步骤 1.下载esp32_win32_msys2_environment_and_toolchain-20181001.zip 2.下载xtensa…

【禅道客户案例】北大软件携手禅道,开启产品化之路新征程

在项目制项目模式下&#xff0c;软件公司根据客户的需求进行短期项目开发&#xff0c;具有灵活、高效、受众面广的优点&#xff0c;在业界得到了广泛的应用。但这种模式也面临诸多挑战&#xff0c;软件公司需要不断地开发新项目来维持业务增长&#xff0c;由于没有自己的产品也…

和comate一起,用JavaScript实现一个简易版五子棋小游戏

前言 五子棋起源于中国&#xff0c;是全国智力运动会竞技项目之一&#xff0c;是一种两人对弈的纯策略型棋类游戏。双方分别使用黑白两色的棋子&#xff0c;下在棋盘直线与横线的交叉点上&#xff0c;先形成五子连珠者获胜。 这次和Baidu Comate智能代码助手共同完成这个小游戏…

课程作业管理系统,基于 SpringBoot+Vue+MySQL 开发的前后端分离的课程作业管理系统设计实现

目录 一. 前言 二. 功能模块 2.1. 管理员功能模块 2.2. 教师功能模块 2.3. 学生功能模块 三. 部分代码实现 四. 源码下载 一. 前言 随着科学技术的飞速发展&#xff0c;社会的方方面面、各行各业都在努力与现代的先进技术接轨&#xff0c;通过科技手段来提高自身的优势…

云计算技术发展趋势详解

云计算最全详解(图文全面总结) 云计算是技术趋势的未来&#xff0c;掌握它至关重要。从基础到高级&#xff0c;本文深入探讨云计算的方方面面&#xff0c;为您提供全面的理解。 云计算 云计算将计算转移到远程数据中心&#xff0c;让用户灵活、经济地访问资源。就像水电一样&…

电商核心内容揭秘50:个性化广告与投放策略

相关系列文章 电商技术揭秘相关系列文章合集&#xff08;1&#xff09; 电商技术揭秘相关系列文章合集&#xff08;2&#xff09; 电商技术揭秘相关系列文章合集&#xff08;3&#xff09; 电商技术揭秘四十一&#xff1a;电商平台的营销系统浅析 电商技术揭秘四十二&#…

Cocos打安卓包打不出来?看看这个

点击上方亿元程序员+关注和★星标 引言 Cocos如何更加顺利地打出安卓包 大家好,相信小伙伴们通过关注亿元程序员,慢慢地进入了游戏开发行业,对游戏开发的认知也逐渐增长。 也有小伙伴通过阅读笔者的文章,成功独立完成了属于自己的游戏,并且成功地上线。 这是值得开心的…

森林消防的新利器:高扬程水泵的应用与优势/恒峰智慧科技

森林是地球上的绿色肺叶&#xff0c;保护森林安全对于维护生态平衡和人类生存环境至关重要。在森林消防领域&#xff0c;高效、快速的灭火设备是保障森林安全的重要武器。近年来&#xff0c;高扬程水泵作为一种新型的消防设备&#xff0c;在森林消防中发挥了重要作用。本文将详…

前端 Android App 上架详细流程 (Android App)

1、准备上架所需要的材料 先在需要上架的官方网站注册账号。提前把手机号&#xff0c;名字&#xff0c;身份证等等材料准备好&#xff0c;完成开发者实名认证&#xff1b;软著是必要的&#xff0c;提前准备好&#xff0c;软著申请时间比较长大概需要1-2周时间才能下来&#xf…

【Qt】demo示例--通过定时器实现时间刷新

【Qt】demo示例--通过定时器实现时间刷新 1.背景2.代码3.运行 1.背景 Qt Creator版本&#xff1a;4.2.0 &#xff0c;如下图&#xff1a; 即安装qt-opensource-windows-x86-msvc2013_64-5.7.1.exe 后自带得Qt编程IDE&#xff1b; 2.代码 项目结构如下&#xff1a; mydial…

ReactFlow的ReactFlow实例事件传参undefined处理状态切换

1.问题 ReactFlow的ReactFlow实例有些事件我们在不同的状态下并不需要&#xff0c;而且有时候传参会出现其它渲染效果&#xff0c;比如只读状态下我们不想要拖拉拽onEdgesChange连线重连或删除的功能。 2.思路 事件名称类型默认值onEdgesChange(changes: EdgeChange[]) >…