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

Shell 脚本基础

一、Shell 简介

1.Shell 的定义与作用

Shell,通常被称为命令行解释器 (Command Line Interpreter),是用户 👤 与 Linux/Unix 操作系统内核进行交互 ↔️ 的“桥梁” 🌉。它扮演着翻译官 🗣️ 的角色:

  • 接收用户输入的命令 (如 ls, cd, mkdir)。
  • 解释这些命令 🧐。
  • 调用操作系统内核执行相应的功能。
  • 将执行结果返回并显示给用户 🖥️。

简单来说,没有 Shell,我们就很难方便地直接与强大的操作系统内核打交道了!💻 (简直无法想象!)

2.常见的 Shell 类型

存在多种不同的 Shell 实现,它们在语法、功能和效率上可能有所差异 ✅:

  • bash (Bourne Again SHell): 这是目前最流行、最常用 ⭐ 的 Shell,是绝大多数 Linux 发行版的默认 Shell。它兼容 sh,并增加了许多新特性(命令历史、命令补全、作业控制等)。我们后续的脚本主要会以 bash 为基础。
  • sh (Bourne Shell): 一个早期且经典 📜 的 Unix Shell,由 Stephen Bourne 开发。许多旧的或要求高兼容性 🤝 的脚本仍会使用它。在很多系统中,/bin/sh 可能是一个指向 bash (或其他 Shell) 的符号链接,但 bash 在以 sh 模式运行时会限制某些功能以保持兼容。
  • zsh (Z Shell): 一个功能极其强大 💪 的 Shell,提供了比 bash 更丰富的功能、更强的自动补全 ✨、主题和插件系统(如 Oh My Zsh),深受许多开发者喜爱 ❤️。
  • ksh (Korn Shell): 由 David Korn 开发,试图融合 sh 的脚本能力和 C Shell 的交互特性。
  • csh/tcsh (C Shell / TENEX C Shell): 语法风格类似 C 语言,交互性较好,但脚本编写方面有时被认为不如 Bourne 系列 Shell。

小提示: 💡 你可以在终端输入 echo $SHELL 来查看你当前正在使用的 Shell 类型。

3.Shell 与 Shell 脚本的区别

这是一个非常基础重要的概念 🤔:

  • Shell (交互式): 指的是你直接在终端(命令行界面)中输入命令,按回车后立即看到执行结果的环境。这是一种你问我答式的交互方式 💬。
# 这就是在交互式 Shell 中输入命令
ls -l
pwd
  • Shell 脚本 (非交互式): 是一个文本文件 📜,里面包含了一系列按照特定顺序编写的 Shell 命令。你可以一次性执行这个脚本文件,让计算机自动完成 🤖 一系列复杂的或重复性的任务,实现自动化。这是一种批量处理的执行方式 ⚙️。
# my_script.sh 文件内容可能像这样:
# #!/bin/bash
# echo "开始执行任务..."
# mkdir temp_dir
# cd temp_dir
# echo "任务完成!"

二、 编写第一个 Shell 脚本

1.脚本文件的创建与命名

创建一个 Shell 脚本非常简单

  1. 使用文本编辑器 打开你喜欢的文本编辑器(如 vim, nano, gedit, VS Code 等)📝。
  2. 编写命令: 在文件中输入你的 Shell 命令。
  3. 保存文件 将文件保存 💾。
    • 命名约定 🏷️: 虽然不是强制性的,但强烈建议将 Shell 脚本文件以 .sh 后缀结尾(例如 my_first_script.sh, backup_data.sh)。这有助于识别文件类型 👀,并且让其他用户(以及未来的你 😅)更容易理解。📄
2.指定解释器 (Shebang)

这是 Shell 脚本中至关重要❗ 的一行! ✨

  • 什么是 Shebang? 脚本文件的第一行内容,格式为 #!解释器路径
  • 作用: 它告诉操作系统内核,当直接执行这个脚本文件时(即使用方法 2:./script.sh 方式),应该调用哪个解释器来处理文件中的命令。 注意: 如果使用方法 1 (bash script.sh) 或方法 3 (source script.sh) 执行,Shebang 行实际上会被忽略 🤷‍♀️,因为解释器已经由用户显式指定 (方法 1) 或就是当前 Shell (方法 3)。
  • 绝对要求: Shebang 行必须是脚本文件的绝对第一行,前面不能有任何字符,包括空格或空行。

常见的 Shebang 选项:

虽然 #!/bin/bash最常见的,尤其是在 Linux 环境中,但还有其他一些常用的选项,选择哪个取决于你的脚本需求和目标运行环境:

  1. #!/bin/bash:

    • 指定使用 Bourne Again Shell (bash) 来执行脚本。
    • 优点: 可以使用 bash 提供的所有扩展功能 💪(如数组、双方括号 [[ ]] 条件测试、函数等)。
    • 缺点: 如果系统没有安装 bash 或者 bash 不在 /bin/bash 路径下,脚本会执行失败 💥。依赖 bash 特性的脚本可移植性稍差
  2. #!/bin/sh:

    • 指定使用 Bourne Shell (sh) 来执行脚本。
    • 优点: 目标是编写符合 POSIX 标准 📜 的脚本,具有更好的可移植性 🌍。在很多系统中,/bin/sh 可能是一个指向 bashdash 或其他兼容 Shell 的符号链接。当 bashsh 模式被调用时,它会禁用一些 bash 特有的扩展。
    • 缺点: 不能使用 bash 等现代 Shell 的扩展语法和功能,编写复杂脚本可能更繁琐 😓。
  3. #!/usr/bin/env <interpreter> (例如 #!/usr/bin/env bash#!/usr/bin/env python):

    • 这是强烈推荐 👍 的一种方式,特别是对于需要分发 📦 的脚本。
    • env 是一个命令,它会在系统的PATH 环境变量 🗺️ 所指定的目录中查找第一个找到的 <interpreter>(如 bash, python, perl, node 等)并使用它来执行脚本。
    • 优点: 极大地提高了可移植性 🚀。不同系统或用户可能将解释器安装在不同的路径下(如 /usr/bin/bash, /usr/local/bin/bash, /opt/bin/bash)。使用 env 可以确保只要解释器在用户的 PATH 中,脚本就能找到它 👌。
    • 缺点: 相比直接指定路径,会有极其微小的额外启动开销(查找解释器)。安全性方面,如果 PATH 被恶意篡改,可能会执行非预期的解释器(但这种情况很少见且通常已有更大的安全问题 😨)。
    • 常见示例:
      • #!/usr/bin/env bash (推荐 ✅ 的 bash 脚本 Shebang)
      • #!/usr/bin/env sh (推荐 ✅ 的可移植 sh 脚本 Shebang)
      • #!/usr/bin/env python3 (用于 Python 3 脚本 🐍)
      • #!/usr/bin/env perl (用于 Perl 脚本 🐪)
      • #!/usr/bin/env node (用于 Node.js 脚本 🟢)
  4. 其他 Shell:

    • #!/bin/zsh#!/usr/bin/env zsh: 指定使用 Z Shell
    • #!/bin/ksh#!/usr/bin/env ksh: 指定使用 Korn Shell

🤔 如何选择 Shebang?

  • 如果你的脚本只在你自己控制的、确定安装了 bash 的 Linux 系统上运行,并且你需要 bash 的特性,#!/bin/bash#!/usr/bin/env bash 都可以。
  • 如果你希望脚本尽可能广泛地兼容 🌍 各种 Unix-like 系统(包括 macOS, BSD 等),并且只使用了 sh 的标准功能,那么使用 #!/bin/sh#!/usr/bin/env sh更安全、更推荐的选择 ✅。
  • 为了最大化可移植性 🚀,通常首选推荐使用 #!/usr/bin/env <interpreter> 的形式 👍。
3.脚本的执行方式

你有三种主要的方式来运行你的 Shell 脚本 ▶️:

  1. 作为解释器的参数执行:
    • 这种方式不需要给脚本文件设置执行权限 🔑。
    • 你直接调用 Shell 解释器 (如 bashsh),并将脚本文件名作为参数传递给它。
    • 优点: 简单直接,无需关心执行权限。
    • 注意: 脚本在子 Shell 🐣 (subshell) 中执行,脚本中定义的变量、函数或对环境的修改(如 cd)在脚本结束后不会影响当前的 Shell 环境 🏠。
    • 命令格式:
bash my_script.sh
# 或者,如果脚本兼容 sh
sh my_script.sh
  1. 赋予执行权限后直接运行:
    • 第一步:添加执行权限 🔑: 使用 chmod 命令给脚本文件添加执行权限 (x)
chmod +x my_script.sh

+x 表示为所有用户(所有者、同组用户、其他用户)添加执行权限。你也可以使用更精细的权限设置,如 chmod u+x my_script.sh 只为文件所有者添加执行权限。

  • 第二步:直接执行:如果脚本文件位于当前目录,你需要使用 ./ 来告诉 Shell 在当前目录下查找并执行该文件:
./my_script.sh
  • 如果脚本文件位于系统的 PATH 环境变量 🗺️ 所包含的目录中,或者你指定了完整的绝对/相对路径,则可以直接使用脚本名或路径执行:
# 假设 /opt/scripts 在 PATH 中,且 my_script.sh 在该目录下
my_script.sh
# 或者使用绝对路径
/path/to/your/script/my_script.sh
# 或者使用相对路径
../scripts/my_script.sh
  • 优点: 这是最标准、常用 ✅ 的执行方式,特别是对于需要分发的工具脚本。Shebang 行 #! 决定了使用哪个解释器。
  • 注意: 脚本同样在子 Shell 🐣 (subshell) 中执行,对当前 Shell 环境没有持久影响 🏠。
  • 为什么需要 ./ 出于安全考虑 🛡️。Linux/Unix 系统默认不会在当前目录下查找可执行文件(除非当前目录被显式添加到了 PATH 环境变量中)。使用 ./ 可以明确指定执行当前目录下的文件,防止意外执行了与系统命令同名的恶意脚本 😈。
  1. 使用 source. 命令在当前 Shell 环境 🏠 中执行: 🎯
    • source 是一个 Shell 内建命令,它的简写形式是一个点 (.)。
    • 这种方式不需要给脚本文件设置执行权限 🔑(只需要读权限 r 即可 📖)。
    • 命令格式:
source my_script.sh
# 或者使用点号 (点号和脚本名之间必须有空格!)
. my_script.sh
  • 核心区别: 核心区别: 使用 source. 执行时,脚本中的命令直接在当前的 Shell 环境中 🏠 执行,而不是启动一个新的子 Shell 🐣。
  • 优点/用途: 这意味着脚本中定义的变量、函数、别名 (alias),或者执行的 cdexport 等命令,在脚本执行完毕后会保留在当前的 Shell 会话中 ✅。因此,它常用于:
    • 加载配置文件 ⚙️ (如 .bashrc, .profile, /etc/profile)。
    • 设置环境变量 🌐 供后续命令使用。
    • 定义可以在当前终端后续使用的函数库 📚。
    • 在脚本中切换目录 (cd) 后,希望停留在那个新目录 👉。
  • 注意: 必须确保脚本内容是你希望在当前环境中执行的,因为它可能直接修改 💥 你当前的工作环境。要谨慎使用!⚠️

🎯 执行方式对比小结:

执行方式需要执行权限 (x)? 🔑在子 Shell 中执行? 🐣对当前 Shell 环境的影响 🏠主要用途 🎯
bash script.sh运行一次性任务,不需改变当前环境
./script.sh (需+x)标准的脚本执行 ✅,分发工具脚本
source script.sh (需r) 📖有 (持久)加载配置/环境/函数 ⚙️,改变当前环境

三、注释与可读性

编写清晰易懂 ✨ 的脚本至关重要,而注释是实现这一目标的关键工具 🛠️。

1.单行注释
  • 在 Shell 脚本中,使用 # 符号来表示单行注释 💬。
  • # 开始,直到该行的末尾,所有的内容都会被 Shell 解释器忽略 🙈,不会被执行。
  • 注释可以单独占一行,也可以放在命令的后面。
#!/bin/bash# 这是一个完整的单行注释,用于说明脚本的目的
# Author: Your Name
# Date: 2025-5-2echo "Hello, World!" # 这也是一个注释,解释这行命令的作用# 下面定义一个变量
USER_NAME="Alice" # 为变量赋值
2.多行注释的实现方式

Shell 本身没有提供像 C 语言 /* ... */ 或 Python """ ... """ 那样的原生多行注释块 🧱 语法。但是,可以通过一些技巧 🎩 来实现类似的效果:

  • 方法一:使用 Here Document (推荐用于注释大段代码或文本)
    将你想要注释掉的内容通过 Here Document (<<LIMITER ... LIMITER) 重定向给一个空命令 : (冒号是一个内置的不做任何事情的命令)。LIMITER 可以是任何不出现在注释内容中的字符串,通常使用 EOFCOMMENT。在起始 LIMITER 前加上单引号 (: <<'EOF') 可以阻止其中的变量和命令替换。
#!/bin/bashecho "这行代码会执行": <<'MY_MULTI_LINE_COMMENT'
这里是第一行注释内容。
这里是第二行注释内容。
这整块代码/文字都不会被执行。
可以包含 $VARIABLES 或 `commands`,因为用了 'MY_MULTI_LINE_COMMENT'。
MY_MULTI_LINE_COMMENTecho "这行代码也会执行"
  • 方法二:连续使用单行注释
    这是最简单直接的方法,只需在每一行注释前都加上 #。对于解释性的多行文字,这种方式也很常用。
#!/bin/bash# ==================================
# 这是一个多行注释块
# 用来详细说明某个复杂函数的逻辑
# 或者记录一些重要的注意事项
# ==================================
calculate_sum() {
# 函数内部的注释
local result=$(($1 + $2)) # 计算总和
echo $result
}sum=$(calculate_sum 10 20)
echo "计算结果是: $sum"
3.良好的注释习惯与代码可读性

注释的终极目标 🏆 是提升代码的可读性 👀 和可维护性 🔧。遵循以下原则:

  • 解释“为什么”(Why 🤔),而不是“干什么”(What ✅): 好的代码(比如有意义的变量名和函数名)本身应该能说明它在做什么。注释应该聚焦于解释为什么选择这种实现方式、背后的逻辑、潜在的陷阱 ⚠️ 或重要的上下文
# 不好的注释 (冗余 🤷)
# i = i + 1  # 把 i 加 1# 好的注释 (解释原因 👍)
# 使用特定的算法ID,因为旧版本存在兼容性问题
algorithm_id=3
  • 保持注释简洁同步 🔄: 注释应该清晰明了,避免冗长。最重要的是,当代码发生变化时,务必更新相关的注释,否则过时的注释比没有注释更糟糕!💣
  • 注释复杂 🤯 或不直观的部分: 对于复杂的正则表达式、算法、重要的业务逻辑或临时的 Workaround,添加注释非常有帮助 🙏。
  • 文件头注释 ራስ: 在脚本文件的开头添加注释块,说明脚本的用途、作者、创建/修改日期、版本号、使用方法等信息,是一个非常好的实践 ⭐。
#!/usr/bin/env bash
# ==============================================================================
# Script Name:    backup_database.sh
# Description:    Performs a daily backup of the production PostgreSQL database.
# Author:         Your Name <your.email@example.com>
# Date Created:   2023-5-2
# Last Modified:  2023-5-2
# Version:        1.2
# Usage:          ./backup_database.sh
# Notes:          Requires pg_dump command and write access to BACKUP_DIR.
#                 Recommend running via cron.
# ==============================================================================
  • 利用空行(代码块外的)缩进 🏗️: 合理地使用空行分隔逻辑块。在非代码块的文本部分,可以使用缩进(如列表项)来组织内容,提高结构清晰度。
  • 写给未来的你 🔮 和他人 🤝: 记住,你写的代码和注释,很可能在几个月或几年后需要被你自己或其他同事阅读和维护。清晰的注释是对未来时间和精力的投资 💰。

四、Shell 脚本基础练习题 🧠✍️

题目一:基础概念

❓ 简述 Shell 在 Linux/Unix 系统中的主要作用是什么?它扮演了什么角色

题目二:Shebang

❓ Shell 脚本的第一行 #!/bin/bash 有什么作用?它通常被称为什么

题目三:执行权限

❓ 如何让一个名为 my_script.sh 的 Shell 脚本文件,能够直接通过 ./my_script.sh 的方式运行?请写出关键命令

题目四:执行方式辨析

❓ 执行 Shell 脚本时,使用 bash script.sh 和 source script.sh (或 . script.sh) 的主要区别是什么?哪种方式会影响当前的 Shell 环境? 🤔

题目五:注释

❓ 在 Shell 脚本中,如何添加单行注释?请给出表示注释的符号

题目六:多行注释模拟

❓ Shell 没有原生的多行注释块语法。请描述一种常用的、基于 Here Document 的方法来模拟多行注释。📝

题目七:Shebang 最佳实践

❓ 为什么通常推荐使用 #!/usr/bin/env bash 而不是直接使用 #!/bin/bash 作为 Shebang? 主要优势是什么? 🚀


参考答案 ✅💡

答案一:

Shell 主要作用是作为用户 👤 与 操作系统内核 之间的命令行解释器。它扮演着翻译官 🗣️ 的角色,负责接收、解释用户的命令,调用内核执行,并将结果返回给用户 🖥️。它是实现与系统交互 ↔️ 的桥梁 🌉。

答案二:

#!/bin/bash 的作用是指定解释器。当脚本被直接执行时 (例如通过 ./script.sh),操作系统会根据这一行找到 /bin/bash 解释器来处理脚本中的命令 ✨。这一行通常被称为 Shebang

答案三:

为了让 my_script.sh 可以通过 ./my_script.sh 直接运行,需要给它添加执行权限 🔑。关键命令是:

chmod +x my_script.sh

答案四:

主要区别在于执行环境

  • bash script.sh 会启动一个新的子 Shell (subshell) 🐣 来执行脚本。脚本中的变量定义、函数、cd 等操作不会影响当前的 Shell 环境 🏠。
  • source script.sh (或 . script.sh) 则是在当前的 Shell 环境 🏠 中直接执行脚本的命令。脚本中定义的变量、函数、cd 等操作会保留在当前 Shell 会话中 ✅。

因此,source 或 . 方式会影响当前的 Shell 环境。

答案五:

在 Shell 脚本中,使用 # 符号来添加单行注释 💬。从 # 到行尾的内容都会被解释器忽略 🙈。

答案六:

一种常用的模拟多行注释的方法是使用 Here Document 并将其重定向给一个空命令 :。格式如下:

: <<'任意分隔符'
这里是第一行注释。
这里是第二行注释。
这整块内容都不会被执行。
任意分隔符

使用单引号包裹起始分隔符 (如 ‘任意分隔符’) 可以阻止其中的变量和命令替换,使其更像纯粹的注释块。

答案七:

推荐使用 #!/usr/bin/env bash 的主要优势在于提高了脚本的可移植性 🚀🌍。

  • #!/bin/bash 写死了 bash 解释器的绝对路径。如果用户的 bash 不在这个位置 (例如在 /usr/local/bin/bash),脚本就无法直接执行 💥。
  • #!/usr/bin/env bash 利用 env 命令在用户的PATH 环境变量 🗺️ 中查找 bash 解释器。只要用户的 PATH 中能找到 bash,无论它安装在哪里,脚本都能正确找到并执行 👌。这使得脚本在不同系统、不同环境下的适应性更强 💪。

相关文章:

Shell 脚本基础

一、Shell 简介 1.Shell 的定义与作用 Shell&#xff0c;通常被称为命令行解释器 (Command Line Interpreter)&#xff0c;是用户 &#x1f464; 与 Linux/Unix 操作系统内核进行交互 ↔️ 的“桥梁” &#x1f309;。它扮演着翻译官 &#x1f5e3;️ 的角色&#xff1a; 接…...

【AI面试准备】元宇宙测试:AI+低代码构建虚拟场景压力测试

介绍元宇宙测试&#xff1a;AI低代码构建虚拟场景压力测试&#xff08;如数字孪生工厂&#xff09;。如何快速掌握&#xff0c;以及在实际工作中如何运用。 目录 **元宇宙测试&#xff1a;AI低代码构建虚拟场景压力测试****一、元宇宙测试的核心挑战与需求**1. **元宇宙测试的独…...

【网络层】之IP协议

网络层之IP协议 网络层的作用IP地址不足的问题私网IP和公网IP网段划分传统的网段划分的方法CIDR网段划分路由器的角色理解运营商的角色子网划分的过程 路由表IP协议介绍报文如何分离、交付 网络层的作用 IP协议是网络层的一种典型协议&#xff0c;只要弄清楚了IP协议的作用&…...

AI编译器对比:TVM vs MLIR vs Triton在大模型部署中的工程选择

引言&#xff1a;大模型部署的编译器博弈 随着千亿参数大模型成为常态&#xff0c;推理延迟优化成为系统工程的核心挑战。本文基于NVIDIA A100与Google TPUv4平台&#xff0c;通过BERT-base&#xff08;110M&#xff09;和GPT-2&#xff08;1.5B&#xff09;的实测数据&#x…...

【dify—10】工作流实战——文生图工具

目录 一、创建工作流 应用 二、安装硅基流动 三、配置硅基流动 四、API测试 &#xff08;1&#xff09;进入API文档 &#xff08;2&#xff09;复制curl代码 &#xff08;3&#xff09;Postman测试API 五、 建立文生图工作流 &#xff08;1&#xff09;建立http请求 &…...

企业级分布式 MCP 方案

飞书原文档链接地址&#xff1a;https://ik3te1knhq.feishu.cn/wiki/D8kSwC9tFi61CMkRdd8cMxNTnpg 企业级分布式 MCP 方案 [!TIP] 背景&#xff1a;现阶段 MCP Client 和 MCP Server 是一对一的连接方式&#xff0c;若当前 MCP Server 挂掉了&#xff0c;那么 MCP Client 便不…...

玩转Docker(一):基本概念

容器技术是继大数据和云计算之后又一炙手可热的技术&#xff0c;而且未来相当一段时间内都会非常流行。 本文将对其基本概念和基本使用做出介绍。包括容器生态系统、容器的原理、怎样运行第一个容器、容器技术的概念与实践、Docker镜像等等 目录 一. 鸟瞰容器生态系统 1. 容器…...

Linux系统安装方式+适合初学者的发行版本

Linux系统安装方式适合初学者发行版—目录 一、Linux系统的安装方式1. 物理机直接安装2. 虚拟机安装3. 双系统安装4. Live USB试用5. 云服务器安装 二、适合初学者的Linux发行版1. Ubuntu2. Linux Mint3. Zorin OS4. Pop!_OS5. Elementary OS6. Fedora7. Manjaro 三、选择建议场…...

开启 Spring AI 之旅:从入门到实战

开启 Spring AI 之旅&#xff1a;从入门到实战 引言 在当今人工智能飞速发展的时代&#xff0c;Spring AI 为开发者们提供了一个强大而便捷的工具&#xff0c;用于在 Spring 生态系统中构建 AI 应用程序。本文将为你提供如何开始使用 Spring AI 的详细指南&#xff0c;帮助你…...

python数据分析(七):Pandas 数据变形与重塑

Pandas 数据变形与重塑全面指南 1. 引言 在数据分析过程中&#xff0c;我们经常需要将数据从一种结构转换为另一种结构&#xff0c;以适应不同的分析需求。Pandas 提供了丰富的数据变形与重塑功能&#xff0c;包括旋转(pivot)、堆叠(stack)、融合(melt)等多种操作。本文将详细…...

SX24C01.UG-PXI程控电阻桥板卡

PXI程控电阻桥板卡 概述 简介 程控电阻桥板卡采用4 个可程控精密调节的电阻臂组成桥式电路&#xff0c;通过计算机PXI总线控制继电器通断进行电阻调节&#xff1b;可根据具体情况&#xff0c;方便地选择不同桥路的连接&#xff1b;程控电阻桥板卡支持“1/4 桥”、“半桥”和…...

Python项目源码69:一键解析+csv保存通达信日线数据3.0

Python项目源码39&#xff1a;学生积分管理系统1.0&#xff08;命令行界面Json&#xff09; Python项目源码38&#xff1a;模拟炒股系统2.0&#xff08;tkinterJson&#xff09; Python项目源码35&#xff1a;音乐播放器2.0&#xff08;Tkintermutagen&#xff09; Python项…...

Conda 与 Spyder 环境管理

前言 作为 Python 科学计算领域的黄金搭档&#xff0c;Anaconda 和 Spyder 为研究人员和数据分析师提供了强大的工作环境。本文将详细介绍如何使用 Conda 管理 Python 环境&#xff0c;并在 Spyder IDE 中灵活切换这些环境&#xff0c;助你打造高效的 Python 开发工作流。 一…...

头皮理疗预约小程序开发实战指南

生活服务类小程序开发正成为互联网创业的热点领域,头皮理疗预约小程序作为其中的细分品类,具有广阔的市场前景和用户需求。基于微信小程序原生开发或uniapp框架,结合Java后端和MySQL数据库,可构建一个功能完善、性能稳定且易于维护的头皮理疗预约平台。本文将从零开始,详细…...

cPanel 的 Let’s Encrypt™ 插件

在 cPanel & WHM 中&#xff0c;推出了一个名为 AutoSSL 的功能。可能有些朋友还不了解 AutoSSL&#xff0c;它是一个能够自动为您的网站申请和安装免费 SSL 证书的工具&#xff0c;这些证书由 Comodo 签发&#xff0c;保证网站的安全性。 AutoSSL 与 Let’s Encrypt Let’…...

《Android 应用开发基础教程》——第十一章:Android 中的图片加载与缓存(Glide 使用详解)

目录 第十一章&#xff1a;Android 中的图片加载与缓存&#xff08;Glide 使用详解&#xff09; &#x1f539; 11.1 Glide 简介 &#x1f538; 11.2 添加 Glide 依赖 &#x1f538; 11.3 基本用法 ✦ 加载网络图片到 ImageView&#xff1a; ✦ 加载本地资源 / 文件 / UR…...

MySQL 中的游标(Cursor)

一、游标的作用 ​​逐行处理数据​​&#xff1a;当需要对查询结果集中的每一行进行特定操作&#xff08;如计算、条件判断、调用其他过程&#xff09;时使用。​​替代集合操作​​&#xff1a;在无法通过单一 SQL 语句完成复杂逻辑时&#xff0c;游标提供逐行处理的能力。​…...

【嵌入式Linux】基于ARM-Linux的zero2平台的智慧楼宇管理系统项目

目录 1. 需求及项目准备&#xff08;此项目对于虚拟机和香橙派的配置基于上一个垃圾分类项目&#xff0c;如初次开发&#xff0c;两个平台的环境变量&#xff0c;阿里云接入&#xff0c;摄像头配置可参考垃圾分类项目&#xff09;1.1 系统框图1.2 硬件接线1.3 语音模块配置1.4 …...

记忆翻牌游戏:认知科学与状态机的交响曲

目录 记忆翻牌游戏&#xff1a;认知科学与状态机的交响曲引言第一章 网格空间拓扑学1.1 自适应网格算法1.2 卡片排布原理 第二章 状态机设计2.1 状态跃迁矩阵2.2 时空关联模型 第三章 记忆强化机制3.1 认知衰减曲线3.2 注意力热力图 第四章 动画引擎设计4.1 翻牌运动方程4.2 粒…...

【业务领域】InfiniBand协议总结

InfiniBand协议总结 InfiniBand协议是什么&#xff1f;Infiniband产生的原因Mellanox公司介绍及其新闻基于TCP/IP的网络与IB网络的比较IB标准的优势什么是InfiniBand网络什么是InfiniBand架构Mellanox IB卡介绍InfiniBand速率发展介绍InfiniBand网络主要上层协议InfiniBand管理…...

使用Java正则表达式检查字符串是否匹配

在Java开发中&#xff0c;正则表达式&#xff08;Regular Expression&#xff0c;简称Regex&#xff09;是处理字符串的强大工具&#xff0c;广泛应用于模式匹配、数据验证和文本处理。Java通过java.util.regex包提供了对正则表达式的支持&#xff0c;包含Pattern和Matcher两个…...

个人健康中枢的多元化AI硬件革新与精准健康路径探析

在医疗信息化领域,个人健康中枢正经历着一场由硬件技术革新驱动的深刻变革。随着可穿戴设备、传感器技术和人工智能算法的快速发展,新一代健康监测硬件能够采集前所未有的多维度生物数据,并通过智能分析提供精准的健康建议。本文将深入探讨构成个人健康中枢的最新硬件技术,…...

Android基础控件用法介绍

Android基础控件用法详解 Android应用开发中,基础控件是构建用户界面的核心元素。本文将详细介绍Android中最常用的基础控件及其用法。 一、TextView(文本显示控件) TextView用于在界面上显示文本信息。 基本用法 <TextViewandroid:id="@+id/textView"andr…...

iO(不可区分混淆)是Web3隐私的圣杯?

1. 引言 iO 是区块链隐私的圣杯吗&#xff1f;本文将探讨&#xff1a; 不可区分混淆&#xff08;indistinguishability obfuscation, iO&#xff09;的局限性iO可能带来的变革iO为何重要&#xff1f;iO是否能真正成为可信硬件的替代方案&#xff1f; 区块链隐私面临的最大挑…...

文章三《机器学习基础概念与框架实践》

文章3&#xff1a;机器学习基础概念与框架实践 ——从理论到代码&#xff0c;用Scikit-learn构建你的第一个分类模型 一、机器学习基础理论&#xff1a;三大核心类型 机器学习是人工智能的核心&#xff0c;通过数据让计算机自动学习规律并做出预测或决策。根据学习方式&#…...

中小企业MES系统概要设计

版本&#xff1a;V1.0 日期&#xff1a;2025年5月2日 一、系统架构设计 1.1 整体架构模式 采用分层微服务架构&#xff0c;实现模块解耦与灵活扩展&#xff0c;支持混合云部署&#xff1a; #mermaid-svg-drxS3XaKEg8H8rAJ {font-family:"trebuchet ms",verdana,ari…...

自动化测试项目1 --- 唠嗑星球 [软件测试实战 Java 篇]

目录 项目介绍 项目源码库地址 项目功能测试 1.自动化实施步骤 1.1 编写测试用例 1.2 自动化测试脚本开发 1.2.1 配置相关环境, 添加相关依赖 1.2.2 相关代码编写 2. 自动化功能测试总结 2.1 弹窗的解决相关问题 2.2 断言的使用和说明 2.3 重新登录问题 项目性能…...

c语言 关键字--目录

1.c语言 关键字 2.typedef 关键字 3.volatile 关键字 4.register 关键字 5.const关键字用法 6.extern关键字...

C语言与指针3——基本数据类型

误区补充 char 的 表示范围0-127 signed char 127 unsigned char 0-255enum不常用&#xff0c;但是常见&#xff0c;这里记录一下。 enum Day {Monday 1,//范围是IntTuesday 2,Wednesday 3 }; enum Day d Monday; switch (d) {case Monday:{printf("%d",Monday);…...

[更新完毕]2025五一杯C题五一杯数学建模思路代码文章教学:社交媒体平台用户分析问题

完整内容请看文章最下面的推广群 社交媒体平台用户分析问题 在问题一中为解决博主在特定日期新增关注数的预测问题&#xff0c;本文构建了基于用户历史行为的二分类模型。首先&#xff0c;从用户对博主的观看、点赞、评论、关注等交互行为中提取统计与时序特征&#xff0c;形成…...

使用Vite创建vue3项目

什么是vite Vite 是新一代构建工具&#xff0c;由 Vue 核心团队开发&#xff0c;提供极快的开发体验。 它利用浏览器原生ES模块导入功能&#xff0c;提供了极快的热模块更新&#xff08;HMR&#xff09;和开发服务器启动速度。 官网&#xff1a;https://vitejs.cn/vite3-cn/…...

Linux管道识

深入理解管道 (Pipes)&#xff1a;连接命令的瑞士军刀 在 Linux 和类 Unix 系统&#xff08;包括 macOS&#xff09;的命令行世界里&#xff0c;管道&#xff08;Pipe&#xff09;是一个极其强大且基础的概念。它允许你将一个命令的输出直接作为另一个命令的输入&#xff0c;像…...

LabVIEW 中VI Server导出 VI 配置

该 LabVIEW VI 展示了在 VI Server 中配置和执行 Exported VIs 的过程&#xff0c;实现对服务器端导出 VI 的远程调用与操作。 ​ 具体过程及模块说明 前期配置&#xff1a;需确保在 LabVIEW 的 “Tools> Options > VI Server > Protocols” 路径下&#xff0c;启用 …...

map和set的遗留 + AVL树(1):

在讲解新的东西之前&#xff0c;我们把上节遗留的问题说一下&#xff1a; 遗留问题&#xff1a; 首先&#xff0c;我们的最上面的代码是一个隐式类型转换&#xff0c;我们插入了一对数据。 我们说了&#xff0c;我们的方括号的底层是insert&#xff0c;当我们调用operator的[…...

Java学习手册:Spring Security 安全框架

一、Spring Security 简介 Spring Security 是一个功能强大且高度可定制的身份验证和访问控制框架&#xff0c;用于保护 Java 应用程序&#xff0c;尤其是基于 Spring 的应用。它构建在 Spring 框架之上&#xff0c;能够轻松地集成到基于 Spring 的应用程序中&#xff0c;包括…...

pip 常用命令及配置

一、python -m pip install 和 pip install 的区别 在讲解 pip 的命令之前&#xff0c;我们有必要了解一下 python -m pip install 和 pip install 的区别&#xff0c;以便于我们在不同的场景使用不同的方式。 python -m pip install 命令使用 python 可执行文件将 pip 模块作…...

C++_STL

C 标准模板库&#xff08;Standard Template Library&#xff0c;STL&#xff09;是一套功能强大的 C 模板类和函数的集合&#xff0c;它提供了一系列通用的、可复用的算法和数据结构。 STL 的设计基于泛型编程&#xff0c;这意味着使用模板可以编写出独立于任何特定数据类型的…...

[FPGA Video] AXI4-Stream Remapper

Xilinx AXI4-Stream Remapper IP (PG379) 详细介绍 概述 Xilinx LogiCORE™ IP AXI4-Stream Remapper 核是一个专为视频处理设计的模块&#xff0c;用于在不同每时钟像素数&#xff08;Pixels Per Clock, PPC&#xff09;要求之间重新映射视频像素。它支持将输入 AXI4-Stream…...

【数据结构】励志大厂版·初阶(复习+刷题):栈与队列

前引&#xff1a;本篇将由小编与大家一起复习 栈 、队列 的知识点&#xff0c;栈、队列的顺序、链式结构各个缺点好处&#xff0c;如何实现、对于一般的增删查找此篇文章一定再详细不过&#xff01;对代码的注释、何时需要判断、特殊情况&#xff0c;白话文版一解到底&#xff…...

pytest——参数化

之前有说过&#xff0c;通过pytest测试框架标记参数化功能可以实现数据驱动测试。数据驱动测试使用的文件主要有以下类型&#xff1a; txt 文件 csv 文件excel 文件json 文件yaml 文件.... 本文主要讲的就是以上几种文件类型的读取和使用 一.txt 文件读取使用 首先创建一个 …...

第7篇:RESTful API设计与安全防护

在前后端分离架构中&#xff0c;RESTful API是系统交互的核心通道。本文将从接口规范设计到安全防护&#xff0c;全面讲解如何在EggJS中构建安全、规范、易用的API系统&#xff0c;并提供完整的解决方案和最佳实践。 一、标准化API接口规范设计 1. RESTful设计原则 核心要素&…...

CSS 架构与命名规范

CSS 架构与命名规范&#xff1a;BEM、SMACSS 与 OOCSS 实战 引言 在前端开发中&#xff0c;随着项目规模的扩大&#xff0c;CSS 代码往往会变得难以维护和扩展。无组织的样式表会导致命名冲突、权重覆盖问题和样式继承混乱&#xff0c;这些问题在团队协作的大型项目中尤为明显…...

实验二 软件白盒测试

实验二 软件白盒测试 某工资计算程序功能如下&#xff1a;若雇员月工作小时超过40小时&#xff0c;则超过部分按原小时工资的1.5倍的加班工资来计算。若雇员月工作小时超过50小时&#xff0c;则超过50的部分按原小时工资的3倍的加班工资来计算&#xff0c;而40到50小时的工资仍…...

【数据结构】String字符串的存储

目录 一、存储结构 1.字符串常量池 2.字符串哈希表 2.1结构 2.2基础存储单位 2.2.1键对象 2.2.2值对象 二、存储过程 1.搜索 2.创建 三、存储位置 四、存储操作 1.new新建 2.intern入池 这是String类的详解&#xff1a;String类变量 一、存储结构 1.字符串常量池…...

LLMs Tokenizer Byte-Pair Encoding(BPE)

1 Byte-Pair Encoding(BPE) 如何构建词典&#xff1f; 准备足够的训练语料;以及期望的词表大小&#xff1b;将单词拆分为字符粒度(字粒度)&#xff0c;并在末尾添加后缀“”&#xff0c;统计单词频率合并方式:统计每一个连续/相邻字节对的出现频率&#xff0c;将最高频的连续字…...

npm,yarn,pnpm,cnpm,nvm,npx包管理器常用命令

前端比较主流的包管理器主要有三个npm&#xff0c;yarn&#xff0c;pnpm 多层级依赖&#xff0c;通常发生在依赖之间存在复杂的版本要求时 包 A 依赖于包 B1.0.0 包 B 依赖于包 C2.0.0 另一个包 D 也依赖于 C3.0.0 一、NPM (Node Package Manager) https://www.npmjs.cn/…...

使用mybatis实例类和MySQL表的字段不一致怎么办

在 MyBatis 中&#xff0c;当 Java 实体类的属性名与数据库表的字段名不一致时&#xff0c;会导致查询结果无法正确映射。以下是几种常见解决方案及代码示例&#xff1a; 1. 使用 resultMap 显式映射&#xff08;推荐&#xff09; 场景&#xff1a;字段名与属性名差异较大&…...

阿里发布新一代通义千问 Qwen3模型

近日&#xff0c;阿里巴巴发布了新一代通义千问 Qwen3 模型&#xff0c;一举登顶全球最强开源模型。 这是国内首个“混合推理模型”&#xff0c;将“快思考”与“慢思考”集成进同一个模型&#xff0c;大大节省算力消耗。 旗舰模型 Qwen3-235B-A22B 在代码、数学、通用能力等…...

React pros比较机制

将 count1作为prop传递给Memoson组件 引用类型情况 虽然list值没有发生变化&#xff0c;但是仍旧重新渲染 解决方法使用useMemo()函数&#xff0c;传递一个空依赖项 // 传递数据为引用类型 比较的是引用 // 使用useMemo函数改写、const list useMemo(()>{return [1,2,3]},[…...

Flowable7.x学习笔记(十七)审批我的待办

前言 前文完成了我的待办的查询功能&#xff0c;本文就在此基础上从源码解读到完成审批任务的功能&#xff0c;审批界面我就先不带表单&#xff0c;直接单纯审批通过&#xff0c;这里需要注意的事&#xff0c;审批的表单其实每个节点都可能需要不同的表单内容&#xff0c;后续要…...