我经常有这样的项目目录布局
project
`-- component-a
| `-- files...
`-- component-b
| `-- files...
`-- component-c
`-- files...
我通常会在其中一个component
目录中工作,因为那是文件所在的位置。当我返回到 shell 时,我通常只需要移动到同级目录,特别是当我需要对每个组件进行一些不可编写脚本的更改时。在这些情况下,我什至不会关心我将要处理的上一个同级目录或下一个同级目录。
我可以定义一个命令prev
,或者next
简单地cd
进入上一个目录或下一个目录(按字母表或其他方式)吗?因为cd ../com<TAB><Arrow keys>
一直打字已经有点过时了。
答案1
不要使用 commandlinefu 解决方案另一个答案:它不安全,而且效率低下。相反,如果您正在使用bash
,只需使用以下功能。为了使它们持久存在,请将它们放入您的.bashrc
.请注意,我使用全局顺序,因为它是内置的且简单。通常是全局顺序是但在大多数语言环境中都是按字母顺序排列的。如果没有可以转到的下一个或上一个目录,您将收到一条错误消息。特别是,如果您尝试next
或prev
在根目录中,您将看到错误/
。
## bash and zsh only!
# functions to cd to the next or previous sibling directory, in glob order
prev () {
# default to current directory if no previous
local prevdir="./"
local cwd=${PWD##*/}
if [[ -z $cwd ]]; then
# $PWD must be /
echo 'No previous directory.' >&2
return 1
fi
for x in ../*/; do
if [[ ${x#../} == ${cwd}/ ]]; then
# found cwd
if [[ $prevdir == ./ ]]; then
echo 'No previous directory.' >&2
return 1
fi
cd "$prevdir"
return
fi
if [[ -d $x ]]; then
prevdir=$x
fi
done
# Should never get here.
echo 'Directory not changed.' >&2
return 1
}
next () {
local foundcwd=
local cwd=${PWD##*/}
if [[ -z $cwd ]]; then
# $PWD must be /
echo 'No next directory.' >&2
return 1
fi
for x in ../*/; do
if [[ -n $foundcwd ]]; then
if [[ -d $x ]]; then
cd "$x"
return
fi
elif [[ ${x#../} == ${cwd}/ ]]; then
foundcwd=1
fi
done
echo 'No next directory.' >&2
return 1
}
¹ 它不处理所有可能的目录名称。 解析ls
输出永远不安全。
²cd
可能不需要非常高效,但 6 个进程有点过多。
答案2
以下函数允许更改到同级目录(bash 函数)
function sib() {
## sib search sibling directories
## prompt for choice (when two or more directories are found, current dir is removed from choices)
## change to directory after selection
local substr=$1
local curdir=$(pwd)
local choices=$(find .. -maxdepth 1 -type d -name "*${substr}*" | grep -vE '^..$' | sed -e 's:../::' | grep -vE "^${curdir##*/}$" | sort)
if [ -z "$choices" ]; then
echo "Sibling directory not found!"
return
fi
local count=$(echo "$choices" | wc -l)
if [[ $count -eq 1 ]]; then
cd ../$choices
return
fi
select dir in $choices; do
if [ -n "$dir" ]; then
cd ../$dir
fi
break
done
}
使用示例:
$ tree
.
├── component-aaa-01
├── component-aaa-02
├── component-bbb-01
├── component-bbb-02
├── component-ccc-01
├── component-ccc-02
└── component-ccc-03
7 directories, 0 files
$ cd component-aaa-01/
$ sib bbb-01
$ pwd
component-bbb-01
$ sib bbb
$ pwd
component-bbb-02
$ sib ccc
1) component-ccc-01
2) component-ccc-02
3) component-ccc-03
#? 3
$ pwd
component-ccc-03
$ sib 01
1) component-aaa-01
2) component-bbb-01
3) component-ccc-01
#? 2
$ pwd
component-bbb-01
答案3
我发现了一个小费命令行fu.com。我将其重新发布到此处以使其更易于查找,并next
在我使用它时添加了解释和命令。
alias prev='cd ../"$(ls -F .. | grep '/' | grep -B1 -xF "${PWD##*/}/" | head -n 1)"'
alias next='cd ../"$(ls -F .. | grep '/' | grep -A1 -xF "${PWD##*/}/" | tail -n 1)"'
神奇之处在于“$(...)”块。它通过管道将一些命令相互传递,如下所示:
ls -F .. | # list items in parent dir; `-F` requests filetype indicators
grep '/' | # select the directories (written as `mydir/`)
grep -B1 -xF "${PWD##*/}/" | # search for the name of the current directory in the output;
# print it and the line preceding it
head -n 1 # the first of those two lines contains the name of the previous sibling