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

news2025/1/10 20:34:31

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 = this
        const 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 information
            Neutralino.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 application
            Neutralino.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>`;
    // 计算js
    showContainerEle.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文档读一遍就基本没问题了

拜!

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

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

相关文章

【C++BFS】959. 由斜杠划分区域

本文涉及知识点 CBFS算法 LeetCode959. 由斜杠划分区域 在由 1 x 1 方格组成的 n x n 网格 grid 中&#xff0c;每个 1 x 1 方块由 ‘/’、‘’ 或空格构成。这些字符会将方块划分为一些共边的区域。 给定网格 grid 表示为一个字符串数组&#xff0c;返回 区域的数量 。 请注…

整理压缩JavaScript文件

你是不是会担心有专业人员通过你写的JavaScript语法来判断你的接口数据&#xff0c;今天就介绍一种可以封装你写的JavaScript的工具 环境&#xff1a;Windows10、node20.11.1、Vue/cli 5.0.8 uglify-js 是一个流行的 JavaScript 压缩工具&#xff0c;用于减少 JavaScript 文件…

angular入门基础教程(二)第一个angular组件

ng中的语法跟vue中是一样的插值语法&#xff0c;其实也是早期vue抄的ng的思路&#xff0c;使用{{variable}}形式&#xff0c;vue借鉴了ng和react&#xff0c;这个我们就不多了。 新建一个子组件 在项目根目录下面&#xff0c;执行 ng g component ./components/UserList这样…

Transformer——逐步详解架构和完整代码搭建

好久没更新博客&#xff0c;后面更新会勤一些。今天想聊一下Transformer&#xff0c;Transformer在NLP和CV领域都有着重要的价值&#xff0c;甚至可以看作是一个基础模型&#xff0c;这篇博客将通过详细代码深入解析Transformer模型总体架构图各个部分的的作用和搭建:论文链接&…

angular入门基础教程(九)依赖注入(DI)

依赖注入 Angular 中的依赖注入&#xff08;DI&#xff09;是框架最强大的特性之一。可以将依赖注入视为 Angular 在运行时为你的应用 提供所需资源的能力。依赖项可以是服务或其他资源。 使用服务的一种方式是作为与数据和 API 交互的方式。为了使服务可重用&#xff0c;应该…

晋升有望,5本易录用的计算机三四区潜力刊,通过率>50%,2个月超速接收,好发

今天模术狮给大家整理了5本让你晋升有望&#xff01;易录用的计算机三区四区潜力刊&#xff0c;通过率&#xff1e;50%&#xff0c;2个月超速接收&#xff0c;好发&#xff01; 1 DATA MINING AND KNOWLEDGE DISCOVERY 期刊简介&#xff1a;数据收集、存储和分发方面的进步产生…

Qt系统机制

Qt系统 Qt文件概述输入输出设备类QFileQFileInfoQt多线程Qt多线程常用API使用Qt多线程 线程安全互斥锁读写锁条件变量信号量 Qt网络QUdpSocketQNetworkDatagram设计一个UDP回显服务器QTcpServerQTcpSocketTcp版本的回显服务器HttpClient核心API Qt 音频Qt视频 Qt文件概述 ⽂件操…

阿里云服务器系统盘扩容后,宝塔面板不显示新容量的问题

1. 安装 growpart 扩展 yum install -y cloud-utils-growpart 2. 运行fdisk -l命令查看磁盘实际大小。 fdisk -l 用于:查看磁盘实际大小 说明:磁盘(/dev/vda)实际大小为:250 G。 3. 运行df -h命令查看磁盘分配大小。 df -h 用于:查看磁盘分配大小 4. 自适应分区扩容 g…

2024 Navicat Premium最新版简体中文版破解激活永久图文详细教程(亲测可用)

1.官网下载&#xff1a;下载地址 2.百度网盘下载&#xff1a;下载地址 3.未安装过的用户可直接跳过该步骤&#xff0c;如果已安装Navicat&#xff0c;记得先卸载干净&#xff0c;防止破解失效&#xff0c;卸载完成后执行补丁压缩包中的Navicat.bat脚本&#xff08;一闪而过表示…

热门景区精准客流统计也能如此简单做到了

在热门景区&#xff0c;游客如织&#xff0c;如何实现精准的客流统计成为了景区管理者关注的重点。令人欣喜的是&#xff0c;如今这一难题已经有了简单而有效的解决方案。 一、景区应用客流统计的原因 首先&#xff0c;热门景区承载着巨大的游客流量&#xff0c;为了确保游客的…

如何使用git拉取gitee上面的项目/代码?(超简单)

一、下载git软件 下载地址&#xff1a;git官网地址 1.点击右边小电脑上的按钮下载 2.选择自己电脑对应的系统 3.基本都是默认&#xff0c;这里需要勾一下就ok 4.正在安装 2.使用git软件 1.如何打开git 找到你想要操作的文件夹&#xff0c;右击open git bash here就可以…

云计算实训16——关于web,http协议,https协议,apache,nginx的学习与认知

一、web基本概念和常识 1.Web Web 服务是动态的、可交互的、跨平台的和图形化的为⽤户提供的⼀种在互联⽹上浏览信息的服务。 2.web服务器&#xff08;web server&#xff09; 也称HTTP服务器&#xff08;HTTP server&#xff09;&#xff0c;主要有 Nginx、Apache、Tomcat 等。…

【参会邀请】第四届区块链技术与信息安全国际会议(ICBCTIS 2024)诚邀相聚江城!

我们诚挚地邀请您参与第四届区块链技术与信息安全国际会议&#xff08;ICBCTIS 2024&#xff09;。本届会议将于2024年8月17日~19日在中国武汉召开。会议将围绕区块链技术与信息安全等相关研究领域&#xff0c;特邀国内外数位在此领域学术卓越的学者专家做相关致辞与报告&#…

一行Python代码实现神奇效果:创意编程实例

文末赠免费精品编程资料~~ 1. 基础中的魔法&#xff1a;打印艺术 目标&#xff1a;用一行代码打印出一个简单的图案&#xff0c;比如心形。 print( .join([**i for i in range(1, 6)] [ *4] [**i for i in range(5, 0, -1)]))解析&#xff1a; 我们利用列表推导式生成两部…

Halcon深度学习分类模型

1.Halcon20之后深度学习支持CPU训练模型&#xff0c;没有money买显卡的小伙伴有福了。但是缺点也很明显&#xff0c;就是训练速度超级慢&#xff0c;推理效果也没有GPU好&#xff0c;不过学习用足够。 2.分类模型是Halcon深度学习最简单的模型&#xff0c;可以用在物品分类&…

说真的,内裤袜子丢进洗衣机比手洗好!内裤袜子洗衣机推荐

内裤和袜子作为日常生活中不可或缺的贴身衣物&#xff0c;其清洁卫生尤为重要&#xff0c;但频繁的洗涤工作往往令人感到繁琐。正是因为这一清洗需求&#xff0c;内裤袜子洗衣机应运而生&#xff0c;它不仅为我们的生活带来了便利&#xff0c;更体现了对个人卫生和生活品质的重…

货拉拉论文入选亚太消费者研究会议及亚太营销国际学术会议

近日,亚太消费者研究会议(AP-ACR)召开。本次会议上,货拉拉和香港中文大学合作就论文《Why Showing Multiple Options Simultaneously Makes Customers Less Picky》(《为什么同步显示多个选项会使消费者变得更不挑剔》)进行主题报告。此前,本篇论文也曾在第二届亚太营销国际学术…

libevent入门篇

文章目录 概述下载编译目录samplehello-world初始化创建监听器处理连接处理信号 build 小结 概述 libevent 和 libev 都是由 c 实现的异步事件库&#xff1b;注册异步事件&#xff0c;检测异步事件&#xff0c;根据事件的触发先 后顺序&#xff0c;调用相对应回调函数处理事件…

命令行使用ADB,不用root,完美卸载小米预装软件

ADB安装与运行 install java 下载安装 注意选择JDK17以上版本 https://www.oracle.com/java/technologies/downloads/#jdk22-windows 选择中间的安装文件下载 编辑系统变量 C:\Program Files (x86)\Java\jdk-22 C:\Program Files (x86)\Java\jdk-22\bin 把C:\Progra…