![제외 및 포함 파일 형식 확장자를 grep에 전달합니다.](https://linux55.com/image/193406/%EC%A0%9C%EC%99%B8%20%EB%B0%8F%20%ED%8F%AC%ED%95%A8%20%ED%8C%8C%EC%9D%BC%20%ED%98%95%EC%8B%9D%20%ED%99%95%EC%9E%A5%EC%9E%90%EB%A5%BC%20grep%EC%97%90%20%EC%A0%84%EB%8B%AC%ED%95%A9%EB%8B%88%EB%8B%A4..png)
쉼표를 구분 기호로 사용하여 파일 형식 확장에 대한 제외 및 포함 정보를 포함하는 bash 함수에 명령줄 인수로 전달되는 두 개의 변수가 있습니다.
excl="el"
excl="el,htm"
incl="texi,org"
제외 및 포함 옵션을 사용 excl
하고 구문 분석 하고 싶습니다 .incl
grep
excl="el,htm"
incl="texi,org"
grep -hir --exclude=\*.{el,htm} --include=\*.{org,texi} "$@"
excl="el"
incl="texi,org"
grep -hir --exclude=\*.el --include=\*.{org,texi} "$@"
답변1
개별 문자열을 추적하려면 배열을 사용해야 합니다. 단일 문자열을 사용하여 여러 값을 저장하면 구분 기호가 포함된 접미사(예: ,v
CVS 및 RCS 파일에 있는 접미사)가 있는 코드를 사용할 수 없습니다 .
exclude=( .el .htm )
include=( .texi .org )
opts=( -h -i -r )
for ext in "${exclude[@]}"; do
opts+=( --exclude="*$ext" )
done
for ext in "${include[@]}"; do
opts+=( --include="*$ext" )
done
grep "${opts[@]}" "$@"
그러면 파일 이름 접미사가 두 개의 배열 exclude
및 에 저장됩니다 include
. 그런 다음 두 배열의 요소를 반복하여 이라는 세 번째 배열에 적절한 옵션을 추가합니다 opts
. 그런 다음 에 대한 호출에서 세 번째 배열을 사용합니다 grep
.
예를 들어 배열을 확장할 때 사용되는 큰따옴표는 "${opts[@]}"
각 개별 배열 요소가 큰따옴표로 묶여 있고 쉘에서 더 이상 분할하거나 와일드카드를 사용할 수 없도록 합니다.
함수로서 포함 및 제외된 파일 이름 접미사 목록을 두 개의 별도 인수로 사용합니다.
call_grep () {
local -n include="$1"
local -n exclude="$2"
shift 2
local opts=( -h -i -r )
local ext
for ext in "${exclude[@]}"; do
opts+=( --exclude="*$ext" )
done
for ext in "${include[@]}"; do
opts+=( --include="*$ext" )
done
grep "${opts[@]}" "$@"
}
스크립트의 주요 부분:
excl=( .el .htm )
incl=( .texi .org )
call_grep incl excl more arguments here
call_grep
이는 두 배열의 이름을 가져오는 함수를 설정합니다 . 첫 번째 배열은 포함할 파일 이름 접미사 배열이고, 두 번째 배열은 제외할 접미사 배열입니다. 함수는 이러한 배열의 이름을 수신하고 이를 사용하여 두 개의 로컬 이름 참조 변수를 설정합니다. grep
Pass as-는 세 번째 매개변수로 시작합니다 .
다시 말하지만 실제 명령줄 구문 분석을 사용하면 다음과 같습니다 call_grep
.
call_grep () {
OPTIND=1
local ext opt
local opts=( -h -i -r )
while getopts 'i:e:' opt; do
case $opt in
i)
local -n include="$OPTARG"
for ext in "${include[@]}"; do
opts+=( --include="*$ext" )
done
;;
e)
local -n exclude="$OPTARG"
for ext in "${exclude[@]}"; do
opts+=( --exclude="*$ext" )
done
;;
*)
echo 'Error in option parsing' >&2
exit 1
esac
done
shift "$(( OPTIND - 1 ))"
grep "${opts[@]}" "$@"
}
이제 함수는 하나 -i
와 하나 -e
의 인수(둘 다 선택 사항)를 허용합니다. 각 옵션에 대한 인수는 포함하거나 제외할 파일 이름 접미사를 포함하는 배열의 이름이어야 합니다.
다음과 같이 사용할 수 있습니다.
excl=( .el .htm )
incl=( .texi .org )
call_grep -i incl -e excl -- more arguments here
--
에 직접 전달되어야 하는 매개변수와 함수의 매개변수를 분리하려면 를 사용해야 합니다 grep
.
grep
쉘 모드나 긴 옵션을 언급하지 않고 간단히 호출하는 방법만 원하는 경우 :
call_grep () {
OPTIND=1
while getopts 'i:e:' opt; do
case $opt in
i) opts+=( --include="*$OPTARG" ) ;;
e) opts+=( --exclude="*$OPTARG" ) ;;
*) echo 'error' >&2; exit 1
esac
done
shift "$(( OPTIND - 1 ))"
grep "${opts[@]}" "$@"
}
반복적으로 사용하여 여러 접미사를 포함 할 수 -i suffix
있으며 마찬가지로 접미사를 제외할 수도 있습니다. 예를 들어,
call_grep -i .texi -e .el -e .htm -i .org -- other arguments for grep here
또는
call_grep -i{.texi,.org} -e{.htm,.el} -- more here for grep
답변2
양식으로 확장 목록을 얻으려면 el,htm
다음을 수행하십시오.유혹을 받다중괄호 확장을 사용하는 것과 유사한 작업을 수행합니다.
eval 'echo grep -hir --exclude=*.{'"$excl"'} "$@"'
그러나 일반적인 주의 사항과는 별도로 eval
, 예를 들어 따옴표가 없는 세미콜론은 명령을 종료하여 구문을 엉망으로 만듭니다. 확장이 하나만 포함되거나 확장이 전혀 없는 경우 $excl
에도 작동하지 않습니다.grep
$excl
{foo}
{}
아니요버팀대 확장으로 처리됩니다. 그러니 중괄호 확장을 잊어버리자.
실제로 위의 Kusalananda의 답변처럼 배열에 매개변수 목록을 작성하게 됩니다.조건부로 인수를 스크립트에 전달
쉼표를 구분 기호로 유지하면서 쉼표로 구분하는 쉬운 방법은 read -a
배열을 생성하는 것이며 옵션 목록을 작성하려면 다른 배열이 필요합니다.
excl="el,htm"
IFS=, read -r -a exts <<< "$excl"
opts=()
for ext in "${exts[@]}"; do
opts+=(--exclude="*.$ext")
done
grep -hir "${opts[@]}" "$@"
주석에서 알 수 있듯이 RCS와 같은 확장은 ,v
점으로 시작하지 않는 다른 확장과 마찬가지로 여기서 문제가 됩니다. 확장자를 문자열로 제공하려면 대신 콜론, 세미콜론 또는 공백을 구분 기호로 허용하고 필요한 경우 사용자에게 구분 기호로 명시적으로 점을 입력하도록 요청할 수 있습니다 :
.
excl=".html:,v"
IFS=: read -r -a exts <<< "$excl"
opts=()
for ext in "${exts[@]}"; do
opts+=(--exclude="*$ext")
done
grep -hir "${opts[@]}" "$@"
물론 구분 기호로 사용하기로 선택한 문자는 확장의 일부가 될 수 없지만 세미콜론과 콜론은 쉼표보다 드물게 사용됩니다.