【学习日记】操作系统-入门知识-个人学习记录
我的学习笔记链接:
MyLinuxProgramming
参考资料
- CSAPP
- 操作系统导论OSTEP √
- APUE
- https://stevens.netmeister.org/631
- 软件调试
- 王道-操作系统
- 操作系统真象还原
- 小林coding-图解系统
- https://xiaolincoding.com
- 嵌入式软件开发笔试面试指南
- Linux是怎样工作的
- 2020 南京大学 “操作系统:设计与实现” (蒋炎岩)
- 【清华 操作系统原理】
- 【【哈工大】操作系统 李治军(全32讲)】
- 【彻底搞懂 进程&线程、进程池&线程池】
操作系统概念
-
特点
- 并发
- 共享
- 虚拟
- 异步
-
Linux
- 多任务、SMP对称多处理、ELF可执行文件、宏内核
-
Windows
- NT、PE可执行文件、混合型内核
进程与线程
-进程
-
进程的状态
- 运行台 就绪态 阻塞态(等待态) 创建态 终止态

- 运行台 就绪态 阻塞态(等待态) 创建态 终止态
-
进程控制块 PCB
- PCB 是进程存在的唯一标志
-
并发:交替进行
-
并行:同时进行 --流水线
-线程
-
线程间可以并发运行且共享相同的地址空间
-
进程中一个线程崩溃,其他线程都崩溃
-
线程是调度的基本单位,进程是资源拥有的基本单位
-
线程的实现
- 用户线程的整个线程管理和调度,操作系统不直接参与
- 用户级线程库函数来完成线程的管理,包括线程的创建、终止、同步和调度等
- 内核线程由操作系统管理程对应的TCB放在操作系统
- 内核线程的创建、终止和管理都由操作系统负责
- 轻量级线程 LWP
-进程调度算法
-
需要考虑的因素
- CPU利用率
- 吞吐量。长作业会降低吞吐量
- 周转时间。进程运行和阻塞时间的总和
- 等待时间。进程处于就绪队列的时间
- 响应时间。交互式系统中衡量调度算法的主要标准
-
FCFS 先来先服务调度算法
- 不可剥夺
- 效率低,适用于CPU繁忙型作业
- 对长作业有利,对短作业不利
-
SJF 短作业优先调度算法
- 对短作业有利
- 相比FCFS降低了平均周转时间
- 可能会导致长作业“饥饿”现象
-
优先级调度算法
- 分为非抢占式和抢占式
- 优先级设置规则:
- 系统进程>用户进程 交互型>非交互型 I/O型>计算型
-
RR 时间片轮转调度算法
- 适用于分时系统,提高交互体验
- 选择适当的时间片很重要
-
MLFQ 多级反馈队列调度算法

- 动态调整优先级和时间片大小
- 多个就绪队列,每个队列不同优先级
- 每个队列进程运行时间片不同
- 每个队列采用FCFS算法
- 按队列优先级调度
- 优点:短作业优先,周转时间短,长作业不会饿死
-协程
- 是一种比线程更加轻量级的存在
- 正如一个进程可以拥有多个线程一样,一个线程也可以拥有多个协程
- 协程不被操作系统内核所管理,而是完全由程序所控制(也就是在用户态执行)
- 协程的优点是性能得到了很大的提升,不像线程切换那样消耗资源
进程间通信 IPC
-管道
-
管道是内核中的一串缓存,无格式且大小受限
-
一次性操作,只支持半双工
-
生产者消费者模式
-
匿名管道。Linux命令行中的 “|”
- 只能用于父子关系的进程间通信
-
命名管道。FIFO
mkfifo myPipe echo "hello" > myPipe cat < myPipe

-消息队列
- 消息队列是内核中的消息链表
- 不适合大数据传输
-共享内存
- 共享内存的机制是拿出一块虚拟地址空间,映射到相同的物理内存中
-信号量
- 信号量是一个整型的计数器
- 用于实现进程间的互斥与同步
- P、V 操作
- P操作使得信号量减1
- V操作使得信号量加1
-信号
- 可参考APUE第10章
- 唯一的异步通信机制
- 两种无法捕捉和忽略的信号:SIGKILL、SEGSTOP
-Socket
int socket(int domain, int type, int protocal)
-
字节流SOCKSTREAM、数据包SOCKDGRAM
-
基于TCP协议的通信示例:

- 1.服务端和客户端初始化socket,得到文件描述符
- 2.服务端调用bind,绑定IP和端口
- 3.服务端调用listen,开始监听
- 4.服务端调用accept,等待客户端连接
- 5.客户端调用connect,向服务端IP和端口发起连接请求
- 6.服务端调用accept,返回用于传输的socket文件描述符
- 7.客户端调用write写入数据
- 8.服务端调用read读取数据
- 9.客户端断开连接时,调用close,那么服务端read读取数据的时候,会读到EOF,待处理完数据后,服务端调用 close,表示连接关闭
- 这里需要注意的是,服务端调用 accept 时,连接成功了会返回一个已完成连接的 socket,后续用来传输数据。
多线程同步
- 多个线程如果竞争共享资源,如果不采取有效的措施,则会造成共享数据的混乱
- 竞争条件。不确定性
- 临界区
-互斥与同步
-
同步举例:「操作 A 应在操作 B 之前执行」,「操作 C 必须在操作 A 和操作 B 都完成之后才能执行」等
-
互斥举例:「操作 A 和操作 B 不能在同一时刻执行」
-
锁
- 原子操作指令 —— 测试和置位(Test-and-Set)
int TestAndSet(int *old_ptr, int new) {int old = *old_ptr;*old_ptr = new;return old; }- 忙等待锁。也称为自旋锁
- 无等待锁。
-
信号量
- P/V操作
// 信号量数据结构 type struct sem_t{int sem; // 资源个数queue_t *q; // 等待队列 } sem_t;// 初始化信号量 void init(sem_t *s, int sem) {s->sem = sem;queue_init(s->q); }// P操作 void P(sem_t *s) {s->sem--;if(s->sem < 0){// 1、保留调用线程CPU现场// 2、将该线程的TCB插入s的等待队列// 3、设置该线程为等待状态// 4、执行调度程序} }// V操作 void V(sem_t *s) {s->sem++;if(s->sem <= 0){// 1、移除s等待队列首元素// 2、将该线程的TCB插入就绪队列// 3、设置该线程为就绪状态} }- 对于两个并发线程,互斥信号量的值仅取 1、0 和 -1 三个值
- 如果互斥信号量为 1,表示没有线程进入临界区
- 如果互斥信号量为 0,表示有一个线程进入临界区
- 如果互斥信号量为 -1,表示一个线程进入临界区,另一个线程等待进入
- 通过互斥信号量的方式,能保证临界区任何时刻只有一个线程在执行,就达到了互斥的效果。
-生产者-消费者问题
-
问题描述:
- 生产者在生成数据后,放在一个缓冲区中
- 消费者从缓冲区取出数据处理
- 任何时刻,只能有一个生产者或消费者可以访问缓冲区
-
互斥与同步的体现:
- 任何时刻只能有一个线程操作缓冲区,操作缓冲区是临界代码,说明需要互斥
- 缓冲区空时,消费者必须等待生产者生成数据;缓冲区满时,生产者必须等待消费者取出数据。说明需要同步
-
需要三个信号量:
- 互斥信号量 mutex:用于互斥访问缓冲区,初始化值为 1
- 资源信号量 fullBuffers:用于消费者询问缓冲区是否有数据,有数据则读取数据,初始化值为 0(表明缓冲区一开始为空);
- 资源信号量 emptyBuffers:用于生产者询问缓冲区是否有空位,有空位则生成数据,初始化值为 n (缓冲区大小)
semaphore mutex = 1; semaphore empty = n; // 空闲缓冲区大小 semaphore full = 0; // 缓冲区初始为空 producer() {while(1){生产数据...P(empty);P(mutex);把数据放入缓冲区...V(mutex);V(full); } } consumer() {while(1){P(full);P(mutex);从缓冲区中取出数据...V(mutex);V(empty);消费数据... } }
-哲学家就餐问题
- 用一个数组 state 来记录每一位哲学家在进程、思考还是饥饿状态(正在试图拿叉子)
- https://leetcode.cn/problems/the-dining-philosophers/description/

-读写者问题
- …
锁
-死锁
- 死锁产生需同时满足的四个条件
-
互斥

-
不剥夺

-
请求并保持

-
循环等待

-
pstack <pid>显示每个线程的栈跟踪信息(函数调用过程)
-避免死锁
- 破坏四个条件之一即可
- 使用资源有序分配法,破坏环路等待条件
- 死锁避免算法:银行家算法
-锁的种类
-
加锁的目的是保证共享资源在任意时间里,只有一个线程访问,避免多线程导致共享数据错乱的问题
-
互斥锁 [独占锁]
- 加锁失败后,线程会释放CPU

- 加锁失败后,线程会释放CPU
-
自旋锁
- 加锁失败后,线程会忙等待
- 相对来说开销比互斥锁小一点
-
读写锁
-
适用于读多写少的场景

-
公平读写锁,防止饥饿问题
-
-
悲观锁
- 悲观锁认为多线程同时修改共享资源的概率比较高,于是很容易出现冲突,所以访问共享资源前先要上锁
-
乐观锁
- 如果多线程同时修改共享资源的概率比较低,就可以采用乐观锁
- 乐观锁假定冲突的概率很低
- 乐观锁的工作方式:
- 先修改完共享资源,再验证这段时间内有没有发生冲突
- 如果没有其他线程在修改资源,操作完成
- 如果发现有其他线程已经修改过这个资源,放弃本次操作
- 乐观锁的应用:在线文档多人编辑、Git、SVN
内存管理
-虚拟内存
-
不同进程的虚拟地址和不同内存的物理地址映射
-
虚拟地址(VA)—> 内存管理单元(MMU) —> [映射到]物理地址(PA)
-
管理方式:内存分段和内存分页
-
内存分段:
- 段表:段号、段基地址、段界限
- 问题:内存碎片、交换效率低
-
内存分页
- 把整个虚拟和物理内存空间分为若干页
- 每一页4kb
- 换出换入 swap out / swap in

- 页表:页号、页内偏移
- 多级页表
- 解决页表过大的问题
- TLB-Translation Lookaside Buffer
- 页表缓存,局部性原理
-
段页式内存管理
- 段号、段内页号和页内位移
- 第一次访问段表,得到页表起始地址;
- 第二次访问页表,得到物理页号;
- 第三次将物理页号与页内位移组合,得到物理地址。

-Linux内存管理

- 程序文件段 .text
- 包括二进制可执行代码
- 已初始化数据段 .data
- 包括静态常量
- 未初始化数据段 .bss
- 包括未初始化的静态变量
- 堆段
- 包括动态分配的内存,从低地址开始向上增长
- 文件映射段
- 包括动态库、共享内存等,从低地址开始向上增长(跟硬件和内核版本有关)
- 栈段
- 包括局部变量和函数调用的上下文等
- 堆和文件映射段的内存动态分配
- 比如使用C标准库的 malloc() 或 mmap() ,就可以分别在堆和文件映射段动态分配内存。
-页面置换算法
-
OPT 最佳置换算法
- 无法实现,但作为评价基准
-
FIFO 先进先出页面置换算法
- Belady异常
-
LRU 最近最久未使用置换算法
- 最新访问放在第一位
- 缓存满了删除末尾(最不经常访问的数据)
-磁盘调度算法
-
FCFS 先来先服务算法
-
SSF 最短寻道时间优先
- 优先选择从当前磁头位置所需寻道时间最短的请求
- 可能产生饥饿:磁头在一小块区域来回移动
-
SCAN 扫描算法(电梯算法)
- 磁头在一个方向上移动,访问所有未完成的请求
- 直到磁头到达该方向上的最后的磁道,才调换方向

-
C-SCAN 循环扫描算法
- 有磁头朝某个特定方向移动时,才处理磁道访问请求
- 返回时直接快速移动至最靠边缘的磁道,也就是复位磁头,并且返回中途不处理任何请求

-
LOOK与C-LOOK算法
- SCAN的优化:LOOK。磁头在移动到「最远的请求」位置,然后立即反向移动
- C-SCAN的优化:C-LOOK。反向移动途中不响应请求
文件系统
-组成
-
一切皆文件

-
Linux文件系统组成:
- 索引节点 inode
- 记录文件的元信息,如 inode 编号、文件大小、访问权限、创建时间、修改时间、数据在磁盘的位置等等
- 索引节点是文件的唯一标识
- 目录项 dentry
- 记录文件的名字、索引节点指针以及与其他目录项的层级关联关系
- 索引节点 inode
-
磁盘读写最小单位是扇区 512B
-
文件系统把多个扇区组成了逻辑块
-
Linux中逻辑块大小为4KB,也就是8个扇区
-虚拟文件系统 VFS
- Linux支持的文件系统分为三类:
- 磁盘文件系统。占用磁盘空间。如Ext 2/3/4
- 内存文件系统。占用内存空间。如/proc /sys
- 网络文件系统。如NFS、SMB
文件的使用
- 文件描述符 fd
- 系统调用。
open(); write(); close(); - 打开文件表中维护打开文件的状态和信息:
- 文件指针
- 文件打开计数器
- 文件磁盘位置
- 访问权限
-文件存储
-
连续空间存放
- 指定起始块位置和长度
- 缺点:磁盘空间碎片、文件长度不易扩展
-
非连续空间存放方式
-
链表方式或索引方式
-
「隐式链表」:文件头要包含「第一块」和「最后一块」的位置,并且每个数据块里面留出一个指针空间,用来存放下一个数据块的位置
-
「显式链接」:把用于链接文件各数据块的指针,显式地存放在内存的一张链接表中,该表在整个磁盘仅设置一张,每个表项中存放链接指针,指向下一个数据块号
- 文件分配表 FAT
-
「索引数据块」:里面存放了指向文件数据块的指针列表(相当于书的目录)
- 没有碎片问题
- 方便创建、扩大、缩小
- 支持顺序读写和随机读写
-
「链式索引块」:链表加索引的组合
- 在索引数据块留出一个存放下一个索引数据块的指针

- 在索引数据块留出一个存放下一个索引数据块的指针
-
「多级索引块」
-
-空闲空间管理
-
空闲表法
-
空闲链表法

-
位图法
-软链接和硬链接
-
硬链接
- 只有删除文件所有硬链接及源文件时,系统才会彻底删除该文件
-
软链接
- 跨文件系统
- 相当于重新创建文件,有独立的inode
- 该文件的内容是另外一个文件的路径
-文件I/O
- 直接与非直接 I/O
- 阻塞与非阻塞 I/O
- I/O 多路复用技术。如
select poll
- I/O 多路复用技术。如
- 同步与异步 I/O
设备管理
-
输入输出设备可分为两大类:
- 块设备(Block Device)
- 字符设备(Character Device)
-
中断有两种:
- 软中断,例如代码调用 INT 指令触发
- 硬件中断,硬件通过中断控制器触发的
-
DMA
- 设备在 CPU 不参与的情况下,能够自行完成把设备 I/O 数据放入到内存

- 设备在 CPU 不参与的情况下,能够自行完成把设备 I/O 数据放入到内存
-
设备驱动程序
-
存储系统I/O软件分层
- 文件系统层、通用块层、设备层

- 文件系统层、通用块层、设备层
相关文章:
【学习日记】操作系统-入门知识-个人学习记录
我的学习笔记链接: MyLinuxProgramming 参考资料 CSAPP操作系统导论OSTEP √APUEhttps://stevens.netmeister.org/631软件调试王道-操作系统操作系统真象还原小林coding-图解系统https://xiaolincoding.com嵌入式软件开发笔试面试指南Linux是怎样工作的2020 南京大…...
ChatGPT自动生成思维导图
🍏🍐🍊🍑🍒🍓🫐🥑🍋🍉 ChatGPT自动生成思维导图 文章目录 🍐问题引入🍐具体操作markmapXmind 🐳结语 …...
count(0)、count(1)和count(*)、count(列名) 的区别
当我们对一张数据表中的记录进行统计的时候,习惯都会使用 count 函数来统计,但是 count 函数传入的参数有很多种,比如 count(1)、count(*)、count(字段) 等。 到底哪种效率是最好的呢?是不是 count(*) 效率最差? 一.…...
python爬虫入门,10分钟就够了,这可能是我见过最简单的基础教学
一、基础入门 1.1什么是爬虫 爬虫(spider,又网络爬虫),是指向网站/网络发起请求,获取资源后分析并提取有用数据的程序。 从技术层面来说就是 通过程序模拟浏览器请求站点的行为,把站点返回的HTML代码/JSON数据/二进制数据&…...
华为OD机试真题 Java 实现【记票统计】【牛客练习题】
一、题目描述 请实现一个计票统计系统。你会收到很多投票,其中有合法的也有不合法的,请统计每个候选人得票的数量以及不合法的票数。 (注:不合法的投票指的是投票的名字不存在n个候选人的名字中!!) 数据范围:每组输入中候选人数量满足 1≤n≤100 ,总票数量满足 1≤…...
.NET并行计算
一段很简答的,模拟多任务并发的测试代码。 private void button_Click(object sender, EventArgs e) { List<Action> actions new List<Action>(); for (int i 0; i < 30; i) { //匿…...
Python:Python编程:金融量化交易
金融量化交易 1. numpy2. scipy3. Pandas3.1 : Series 3.2: DataFrame代码示例 在金融量化交易中,下面几个模块是应用的比较广泛的 numpy (Numberic Python) : 提供大量的数值编程工具,可以方便的处理:向量矩阵等运算,…...
「HTML和CSS入门指南」canvas 标签详解
什么是 canvas 标签? 在 HTML 中,canvas 标签用于在网页中绘制图形、动画和其他复杂的视觉效果。它是一个独立的标签,并且可以使用 JavaScript 来操纵和渲染其内容。使用 canvas 标签可以帮助您创造交互性更强、生动更具吸引力的用户界面和体验。 canvas 标签的基本语法 以…...
【JS】1699- 重学 JavaScript API - WebSockets API
❝ 前期回顾: 1. Page Visibility API 2. Broadcast Channel API 3. Beacon API 4. Resize Observer API 5. Clipboard API 6. Fetch API 7. Performance API 8. Web Storage API ❞ WebSockets API 提供了一种在客户端和服务器之间建立持久连接的机制,使…...
String s = new String(“xyz“) 创建了几个对象?
这个问题相信每个学习 java 的同学都不陌生,作为一个经典的面试题,到现在工作这么多年了我真是认为挺操蛋的一个问题,在网上到现在你仍然可以看见很多讨论这个问题的人,其中不乏工作很多年的人都有争论,我认为还是有必…...
STL库(1)
STL库(1) vectorvector介绍vector使用初始化元素访问内存扩容插入删除 listlist介绍初始化,元素访问插入删除元素 vector和list区别 vector vector介绍 vector是可以改变大小的数组的容器。其内存结构和数组一样,使用连续的存储…...
玻璃制品行业丨外贸业务管理难点及解决方案
玻璃作为一种重要的建筑材料,在国际贸易中一直占有一定的份额。随着国外市场需求量的不断增加,对玻璃制品的技术含量要求越来越高,需要在研发方面的投入也逐步加大。由于国际市场竞争激烈,想要做玻璃制品行业的外贸公司࿰…...
Spring Boot如何实现自定义Spring Boot启动器
Spring Boot如何实现自定义Spring Boot启动器 在Spring Boot中,启动器(Starter)是一组依赖项的集合,它们一起提供了某个特定的功能。使用Spring Boot启动器可以让我们更加方便地集成第三方库和框架,并且可以避免版本冲…...
【面试题HTTP中的两种请求方法】GET 和 POST 有什么区别?
GET 和 POST 有什么区别? 1.相同点和最本质的区别1.1 相同点1.2 最本质的区别 2.非本质区别2.1 缓存不同2.2 参数长度限制不同2.3 回退和刷新不同2.4 历史记录不同2.5 书签不同 总结代码示例 GET 和 POST 是 HTTP 请求中最常用的两种请求方法,在日常开发…...
Allegro16.6详细教程(三)
確定Pad的層面 (1)用Single layer mode開關來控制pad type 勾選Single layer mode,則pad為單面孔,比如SMD 不勾選Single layer mode,則pad為通孔,比如:via (2)用滑鼠左鍵點選BEGIN LAYER彈出下面3個欄位 Regular, Thermal Relief, Anti Pad;Regular用於正片,Thermal R…...
Python3数据分析与挖掘建模(6)离散分布分析示例
1. 离散分布分析示例 相关库: pandas详细用法 numpy详细用法 1.1 引入算法库 # 引入 pandas库 import pandas as pd # 引入 numpy库 import numpy as np# 读取数据 dfpd.read_csv("data/HR.csv")# 查看数据 df Out[6]: satisfaction_level last_eval…...
汇编语言程序设计基础知识二
五、顺序结构 1、程序设计的步骤 1、分析问题 2、建立数据模型 3、设计算法 4、编制程序 5、上机调试 2、流程图的应用 3、程序的基本控制结构 1、顺序结构:程序顺序执行,不发生跳转 2、分支结构:程序在执行过程中发生跳转 3、循环…...
一文详解!Robot Framework Selenium UI自动化测试入门篇
目录 前言: 自动化框架的选择 测试环境的搭建 导入Selenium2Library包 关键字是什么? 创建测试用例 前言: 自动化测试的重要性越来越受到人们的重视,因为它可以提高测试效率、降低测试成本并减少人为错误的出现。为了满足这…...
Java 9 模块化系统详解
Java 9 模块化系统详解 一、简介1. 引入模块化系统原因2. 模块化系统带来的优势和挑战3. 模块化关键概念 二、模块化基础1. 模块化源代码结构规范2. 模块定义与描述符3. 打包可执行模块 三、模块化系统的高级特性1. 模块发现与解决依赖2. 模块化升级与替换3. 模块化动态访问 四…...
Windows定时执行Python脚本
在Linux环境下我们可以使用crontab工具来定时的执行脚本,可以很轻松的管理各个虚拟环境下的py文件在Windows上可以使用任务计划程序来定时执行我们的脚本 关于这个的基本使用可以查看我前面的博客 https://blog.csdn.net/wyh1618/article/details/125725967?spm10…...
基于算法竞赛的c++编程(28)结构体的进阶应用
结构体的嵌套与复杂数据组织 在C中,结构体可以嵌套使用,形成更复杂的数据结构。例如,可以通过嵌套结构体描述多层级数据关系: struct Address {string city;string street;int zipCode; };struct Employee {string name;int id;…...
Docker 离线安装指南
参考文章 1、确认操作系统类型及内核版本 Docker依赖于Linux内核的一些特性,不同版本的Docker对内核版本有不同要求。例如,Docker 17.06及之后的版本通常需要Linux内核3.10及以上版本,Docker17.09及更高版本对应Linux内核4.9.x及更高版本。…...
VB.net复制Ntag213卡写入UID
本示例使用的发卡器:https://item.taobao.com/item.htm?ftt&id615391857885 一、读取旧Ntag卡的UID和数据 Private Sub Button15_Click(sender As Object, e As EventArgs) Handles Button15.Click轻松读卡技术支持:网站:Dim i, j As IntegerDim cardidhex, …...
【人工智能】神经网络的优化器optimizer(二):Adagrad自适应学习率优化器
一.自适应梯度算法Adagrad概述 Adagrad(Adaptive Gradient Algorithm)是一种自适应学习率的优化算法,由Duchi等人在2011年提出。其核心思想是针对不同参数自动调整学习率,适合处理稀疏数据和不同参数梯度差异较大的场景。Adagrad通…...
练习(含atoi的模拟实现,自定义类型等练习)
一、结构体大小的计算及位段 (结构体大小计算及位段 详解请看:自定义类型:结构体进阶-CSDN博客) 1.在32位系统环境,编译选项为4字节对齐,那么sizeof(A)和sizeof(B)是多少? #pragma pack(4)st…...
【解密LSTM、GRU如何解决传统RNN梯度消失问题】
解密LSTM与GRU:如何让RNN变得更聪明? 在深度学习的世界里,循环神经网络(RNN)以其卓越的序列数据处理能力广泛应用于自然语言处理、时间序列预测等领域。然而,传统RNN存在的一个严重问题——梯度消失&#…...
实现弹窗随键盘上移居中
实现弹窗随键盘上移的核心思路 在Android中,可以通过监听键盘的显示和隐藏事件,动态调整弹窗的位置。关键点在于获取键盘高度,并计算剩余屏幕空间以重新定位弹窗。 // 在Activity或Fragment中设置键盘监听 val rootView findViewById<V…...
佰力博科技与您探讨热释电测量的几种方法
热释电的测量主要涉及热释电系数的测定,这是表征热释电材料性能的重要参数。热释电系数的测量方法主要包括静态法、动态法和积分电荷法。其中,积分电荷法最为常用,其原理是通过测量在电容器上积累的热释电电荷,从而确定热释电系数…...
JavaScript基础-API 和 Web API
在学习JavaScript的过程中,理解API(应用程序接口)和Web API的概念及其应用是非常重要的。这些工具极大地扩展了JavaScript的功能,使得开发者能够创建出功能丰富、交互性强的Web应用程序。本文将深入探讨JavaScript中的API与Web AP…...
探索Selenium:自动化测试的神奇钥匙
目录 一、Selenium 是什么1.1 定义与概念1.2 发展历程1.3 功能概述 二、Selenium 工作原理剖析2.1 架构组成2.2 工作流程2.3 通信机制 三、Selenium 的优势3.1 跨浏览器与平台支持3.2 丰富的语言支持3.3 强大的社区支持 四、Selenium 的应用场景4.1 Web 应用自动化测试4.2 数据…...
