grep: відобразити ім'я файлу один раз, а потім відобразити контекст з номерами рядків


16

У нашому вихідному коді є коди помилок, розкидані по всьому. Знайти їх легко за допомогою grep, але я хотів би find_codeвиконати функцію bash, яку я можу виконати (напр. find_code ####), Яка забезпечить вихід у цих рядках:

/home/user/path/to/source.c

85     imagine this is code
86     this is more code
87     {
88         nicely indented
89         errorCode = 1111
90         that's the line that matched!
91         ok this block is ending
92     }
93 }

Ось що я зараз маю:

find_code()
{
    # "= " included to avoid matching unrelated number series
    # SRCDIR is environment variable, parent dir of all of projects
    FILENAME= grep -r "= ${1}" ${SRCDIR}
    echo ${FILENAME}
    grep -A5 -B5 -r "= ${1}" ${SRCDIR} | sed -e 's/.*\.c\[-:]//g'
}

Проблеми:

1) Тут не вказані номери рядків

2) він відповідає лише .c вихідним файлам. У мене виникають проблеми з підключенням sed до файлів .c, .cs, .cpp та інших вихідних файлів. Ми, однак, використовуємо C, тому просто збігаємося - або: (символи, які grep додають до імені файлу перед кожним рядком коду), відповідають object->pointersі переплутують все.

Відповіді:


11

Я б змінив кілька речей.

find_code() { 
    # assign all arguments (not just the first ${1}) to MATCH
    # so find_code can be used with multiple arguments:
    #    find_code errorCode
    #    find_code = 1111
    #    find_code errorCode = 1111
    MATCH="$@" 

    # For each file that has a match in it (note I use `-l` to get just the file name
    # that matches, and not the display of the matching part) I.e we get an output of:
    #
    #       srcdir/matching_file.c
    # NOT:
    #       srcdir/matching_file.c:       errorCode = 1111
    #
    grep -lr "$MATCH" ${SRCDIR} | while read file 
    do 
        # echo the filename
        echo ${file}
        # and grep the match in that file (this time using `-h` to suppress the 
        # display of the filename that actually matched, and `-n` to display the 
        # line numbers)
        grep -nh -A5 -B5 "$MATCH" "${file}"
    done 
}

Я пристосував це назад до моїх специфікацій - просто хочу шукати коди помилок. Отже MATCH="= ${1}". Я також додав, --include=*.c --include=*.cpp --include=*.java --include=*.csщоб обмежити пошук вихідних файлів. Спасибі!
TravisThomas

1
Добре о, радий, що вам вдалося добре налагодити свої потреби :)
Drav Sloan

3

Ви можете використовувати findдва -execс, другий один буде виконуватися тільки тоді , коли перший з них є успішним, наприклад , пошук тільки .cpp, .cі .csфайли:

find_code() {
find ${SRCDIR} -type f \
\( -name \*.cpp -o -name \*.c -o -name \*.cs \) \
-exec grep -l "= ${1}" {} \; -exec grep -n -C5 "= ${1}" {} \;
}

тому перший grepдрукує назви файлів, які містять ваш візерунок, а другий друкує відповідні рядки + контекст, пронумерований, з відповідних файлів.

Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.