मैं पिछली / अगली सिबलिंग डायरेक्टरी को कैसे सीडी कर सकता हूं?


10

मैं अक्सर इस तरह एक परियोजना निर्देशिका लेआउट है

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

मैं आमतौर पर componentनिर्देशिकाओं में से एक में काम कर रहा हूँ , क्योंकि जहाँ फाइलें हैं। जब मैं शेल पर लौटता हूं, तो मुझे अक्सर सिबलिंग डायरेक्टरी में जाने की जरूरत होती है, खासकर तब जब मुझे हर कंपोनेंट में कुछ गैर-स्क्रिप्ट करने योग्य बदलाव करने की जरूरत होती है। उन मामलों में, मुझे यह भी परवाह नहीं होगी कि पिछली भाई-बहन की निर्देशिका क्या है जो मैं काम करूंगी, या अगली भाई-बहन निर्देशिका।

क्या मैं एक कमांड को परिभाषित कर सकता हूं prevया nextजो cdमुझे केवल पिछली निर्देशिका में, या अगली निर्देशिका (वर्णमाला या जो भी हो) में परिभाषित कर सकता है ? क्योंकि cd ../com<TAB><Arrow keys>हर समय टाइप करना थोड़ा पुराना होता जा रहा है।

जवाबों:


8

अन्य उत्तर से कमांडलाइनफ्यू समाधान का उपयोग न करें : यह असुरक्षित और अक्षम है। इसके बजाय, यदि आप उपयोग कर रहे हैं, तो 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
}

Names यह सभी संभावित निर्देशिका नामों को नहीं संभालता है। पार्सिंग lsआउटपुट कभी भी सुरक्षित नहीं होता है

² cdशायद बहुत कुशल होने की जरूरत नहीं है, लेकिन 6 प्रक्रियाओं थोड़ा अधिक है।


1
मैं वास्तव में zsh का उपयोग करता हूं, लेकिन यदि आप अगले फ़ंक्शन के लिए पहले लूप में पहले टेस्ट को बदलते हैं [[ -n $foundcwd ]]तो आपका उत्तर बश और zsh के तहत समान रूप से काम करता है। बहुत अच्छा, और इसे लिखने के लिए धन्यवाद।
एस्टीस

4

निम्नलिखित फ़ंक्शन सिबलिंग डायरेक्टरी (बैश फ़ंक्शन) में बदलने की अनुमति देता है

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

बहुत अच्छा। मैं jw013 के उत्तर को स्वीकार किए जाने के रूप में रख रहा हूं, क्योंकि मेरा सवाल और उपयोग का मामला था 'मैं अगले भाई-बहन को स्थानांतरित करना चाहता हूं, और मुझे परवाह नहीं है कि इसे क्या कहा जाता है'; लेकिन यह उपयोगी है, भी, और निफ्टी, तो एक उत्थान है।
एस्टीस

2

मुझे Commandlinefu.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

2
आपके द्वारा पाई गई कमांड में कई समस्याएं हैं। मैंने इसे सबसे प्रबल लोगों को सही करने के लिए संपादित किया है: यह निर्देशिका नाम को एक grep पैटर्न के रूप में मान रहा था और इसे एक विकल्प से मेल खाने की अनुमति देता था; और यह निर्देशिका का नाम खोल विस्तार (शब्द बंटवारे और ग्लोबिंग) दो बार (: हमेशा के आसपास चर और आदेश प्रतिस्थापन दोहरे उद्धरण चिह्नों का उपयोग गुजरना पड़ा "$foo", "$(foo)")। इसके अलावा, का आउटपुट पार्स करना lsअविश्वसनीय है , यह बिना नाम वाले वर्णों वाले फ़ाइल नामों के साथ विफल हो सकता है।
गिल्स एसओ- बुराई को रोकें '
हमारी साइट का प्रयोग करके, आप स्वीकार करते हैं कि आपने हमारी Cookie Policy और निजता नीति को पढ़ और समझा लिया है।
Licensed under cc by-sa 3.0 with attribution required.