15

我很惊讶以前没有被问到,但是……</p>

和有什么区别

别名alias EXPORT='alias'

函数function exporter() { echo $EXPORT }

出口export ALIAS='export'

就此而言...

alias export=$(function) (j/k)

bash(zsh等人)

具体来说,我最感兴趣的是了解

alias this=that

export that=this

我有这两种形式......到处都是- 并且宁愿停止任意选择一种,而不是另一种。

我确信在某处有一个很好的参考“unix shell 的范围和用例”......但我想我会在这里以righteous-canonicalicism的名义发布这个问题。

4

2 回答 2

27

你问的是两种截然不同的事物:别名和函数定义了像命令一样的事物;export标记要导出到子进程的变量。让我先来看看类似命令的东西:

别名 ( alias ll='ls -l') 定义命令的简写。它们旨在用于交互使用(它们实际上在 shell 脚本中默认被禁用),并且简单但不灵活。例如,您在别名之后指定的任何参数都会简单地附加到命令的末尾;如果你想要类似的东西alias findservice='grep "$1" /etc/services',你不能这样做,因为$1在这里没有做任何有用的事情。

函数就像一个更灵活、更强大的别名版本。函数可以接受和处理参数,包含循环、条件、here-documents 等……基本上,你可以用 shell 脚本做的任何事情都可以在函数中完成。请注意,定义函数的标准方法实际上并不使用关键字function,只是在名称后面加上括号。例如:findservice() { grep "$1" /etc/services; }

好的,现在转到 shell 变量。在开始之前export,我需要谈谈未导出的变量。基本上,您可以将变量定义为具有某些(文本)值,然后如果您通过$variablename它引用变量,它将被替换到命令中。这在两个方面与别名或函数不同:别名或函数只能作为命令中的第一个单词出现(例如,ll filename将使用别名ll,但echo ll不会),并且必须使用$(echo $foo将使用变量 foo显式调用变量,但echo foo不会)。更根本的是,别名和函数旨在包含可执行代码(命令、shell 语法等),而变量旨在存储不可执行的数据。

(顺便说一句,您几乎应该总是将变量引用放在双引号内——也就是说,使用echo "$foo"而不是 just echo $foo。没有双引号,变量的内容会以一种有点奇怪的方式被解析,这往往会导致错误。)

还有一些“特殊”的shell 变量,它们由shell 自动设置(例如$HOME),或者影响shell 的行为(例如$PATH,控制它在哪里查找可执行命令),或者两者兼而有之。

ed 变量在export当前 shell 中都可用,也可以传递给任何子进程(子shell、其他命令等)。例如,如果我这样做LC_ALL=en_US.UTF-8,则告诉我当前的 shell 使用“en_US.UTF-8”语言环境设置。另一方面,如果我这样做export LC_ALL=en_US.UTF-8,将告诉当前 shell以及它执行的所有子进程和命令使用该语言环境设置。

请注意,shell 变量可以单独标记为导出,而不是定义它,并且一旦导出,它就会保持导出状态。例如,$PATHis (据我所知)总是导出的,因此PATH=/foo:/bar具有相同的效果export PATH=/foo:/bar(尽管后者可能是首选,以防万一 $PATH 不知何故尚未导出)。

通过使用赋值作为命令的前缀,也可以将变量导出到特定命令而不在当前 shell 中定义它。例如LC_ALL=en_US.UTF-8 sort filename,将告诉sort命令使用“en_US.UTF-8”语言环境设置,但不将其应用于当前 shell(或任何其他命令)。

于 2014-04-19T18:09:50.450 回答
1

TL;博士:

  1. 您问题中实体的外壳评估顺序(每个 POSIX)是:
    aliases --> variables --> command substitutions --> special built-ins --> functions --> regular built-ins
  2. 别名不会在子 shell 中持续存在,但可以使用export命令使变量(以及 Bash 中的函数)保持不变。
  3. 可以通过编写与常规内置函数同名的函数来覆盖常规内置函数(因为函数在常规内置函数之前扩展)。(注意:如果您尝试向常规内置函数添加功能,请在函数定义中调用内置command函数,以免意外创建递归函数。
  4. 可以使用(特殊内置)readonly命令将变量设置为只读,但别名不能。

用例:

  1. 如果您需要跨子外壳使用变量,请导出变量。
  2. 如果您不希望它在父 shell 的生命周期内更改,请创建一个变量readonly(一旦执行,这不能用 撤消unset;您必须重新启动父 shell)。
  3. 如果您想覆盖或向常规内置函数添加功能,请使用函数。

注意: 如果您想确定您使用的是特殊的或常规的内置函数,而不是其他人的函数,请使用 builtin the_builtin或者如果 shell 不支持该 builtin 命令,请使用 POSIX 命令 command -p the_builtin其中 -p 开关告诉命令默认情况下使用 shell 附带的 $PATH(以防用户覆​​盖了路径)。

注意: 可以使变量充当别名,该别名也可以跨子外壳持续存在并且无法更改。例如,

#! /bin/sh

my_cmd='ls -al'
export my_cmd
readonly my_cmd

会表现得像

#! /bin/sh

alias my_cmd='ls -al'

只要

  1. my_cmd 不使用双引号(即 ${my_cmd}NOT "${my_cmd}",因此它不被视为单个字符串,并且
  2. IFS 是标准 space-tab-newline 而不是切换到其他东西,因此元素 my_cmd 是通配符的,并且由空格分隔的每个部分都被评估为单个标记(否则它将被评估为单个字符串)。

每个 shell(例如bash, zsh, ksh,yash等)都有些不同,因此请务必查看它的参考手册(它们每个都以独特的方式实现 POSIX,或者有时根本不实现)。

于 2021-06-05T18:58:45.157 回答