主要问题:如何获得declare -F
当前 shell 中的值与 shell 刚刚启动时的值之间的增量(下面的前两个命令)。$(declare -F)
并没有解决问题,因为子 shell 是 shell 进程的副本。子公司:为什么下面第三个命令没有输出任何内容?
$ exec env -i bash
$ declare -F
declare -f ShowInstallerIsoInfo
declare -f __expand_tilde_by_ref
declare -f __get_cword_at_cursor_by_ref
declare -f __load_completion
declare -f __ltrim_colon_completions
declare -f __parse_options
declare -f __reassemble_comp_words_by_ref
declare -f _allowed_groups
declare -f _allowed_users
declare -f _available_interfaces
declare -f _bashcomp_try_faketty
declare -f _bq_completer
declare -f _cd
declare -f _cd_devices
declare -f _command
declare -f _command_offset
declare -f _complete_as_root
declare -f _completer
declare -f _completion_loader
declare -f _configured_interfaces
declare -f _count_args
declare -f _dvd_devices
declare -f _expand
declare -f _filedir
declare -f _filedir_xspec
declare -f _fstypes
declare -f _get_comp_words_by_ref
declare -f _get_cword
declare -f _get_first_arg
declare -f _get_pword
declare -f _gids
declare -f _have
declare -f _included_ssh_config_files
declare -f _init_completion
declare -f _installed_modules
declare -f _ip_addresses
declare -f _kernel_versions
declare -f _known_hosts
declare -f _known_hosts_real
declare -f _longopt
declare -f _mac_addresses
declare -f _minimal
declare -f _modules
declare -f _ncpus
declare -f _open_files_for_editing
declare -f _parse_help
declare -f _parse_usage
declare -f _pci_ids
declare -f _pgids
declare -f _pids
declare -f _pnames
declare -f _python_argcomplete
declare -f _quote_readline_by_ref
declare -f _realcommand
declare -f _rl_enabled
declare -f _root_command
declare -f _service
declare -f _services
declare -f _shells
declare -f _signals
declare -f _split_longopt
declare -f _sysvdirs
declare -f _terms
declare -f _tilde
declare -f _uids
declare -f _upvar
declare -f _upvars
declare -f _usb_ids
declare -f _user_at_host
declare -f _usergroup
declare -f _userland
declare -f _variables
declare -f _xfunc
declare -f _xinetd_services
declare -f dequote
declare -f quote
declare -f quote_readline
$ "$SHELL" -c 'declare -F'
其他:
$ uname -a
Linux elitebook 6.7.3-arch1-2 #1 SMP PREEMPT_DYNAMIC Fri, 02 Feb 2024 17:03:55 +0000 x86_64 GNU/Linux
$ bash --version
GNU bash, version 5.2.26(1)-release (x86_64-pc-linux-gnu)
$ echo $SHELL
/bin/bash
更新:
干净状态是一个空函数列表,如输出所示
bash -c 'declare -F'
根据第一个命令,我预计第二个命令会输出两个函数而不是零。
$ grep -F -f <(declare -F | cut --delimiter=' ' --fields=3) ~/.bashrc | grep -v -E '^#' | wc -l
2
$ bash -c 'source ~/.bashrc; declare -F'
更新2
我期望第二个输出两个函数而不是零。
采购不生效的原因可能是:
$ cat ~/.bashrc
#
# ~/.bashrc
#
# If not running interactively, don't do anything
[[ $- != *i* ]] && return
解决方法确实输出两个函数。
$ bash -c 'source <(grep -v '\''!= *i*'\'' ~/.bashrc); compgen -A function' 2>/dev/null
答案1
如果您想知道自 shell 启动并获取启动文件以来函数列表中发生了什么变化,因为在 bash 中,函数名称无论如何都不能包含换行符,您可以使用compgen -A function
每行一个转储函数列表,并且comm
比较不同时间点的转储(需要在同一区域设置中排序)。
添加:
initial_functions=$(compgen -A function | LC_ALL=C sort)
在你的末尾~/.bashrc
,然后运行:
LC_ALL=C comm -13 <(printf '%s\n' "$initial_functions") \
<(compgen -A function | LC_ALL=C sort)
查看此后添加了哪些内容。
LC_ALL=C comm -23 <(printf '%s\n' "$initial_functions") \
<(compgen -A function | LC_ALL=C sort)
对于那些已经被删除的。
zsh 等效项是:
initial_functions=( ${(k)functions} )
() { print -rC1 -- ${argv:|initial_functions}; } ${(k)functions}
() { print -rC1 -- ${initial_functions:|argv}; } ${(k)functions}
无论如何,请注意,像 zsh 这样的 bash 本身并不定义任何函数,因此清洁状态是一个空函数列表,如 的输出所示bash -c 'declare -F'
。
您可以看到的所有函数要么是从环境中导入的(因为支持通过环境变量bash
导出函数),要么是在 bash 代码中定义的,该代码自启动以来一直被解释,可能是通过文件或任何BASH_FUNC_funcname%%=
export -f
bash
$BASHENV
bash 启动文件,其中一些可能由您的操作系统提供,或来自那里的文件,例如来自bash-completion
第三方项目你似乎正在使用它。
然而,zsh 提供了大量的函数,您可以将它们作为示例函数加载或自动加载,或者支持它的几个子系统,例如完成(在 zsh 中不是一个单独的项目)、FTP、行编辑器、日历、提示配置、文档,以及一些有用的工具,例如zargs
,,...zmv
zcalc