내가 원하는 것은 여기와 거의 똑같지만 결과의 형식을 "줄 번호, 구분 기호, 파일 이름, 줄 바꿈"으로 지정하여 줄 번호가 파일 이름 뒤가 아닌 줄 시작 부분에 나타나도록 하고 싶습니다. 일치하는 항목이 포함된 행은 표시되지 않습니다.
이 형식이 선호되는 이유는 다음과 같습니다.
- (ㅏ)파일 이름은 길고 비밀스러울 수 있으며 도구에서 파일 이름을 줄 번호와 구분하는 데 사용하는 구분 기호를 포함합니다. 이는 파일 내의 패턴에도 동일한 구분 기호가 포함될 수 있기 때문에 awk로 이를 수행하는 것이 매우 어렵습니다. 또한 줄 시작 부분의 줄 번호는 파일 이름 뒤에 나타나는 줄 번호보다 더 잘 정렬됩니다. 이 필수 형식이 필요한 또 다른 이유는
(둘)패턴과 일치하는 줄이 너무 길어서 stdout에 표시된 출력에서 한 줄당 한 줄 속성을 엉망으로 만들 수 있습니다. 파일에 저장하고 다음과 같은 도구를 사용하는 것보다 stdout에서 출력을 보는 것이 더 좋습니다. vi를 사용하면 출력 파일의 각 줄을 한 줄씩 볼 수 있습니다.
이제 요청을 하였으므로 다음 사항을 고려하십시오.
제가 사용하고 있는 Linux 호스트에 Ack가 설치되어 있지 않아 사용할 수 없습니다.
다음을 수행하면 셸은
find .
"find ."를 실행하고 현재 작업 디렉터리에서 시작하여 재귀적으로 내려가는 절대 경로 목록으로 바꿉니다.grep -n PATTERN $(find .)
그런 다음 -n은 줄 번호를 인쇄하지만 원하는 위치는 인쇄하지 않습니다. 또한 어떤 이유로 디렉토리 이름에 패턴이 포함되어 있으면 grep이 해당 패턴을 포함하는 일반 파일과 일치하는지 이해가 되지 않습니다. 이것은 내가 원하는 것이 아니므로 다음을 사용합니다.
grep -n PATTERN $(find . -type f)
또한 find 출력이 grep에 동적으로 전달되도록 이 명령을 변경하고 싶습니다. 먼저 절대 경로의 전체 목록을 작성한 다음 대부분을 grep에 전달하는 대신 목록을 작성할 때 각 행을 grep에 전달하는 편이 낫기 때문에 다음을 시도했습니다.
find . -exec grep -n PATTERN '{}' \;
에 따르면 이것은 올바른 구문인 것 같지만
man page
이 명령을 실행하면 Bash 쉘이 약 100배 느리게 실행되므로 이는 올바른 접근 방식이 아닙니다.
내가 설명한 내용을 고려하면 어떻게 이 명령과 유사한 작업을 수행하고 원하는 형식을 얻을 수 있습니까? 문제의 게시물과 관련된 질문을 나열했습니다.
답변1
grep 사용
를 사용하는 대신 파일 시스템 반복으로 -r
전환 할 수 없는 이유는 무엇입니까 ? 나는 또한 스위치 대신 2개의 추가 스위치를 사용할 것입니다 .grep
find
-n
$ grep -rHn PATTERN <DIR> | cut -d":" -f1-2
예시 #1
$ grep -rHn PATH ~/.bashrc | cut -d":" -f1-2
/home/saml/.bashrc:25
세부 사항
-r
- 파일 + 디렉토리를 재귀적으로 검색-H
- 일치하는 경우 파일 이름을 인쇄합니다(보다 덜 제한적). 즉, 다른 스위치와 함께-l
사용됩니다.grep
-n
- 일치하는 줄 번호 표시
예시 #2
$ grep -rHn PATH ~/.bash* | cut -d":" -f1-2
/home/saml/.bash_profile:10
/home/saml/.bash_profile:12
/home/saml/.bash_profile_askapache:99
/home/saml/.bash_profile_askapache:101
/home/saml/.bash_profile_askapache:118
/home/saml/.bash_profile_askapache:166
/home/saml/.bash_profile_askapache:218
/home/saml/.bash_profile_askapache:250
/home/saml/.bash_profile_askapache:314
/home/saml/.bash_profile_askapache:2317
/home/saml/.bash_profile_askapache:2323
/home/saml/.bashrc:25
찾기 사용
$ find . -exec sh -c 'grep -Hn PATTERN "$@" | cut -d":" -f1-2' {} +
예
$ find ~/.bash* -exec sh -c 'grep -Hn PATH "$@" | cut -d":" -f1-2' {} +
/home/saml/.bash_profile:10
/home/saml/.bash_profile:12
/home/saml/.bash_profile_askapache:99
/home/saml/.bash_profile_askapache:101
/home/saml/.bash_profile_askapache:118
/home/saml/.bash_profile_askapache:166
/home/saml/.bash_profile_askapache:218
/home/saml/.bash_profile_askapache:250
/home/saml/.bash_profile_askapache:314
/home/saml/.bash_profile_askapache:2317
/home/saml/.bash_profile_askapache:2323
/home/saml/.bashrc:25
정말 사용하고 싶다면 .find 파일을 사용하여 파일을 찾은 후 find
이렇게 하면 됩니다 .grep
find
답변2
grep -n PATTERN `find . -type f`
이는 명령 대체의 출력이 공백으로 구분된 파일 이름 와일드카드 패턴 목록으로 해석되기 때문에 좋지 않습니다. 파일 이름에 공백이나 그 중 하나가 포함된 경우 \[*?
이 스니펫은 작동하지 않습니다 . 또한 일치하는 파일이 많으면 명령줄이 너무 길어지게 됩니다.
find . -exec grep -n PATTERN '{}' \;
이는 훌륭하고 안정적이지만 grep
파일당 한 번 호출됩니다. 그래서 속도가 너무 느립니다.
-exec … {} +
최대한 많은 파일을 일괄적으로 실행하는 명령 입니다. 마지막 배치(또는 이론적으로 다른 배치)는 단일 파일로 구성될 수 있으므로 grep
파일 이름이 인쇄되지 않습니다. -H
항상 파일 이름을 인쇄하는 옵션을 전달하거나 매개변수를 추가합니다 (일치하는 항목은 포함하지 않지만 최소한 두 개가 표시되도록 /dev/null
합니다). grep
파일 이름).
find . -type f -exec grep -Hn PATTERN {} +
GNU grep에는 일치하는 줄 번호를 인쇄하는 옵션이 없지만 일치하는 줄 텍스트를 인쇄하는 옵션은 없습니다. sed를 사용하여 일치하는 텍스트를 제거하고 줄 번호를 파일 이름으로 바꿀 수 있습니다.
find . -type f -exec grep -Hn PATTERN {} + | sed 's/^\([^:]*\):\([^:]*\):.*/\2:\1/'
줄 번호를 오른쪽으로 정렬하려면 awk가 제가 생각할 수 있는 어떤 대안보다 훨씬 간단합니다.
find . -type f -exec grep -Hn PATTERN {} + | awk -F : '{printf "%8d:%s", $2, $1}'
grep 대신 awk에서 일치를 수행하면 더 많은 제어권을 얻을 수 있습니다. Awk는 보다 일반적인 해석 언어 도구이기 때문에 속도가 느린 경향이 있습니다. 한 가지 이점은 grep의 출력이 모호해질 수 있는 콜론이나 개행 문자가 포함된 파일 이름을 처리하는 방법을 선택할 수 있다는 것입니다. 다음 코드 조각은 awk를 사용하여 포함된 파일 이름 :
(개행도 포함하지만 이러한 파일 이름의 경우 모호한 출력을 생성함)을 검색하고 처리합니다. awk를 사용하는 것에 주의하세요확장 정규식, 예를 들어 grep -E
(사소한 변경 사항이 있지만 grep 또는 awk 구현 간의 변경 사항 이상은 아닙니다).
find . -type f -exec awk '/PATTERN/ {printf "%d:", FNR; print FILENAME}' {} +