如何在 Zsh 中使用参数扩展去除文件扩展名

本文介绍在 zsh 脚本中高效去除文件扩展名的方法,重点讲解内置的参数扩展语法(如 `${1%.java}` 和 `${1%.*}`),避免依赖外部命令或正则表达式,兼顾简洁性、可移植性与安全性。

在 Zsh(注意:正确名称为 Zsh,而非“zhell”)中,处理文件名时无需调用 sed、basename 或启用复杂正则引擎——Zsh 原生支持强大且高效的参数扩展(Parameter Expansion),专为这类字符串操作而优化。

以你的 jcr 函数为例:

jcr() {
  javac "$1"
  java "${1%.java}"  # ✅ 安全、高效、标准写法
}

这里 ${1%.java} 使用了 后缀删除(suffix removal) 语法:% 表示从变量值末尾开始匹配并移除最短匹配的模式;.java 是字面量后缀。当 $1 为 Hell

o.java 时,结果即为 Hello。

更通用的做法是移除任意扩展名(不限于 .java):

jcr() {
  local base="${1%.*}"  # 移除最后一个点及其后所有字符(如 file.txt → file,archive.tar.gz → archive.tar)
  javac "$1"
  java "$base"
}

⚠️ 注意事项:

  • 使用 "${1%.*}" 而非 ${1%.*}:始终用双引号包裹参数扩展,防止空格、通配符等引发意外词分割或路径展开。
  • % 匹配最短后缀,%% 匹配最长后缀(例如 file.tar.gz 中,${f%.*} → file.tar,${f%%.*} → file)。
  • 若文件名不含点(如 Main),${1%.*} 会原样返回 Main,行为安全可靠。
  • 不推荐 ${1//.java}(全局替换):它会错误地将 My.java.java 变成 My,且无法处理其他扩展(如 .class),语义不准确。

✅ 最佳实践示例(健壮版):

jcr() {
  [[ -n "$1" ]] || { echo "Error: missing filename" >&2; return 1; }
  [[ -f "$1" ]] || { echo "Error: '$1' not found" >&2; return 1; }

  local src="$1"
  local base="${src%.*}"

  if ! javac "$src"; then
    echo "Compilation failed." >&2
    return 1
  fi

  java "$base"
}

总结:Zsh 的参数扩展是轻量、快速、POSIX 兼容(Bash 也支持 %/%%)的核心特性。掌握 ${var%pattern}(删后缀)、${var#pattern}(删前缀)等语法,比正则更简洁、更安全,是编写高质量 shell 工具的必备技能。官方参考见 Zsh Expansion Documentation。