스크립트 자체 내의 Bash 스크립트 파일 이름을 어떻게 확인할 수 있습니까?
내 스크립트가 파일 runme.sh
에있는 것처럼 하드 코딩없이 "Your running runme.sh"메시지를 표시하려면 어떻게해야합니까?
스크립트 자체 내의 Bash 스크립트 파일 이름을 어떻게 확인할 수 있습니까?
내 스크립트가 파일 runme.sh
에있는 것처럼 하드 코딩없이 "Your running runme.sh"메시지를 표시하려면 어떻게해야합니까?
답변:
me=`basename "$0"`
일반적으로 원하는 것이 아닌 symlink 1을 읽으 려면 (일반적으로 사용자를 혼동하지 않으려는 경우) 다음을 시도하십시오.
me="$(basename "$(test -L "$0" && readlink "$0" || echo "$0")")"
IMO는 혼란스러운 결과를 낳습니다. "foo.sh를 실행했지만 bar.sh를 실행 중입니다!? 버그 여야합니다!" 또한 다른 이름의 심볼릭 링크를 사용하는 목적 중 하나는 이름에 따라 다른 기능을 제공하는 것입니다 (일부 플랫폼에서는 gzip 및 gunzip을 생각하십시오).
1 에,이다 해결 심볼릭 링크 등 사용자가 실행 때 foo.sh
실제로 심볼릭 링크에 bar.sh
, 당신은 확인 된 이름을 사용하고자하는 bar.sh
것이 아니라 foo.sh
.
$0
첫 번째 예에서는 단어 분할이 적용됩니다. 3. $0
basename, readlink 및 echo에 전달되어 명령 행 스위치로 처리 될 수있는 위치 . me=$(basename -- "$0")
가독성을 희생하면서 대신 또는 훨씬 더 효율적으로 제안합니다 me=${0##*/}
. me=$(basename -- "$(readlink -f -- "$0")")
gnu 유틸리티를 가정하면 심볼릭 링크의 경우 그렇지 않으면 여기에 쓰지 않을 매우 긴 스크립트가됩니다.
# ------------- SCRIPT ------------- #
#!/bin/bash
echo
echo "# arguments called with ----> ${@} "
echo "# \$1 ----------------------> $1 "
echo "# \$2 ----------------------> $2 "
echo "# path to me ---------------> ${0} "
echo "# parent path --------------> ${0%/*} "
echo "# my name ------------------> ${0##*/} "
echo
exit
# ------------- 호출 ------------- ## ------------- 호출 ------------- #
# 다음 줄에서 첫 번째 인수는 double 내에서 호출됩니다. # 다음 줄에서 첫 번째 인수는 double 내에서 호출됩니다.
두 단어를 포함하므로 # 및 작은 따옴표두 단어를 포함하므로 # 및 작은 따옴표
$ /misc/shell_scripts/check_root/show_parms.sh " 'hello there'" " 'william'"/ misc / shell_scripts / check_root / show_parms . sh " '안녕하세요'" " 'william'"
# ------------- 결과 ------------- ## ------------- 결과 ------------- #
---> 'hello there' 'william'와 함께 호출 된 인수---> 'hello there' 'william'와 함께 호출 된 인수
# $ 1 ----------------------> '안녕하세요'# $ 1 ----------------------> '안녕하세요'
# $ 2 ----------------------> 'william'# $ 2 ----------------------> 'william'
# 나에게 경로 --------------> /misc/shell_scripts/check_root/show_parms.sh# 나에게 경로 --------------> /misc/shell_scripts/check_root/show_parms.sh
# 부모 경로 -------------> / misc / shell_scripts / check_root# 부모 경로 -------------> / misc / shell_scripts / check_root
# 내 이름 -----------------> show_parms.sh# 내 이름 -----------------> show_parms.sh
# ------------- 끝 ------------- ## ------------- 끝 ------------- #
show_params
선택적 확장자가없는 이름 즉, 어떻게해야 합니까?
${0##*/}
. GitBash를 사용하여 테스트했습니다.
와 bash는> = 3 개 다음 작품 :
$ ./s
0 is: ./s
BASH_SOURCE is: ./s
$ . ./s
0 is: bash
BASH_SOURCE is: ./s
$ cat s
#!/bin/bash
printf '$0 is: %s\n$BASH_SOURCE is: %s\n' "$0" "$BASH_SOURCE"
dirname $BASE_SOURCE
"를 사용 하여 스크립트가있는 디렉토리를 쉽게 얻을 수 있습니다.
$BASH_SOURCE
스크립트를 소싱 할 때 정답을 제공합니다.
그러나 여기에는 경로가 포함되므로 스크립트 파일 이름 만 얻으려면 다음을 사용하십시오.
$(basename $BASH_SOURCE)
. <filename> [arguments]
, $0
발신자 쉘의 이름을 줄 것입니다. 적어도 OSX에서는 확실합니다.
스크립트 이름에 공백이 있으면 MacOS 를 사용 "$0"
하거나 "$(basename "$0")"
MacOS 에서보다 강력한 방법을 사용 하십시오 "$(basename \"$0\")"
. 이렇게하면 이름이 엉망이되거나 해석되지 않습니다. 일반적으로 쉘에서 변수 이름을 항상 큰 따옴표로 묶는 것이 좋습니다.
Chris Conway에 대답하려면 Linux에서 적어도 다음을 수행하십시오.
echo $(basename $(readlink -nf $0))
readlink는 기호 링크의 값을 인쇄합니다. 심볼릭 링크가 아닌 경우 파일 이름을 인쇄합니다. -n은 개행을 인쇄하지 않도록 지시합니다. -f는 링크를 완전히 따르도록 지시합니다 (심볼릭 링크가 다른 링크에 대한 링크 인 경우 해당 링크도 해결합니다).
파일이 소스인지 스크립트로 실행되는지에 관계 없이이 줄이 항상 작동하는 것으로 나타났습니다.
echo "${BASH_SOURCE[${#BASH_SOURCE[@]} - 1]}"
readlink
위의 경로에서 심볼릭 링크를 사용하려면 재귀 적으로 또는 비재 귀적으로 사용하십시오.
one-liner가 작동하는 이유는 BASH_SOURCE
환경 변수와 관련 변수를 사용하여 설명합니다 FUNCNAME
.
BASH_SOURCE
멤버가 FUNCNAME 배열 변수에서 해당 쉘 함수 이름이 정의 된 소스 파일 이름 인 배열 변수입니다. 쉘 함수 $ {FUNCNAME [$ i]}은 $ {BASH_SOURCE [$ i]} 파일에 정의되어 있으며 $ {BASH_SOURCE [$ i + 1]}에서 호출되었습니다.
기능 명
현재 실행 호출 스택에있는 모든 쉘 함수의 이름을 포함하는 배열 변수입니다. 인덱스가 0 인 요소는 현재 실행중인 쉘 함수의 이름입니다. 맨 아래 요소 (인덱스가 가장 높은 요소)는 "main"입니다. 이 변수는 쉘 기능이 실행될 때만 존재합니다. FUNCNAME에 대한 지정은 적용되지 않으며 오류 상태를 리턴합니다. FUNCNAME을 설정하지 않으면 나중에 다시 설정하더라도 특수 속성이 손실됩니다.
이 변수는 BASH_LINENO 및 BASH_SOURCE와 함께 사용할 수 있습니다. FUNCNAME의 각 요소에는 호출 스택을 설명하기 위해 BASH_LINENO 및 BASH_SOURCE에 해당 요소가 있습니다. 예를 들어 $ {FUNCNAME [$ i]}은 (는) 행 번호 $ {BASH_LINENO [$ i]}의 $ {BASH_SOURCE [$ i + 1]} 파일에서 호출되었습니다. 발신자 기본 제공은이 정보를 사용하여 현재 호출 스택을 표시합니다.
[출처 : 배쉬 매뉴얼]
a
( a
의 내용이 있다고 가정)로 작동하면 경로 echo "${BASH_SOURCE[${#BASH_SOURCE[@]} - 1]}"
가 제공됩니다 a
. 당신은 스크립트를 작성하지만 만약 b
에 source a
그것의 실행 ./b
, 그것은 돌아갑니다 b
'의 경로를.
이 답변은 상황에 따라 맞지만 'source'키워드를 사용하여 다른 스크립트에서 스크립트를 실행하면 동일한 쉘에서 실행되므로 여전히 문제가 있습니다. 이 경우 호출 스크립트의 $ 0를받습니다. 그리고이 경우 스크립트 자체의 이름을 얻는 것이 가능하지 않다고 생각합니다.
이것은 엣지 케이스이며 너무 심각하게 생각해서는 안됩니다. 다른 스크립트에서 직접 ( 'source'없이) 스크립트를 실행하면 $ 0을 사용하는 것이 좋습니다.
Re : 위의 Tanktalus (허용됨) 대답은 약간 더 깨끗한 방법입니다.
me=$(readlink --canonicalize --no-newline $0)
스크립트가 다른 bash 스크립트에서 제공 되었다면 다음을 사용할 수 있습니다.
me=$(readlink --canonicalize --no-newline $BASH_SOURCE)
귀하의 목표가 사용자에게 피드백을 제공하는 것이 목적이라면 심볼릭 링크를 역 참조하는 것이 혼란 스러울 것입니다.하지만 정식 이름을 스크립트 또는 다른 파일로 가져와야하는 경우가 있습니다.
source
d 스크립트 이름에 감사드립니다 .
쉘 스크립트를 호출하면
/home/mike/runme.sh
$ 0은 이름입니다
/home/mike/runme.sh
기본 이름 $ 0은 기본 파일 이름을 가져옵니다.
runme.sh
이 기본 이름을 다음과 같은 변수에 넣어야합니다.
filename=$(basename $0)
추가 텍스트를 추가하십시오
echo "You are running $filename"
그래서 당신의 스크립트는
/home/mike/runme.sh
#!/bin/bash
filename=$(basename $0)
echo "You are running $filename"
this="$(dirname "$(realpath "$BASH_SOURCE")")"
이렇게하면 심볼릭 링크가 해결되고 (실제로 처리) 공백을 처리하고 (큰 따옴표로 처리), 소스 (. ./myscript)를 제공하거나 다른 스크립트 ($ BASH_SOURCE가 처리)에서도 현재 스크립트 이름을 찾습니다. 결국 재사용하거나 다른 곳에서 쉽게 복사하기 위해 환경 변수에 저장하는 것이 좋습니다 (this =) ...
realpath
는 내장 BASH 명령이 아닙니다. 특정 배포판에서만 사용할 수있는 독립 실행 형 실행 파일
$0
질문에 대답하지 않습니다. 데모 :
$ 고양이 script.sh #! / bin / sh 에코`기본 이름 $ 0` $ ./script.sh script.sh ln script.sh linktoscript $ ./linktoscript 링크
어떻게 ./linktoscript
인쇄 할 수 script.sh
있습니까?
[편집] 위의 주석에서 @ephemient에 따르면 심볼릭 링크가 고안된 것처럼 보일지 모르지만 $0
파일 시스템 리소스를 나타내지 않도록 바이올린을 피울 수 있습니다. OP는 그가 원하는 것에 대해 약간 모호합니다.
Bill Hernandez 덕분에 정보. 내가 선호하는 환경 설정을 추가했습니다.
#!/bin/bash
function Usage(){
echo " Usage: show_parameters [ arg1 ][ arg2 ]"
}
[[ ${#2} -eq 0 ]] && Usage || {
echo
echo "# arguments called with ----> ${@} "
echo "# \$1 -----------------------> $1 "
echo "# \$2 -----------------------> $2 "
echo "# path to me ---------------> ${0} " | sed "s/$USER/\$USER/g"
echo "# parent path --------------> ${0%/*} " | sed "s/$USER/\$USER/g"
echo "# my name ------------------> ${0##*/} "
echo
}
건배
짧고 명확하며 간단합니다. my_script.sh
#!/bin/bash
running_file_name=$(basename "$0")
echo "You are running '$running_file_name' file."
넣어 :
./my_script.sh
You are running 'my_script.sh' file.
DIRECTORY=$(cd `dirname $0` && pwd)
다른 스택 오버플로 질문에서 위의 내용을 얻었습니다 .Bash 스크립트가 저장된 디렉토리를 알 수 있습니까? 이 주제에도 도움이된다고 생각합니다.
저는 여기에서 영감을 해낸 것입니다 Dimitre Radoulov 의 대답 (나는 방법에 의해 upvoted한다) .
script="$BASH_SOURCE"
[ -z "$BASH_SOURCE" ] && script="$0"
echo "Called $script with $# argument(s)"
스크립트 호출 방식에 관계없이
. path/to/script.sh
또는
./path/to/script.sh
이 같은 것?
export LC_ALL=en_US.UTF-8
#!/bin/bash
#!/bin/sh
#----------------------------------------------------------------------
start_trash(){
ver="htrash.sh v0.0.4"
$TRASH_DIR # url to trash $MY_USER
$TRASH_SIZE # Show Trash Folder Size
echo "Would you like to empty Trash [y/n]?"
read ans
if [ $ans = y -o $ans = Y -o $ans = yes -o $ans = Yes -o $ans = YES ]
then
echo "'yes'"
cd $TRASH_DIR && $EMPTY_TRASH
fi
if [ $ans = n -o $ans = N -o $ans = no -o $ans = No -o $ans = NO ]
then
echo "'no'"
fi
return $TRUE
}
#-----------------------------------------------------------------------
start_help(){
echo "HELP COMMANDS-----------------------------"
echo "htest www open a homepage "
echo "htest trash empty trash "
return $TRUE
} #end Help
#-----------------------------------------------#
homepage=""
return $TRUE
} #end cpdebtemp
# -Case start
# if no command line arg given
# set val to Unknown
if [ -z $1 ]
then
val="*** Unknown ***"
elif [ -n $1 ]
then
# otherwise make first arg as val
val=$1
fi
# use case statement to make decision for rental
case $val in
"trash") start_trash ;;
"help") start_help ;;
"www") firefox $homepage ;;
*) echo "Sorry, I can not get a $val for you!";;
esac
# Case stop