如何 cd 到上一个/下一个同级目录?

如何 cd 到上一个/下一个同级目录?

我经常有这样的项目目录布局

project
`-- component-a
|   `-- files...
`-- component-b
|   `-- files...
`-- component-c
    `-- files...

我通常会在其中一个component目录中工作,因为那是文件所在的位置。当我返回到 shell 时,我通常只需要移动到同级目录,特别是当我需要对每个组件进行一些不可编写脚本的更改时。在这些情况下,我什至不会关心我将要处理的上一个同级目录或下一个同级目录。

我可以定义一个命令prev,或者next简单地cd进入上一个目录或下一个目录(按字母表或其他方式)吗?因为cd ../com<TAB><Arrow keys>一直打字已经有点过时了。

答案1

不要使用 commandlinefu 解决方案另一个答案:它不安全,而且效率低下。相反,如果您正在使用bash,只需使用以下功能。为了使它们持久存在,请将它们放入您的.bashrc.请注意,我使用全局顺序,因为它是内置的且简单。通常是全局顺序但在大多数语言环境中都是按字母顺序排列的。如果没有可以转到的下一个或上一个目录,您将收到一条错误消息。特别是,如果您尝试nextprev在根目录中,您将看到错误/

## 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

相关内容