当前位置: 首页 > news >正文

Neutralinojs教程项目实战初体验(踩坑指南),干翻 electron

Neutralinojs 项目实战初体验(踩坑指南),干翻 electron

Neutralinojs 官方文档

卧槽卧槽,!这个年轻人居然用浏览器把电脑关机了_哔哩哔哩_bilibili正是在下

本教程搭建的是纯原生项目,没有和其它前端框架绑定。

如果反响还不错的话,就出集成框架(vue3)的内容。(集成react官方已提供)

本文将会搭建一个桌面端程序如图所示

本项目实现了一个所见即所得的网页实时操作页面,旨在演示一些如何搭建项目,并介绍Neutralinojs中的一些常见api,包括IPC消息通讯,托盘,消息通知等

在这里插入图片描述
在这里插入图片描述

什么是 Neutralinojs?

Neutralinojs 是一个轻量级可移植的桌面应用程序开发框架适用于 Linux、macOS 和 Windows 的预构建 x64 二进制文件。它可以让你使用 JavaScript, HTML 和 CSS 开发轻量的跨平台桌面应用。 您可以使用任何编程语言(通过扩展 IPC)扩展 Neutranojs,并将 NeutranoJS 用作任何代码的一部分(通过子进程 IPC)。

在 Electron 和 NWjs 中,你必须安装 Node.js 和成百上千的依赖库。内嵌的 Chromium 和 Node 使简单的应用也变的很臃肿。 Neutralizojs 提供了一个轻量级和可移植的 SDK,它是 Electron 和 NW.js 的替代品。 Neutralizojs不捆绑 Chromium,而是在操作系统中使用现有的 web 浏览器库(例如在 Linux 中使用 gtk-webkit2)。 Neutralizojs 为本机操作实现了 WebSocket 连接,并嵌入了一个静态 web 服务器来提供 web 内容。 此外,它还为开发人员提供了一个内置的 JavaScript 客户端库。

在使用neu cli运行neu run时候会提供一个websocket服务,我扒代码的时候发现的,如图(neu cli源码片段)

请添加图片描述

neutralinojs 是官方提供的一个 JavaScript 客户端库(也称为 Neutralino.js)供开发人员进行交互,他是如何与neu cli进行交互的呢???

没错是websocket,neutralinojs会发起websocket连接,进行发送和接收数据,如图(neutralinojs源码片段)请添加图片描述

!!!!于是,这个框架的工作原理就很清楚了

当我们使用脚手架的时候执行neu run或运行neu build打包后的可执行exe文件的时候,后台会默认启动一个websocket服务,前端页面使用neutralinojs 会发起websocket连接,前端需要调用底层代码的时候,就发送websocket给后台,后台调用系统级别的api,执行相应的操作,然后返回对应的操作结果(可能是对象,或者其它类型的数据)

提供多种模式

  • window

    Neutralinojs 应用程序将在本机窗口上运行。该窗口将使用用户的操作系统主题。 此模式是跨平台应用程序开发的不错选择。

  • browser

    Neutralinojs 应用程序将使用用户的默认浏览器来加载应用程序。 因此,您可以使用本机操作构建 Web 应用程序。您通常无法访问操作 通过 Web 浏览器提供系统级功能。但是,Neutralinojs 浏览器模式可帮助您制作可以 使用所需的安全控制访问操作系统层。

  • cloud

    此模式将 Neutralinojs 进程作为后台服务器运行。 您将能够将应用程序公开到公共网络或 Internet。可以实现手机远程操作电脑客户端功能,类似ssh

    卧槽卧槽,!这个年轻人居然用浏览器把电脑关机了_哔哩哔哩_bilibili正是在下

  • chrome

    Neutralinojs 应用程序将作为 Chrome 应用程序运行。该框架使用以下 Chrome 命令行 使 Web 应用程序看起来更像本机应用程序的参数。

提供各种原生 api

  • 系统信息
  • 剪切板
  • 文件系统
  • os

可选开放的权限

用户可以在neutralino.config.json中,配置设置开发的系统原生 api 权限

全局变量

用户可以在neutralino.config.json中,配置各种全局变量,统一配置管理

自动更新

  • 提供简单易操作的更新功能

  • 打包大小 neu build

    • 空项目打包只有 2.45M(electron40-50M)!!!
  • 打包速度

    • 毫秒级 (空项目)
    • 快!!!!按下键盘就打包完了!!!!
  • 项目运行速度 neu run

    • 基本秒起(空项目)

因为某些不可抗的原因,按照官方示例步骤,一步一步去搭建项目有时候会报错/(ㄒ o ㄒ)/~~

所以下面是遇到错误的一些解决方案

初始化项目(开始踩坑)

我们将搭建一个和框架的无关的程序

1. 安装 neu CLI

neu CLI 是用来创建 Neutralionjs 程序的脚手架程序

npm install -g @neutralinojs/neu

如果您不想进行全局安装,请将 neu CLI 与 npx 一起使用。

npx @neutralinojs/neu <command>

2. 创建新应用

输入以下命令以搭建新应用的基架

neu create newdemo

很多人在执行到 neu create newdemo 的时候卡住不动了如图

请添加图片描述

这是由于某些不可抗力,导致某些镜像下载不下来
如何解决呢??

如果下载没有问题的话(恭喜)会有这样的提示

请添加图片描述

你的项目结构是这样的!!! 执行neu run就可以启动项目了!!

请添加图片描述

我们先来认识一下,项目目录下面的文件是干什么的

  • .github

    • FUNDING.yml 作者用来拉赞助的
  • .tmp(前面介绍过)

    • **.**各种临时文件
  • bin(前面介绍过)

    • **.**
    • 存放的是一些用于启动 Neutralinojs 应用程序的主可执行程序,或者是一些辅助工具。用于配置或启动 Neutralinojs 应用,或者用于构建和打包应用程序。
    • 由于 Neutralinojs 旨在支持多个操作系统,因此 bin 文件夹可能包含针对不同操作系统(如 Windows、macOS、Linux)的特定二进制文件。
    • 用于存放与运行和构建应用程序相关的二进制文件和脚本。
  • resources

    • icons 图片 图标
    • js
      • **.js各种需要用到的js文件按
      • **.d.ts各种类型声明文件
    • index.html 项目主入口
    • index.css 样式
  • .gitignore git忽略文件

  • LICENSE 许可证书

  • neutralino.config.json 配置文件!!!!!!!非常重要,需要细说

  • neutralinojs.log 日志

3.如果下载一直卡顿,并且项目结构不是上面图片这样目录结构,或者缺失东西,项目跑步起来,那么请按照下面的步骤操作!!!(踩坑吧!!!!)

停止下载(很多时候是网络问题,多试一下,实在不行才进行下面的操作)

此时本地目录会存在一个 newdemo 文件夹,里面可能包含以下这些内容

请添加图片描述

和完整项目对比,明显会发现多了和少了很多文件

这时候我们就需要删除多余的文件,补充缺少的文件!!!!

删除.tmp目录
  • 这个文件是一个缓存临时文件夹的地方,会存在大量缓存内容,只会在搭建项目中途出现,网友们的这个文件夹中的内容应该是各不相同,因为不知道网络会卡在构建项目的某个步骤,所以这个临时缓存文件夹内容是不一样的。

  • 你咋知道.temp目录干啥的???答:我是在分析neu-cli脚手架源码的时候发现的,它会进行以下三步操作,每一步操作都会在临时文件夹里面产生一些东西,所以你会卡在什么奇怪的地方是不确定的请添加图片描述

新建bin文件夹
  • 你咋知道bin文件的???答:我在分析neu-cli脚手架源码的时候发现的,请添加图片描述

    • 介绍一下:bin这个文件目录里面存放的是一些用于启动 Neutralinojs 应用程序的主可执行程序,或者是一些辅助工具。用于配置或启动 Neutralinojs 应用,或者用于构建和打包应用程序。
    • 由于 Neutralinojs 旨在支持多个操作系统,因此 bin 文件夹可能包含针对不同操作系统(如 Windows、macOS、Linux)的特定二进制文件。
    • 用于存放与运行和构建应用程序相关的二进制文件和脚本。
  • 长这样请添加图片描述

  • 你要问了?这些文件从哪里来的????

    • 首先到neutralinojs官方项目地址里面GitHub - neutralinojs/neutralinojs: Portable and lightweight cross-platform desktop application development framework

    • 长这样请添加图片描述

    • 然后点这里,找到官方发布的二进制文件请添加图片描述

    • 我这里最新是5.2.0(2024/07/29已经是5.3.0版本了,支持背景透明)版本点击下载解压就可以了请添加图片描述

    • 压缩包里面就是这样请添加图片描述

    • 把里面的文件全部放在bin文件目录就可以了!!!!!

添加neutralino.js
  • neutralinojs 是官方提供的一个 JavaScript 客户端库(也称为 Neutralino.js)供开发人员进行交互

  • 正常情况下会有这个文件目录请添加图片描述

  • 这个目录下面存放我们的自己写的项目代码

  • 如果下载存在问题会缺失js文件目录下面的neutralino.jsneutralino.d.ts

  • 他们俩分别是我们开发中要使用到的和它的类型声明文件

  • 你又要问了!!!如何获取呢????

  • 访问地址GitHub - neutralinojs/neutralino.js: JavaScript API for Neutralinojs
    请添加图片描述

  • 执行命令
    npm install @neutralinojs/lib
    # --- or ---
    yarn add @neutralinojs/lib
    
  • 然后再node_modules文件目录下面就能找到他们俩了,把他俩放在js目录下面,就ok了请添加图片描述

启动项目!!!!

经过上面的步骤,我们的代码目录结构就成这样了

请添加图片描述

然后在控制台执行命令

neu run

出现一下画面

请添加图片描述

!!!!!恭喜你,项目启动成功

开始实战项目

认识项目目录

经过前面的步骤,我们已经可以把项目完美的启动起来了。

现在我们先来认识一下,项目目录下面的文件是干什么的

  • .github

    • FUNDING.yml 作者用来拉赞助的
  • .tmp(前面介绍过)

    • **.**各种临时文件
  • bin(前面介绍过)

    • **.**
    • 存放的是一些用于启动 Neutralinojs 应用程序的主可执行程序,或者是一些辅助工具。用于配置或启动 Neutralinojs 应用,或者用于构建和打包应用程序。
    • 由于 Neutralinojs 旨在支持多个操作系统,因此 bin 文件夹可能包含针对不同操作系统(如 Windows、macOS、Linux)的特定二进制文件。
    • 用于存放与运行和构建应用程序相关的二进制文件和脚本。
  • resources

    • icons 图片 图标
    • js
      • **.js各种需要用到的js文件
      • **.d.ts各种类型声明文件
    • index.html 项目主入口
    • index.css 样式
  • .gitignore git忽略文件

  • LICENSE 许可证书

  • neutralino.config.json 配置文件!!!!!!!非常重要,需要细说

  • neutralinojs.log 日志

neutralino.config.json详解
{"$schema": "https://raw.githubusercontent.com/neutralinojs/neutralinojs/main/schemas/neutralino.config.schema.json",// 配置schema。。没研究过"applicationId": "js.neutralino.sample",// 应用id"version": "1.0.0",// 版本号"defaultMode": "window",// 默认模式 有四种模式 对应下面的 modes属性"port": 0,// 开发端口"documentRoot": "/resources/",// 对应项目文件中的文件目录"url": "/",// url相对路径"enableServer": true,// 能够启动服务"enableNativeAPI": true,// 支持原生api"tokenSecurity": "one-time",//【cloud模式】 one-time服务器只发送一次token,客户端将保留,其它客户端访问的时候将报错,推荐;如果不传,所有客户端都能访问;您可以使用该身份验证详细信息从外部进程连接到 Neutralinojs WebSocket 作为 IPC 机制"exportAuthInfo":true,// 将身份验证详细信息导出到文件中。${NL_PATH}/.tmp/auth_info.json //您可以使用该身份验证详细信息从外部进程连接到 Neutralinojs WebSocket 作为 IPC 机制"logging": {// 日志"enabled": true,"writeToLogFile": true},// 原生api允许列表 !!!!!!!这个很重要,需要自己设置开放那些权限,官方文档有对这些权限的介绍  "nativeAllowList": ["app.*","os.*","debug.log"],
// 项目中可以使用到的全局变量  "globalVariables": {"TEST1": "Hello","TEST2": [2,4,5],"TEST3": {"value1": 10,"value2": {}}},// 可选的模式,使用不同的模式,项目使用的效果也不一样"modes": {
// 窗口设置  "window": {"title": "test1","width": 800,"height": 500,"minWidth": 400,"minHeight": 200,"center": true,"fullScreen": false,"alwaysOnTop": false,"icon": "/resources/icons/appIcon.png","enableInspector": true,"borderless": false,"maximize": false,"hidden": false,"resizable": true,"exitProcessOnClose": false,"trasparent":true,// 窗口透明,5.3v版本开始支持(我现在是5.2v,快了)"nativeAllowList": ["app.*"]// 允许的权限},"browser": {// 全局变量"globalVariables": {"TEST": "Test value browser"},// 锁定权限列表"nativeBlockList": ["filesystem.*"]},"cloud": {// 请求路径"url": "/resources/#cloud","nativeAllowList": ["app.*"]},"chrome": {"width": 800,"height": 500,"args": "--user-agent=\"Neutralinojs chrome mode\"",// 锁定的权限"nativeBlockList": ["filesystem.*","os.*"]}},// 脚手架相关的"cli": {"binaryName": "test1","resourcesPath": "/resources/","extensionsPath": "/extensions/","clientLibrary": "/resources/js/neutralino.js","binaryVersion": "5.2.0",//!!!!!!!!!!!!!!!!!!!!!!!!!!!这里版本对应的二进制文件的版本,一定要是同一个版本"clientVersion": "5.2.0"//!!!!!!!!!!!!!!!!!!!!!!!!!!!!这里版本对应的二进制文件的版本,一定要是同一个版本}
}

具体更详细内容,请参考官方文档Introduction | Neutralinojs

resources目录详解
  • icons图标图片
  • js
    • main.js业务逻辑代码
    • neutralino.js项目核心库(neutralino支持的各种api都在它身上)
    • neutralino.d.ts项目核心库类型声明文件
  • index.html应用界面
  • styles.css样式

初始化代码

  • main.js

    • 先清空,后面我们交互逻辑都会放在这个文件里面
  • index.html

    • <!DOCTYPE html>
      <html><head><meta charset="UTF-8" /><title>所见即所得</title><link rel="shortcut icon" href="./icon.ico" /><link rel="stylesheet" href="./styles.css" /></head><body><!-- <div id="neutralinoapp"><h1>嘻嘻嘻嘻嘻嘻嘻嘻neutralino.js嘻嘻嘻嘻嘻嘻嘻嘻嘻</h1><div id="info"></div><img src="/icons/logo.gif" alt="Neutralinojs" /></div><div style="display: flex"></div> --><div class="contain"><div class="left"><div class="box"><div class="title">HTML</div><textarea id="html-val"></textarea></div><div class="box"><div class="title">CSS</div><textarea id="css-val"></textarea></div><div class="box"><div class="title">JS</div><textarea id="js-val"></textarea></div></div><div class="right"><div class="box"><div class="title">实际效果</div><iframe id="show-container"></iframe></div><div class="box"><div class="title">控制台</div><div id="log-val"></div></div></div></div><script src="./js/neutralino.js"></script><script src="./js/index.js"></script><!-- Your app's source files --><!-- <script src="/js/main.js"></script> --></body>
      </html>
  • styles.css

    • * {margin: 0;padding: 0;box-sizing: border-box;
      }
      html,
      body {width: 100%;height: 100%;
      }
      .contain {width: 100%;height: 100%;display: flex;.left {color: white;min-width: 350px;width: 30%;height: 100%;background-color: white;padding: 0 20px;.box {width: 100%;height: 33%;display: flex;flex-direction: column;.title {background-color: silver;height: 10%;text-align: center;}textarea {background-color: black;color: white;padding: 20px;width: 100%;height: 90%;font-size: 20px;}}}.right {width: 70%;height: 100%;display: flex;flex-direction: column;.box {width: 100%;height: 50%;background-color: white;padding: 0 20px;.title {background-color: silver;height: 10%;text-align: center;}iframe {background-color: white;width: 100%;height: 89%;}#log-val::before {color: white;white-space: pre-wrap;content: '控制台输出 : ';}#log-val {height: 89%;overflow-y: scroll;color: rgb(240, 240, 132);background-color: black;width: 100%;padding: 10px;content: 'gege';}}}
      }
      

经过上面的初始化步骤,你的界面会变成这样!!!!!请添加图片描述

main.js实现核心业务代码(建议提前去官网看看api文档)

main.js

// 防抖函数
let debounce = (func, delay) => {let timeout = null;return function () {const _this = thisconst args = [...arguments]if (timeout) {clearTimeout(timeout)}timeout = setTimeout(() => {func.apply(_this, args)}, delay)};
};
// 节流函数
let throttle = (func, wait) => {let previous = Date.now();return function () {let now = Date.now();let context = this;let args = arguments;if (now - previous > wait) {func.apply(context, args);previous = now;}}
};// console.log(Neutralino, 'Neutralino')
// 设置系统右下角托盘
const setTray = () => {// 托盘只在window模式下支持if (NL_MODE != "window") {console.log("INFO: 托盘只在window模式下支持.");return;}// 托盘项let tray = {icon: "/resources/icons/trayIcon.png",menuItems: [{ id: "openBaidu", text: "打开百度" },// 分割线条{ id: "SEP", text: "-" },{ id: "VERSION", text: "Get version" },// 分割线条{ id: "SEP", text: "-" },{ id: "提示文字", text: "提示文字" },// 分割线条{ id: "SEP", text: "-" },{ id: "QUIT", text: "Quit" }]};// 设置系统托盘Neutralino.os.setTray(tray);
}
// 托盘右击事件
const onTrayMenuItemClicked = (event) => {switch (event.detail.id) {case "VERSION":// Display version informationNeutralino.os.showMessageBox("Version information",`Neutralinojs server: v${NL_VERSION} | Neutralinojs client: v${NL_CVERSION}`);break;case "openBaidu":// 打开指定网页Neutralino.os.open("https://www.baidu.com");break;case "提示文字":Neutralino.os.showMessageBox("我是提示标题", `我是提示文字内容`);break;case "QUIT":// Exit the applicationNeutralino.app.exit();break;}
}
// 窗口关闭事件
const onWindowClose = () => {Neutralino.app.exit();
}
/*** 万物起源,先调用init方法,然后才能使用任何 原生 API 函数!!!!* 执行init方法的时候,内部会发起websocket连接,打通交互逻辑*/
Neutralino.init()
// 监听 客户端库与 Neutralino 服务器连接成功触发事件。
Neutralino.events.on('ready', async () => {// 弹出一个提示窗口Neutralino.os.showMessageBox('red润提醒您', 'Hello Neutralinojs');// 从剪切板获取数据// let format = await Neutralino.clipboard.getFormat();// console.log(`Format: ${format}`);// 给剪切板设置文字// await Neutralino.clipboard.writeText('Test value');// 给剪切板设置图片// let image = prepareClipboardImage();// await Neutralino.clipboard.writeImage(image);// 从剪切板获取文字// let clipboardText = await Neutralino.clipboard.readText();// console.log(`Text: ${clipboardText}`);// // 从剪切板获取图片// let clipboardImage = await Neutralino.clipboard.readImage();// console.log(`Image: ${clipboardImage}`);// 选择提示框// let button = await Neutralino.os//     .showMessageBox('Confirm',//         'Are you sure you want to quit?',//         'YES_NO', 'QUESTION');// if (button == 'YES') {//     Neutralino.app.exit();// }// 右下角提示,目前win11支持异常,已经有人提了pr,待官方更新中// await Neutralino.os.showNotification('Hello world', 'It works! Have a nice day');// // 错误类型的提示// await Neutralino.os.showNotification('Oops :/', 'Something went wrong', 'ERROR');});
// 苹果系统没有托盘
if (NL_OS != "Darwin") {setTray();
}
// 监听用户单击托盘时候触发
Neutralino.events.on("trayMenuItemClicked", onTrayMenuItemClicked);
// 监听用户关闭窗口时触发事件。
Neutralino.events.on("windowClose", onWindowClose);// 获取配置文件信息
// Neutralino.app.getConfig().then(res => {
//     console.log(res, 'getConfig')
// })
// 获取windows系统环境变量
// Neutralino.os.getEnvs().then(res => {
//     console.log(res, 'getEnvs')
// })
// 输出系统信息
console.log(`APPID:${NL_APPID}PORT:${NL_PORT}OS:${NL_OS}SERVER:v${NL_VERSION}CLIENT:v${NL_CVERSION}MODE:${NL_MODE}
`);const htmlValEle = document.querySelector("#html-val");
const cssValEle = document.querySelector("#css-val");
const jsValEle = document.querySelector("#js-val");
const logValEle = document.querySelector('#log-val')
/*** @type {HTMLIFrameElement}*/
const showContainerEle = document.querySelector('#show-container')
// 初始化值
let htmlVal = htmlValEle.value = "<button onclick='test()'>hello world</button>";
let cssVal = cssValEle.value = "h1{color:red;}";
let jsVal = jsValEle.value = "function test(){alert('12334');}";
// 控制台记录行数
let countLine = 1;const run = (htmlVal, cssVal, jsVal) => {// 界面和样式showContainerEle.contentDocument.body.innerHTML = htmlVal + `<style>${cssVal}</style>`;// 计算jsshowContainerEle.contentWindow.eval(jsVal);// 拦截console,自定义自己的逻辑showContainerEle.contentWindow.console.log = (val) => {logValEle.innerText = logValEle.innerText + "\n第" + countLine + "行:" + val;countLine++;}
}run(htmlVal, cssVal, jsVal)
// 节流稳定输出
const htmlFunc = throttle((e, d) => {// oldconsole(e.target.value, 'e,d')htmlVal = e.target.value;run(htmlVal, cssVal, jsVal)
}, 100)
const cssFunc = throttle((e, d) => {// oldconsole(e.target.value, 'e,d')cssVal = e.target.value;run(htmlVal, cssVal, jsVal)
}, 100)
// 防抖防止拼写错误的时候,频繁提示错误
const jsFunc = debounce((e, d) => {// oldconsole(e.target.value, 'e,d')jsVal = e.target.value;run(htmlVal, cssVal, jsVal)
}, 1000)// 监听事件
htmlValEle.addEventListener('input', htmlFunc)
cssValEle.addEventListener('input', cssFunc)
jsValEle.addEventListener('input', jsFunc)

最终效果!!!

请添加图片描述

完结。

还有很多东西没讲完,比如比如和前端框架整合vue,react,自动更新,实现远程控制等。如果需要学习这块的人比较多的话,就更新。

这个库相对于electron还是简单许多,把官方api文档读一遍就基本没问题了

拜!

相关文章:

Neutralinojs教程项目实战初体验(踩坑指南),干翻 electron

Neutralinojs 项目实战初体验&#xff08;踩坑指南&#xff09;&#xff0c;干翻 electron Neutralinojs 官方文档 卧槽卧槽&#xff0c;&#xff01;这个年轻人居然用浏览器把电脑关机了_哔哩哔哩_bilibili正是在下 本教程搭建的是纯原生项目&#xff0c;没有和其它前端框架…...

【轻松拿捏】Java-List、Set、Map 之间的区别是什么?

List、Set、Map 之间的区别是什么&#xff1f; 一、List 二、Set 三、Map &#x1f388;边走、边悟&#x1f388;迟早会好 一、List 有序性&#xff1a;List 保持元素的插入顺序&#xff0c;即元素按添加的顺序存储和访问。允许重复&#xff1a;List 可以包含重复的元素。…...

用户史订单查询业务

文章目录 概要整体架构流程技术细节小结 概要 在电商、金融、物流等行业中&#xff0c;用户历史订单查询是一项常见的业务需求。这项功能允许用户查看他们过去的交易记录&#xff0c;包括但不限于购买的商品、服务详情、交易金额、支付状态、配送信息等。对于企业而言&#xf…...

第8节课:CSS布局与样式——掌握盒模型与定位的艺术

目录 盒模型&#xff1a;网页布局的基础盒模型的属性盒模型的示例 定位&#xff1a;控制元素位置定位的类型定位的示例 实践&#xff1a;使用CSS布局创建响应式网页结语 CSS布局是网页设计中的基石&#xff0c;它决定了网页元素的排列和分布。盒模型和定位是CSS布局中的两个核心…...

electron 主进程和渲染进程

最近在整理electron 相关的项目问题&#xff0c;对自己来说也是温故知新&#xff0c;也希望能对小伙伴们有所帮助&#xff0c;大家共同努力共同进步。加油&#xff01;&#xff01;&#xff01;&#xff01; 虽然最近一年前端大环境不好&#xff0c;但是大家还是要加油鸭&#…...

redis的高可用及性能管理和雪崩

redis的高可用 redis当中&#xff0c;高可用概念更宽泛一些。 除了正常服务以外&#xff0c;数据量的扩容&#xff0c;数据安全。 实现高可用的方式&#xff1a; 1、持久化 最简单的高可用方法&#xff0c;主要功能就是备份数据。 把内存当中的数据保存到硬盘当中。 2、主…...

php基础语法

文章目录 1. PHP(1) 安装php 2. 基础语法(1) 格式(2) 输出语法(3) 注释(4) 变量(无变量类型自动识别)(5) 输入获取(6) 定界符(7) 换行 3. 基本数据类型(1) 字符串(2) 整数(3). 浮点数(4). boolean类型(5). 数组(6). null值 4. 运算符(1) 算术运算符(2) 比较运算符(3) 逻辑运算符…...

js抓取短信验证码发送

油猴(Tampermonkey)是一个流行的浏览器扩展,它允许用户在浏览器中运行自定义的JavaScript脚本。下面是一个简单的示例脚本,用于收集网站上发送短信验证码的API请求,并以JSON格式存储这些信息。请注意,这个脚本需要根据实际网站的API请求进行调整,因为不同的网站可能有不…...

视频怎么加密?常见的四种视频加密方法和软件

视频加密是一种重要的技术手段&#xff0c;用于保护视频内容不被未经授权的用户获取、复制、修改或传播。在加密过程中&#xff0c;安企神软件作为一种专业的加密工具&#xff0c;可以发挥重要作用。 以下将详细介绍如何使用安企神软件对视频进行加密&#xff0c;并探讨视频加密…...

聚焦全局应用可用性的提升策略,详解GLSB是什么

伴随互联网的快速发展和全球化趋势的深入&#xff0c;企业对网络应用的需求日渐增长。为满足全球范围内用户大量的访问需求&#xff0c;同时解决容灾、用户就近访问以及全球应用交付等问题&#xff0c;GLSB&#xff08;全局负载均衡&#xff09;也因此应运而生。那么GLSB是什么…...

无水印下载视频2——基于tkinter完成头条视频的下载

在数字化时代的浪潮中&#xff0c;视频内容以其丰富性和便捷性&#xff0c;逐渐成为了我们获取信息和娱乐的重要途径。尤其是在短视频平台上&#xff0c;各种创意十足、内容精彩的视频层出不穷&#xff0c;更是吸引了数以亿计的用户。然而&#xff0c;随着视频内容的增加&#…...

Java学习Day17:基础篇7

继承 Java中的继承是面向对象编程中的一个核心概念&#xff0c;它允许我们定义一个类&#xff08;称为子类或派生类&#xff09;来继承另一个类&#xff08;称为父类或基类&#xff09;的属性和方法。继承提高了代码的复用性&#xff0c;使得我们不必从头开始编写所有的代码&a…...

Vue3 Pinia的创建与使用代替Vuex 全局数据共享 同步异步

介绍 提供跨组件和页面的共享状态能力&#xff0c;作为Vuex的替代品&#xff0c;专为Vue3设计的状态管理库。 Vuex&#xff1a;在Vuex中&#xff0c;更改状态必须通过Mutation或Action完成&#xff0c;手动触发更新。Pinia&#xff1a;Pinia的状态是响应式的&#xff0c;当状…...

手撕数据结构02--二分搜索(附源码)

一、理论基础 二分搜索&#xff0c;也称折半搜索、对数搜索&#xff0c;是一种在有序数组中查找某一特定元素的搜索算法。 二分搜索是一种高效的查找算法&#xff0c;适用于在已排序的数组中查找特定元素。它的基本思想是通过不断将搜索区间对半分割&#xff0c;从而快速缩小…...

单片机工程师继续从事硬件设计还是涉足 Linux 开发?

在开始前刚好我有一些资料&#xff0c;是我根据网友给的问题精心整理了一份「linux的资料从专业入门到高级教程」&#xff0c;点个关注在评论区回复“666”之后私信回复“666”&#xff0c;全部无偿共享给大家&#xff01;&#xff01;&#xff01; 怎么说呢&#xff0c;感觉绝…...

《昇思25天学习打卡营第25天|第28天》

今天是打卡的第二十八天&#xff0c;实践应用篇中的计算机视觉中Vision Transformer图像分类。 从Vision Transformer&#xff08;ViT&#xff09;简介开始了解&#xff0c;模型结构&#xff0c;模型特点&#xff0c;实验的环境准备和数据读取&#xff0c;模型解析&#xff08…...

Flutter Dio网络请求报错FormatException: Unexpected character

最近开发Flutter项目&#xff0c;网络请求采用的是Dio框架&#xff0c;在发起网络请求的时候报错&#xff1a; 网络请求返回的数据为&#xff1a; var returnCitySN {"cip": "127.0.0.1", "cid": "00", "cname": "未…...

关于@JsonSerialize序列化与@JsonDeserialize反序列化注解的使用(密码加密与解密举例)

注&#xff1a;另一种方式参考 关于TableField中TypeHandler属性&#xff0c;自定义的类型处理器的使用&#xff08;密码加密与解密举例&#xff09;http://t.csdnimg.cn/NZy4G 1.简介 1.1 序列化与反序列化 学习注解之前&#xff0c;我们可以先了解一下什么是序列化与反序列…...

第二届世界科学智能大赛逻辑推理赛道:复杂推理能力评估 #大模型技术之逻辑推理方向 #Datawhale #夏令营 <二>

第二届世界科学智能大赛逻辑推理赛道&#xff1a;复杂推理能力评估 #大模型技术之逻辑推理方向 #Datawhale #夏令营-CSDN博客 这里在上一篇的基础上&#xff0c;已经充分理解了一遍baseline的流程&#xff0c;并修复了一些后处理的问题&#xff0c;包括答案抽取&#xff0c;中间…...

C# 关于Linq延迟查询

demo: int Count 0;string[] obj { "item1", "item2", "item3", "item4", "item5", "item6" };var query obj.Where(item > IsTrue(item));// 第一次遍历foreach (var item in query){Console.WriteLine(it…...

Navicat For Mysql连接Mysql8.0报错:客户端不支持服务器请求的身份验证协议

windows通过navicat连接本地mysql时报错:Client does not support authentication protocol requested by server; consider upgrading MySQL client 一、问题原因二、解决方法1--失败1. 连接mysql客户端2. 修改加密方式3.正确的解决方法1.查找my.ini文件2.修改my.ini文件3.重…...

以西门子winCC为代表的组态界面,还是有很大提升空间的。

组态界面向来都是功能为主&#xff0c;美观和体验性为辅的&#xff0c;这也导致了国内的一些跟随者如法炮制&#xff0c;而且很多操作的工程师也是认可这重模式&#xff0c;不过现在一些新的组态软件可是支持精美的定制化界面&#xff0c;还有3D交互效果&#xff0c;这就是确实…...

HomeServer平台选择,介绍常用功能

​​ 平台选择 HomeServer 的性能要求不高&#xff0c;以下是我的硬件参数&#xff0c;可供参考&#xff1a; ‍ 硬件&#xff1a; 平台&#xff1a;旧笔记本CPU&#xff1a;i5 4210u内存 8G硬盘&#xff1a;128G 固态做系统盘&#xff0c;1T1T 机械盘组 RAID1 做存储。硬…...

记录一个k8s集群zookeeper部署过程

由于网管中心交维要求必须是支持高可用配置&#xff0c;原先单节点的zookeeper不被允许。所以在k8s集群中做了一个高可用版本的zookeeper。 期间有点小波折&#xff0c;官方给的镜像版本太老&#xff0c;业务不支持&#xff0c;所以手动做了下处理&#xff0c;重新打了一个镜像…...

TapData 信创数据源 | 国产信创数据库 TiDB 数据迁移指南,加速国产化进程,推进自主创新建设

随着国家对自主可控的日益重视&#xff0c;目前在各个行业和区域中面临越来越多的国产化&#xff0c;采用有自主知识产权的国产数据库正在成为主流。长期以来&#xff0c;作为拥有纯国产自研背景的 TapData&#xff0c;自是非常重视对于更多国产信创数据库的数据连接器支持&…...

开始写人工智能

文章目录 概述 概述 开始写人工智能模块。既然决定开始写这些&#xff0c;那就开始吧&#xff01;...

盘点.软件测试模型

软件开发模型   软件开发模型(Software Development Model)是指软件开发全部过程、活动和任务的结构框架。软件开发包括需求、设计、编码和测试等阶段&#xff0c;有时也包括维护阶段。 软件开发模型能清晰、直观地表达软件开发全过程&#xff0c;明确规定了要完成的主要活动…...

燃气安全无小事,一双专业劳保鞋让你步步安心!

燃气作为我们日常生活中不可或缺的能源之一&#xff0c;为我们的生活提供了极大便利&#xff0c;其安全性往往被忽视在忙碌的日常生活背后。然而&#xff0c;燃气事故一旦发生&#xff0c;后果往往不堪设想&#xff0c;轻则财产损失&#xff0c;重则危及生命。因此&#xff0c;…...

springboot校园服装租赁系统-计算机毕业设计源码30824

目 录 摘要 1 绪论 1.1 研究背景与意义 1.2国内外研究现状 1.3论文结构与章节安排 2 校园服装租赁系统分析 2.1 可行性分析 2.1.1 技术可行性分析 2.1.2 经济可行性分析 2.1.3 法律可行性分析 2.2 系统功能分析 2.2.1 功能性分析 2.2.2 非功能性分析 2.3 系统用例…...

线性回归和逻辑回归揭示数据的隐藏模式:理论与实践全解析

机器学习之线性回归和逻辑回归 1. 简介1.1 机器学习概述1.2 监督学习的定义与重要性1.3 线性回归和逻辑回归在监督学习中的作用1.3.1 线性回归1.3.2 逻辑回归 2. 线性回归&#xff08;Linear Regression&#xff09;2.1 定义与目标2.1.1 回归问题的定义2.1.2 预测连续目标变量 …...

掌握采购询价软件:高效比较供应商报价的技巧

在企业运营中&#xff0c;获取所需的产品往往是一项复杂且耗时的任务&#xff0c;这涉及多个环节和流程。然而&#xff0c;借助电子采购询价&#xff08;RFQ&#xff09;系统&#xff0c;许多原本需要采购员手动完成的任务可以自动化运行&#xff0c;从而提高了效率。 那么问题…...

AMQP-核心概念-终章

本文参考以下链接摘录翻译&#xff1a; https://www.rabbitmq.com/tutorials/amqp-concepts 连接&#xff08;Connections&#xff09; AMQP 0-9-1连接通常是长期保持的。AMQP 0-9-1是一个应用级别的协议&#xff0c;它使用TCP来实现可靠传输。连接使用认证且可以使用TLS保护…...

在WPF中使用WebView2详解

Microsoft Edge WebView2 Microsoft Edge WebView2 控件允许在本机应用中嵌入 web 技术(HTML、CSS 以及 JavaScript)。 WebView2 控件使用 Microsoft Edge 作为绘制引擎&#xff0c;以在本机应用中显示 web 内容。 使用 WebView2 可以在本机应用的不同部分嵌入 Web 代码&…...

僵尸进程的例子

以下是一个简单的C语言程序示例&#xff0c;该程序将创建一个子进程&#xff0c;然后子进程退出&#xff0c;但是父进程不会调用wait()或waitpid()来回收子进程的状态&#xff0c;从而使得子进程成为僵尸进程。 #include <stdio.h> #include <stdlib.h> #include …...

消息中间件分享

消息中间件分享 1 为什么使用消息队列2 消息队列有什么缺点3 如何保证消息队列的高可用4 如何处理消息丢失的问题?5 如何保证消息的顺序性1 为什么使用消息队列 解耦、异步、削峰 解耦 不使用中间件的场景 使用中间件的场景 异步 不使用中间件 使用中间件 削峰 不使…...

12. kubernetes调度——污点Taint和容忍Toleration

kubernetes调度——污点Taint和容忍Toleration 一、通过节点属性调度1、节点名称2、节点标签2.1 查看节点标签2.2 添加标签2.3 修改标签2.4 删除标签2.5 通过节点标签进行调度 二、污点Taint和容忍Toleration1、污点Taint1.1 查看Master节点的污点1.2 添加污点1.3 删除污点 2、…...

第100+18步 ChatGPT学习:R实现SVM分类

基于R 4.2.2版本演示 一、写在前面 有不少大佬问做机器学习分类能不能用R语言&#xff0c;不想学Python咯。 答曰&#xff1a;可&#xff01;用GPT或者Kimi转一下就得了呗。 加上最近也没啥内容写了&#xff0c;就帮各位搬运一下吧。 二、R代码实现SVM分类 &#xff08;1&a…...

react函数学习——useState函数

在 React 中&#xff0c;useState 是一个钩子&#xff08;hook&#xff09;&#xff0c;用于在函数组件中添加状态管理功能。它返回一个数组&#xff0c;包含两个元素&#xff1a; 当前状态值&#xff08;selectedValue&#xff09;&#xff1a;这是状态的当前值。更新状态的函…...

方天云智慧平台系统 GetCompanyItem SQL注入漏洞复现

0x01 产品简介 方天云智慧平台系统,作为方天科技公司的重要产品,是一款面向企业全流程的业务管理功能平台,集成了ERP(企业资源规划)、MES(车间执行系统)、APS(先进规划与排程)、PLM(产品生命周期)、CRM(客户关系管理)等多种功能模块,旨在通过云端服务为企业提供…...

C语言同时在一行声明指针和整型变量

如果这么写&#xff0c; int *f, g; 并没有声明2个指针&#xff0c;编译器自己会识别&#xff0c;f是一个指针&#xff0c;g是一个整型变量&#xff1b; void CTszbView::OnDraw(CDC* pDC) {CTszbDoc* pDoc GetDocument();ASSERT_VALID(pDoc);// TODO: add draw code for nat…...

thinkphp框架远程代码执行

一、环境 vulfocus网上自行下载 启动命令&#xff1a; docker run -d --privileged -p 8081:80 -v /var/run/docker.sock:/var/run/docker.sock -e VUL_IP192.168.131.144 8e55f85571c8 一定添加--privileged不然只能拉取环境首页不显示 二、thinkphp远程代码执行 首页&a…...

【公式】博弈论中的核心算法:纳什均衡公式解析

博弈论中的核心算法:纳什均衡公式解析 纳什均衡的基本概念 纳什均衡是博弈论中的一个核心概念,它描述了一个博弈中所有参与者都无法通过单方面改变自己的策略来增加收益的状态。在纳什均衡状态下,每个参与者的策略都是对其他参与者策略的最优反应。纳什均衡的公式可以表示…...

计算机网络面试题2

WebSocket相关知识 什么是WebSocket? WebSocket是一种基于TCP连接的全双工通信协议&#xff0c;即客户端和服务器可以同时发送和接收数据 WebSocket和HTTP有什么区别&#xff1f; 1.WebSocket是双向通信协议&#xff0c;HTTP是单向通信协议 2.WebSocket使用ws://或者wss:/…...

Linux网络——深入理解传入层协议TCP

目录 一、前导知识 1.1 TCP协议段格式 1.2 TCP全双工本质 二、三次握手 2.1 标记位 2.2 三次握手 2.3 捎带应答 2.4 标记位 RST 三、四次挥手 3.1 标记位 FIN 四、确认应答(ACK)机制 五、超时重传机制 六 TCP 流量控制 6.1 16位窗口大小 6.2 标记位 PSH 6.3 标记…...

快速搞定分布式RabbitMQ---RabbitMQ进阶与实战

本篇内容是本人精心整理&#xff1b;主要讲述RabbitMQ的核心特性&#xff1b;RabbitMQ的环境搭建与控制台的详解&#xff1b;RabbitMQ的核心API&#xff1b;RabbitMQ的高级特性;RabbitMQ集群的搭建&#xff1b;还会做RabbitMQ和Springboot的整合&#xff1b;内容会比较多&#…...

5万字长文吃透快手大数据面试题及参考答案(持续更新)

目录 Flink为什么用aggregate()不用process() 为什么使用aggregate() 为什么不用process() 自定义UDF, UDTF实现步骤,有哪些方法?UDTF中的ObjectInspector了解吗? 自定义UDF实现步骤 自定义UDTF实现步骤 UDTF中的ObjectInspector Spark Streaming和Flink的区别 Flu…...

WordPress原创插件:启用关闭经典编辑器和小工具

WordPress原创插件&#xff1a;启用关闭经典编辑器和小工具 主要功能 如图所示&#xff0c;用于启用或禁用经典编辑器和经典小工具&#xff0c;以替代Gutenberg编辑器。 插件下载 https://download.csdn.net/download/huayula/89592822...

萝卜快跑:自动驾驶的先锋与挑战

萝卜快跑&#xff1a;自动驾驶的先锋与挑战 萝卜快跑作为自动驾驶领域的重要参与者&#xff0c;被视为自动驾驶的先锋。它代表了自动驾驶技术在实际应用中的重要突破&#xff0c;为人们的出行方式带来了革新。萝卜快跑的发展展示了自动驾驶技术的巨大潜力&#xff0c;如提高交通…...

得到xml所有label 名字和数量 get_xml_lab.py,get_json_lab.py

import os import xml.etree.ElementTree as ETrootdir2 r"F:\images3\xmls" file_list os.listdir(rootdir2) # 列出文件夹下所有的目录与文件# 初始化字典 classes_dict {}for file_name in file_list:path os.path.join(rootdir2, file_name)if os.path.isfi…...

数据结构算法-排序(二)

插入排序 插入排序核心 假设数组中的一部数据已经排好序&#xff0c;要插入的数据和这些数据进行比较&#xff0c;直到找到合适的位置插入新数据。 插入排序步骤 插入排序主要有以下步骤构成&#xff1a; 假设有序&#xff0c;我们假设**a[0]**已经排好序待插入的数据为a[j]…...