【实用工具】谷歌浏览器插件开发指南

谷歌浏览器插件开发指南涉及以下几个方面:

1. 开发环境准备:首先需要安装Chrome浏览器和开发者工具。进入Chrome应用商店,搜索“Extensions Reloader”和“Manifest Viewer”两个插件进行安装,这两个插件可以方便开发和调试。

2. 创建插件:创建插件的方式有两种。第一种是从零开始创建,需要编写插件的各种文件,包括manifest.json、popup.html等。第二种是使用生成工具,如Yeoman,它可以自动生成插件文件和代码结构。

3. 编写插件代码:插件代码可以使用HTML、CSS、JavaScript等,需要根据插件的功能进行编写。

4. 调试和测试:在Chrome浏览器中可以使用开发者工具进行调试和测试,可以查看插件的运行状态,以及对代码进行修改和调试。

5. 发布和分发:发布插件需要一个Google开发者账号,并进行相应的审核和测试。插件发布后可以通过Chrome应用商店进行分发和安装。

总的来说,谷歌浏览器插件开发需要学习HTML、CSS、JavaScript等相关知识,并掌握Chrome插件开发的基本流程和技能。

Chrome插件本质上也是一个web页面的功能开发,因此需要熟悉以下基本的技能:

  • HTML:页面内容标记。
  • CSS:页面样式设置。
  • JavaScript:处理页面逻辑的脚本。
  • WebPlatformAPI:web平台的标准API。

下面开始正题。

Chrome插件基础概念

ChromeAPI

ChromeAPI是Chrome浏览器提供的JavaScriptAPI,在插件开发中使用这些API可以调用Chrome浏览器的提供的诸多功能,完成我们定制化的需求。

manifest.json

也叫插件清单,可以把它理解为整个插件的配置文件。manifest文件一定要放在根目录下,不可或缺,主要记录了插件的重要元数据、资源定义、权限声明,以及指定要在后台运行和页面运行的文件等。

// manifest.json文件{"manifest_version": 3,"name": "Reading time","version": "1.0","description": "Add the reading time to Chrome Extension documentation articles"
}

其中manifest_version​、name​、version​是必须的。

Service worker

也就是后台服务,主要是负责处理和监听浏览器的各类事件。后台服务可以使用所有ChromeAPI,但是不能直接与网页内容交互。

要使用后台服务,需要先在manifest文件中注册background:

{..."background": {"service_worker": "background.js"}
}

后台服务在浏览器运行起来后,就会一直在后台运行back.js脚本。

Content Script

也叫内容脚本。上面提到后台服务无法直接与网页内容进行交互,而内容脚本便来接替了这部分工作。内容脚本可以读取、修改或注入页面DOM,也可以使用一部分的ChromeAPI,但是不可使用的那部分ChromeAPI可以通过与后台服务的通讯来完成数据或消息的传递交互。

要使用内容脚本,需要先在manifest文件中注册content_scripts:

{..."content_scripts": [{"js": ["scripts/content.js"],"matches": ["https://developer.chrome.com/docs/extensions/*","https://developer.chrome.com/docs/webstore/*"]}]
}

其中,matches表示匹配的网站,js表示执行的脚本。即如果网站地址在matches列表中,则会执行js列表中的脚本。

pages

页面包含了popup的弹窗页面、option页面以及其他页面,这些页面也是就是前端开发中的HTML文件。这些页面都可以访问ChromeAPI。

以上便是Chrome插件开发中很重要且必要的几个概念,也是一个插件工程中主要的几个文件,因此请务必记住这几个概念。接下来,我们将一步一步开发一个插件,来逐步的了解Chrome插件开发的工作。

实际案例

插件开发

需求

开发前,我们先来看看这个插件需要完成那些功能。

1、阅读时间:

    • 在指定的页面上某个位置显示出当前页面预计阅览完成所需要的时间。此处我们已百度百科页面为例,在页面开头位置显示当前页面阅览完成大概需要长时间。
    • 阅读时间的预估规则我们这里简单计算下,使用页面文本的字数除以每分钟200个文字,并取整表示。

2、聚焦模式:

    • 对于某些页面的侧边栏、广告栏等区域做隐藏,实现聚焦主体内容的阅读。

3、标签管理:

    • 组织扩展文档选项卡,实现对特定域名的标签页组合成标签组。

阅读时间功能开发

思路:在内容脚本中使用js统计当前页面的文字数count,并假设阅读速度是500个/分钟,计算出预估时间:

time=count/500

先配置manifest文件:

{"manifest_version": 3,  // 版本号,目前2版本已经停止支持,推广的是3版本"name": "toolKits",  // 插件名称"description": "小工具集",  // 插件描述"version": "1.0",  // 插件开发的版本号,自定义即可"icons": {  // 插件的图标"16": "images/icon-16.png","32": "images/icon-32.png","48": "images/icon-48.png","128": "images/icon-128.png"},"content_scripts": [  // 内容脚本注册{"js": ["scripts/baikeReadTime.js"],"matches": ["https://baike.baidu.com/item/*"]}]
}

从配置中可得知内容脚本的注册规则:当网址匹配上“https://baike.baidu.com/item/*”时,运行scripts/baikeReadTime.js脚本。

编写内容脚本,内容脚本是往页面中注入预估的阅读时间元素,代码如下:

// baikeReadTime.js// 通过检查百度百科页面元素,确定大部分内容都是在class="main-content J-content"的div标签中,因此先获取到内容标签的dom
const mainContent = document.getElementsByClassName("main-content J-content")[0]// 如果对应内容dom存在
if (mainContent) {const text = mainContent.textContent;  // 获取内容dom中所有的文本内容const chineseWordRegExp = /[\u4e00-\u9fa5]/g;  // 通过正则表达式匹配出所有的汉字const chineseWords = text.matchAll(chineseWordRegExp);const count = [...chineseWords].length;  // 计算汉字长度const readTime = Math.round(count / 500);  // 计算预计阅读时间const badge = document.createElement('span');  // 创建一个<span>标签badge.classList.add('collect-text');  // 给<spn>标签添加样式 样式尽量跟周围的标签样式一致badge.textContent = `⏱️ 总字数约${count}, 预计阅读耗时:${readTime} 分钟`;  // 给<span>标签添加文本内容const topTool = document.getElementsByClassName("top-tool")[0]  // 找到class="top-tool"的domtopTool.insertAdjacentElement('beforebegin', badge);  // 把<span>标签插入到topTool的dom前面
}

演示:打开Chrome浏览器,安装插件后,打开任意百度百科的内容页,即可看到我们注入到页面的预估阅读时间元素。比如我们打开百度百科,搜索“亚运会”,页面展示图待更新。

聚焦模式功能开发

还是以百度百科页面为例,我们通过插件实现去掉右侧边栏和底部广告栏的信息

思路:分析页面元素,找到右侧边栏和底部广告栏的元素,再通过内容脚本对其进行样式隐藏即可。

1、首先给插件的图标设置一个徽标,用于展示聚焦模式的开关状态。

因为聚集模式的状态应该是在浏览器打开的时候就显示,因此设置状态徽标的逻辑应该放在background.js脚本中。具体代码如下:

chrome.runtime.onInstalled.addListener(() => {chrome.action.setBadgeText({ "text": "OFF" })
})

调用ChromeAPI,给插件设置徽标。大致意思就是添加一个安装事件的监控器,当插件安装后就给插件设置一个徽标,内容是“OFF”。

监听插件的点击事件。当点击插件的时候,判断当前浏览器标签页(tab页)的网址,如果符合要求,就将插件徽标设置为“ON”。继续在background.js脚本中添加以下代码:

...
const baike = "baike.baidu.com/item/"chrome.action.onClicked.addListener(async (tab) => {if (tab.url.includes(baike)) {  // 如果标签页的网址中包含baike的网址,则将插件徽标设置为“ON”const prevState = await chrome.action.getBadgeText({ tabId: tab.id })const nextState = prevState === "ON" ? "OFF" : "ON"await chrome.action.setBadgeText({tabId: tab.id,text: nextState})}})

以上两步使用到了Chrome的activeTab权限和scripting权限,因此我们需要再manifest文件中配置权限声明:

{..."permissions": ["activeTab","scripting"],...
}

此时,重新加载插件后,如果当前标签页是百科页面,点击插件,就能看到切换徽标状态的效果的。

2、注入css,隐藏元素。

当徽标状态为ON时,给百科页面注入css,隐藏掉右侧边栏。继续在background.js脚本中添加以下代码:

...
chrome.action.onClicked.addListener(async (tab) => {if (tab.url.includes(baike)) {...if (nextState === "ON") {await chrome.scripting.insertCSS({files: ["focus-mode.css"],target: { tabId: tab.id },});} else if (nextState === "OFF") {await chrome.scripting.removeCSS({files: ["focus-mode.css"],target: { tabId: tab.id },});}}})

判断徽标状态,如果是“ON”,则往目标标签页插入css文件,否则就移除掉插入到目标页的css文件。上例中插入了focus-mode.css

编写注入的css文件,已达到隐藏侧边栏和底部广告栏的目的。通过分析网页的html代码可知,class="side-content"的标签即为侧边栏的元素标签,id="tashuo_bottom"的标签即为底部广告栏的标签,因此代码如下:

// focus-mode.css.content {.side-content {display: none;}
}#tashuo_bottom {display: none;
}

3、演示:

浏览器中重新加载插件后,在百科页点击插件,就能看到默认插件的聚集模式是“OFF”状态,点击后徽标更新为“ON”状态,且页面的侧边栏和底部栏都消失了,图待更新。

标签管理功能开发

上面两个案例,我们分别体验了内容脚本和后台服务脚本的能力,接下来我们看看插件打开的页面相关的能力。

该功能我们期望点击插件,能打开一个弹窗页,在这个弹窗页中展示当前浏览器窗口的所有标签页,然后我们可以选择某些标签页将他们放在一个标签组中。

思路:实现popup页面,并在页面中展示当前浏览器窗口的所有标签页,然后页面中有按钮,可创建一个标签组,并将选中的标签页放到这个标签组中。

1、首先我们需要一个popup页面

html代码如下:

# popup.html<!DOCTYPE html>
<html lang="en"><head><meta charset="UTF-8" /><meta http-equiv="X-UA-Compatible" content="IE=edge" /><meta name="viewport" content="width=device-width, initial-scale=1.0" /><link rel="stylesheet" href="./popup.css" /></head><body><template id="li_template"><li><a><h3 class="title">Tab Title</h3><p class="pathname">Tab Pathname</p></a></li></template><h1>Google Dev Docs</h1><button>新建/取消组合</button><ul></ul><script src="./popup.js" type="module"></script></body>
</html>

需要注意的是:这个popup.html无法运行内联 JavaScript,因此我们只能通过导入的方式,引入JavaScript脚本。

css代码如下:

# popup.cssbody {width: 20rem;
}ul {list-style-type: none;padding-inline-start: 0;margin: 1rem 0;
}li {padding: 0.25rem;
}
li:nth-child(odd) {background: #80808030;
}
li:nth-child(even) {background: #ffffff;
}h3,
p {margin: 0;
}

2、点击插件,触发弹窗

要实现点击插件就能打开popup弹窗页,有两中方式:

方法1:可以在manifest文件中的action字段中配置默认弹窗,代码如下:

{..."action": {"default_popup": "popup.html"},...
}

该配置意思是点击插件时默认打开的弹窗就是“popup.html”文件。

方法2:后台脚本中通过监听点击事件,然后调用ChromeActionAPI的setPopup​方法实现。本插件因为之前实现了百度百科页面的聚焦模式功能,因此采用这种方式触发弹窗。对之前的代码做了些调整,具体如下:

// background.js// 监听运行时和安装完成的事件
chrome.runtime.onInstalled.addListener(async () => {// 获取当前浏览器窗口激活的标签页tabs = await chrome.tabs.query({active: true, currentWindow: true})// 如果标签页的url中包含百度百科的地址,就设置徽标,否则不设置徽标if(tabs[0].url.includes("baike.baidu.com/item/")){chrome.action.setBadgeText({ "text": "OFF" })}
})// 监听点击事件
chrome.action.onClicked.addListener(async (tab) => {// 如果当前页面是百度百科的页面,就更新徽标状态,并完成聚焦模式的相关功能if (tab.url.includes("baike.baidu.com/item/")) {const prevState = await chrome.action.getBadgeText({ tabId: tab.id })const nextState = prevState === "ON" ? "OFF" : "ON"await chrome.action.setBadgeText({tabId: tab.id,text: nextState})if (nextState === "ON") {await chrome.scripting.insertCSS({files: ["focus-mode.css"],target: { tabId: tab.id },});} else if (nextState === "OFF") {await chrome.scripting.removeCSS({files: ["focus-mode.css"],target: { tabId: tab.id },});}}// 如果当前页是菜鸟教程网站的页面,就弹出popup弹窗else if(tab.url.includes("www.runoob.com/")){chrome.action.setPopup({ popup: "popup/popup.html" })}})

tips:如果在manifest文件中配置了点击默认弹窗页面,那么action.onClicked事件将不会生效。

3、将所有的标签页的title和path展示在popup弹窗页中

交互逻辑通过popup.js实现
3.1 调用ChromeAPI查询当前浏览器窗口中的所有标签页:

// poppu.js// 查询符合条件的所有tabs
const tabs = await chrome.tabs.query({currentWindow: true,url: ["https://www.runoob.com/*"]
});
...

3.2 将标签页列表插入到popup页面的元素中:

...
// 按照默认顺序对tabs列表中的元素进行排序
const collator = new Intl.Collator();
tabs.sort((a, b) => collator.compare(a.title, b.title));const template = document.getElementById("li_template");  // 从页面获取模版节点
const elements = new Set();
for (const tab of tabs) {const element = template.content.firstElementChild.cloneNode(true);  // 复制模版第一个节点const title = tab.title.split("-")[0].trim();  // 获取tab中的标题const pathname = new URL(tab.url).pathname.slice("/docs".length);  // 获取tab中的路径element.querySelector(".title").textContent = title;  // 复制出来的节点中class=“title”的子节点,写入内容element.querySelector(".pathname").textContent = pathname;  // 复制出来的节点中class=“pathname”的子节点,写入内容element.querySelector("a").addEventListener("click", async () => {  // 复制出来的节点中<a>标签子节点添加点击事件// 点击的时候,更新对应的tab和窗口为激活状态await chrome.tabs.update(tab.id, { active: true });await chrome.windows.update(tab.windowId, { focused: true });});elements.add(element);
}
document.querySelector("ul").append(...elements);

以上代码中使用到了ChromeTabsAPI,该API 中的许多方法无需请求任何权限即可使用。但是,如果要访问标签页的标题和 URL这些敏感属性,就需要声明授权许可。如果我们请求“标签页”权限,导致所有标签页的敏感属性都有权限访问,但是由于我们仅仅管理特定网站的选项卡,因此我们请求更小范围的host权限。

host权限可以授权我们获取指定网站的敏感信息(包含title和URL),以此来缩小权限范围,进而保护用户隐私。在manifest文件中配置host权限:

{..."host_permissions": ["https://www.runoob.com/*"],...
}

注意:如果插件中使用了tabs权限或host权限,那么用户在安装插件时都会有弹窗提示。如下图:

3.3 ​对标签页进行分组

该功能需要使用到ChromeTabGroupsAPI,该API将允许插件对标签组进行命名和设置颜色。同样的,要使用该API也需要在manifest文件中声明权限:

{..."permissions": [..."tabGroups"]...
}

3.4 实现按钮的交互

在 popup.js 中,创建一个按钮,该按钮将使用 tabs.group() 对所有选项卡进行分组并将它们移动到当前窗口中。具体代码如下:

...
const button = document.querySelector("button");  // 获取button标签
// 给button标签添加点击事件
button.addEventListener("click", async () => {const tabIds = tabs.map(({ id }) => id);  // 解构出tabs中每个对象的id属性,并组合成tabIdsconst group = await chrome.tabs.group({ tabIds });  // 将tabIds新建成标签组await chrome.tabGroups.update(groupId, { title: "菜鸟", color: "yellow" });  // 更新标签组的标题和颜色
});

如果标签页已经在分组中,则移出分组

要将标签页移出分组需要使用到ChomeTabsAPI,因此要在manifest文件中声明tabs权限:

{..."permissions": [..."tabGroups","tabs"]...
}

因为在点击按钮时需要先判断是否已经有标签组了,所以对popo.js做如下改造:

...
const button = document.querySelector("button");  // 获取添加组合的button标签
// 给button标签添加点击事件
button.addEventListener("click", async () => {const tabGroups = await chrome.tabGroups.query({ title: "菜鸟" }) // 找到菜鸟标签组// 如果菜鸟标签组存在,则取消标签组,否则添加菜鸟标签组if (tabGroups.length) {const tabs = await chrome.tabs.query({ groupId: tabGroups[0].id })const tabIds = tabs.map(({ id }) => id)await chrome.tabs.ungroup(tabIds)} else {const tabIds = tabs.map(({ id }) => id);  // 解构出tabs中每个对象的id属性,并组合成tabIdsconst groupId = await chrome.tabs.group({ tabIds });  // 将tabIds新建成标签组await chrome.tabGroups.update(groupId, { title: "菜鸟", color: "yellow" });  // 更新标签组的标题和颜色}
});

------但是在调试过程中发现,后台服务中的点击事件存在一些Bug,点击事件中如果有根据网站来判断弹哪个页面的话,第一个条件规则触发后,后面的条件规则将不会覆盖第一个条件规则。目前没有找到原因,后续再继续研究下官方文档。

参考文档:zhuanlan.zhihu.com/p/655456499

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

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

相关文章

Win11 安装 Vim

安装包&#xff1a; 链接&#xff1a;https://pan.baidu.com/s/1Ru7HhTSotz9mteHug-Yhpw?pwd6666 提取码&#xff1a;6666 双击安装包&#xff0c;一直下一步。 配置环境变量&#xff1a; 先配置系统变量中的path&#xff1a; 接着配置用户变量&#xff1a; 在 cmd 中输入…

图的广度遍历-邻接矩阵实现

description 本题要求实现邻接矩阵存储图的广度优先遍历。 函数接口定义&#xff1a; void BFS(MGraph G,Vertex i); 其中MGraph是邻接矩阵存储的图&#xff0c;定义如下&#xff1a; #define MaxVertexNum 10 /定义最大顶点数/ typedef int Vertex;/* 用顶点下标表示顶点,…

这可能是最全的反爬虫及应对方案,再也不怕爬不到数据了

一、什么是反爬虫 网络爬虫&#xff0c;是一个自动提取网页的程序&#xff0c;它为搜索引擎从万维网上下载网页&#xff0c;是搜索引擎的重要组成。但是当网络爬虫被滥用后&#xff0c;互联网上就出现太多同质的东西&#xff0c;原创得不到保护。于是&#xff0c;很多网站开始…

Java项目:高仿12306

github地址&#xff1a;nageoffer/12306: &#x1f525; 官方推荐 &#x1f525; 大学春招、秋招、应届项目&#xff0c;SpringBoot3 Java17 SpringCloud Alibaba Vue3 等技术架构&#xff0c;完成高仿铁路 12306 用户 抢票 订单 支付服务&#xff0c;帮助学生主打就业的…

网站强制跳转至国家反诈中心该怎么办?怎么处理?如何解封?

在互联网环境中&#xff0c;网站安全是非常重要的。然而&#xff0c;在实际操作过程中&#xff0c;不少网站可能因内容问题、技术安全漏洞等原因被迫下线甚至跳转至国家反诈骗中心网址。面对这一严峻问题&#xff0c;我们如何有效解决&#xff0c;让网站恢复运行并解除强制跳转…

在排序数组中查找元素的第一个和最后一个位置

给你一个按照非递减顺序排列的整数数组 nums&#xff0c;和一个目标值 target。请你找出给定目标值在数组中的开始位置和结束位置。 如果数组中不存在目标值 target&#xff0c;返回 [-1, -1]。 你必须设计并实现时间复杂度为 O(log n) 的算法解决此问题。 示例 1&#xff1a…

解决前端二进制流下载的文件(例如:excel)打不开的问题

1. 现在后端请求数据后&#xff0c;返回了一个二进制的数据&#xff0c;我们要把它下载下来。 这是响应的数据&#xff1a; 2. 这是调用接口的地方&#xff1a; uploadOk(){if(this.files.length 0){return this.$Message.warning("请选择上传文件&#xff01;&#xff…

Multi Label Classification with Missing Labels(MLML)的几种loss设计

多标签学习这个方向问题比较多&#xff0c;可以参考多标签学习的新趋势&#xff08;2021 Survey TPAMI&#xff09; 和 部分标签学习和缺失标签学习到底什么异同&#xff1f; 这两篇偏综述性质的解释。本文重点解释下面几个重点问题&#xff1a; Multi Label Classification w…

第十五章 类和对象——友元

生活中你的家有客厅(Public)&#xff0c;有你的卧室(Private) 客厅所有来的客人都可以进去&#xff0c;但是你的卧室是私有的&#xff0c;也就是说只有你能进去 但是呢&#xff0c;你也可以允许你的好闺蜜好基友进去。 在程序里&#xff0c;有些私有属性 也想让类外特殊的一些…

【物联网】STM32的中断机制不清楚?看这篇文章就足够了

在嵌入式系统中&#xff0c;中断是一种重要的机制&#xff0c;用于处理来自外部设备的异步事件。STM32系列微控制器提供了强大的中断控制器&#xff0c;可以方便地处理各种外部中断和内部中断。本文将详细介绍STM32中断的结构和使用方法。 文章目录 1. 什么叫中断2. 中断优先级…

Jmeter基础篇

1.性能测试指标 【虚拟用户数】&#xff1a;线程用户 【并发数】&#xff1a;指在某一时间&#xff0c;一定数量的虚拟用户同时对系统的某个功能进行交互&#xff0c;一般通过集合点实现 【事务】:事务代表一个完整的功能&#xff0c;一个接口可以是事务&#xff0c;多个接口…

linux——进程间通信——管道

✅<1>主页&#xff1a;&#xff1a;我的代码爱吃辣 &#x1f4c3;<2>知识讲解&#xff1a;Linux——进程间通信——管道通信 ☂️<3>开发环境&#xff1a;Centos7 &#x1f4ac;<4>前言&#xff1a;进程间通信&#xff08;InterProcess Communication&…

Matlab杂项记录

文章目录 其他do nothing command in matlab代码格式化在同一个m文件中写多个独立的功能函数改变启动时的默认文件夹博文链接 table使用 其他 do nothing command in matlab disp() % Does nothing but allows developer to set a breakpoint here.代码格式化 Matlab编辑器具…

Python的函数

近期遇到了一个没怎么看懂的Python函数的形式。 def twoSum(self, nums: List[int], target: int) -> List[int]: 后来上网查了资料。

(二)激光线扫描-相机标定

1. 何为相机标定? 当相机拍摄照片时,我们看到的图像通常与我们实际看到的不完全相同。这是由相机镜头引起的,而且发生的频率比我们想象的要高。 这种图像的改变就是我们所说的畸变。一般来说,畸变是指直线在图像中出现弯曲或弯曲。 这种畸变我们可以通过相机标定来进行解…

阿里云服务器地域和可用区查询表_地域可用区大全

阿里云服务器地域和可用区有哪些&#xff1f;阿里云服务器地域节点遍布全球29个地域、88个可用区&#xff0c;包括中国大陆、中国香港、日本、美国、新加坡、孟买、泰国、首尔、迪拜等地域&#xff0c;同一个地域下有多个可用区可以选择&#xff0c;阿里云服务器网分享2023新版…

【AntDesign】封装全局异常处理-全局拦截器

[toc] 场景 本文前端用的是阿里的Ant-Design框架&#xff0c;其他框架也有全局拦截器&#xff0c;思路是相同&#xff0c;具体实现自行百度下吧 因为每次都需要调接口&#xff0c;都需要单独处理异常情况&#xff08;code !0&#xff09;&#xff0c;因此前端需要对后端返回的…

【Linux】Linux常用命令—文件管理(上)

创作不易&#xff0c;本篇文章如果帮助到了你&#xff0c;还请点赞 关注支持一下♡>&#x16966;<)!! 主页专栏有更多知识&#xff0c;如有疑问欢迎大家指正讨论&#xff0c;共同进步&#xff01; &#x1f525;c系列专栏&#xff1a;C/C零基础到精通 &#x1f525; 给大…

基于python的网络爬虫搜索引擎的设计

项目介绍 随着互联网的飞速发展&#xff0c;web已经成为人们主要的检索&#xff0c;和发布的主要平台&#xff0c;在海量的数据中如何快速&#xff0c;准确的找到用户所需要的信息成为人们当前所需求的&#xff0c;而网络爬虫就是为了满足这一需要而产生的研究领域。在现实中我…

【typescript】面向对象(下篇),包含接口,属性的封装,泛型

假期第八篇&#xff0c;对于基础的知识点&#xff0c;我感觉自己还是很薄弱的。 趁着假期&#xff0c;再去复习一遍 面向对象&#xff1a;程序中所有的操作都需要通过对象来完成 计算机程序的本质就是对现实事物的抽象&#xff0c;抽象的反义词是具体。比如照片是对一个具体的…