网络套件字(理论知识)

一、IP地址和目的IP地址

上次说到IP地址是为了是为了让信息正确的从原主机传送到目的主机,而原IP地址和目的IP地址就是用于标识两个主机的,既然叫做地址必然有着路径规划的作用,而路径规划最重要的就是,从哪来到哪去,现在在哪下一步去哪?

比如我要从山西北部骑行到江苏南京,那么我的源地址就是山西,目的地就是南京,第一步山西->河北:第二步河北->山东;第三步山东->江苏;

从上面我们看到的一个简单的路径规划,可以看到,出发地和目的地是不变的而当前位置和下一步的位置是一直在变化的,而IP数据包头部中, 有两个IP地址, 分别叫做源IP地址, 和目的IP地址。

在数据进行传输之前,会先自顶向下贯穿网络协议栈完成数据的封装(每一层会根据对应的协议添加报头),其中在网络层封装的IP报头当中就涵盖了源IP地址和目的IP地址。而除了源IP地址和目的IP地址之外,还有源MAC地址和目的MAC地址的概念。

二、理解源MAC地址和目的MAC地址

2.1MAC地址

MAC地址(Media Access Control Address)是一个用于识别网络设备的唯一标识符。每个网络设备(如计算机、手机、路由器等)都有一个独特的MAC地址。MAC地址通常是由48位二进制数字组成,通常以十六进制表示。MAC地址由厂商在生产设备时分配,分为两部分:前24位是厂商标识符,后24位是设备标识符。MAC地址在数据链路层(OSI模型中的第二层)使用,用于在局域网中唯一标识设备。MAC地址的作用类似于身份证号码,用于在网络中确定设备的身份和位置。

 通常数据的传输是跨局域网的,数据在传输过程中会经过若干个路由器,

而在上篇博客中提到路由器是看作在TCP/IP五层(或四层)模型中的网络层。

而当数据在局域网中传输时,就需要使用到数据链路层,而在该层要使用的就是MAC地址。

2.2源MAC地址和目的MAC地址

当数据在局域网中传输时,数据帧会包含发送者和接收者的MAC地址。网络设备根据目标MAC地址来决定是否接收该数据帧,如果目标MAC地址与自身的MAC地址匹配,设备就会接收并处理该数据帧。

 上面提到的发送者和接收者的MAC地址其实就是源MAC地址和目的MAC地址。

源MAC地址和目的MAC地址是包含在链路层的报头当中的,而MAC地址实际只在当前局域网内有效,因此当数据跨网络到达另一个局域网时,其源MAC地址和目的MAC地址就需要发生变化,因此当数据达到路由器时,路由器会将该数据当中链路层的报头去掉,然后再重新封装一个报头,此时该数据的源MAC地址和目的MAC地址就发生了变化。

这样做就可以让局域网技术不同的局域网之间经行通信,就像上图一样即使局域网的技术不同(一个是以太网,一个是令牌环网)但是数据是在进入局域网的时候才会添加数据链路层的报头,如下图。

 

2.3数据传输的两套地址

  • 一套是源IP地址和目的IP地址,这两个地址在数据传输过程中基本是不会发生变化的(存在一些特殊情况,比如在数据传输过程中使用NET技术,其源IP地址会发生变化,但至少目的IP地址是不会变化的)。
  • 另一套就是源MAC地址和目的MAC地址,这两个地址是一直在发生变化的,因为在数据传输的过程中路由器不断在进行解包和重新封装。

三、端口号

知道了消息如何在两台不同的主机之间传递,那么当消息传递到另一台主机后,如何知道该消息是发送给主机上哪一个应用呢?

比如QQ之间进行通讯,,可以看作是两个不同主机之间进程之间的通讯,主机与主机之间通过ip地址不走错,而进程带有一个端口号,每个主机都有独一无二的IP而一台主机上的每个进程都有唯一的端口号。

端口号 (port) 是传输层协议的内容
  • 端口号是一个2字节16位的整数;
  • 端口号用来标识一个进程, 告诉操作系统, 当前的这个数据要交给哪一个进程来处理;
  • IP地址 + 端口号能够标识网络上的某一台主机的某一个进程;
  • 一个端口号只能被一个进程占用

既然提到了进程我们都知道进程pid是用于标识唯一进程的那么他们有什么关系呢?

虽然进程PID和端口号都是用于唯一标识某种资源(进程或网络服务),但它们之间并没有直接的关联。在实际的网络通信中,操作系统会维护一个端口号与进程之间的映射关系,使得特定端口号的数据能够被正确路由到相应的进程。通常,网络服务启动时会绑定到一个特定的端口号,并且在运行期间会监听该端口,从而等待传入的连接请求或数据包。

传输层协议(TCP和UDP)的数据段中有两个端口号, 分别叫做源端口号和目的端口号. 就是在描述 "数据是谁发的, 要发给谁";
综上 网络通信的本质就是进程间经行通信

四、浅谈UDP/TCP

在前面我们简单谈了在数据链路层(MAC)和传输层(IP)中十分重要的概念,通过这两层,我们已经能将数据从一台主机传输到另一台主机。但是数据的安全性无法保证,而数据;链路层就是用于为应用层提供可靠的、端到端的数据传输服务,隐藏了网络通信的细节,使得应用程序能够简单地进行数据交换而不需要关心底层网络的细节。传输层的核心协议包括TCP和UDP,它们提供了不同级别的服务,满足不同应用场景的需求。

3.1UDP

UDP(User Datagram Protocol)用户数据报协议是一种无连接的、不可靠的传输协议,属于网络通信协议簇的一部分。与TCP不同,UDP不提供可靠的数据传输和错误恢复机制,而是专注于在网络上传输数据包,提供简单的数据传输服务。

 以下是UDP的一些特点:

  1. 无连接性(Connectionless):UDP是一种无连接的协议,发送数据之前不需要建立连接,也不维护连接状态。

  2. 不可靠性(Unreliable):UDP不提供数据包的可靠传输,数据包可能会丢失、重复或顺序错乱。

  3. 轻量级(Lightweight):UDP的头部开销比TCP小,不需要维护连接状态,因此具有较低的延迟。

  4. 面向数据报(Datagram-Oriented):UDP以数据报为单位进行数据传输,每个数据报独立于其他数据报。

  5. 不提供拥塞控制(No Congestion Control):UDP不提供拥塞控制机制,数据包可能会因为网络拥塞而丢失。

注意 上面的不可靠性只是一个相对的概念并不是说他不可靠就一定数据会出错。

UDP常用于对实时性要求较高、数据量较小、传输延迟较低的应用场景,例如音频和视频流传输、DNS查询、实时游戏等。由于其简单和高效的特性,UDP在一些特定的网络应用中具有重要的作用。比如我们经常玩的王者荣耀和原神都是udp链接

3.2TCP

TCP(Transmission Control Protocol)是一种面向连接的、可靠的、基于字节流的传输层协议,是互联网中最常用的协议之一。它提供了可靠的数据传输和错误恢复机制,用于在网络中进行端到端的数据传输。

 以下是TCP的一些主要特点:

  1. 面向连接(Connection-Oriented):在数据传输之前,TCP需要先建立连接,然后再进行数据传输,传输完成后再关闭连接。这种连接的建立和释放过程确保了数据的可靠传输。

  2. 可靠性(Reliability):TCP通过使用序号、确认和重传机制来确保数据的可靠传输。接收方会确认接收到的数据,并在需要时请求重传丢失的数据,从而保证数据的完整性和按序传输。

  3. 流量控制(Flow Control):TCP使用滑动窗口机制进行流量控制,确保发送方和接收方之间的数据传输速率匹配,防止数据发送过快导致接收方缓冲区溢出。

  4. 拥塞控制(Congestion Control):TCP使用拥塞窗口和拥塞避免机制来进行拥塞控制,防止网络拥塞导致的数据丢失和传输延迟增加。

  5. 面向字节流(Byte-Oriented):TCP是一种面向字节流的协议,它不保留消息的边界,而是将数据视为一连串的字节流进行传输。

  6. 全双工通信(Full Duplex Communication):TCP连接是全双工的,允许双方同时发送和接收数据。

TCP被广泛用于诸如网页浏览、文件传输、电子邮件和远程登录等应用中,它的可靠性和稳定性使得它成为互联网通信的重要基础。 

五、socket

在套接字编程中,常常将IP地址和端口号结合起来表示一个通信的端点,这种组合称为套接字地址。因此,可以说IP地址和端口号一起构成了一个套接字地址。然而,严格来说,套接字是操作系统中的一个抽象概念,用于表示网络通信的端点,而IP地址和端口号只是套接字地址的组成部分,用于确定通信的目的地或来源。因此,套接字通常是由IP地址、端口号和协议类型(如TCP或UDP)一起确定的.

5.1socket编程接口

在C语言中,使用套接字(socket)进行网络编程时,常见的编程接口包括:

 socket(): 创建一个套接字,返回套接字描述符。

int socket(int domain, int type, int protocol);
  • domain: 地址族,如 AF_INET(IPv4)或 AF_INET6(IPv6)。
  • type: 套接字类型,如 SOCK_STREAM(流套接字,TCP)或 SOCK_DGRAM(数据报套接字,UDP)。
  • protocol: 协议类型,通常为 0,表示由系统自动选择。

bind(): 将套接字与特定的IP地址和端口号绑定。

int bind(int sockfd, const struct sockaddr *addr, socklen_t addrlen);
  • sockfd: 套接字描述符。
  • addr: 指向要绑定的 sockaddr 结构体的指针。
  • addrlen: 地址结构体的大小。

listen(): 在服务器端开始监听连接请求。

int listen(int sockfd, int backlog);
  • sockfd: 套接字描述符。
  • backlog: 允许的连接等待队列的最大长度。

accept(): 接受客户端的连接请求,并创建一个新的套接字用于与客户端进行通信。

int accept(int sockfd, struct sockaddr *addr, socklen_t *addrlen);
  • sockfd: 服务器套接字描述符。
  • addr: 用于存储客户端地址信息的 sockaddr 结构体。
  • addrlen: 指向存储客户端地址长度的变量的指针。

connect(): 连接到服务器。

int connect(int sockfd, const struct sockaddr *addr, socklen_t addrlen);
  • sockfd: 套接字描述符。
  • addr: 服务器地址的 sockaddr 结构体指针。
  • addrlen: 地址结构体的大小。

send() / recv(): 发送和接收数据。

ssize_t send(int sockfd, const void *buf, size_t len, int flags);
  • sockfd: 套接字描述符。
  • buf: 要发送的数据的缓冲区。
  • len: 要发送的数据的字节数。
  • flags: 发送标志,通常为 0
ssize_t recv(int sockfd, void *buf, size_t len, int flags);
  • sockfd: 套接字描述符。
  • buf: 接收数据的缓冲区。
  • len: 缓冲区长度。
  • flags: 接收标志,通常为 0

sendto() / recvfrom(): 用于在无连接的套接字上发送和接收数据报

ssize_t sendto(int sockfd, const void *buf, size_t len, int flags,const struct sockaddr *dest_addr, socklen_t addrlen);
  • sockfd: 套接字描述符。
  • buf: 要发送的数据的缓冲区。
  • len: 要发送的数据的字节数。
  • flags: 发送标志,通常为 0
  • dest_addr: 目标地址的 sockaddr 结构体指针,用于指定数据发送的目标地址。
  • addrlen: 目标地址结构体的大小。
ssize_t recvfrom(int sockfd, void *buf, size_t len, int flags,struct sockaddr *src_addr, socklen_t *addrlen);
  • sockfd: 套接字描述符。
  • buf: 接收数据的缓冲区。
  • len: 缓冲区长度。
  • flags: 接收标志,通常为 0
  • src_addr: 用于存储发送方地址信息的 sockaddr 结构体指针。
  • addrlen: 指向存储发送方地址长度的变量的指针。

close(): 关闭套接字。

int close(int sockfd);

 sockfd: 要关闭的套接字描述符

5.2sockaddr结构

socket API 是一层抽象的网络编程接口 , 适用于各种底层网络协议 , IPv4 IPv6, 以及后面要讲的 UNIX Domain Socket. 然而 , 各种网络协议的地址格式并不相同,

套接字不仅支持跨网络的进程间通信(网络套接字),还支持本地的进程间通信(域间套接字)。在进行跨网络通信时我们需要传递的端口号和IP地址,而本地通信则不需要。网络的设计者想要把跨网络通信和本地通信进行大一统,因此套接字提供了sockaddr_in结构体和sockaddr_un结构体,其中sockaddr_in结构体是用于跨网络通信的(网络套接字),而sockaddr_un结构体是用于本地通信的(域间套接字)。
为了让套接字的网络通信和本地通信能够使用同一套函数接口,于是就出现了sockeaddr结构体,该结构体与sockaddr_in和sockaddr_un的结构都不相同,但这三个结构体头部的16个比特位都是一样的,这个字段叫做协议家族。

​ 

最上面那个是表示是那种套件字(.sin_family)然后是端口号(.sin_port)和IP地址(.sin_addr)

此时当我们在传递在传参时,就不用传入sockeaddr_in或sockeaddr_un这样的结构体,而统一传入sockeaddr这样的结构体。在设置参数时就可以通过设置协议家族这个字段,来表明我们是要进行网络通信还是本地通信,在这些API内部就可以提取sockeaddr结构头部的16位进行识别,如果前16为地址类型是AD_INET,就是网络间通信,如果地址类型是AD_UNIX,就是本地间通信。如上我们就通过通用sockaddr结构,将套接字网络通信和本地通信的参数类型进行了统一。

注意 

  • IPv4和IPv6的地址格式定义在netinet/in.h中,IPv4地址用sockaddr_in结构体表示,包括16位地址类型, 16 位端口号和32位IP地址.
  • IPv4、IPv6地址类型分别定义为常数AF_INET、AF_INET6. 这样,只要取得某种sockaddr结构体的首地址, 不需要知道具体是哪种类型的sockaddr结构体,就可以根据地址类型字段确定结构体中的内容.
  • socket API可以都用struct sockaddr *类型表示, 在使用的时候需要强制转化成sockaddr_in; 这样的好处是程序的通用性, 可以接收IPv4, IPv6, 以及UNIX Domain Socket各种类型的sockaddr结构体指针做为 参数; 

我们可以包含如下四个头文件查看或使用sockaddr、sockaddr_in、in_addr的相关信息:

#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>

 sockaddr 结构

sockaddr_in 结构
虽然 socket api 的接口是 sockaddr, 但是我们真正在基于 IPv4 编程时 , 使用的数据结构是 sockaddr_in; 这个结构里主要有三部分信息: 地址类型 , 端口号 , IP 地址

 in_addr结构

 in_addr用来表示一个IPv4IP地址. 其实就是一个32位的整数。

在这个里面我们并没有看到sin_family这个部分,事实上这个就是第二个图片240那行那个在 sockaddr_in 结构体中,sin_family 是 __SOCKADDR_COMMON(sin_) 的一部分。这个设计是为了确保不同的套接字地址结构(例如,IPv4、IPv6等)在内部布局上是一致的,以便于通用的套接字地址处理。

##可以把位于它两边的符号合成一个符号。它允许宏定义从分离的文本片段创建标识符

注: 这样的连接必须产生一个合法的标识符。否则其结果就是未定义的。 所以上面直接被替换成sin_family  

 六、网络字节序

 我们都知道不同的计算机在内存存储中存在大小端问题

  • 大端模式: 数据的高字节内容保存在内存的低地址处,数据的低字节内容保存在内存的高地址处。
  • 小端模式: 数据的高字节内容保存在内存的高地址处,数据的低字节内容保存在内存的低地址处。

磁盘文件中的多字节数据相对于文件中的偏 移地址也有大端小端之分, 网络数据流同样有大端小端之分. 那么如何定义网络数据流的地址呢?

  • 发送主机通常将发送缓冲区中的数据按内存地址从低到高的顺序发出;
  • 接收主机把从网络上接到的字节依次保存在接收缓冲区中,也是按内存地址从低到高的顺序保存;
  • 因此,网络数据流的地址应这样规定:先发出的数据是低地址,后发出的数据是高地址.
  • TCP/IP协议规定,网络数据流应采用大端字节序,即低地址高字节.
  • 不管这台主机是大端机还是小端机, 都会按照这个TCP/IP规定的网络字节序来发送/接收数据;
  • 如果当前发送主机是小端, 就需要先将数据转成大端; 否则就忽略, 直接发送即可;

 

为使网络程序具有可移植性 , 使同样的 C 代码在大端和小端计算机上编译后都能正常运行 , 可以调用以下库函数做网络字节序和主机字节序的转换。

 

  • 这些函数名很好记,h表示host,n表示network,l表示32位长整数,s表示16位短整数。
  • 例如htonl表示将32位的长整数从主机字节序转换为网络字节序,例如将IP地址转换后准备发送。
  • 如果主机是小端字节序,这些函数将参数做相应的大小端转换然后返回 ;
  • 如果主机是大端字节序,这些 函数不做转换,将参数原封不动地返回

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

机器人学、机器视觉与控制 上机笔记(第一版译文版 2.1章节)

机器人学、机器视觉与控制 上机笔记&#xff08;第一版译文版 2.1章节&#xff09; 1、前言2、本篇内容3、代码记录3.1、新建se23.2、生成坐标系3.3、将T1表示的变换绘制3.4、完整绘制代码3.5、获取点*在坐标系1下的表示3.6、相对坐标获取完整代码 4、结语 1、前言 工作需要&a…

简单说网络:TCP+UDP

TCP和UPD: (1)都工作在传输层 (2)目的都是在程序之中传输数据 (3)数据可以是文本、视频或者图片(对TCP和UDP来说都是一堆二进制数没有太大区别) 一、区别:一个基于连接一个基于非连接 将人与人之间的通信比喻为进程和进程之前的通信:基本上有两种方式(1)写信;(2)打电话;这…

Docker容器化K8s集群部署教程(一键部署sheel脚本)

本文通过脚本&#xff0c;可以快速地部署和配置Kubernetes环境&#xff0c;省去了各插件手动部署、配置的繁琐过程。 先看最终结果&#xff1a; [rootlocalhost home]# kubectl get node NAME STATUS ROLES AGE VERSION k8smaster Ready control-p…

LlamaIndex 入门实战

文章目录 LlamaIndex 入门实战1. 基本概念2. 优劣势分析3. 简单代码示例4. Index持久化5. 使用场景6. 总结 LlamaIndex 入门实战 LlamaIndex是一个连接大型语言模型&#xff08;LLMs&#xff09;与外部数据的工具&#xff0c;它通过构建索引和提供查询接口&#xff0c;使得大模…

Java学习17-- super类

重点&#xff1a;super类 & 方法重写 super类 super指的是本级的上一级&#xff0c;即father class父类 很好理解&#xff0c;比如Person class>Student class 当前在Student class执行&#xff0c;那么就写this.xxx 需要在Student程序里面调用Person&#xff0c;那就…

HarmonyOS应用/服务发布:打造多设备生态的关键一步

目前 前言HarmonyOS 应用/服务发布的重要性使用HarmonyOS 构建跨设备的应用生态前期准备工作简述发布流程生成签名文件配置签名信息编译构建.app文件上架.app文件到AGC结束语 前言 随着智能设备的快速普及和多样化&#xff0c;以及编程语言的迅猛发展&#xff0c;构建一个无缝…

大数据 - Spark系列《四》- Spark分布式运行原理

Spark系列文章&#xff1a; 大数据 - Spark系列《一》- 从Hadoop到Spark&#xff1a;大数据计算引擎的演进-CSDN博客 大数据 - Spark系列《二》- 关于Spark在Idea中的一些常用配置-CSDN博客 大数据 - Spark系列《三》- 加载各种数据源创建RDD-CSDN博客 目录 &#x1f360;…

sqli-labs-master靶场训练笔记(38-53|boss战)

2024.2.4 level-38 &#xff08;堆叠注入&#xff09; 这题乍一看感觉又是来卖萌的&#xff0c;这不是和level-1一模一样吗 然后仔细看了一下源代码&#xff0c;根据 mysqli_multi_query 猜测这题的本意应该是堆叠注入 mysqli_multi_query() 是 PHP 中用于执行多个 SQL 查…

品牌如何营造生活感氛围?媒介盒子分享

「生活感」简而言之是指人们对生活的感受和意义&#xff0c;它往往没有充斥在各种重要的场合和事件中&#xff0c;而是更隐藏在细碎平凡的生活场景中。在营销越来越同质化的当下&#xff0c;品牌应该如何打破常规模式&#xff0c;洞察消费情绪&#xff0c;找到更能打动消费者心…

2023:AI疯狂进化年

嘿&#xff0c;大家好&#xff01;让我们一起来回顾一下这疯狂的 2023 年吧&#xff01;记得那个二月初吗&#xff1f;ChatGPT 上线了&#xff0c;然后呢&#xff1f;短短两个月&#xff0c;用户数量就像火箭一样突破了 1 亿&#xff01;这速度&#xff0c;简直比超级赛亚人还快…

前端JavaScript篇之对执行上下文的理解

目录 对执行上下文的理解创建执行上下文 对执行上下文的理解 当我们在执行JavaScript代码时&#xff0c;JavaScript引擎会创建并维护一个执行上下文栈来管理执行上下文。执行上下文有三种类型&#xff1a;全局执行上下文、函数执行上下文和eval函数执行上下文。 在写代码的时…

(6)【Python/机器学习/深度学习】Machine-Learning模型与算法应用—使用Adaboost建模及工作环境下的数据分析整理

目录 一、为什么要使用Adaboost建模? 二、泰坦尼克号分析(工作环境) (插曲)Python可以引入任何图形及图形可视化工具 三、数据分析 四、模型建立 1、RandomForestRegressor预测年龄 2、LogisticRegression建模 引入GridSearchCV 引入RandomizedSearchCV 3、Deci…

第三百一十回

我们在上一章回中介绍了"再谈ListView中的分隔线"&#xff0c;本章回中将介绍showMenu的用法.闲话休提&#xff0c;让我们一起Talk Flutter吧。 1. 概念介绍 我们在第一百六十三回中介绍了showMenu相关的内容&#xff0c;它主要用来显示移动PopupMenu在页面中的位置…

C语言第二十一弹---指针(五)

✨个人主页&#xff1a; 熬夜学编程的小林 &#x1f497;系列专栏&#xff1a; 【C语言详解】 【数据结构详解】 转移表 1、转移表 总结 1、转移表 函数指针数组的用途&#xff1a;转移表 举例&#xff1a;计算器的⼀般实现&#xff1a; 假设我们需要做一个能够进行加减…

CoreSight学习笔记

文章目录 1 Components1.1 ROM Table 2 使用场景2.1 Debug Monitor中断2.1.1 参考资料 2.2 Programming the cross halt2.2.1 编程实现2.2.2 参考资料 2.3 CTI中断2.3.1 编程实现2.3.1.1 准备工作2.3.1.2 触发中断2.3.1.3 中断响应 2.3.2 参考资料 1 Components 1.1 ROM Table…

rust语言tokio库底层原理解析

目录 1 rust版本及tokio版本说明1 tokio简介2 tokio::main2.1 tokio::main使用多线程模式2.2 tokio::main使用单线程模式 3 builder.build()函数3.1 build_threaded_runtime()函数新的改变功能快捷键合理的创建标题&#xff0c;有助于目录的生成如何改变文本的样式插入链接与图…

国产三维剖面仪—MPAS-100相控参量阵浅地层剖面仪

最近声学所东海站邹博士发来了他们最新的浅地层剖面仪—MPAS-100相控参量阵浅地层剖面仪的资料&#xff0c;市场型号GeoInsight&#xff0c;委托Ocean Physics Technology公司销售&#xff0c;地大李师兄的公司负责技术支持。 MPAS-100相控参量阵浅地层剖面仪就是俗称的三维浅…

git安装配置

1、下载安装 下载地址 2、配置git用户 git config --global user.name "yw" git config --global user.email "88888qq.com" 3、git init 初始化 4、生成ssh密钥 mkdir .ssh //创建文件夹cd .ssh //进入新建文件夹 ssh-keygen -t rsa // 输入密钥文…

Uniapp真机调试:手机端访问电脑端的后端接口解决

Uniapp真机调试&#xff1a;手机端访问电脑端的后端接口解决 1、前置操作 HBuilderX -> 运行 -> 运行到手机或模拟器 -> 运行到Android App基座 少了什么根据提示点击下载即可 使用数据线连接手机和电脑 手机端&#xff1a;打开开发者模式 -> USB调试打开手机端&…

使用 WMI 查询安全软件信息

在这篇文章中&#xff0c;我们将详细介绍如何使用 Windows Management Instrumentation (WMI) API 来查询当前计算机上安装的安全软件的基本信息。我们将分析代码的各个部分&#xff0c;并解释每个步骤所涉及的技术和原理。 一、什么是 WMI&#xff1f; WMI 是 Windows Manag…