Чтение онлайн

ЖАНРЫ

Основы программирования в Linux
Шрифт:

tbreak -- Set a temporary breakpoint

tcatch -- Set temporary catchpoints to catch events

thbreak -- Set a temporary hardware assisted breakpoint

watch -- Set a watchpoint for an expression

Type "help" followed by command name for full documentation.

Type "apropos word" to search for commands related to "word".

Command name abbreviations are allowed if unambiguous.

Установите точку останова в строке 21 и выполните программу:

$ gdb debug4

(gdb) break 21

Breakpoint 1 at 0x8048427: file debug4.c, line 21.

(gdb) run

Starting program: /home/neil/BLP4e/chapter10/debug4

Breakpoint 1, sort (a=0x804a040, n=5) at debug4.c:21

21 /* 21 */ s = 0;

Вы

можете вывести значение массива и затем с помощью команды
cont
разрешить программе продолжить выполнение. Это позволит программе выполняться до тех пор, пока она не натолкнется на следующую точку останова, в нашем случае это снова строка 21. В любой момент времени может быть активно несколько точек останова:

(gdb) print array[0]

$1 = (data = "bill", '\0' <repeats 4091 times>, key = 3)

Для вывода нескольких последовательных элементов массива можно применить конструкцию

@<число>
, чтобы заставить gdb вывести указанное количество элементов массива. Для того чтобы вывести все пять элементов, можно использовать следующую команду:

(gdb) print array[0]@5

$2 = {{data = "bill", '\0' <repeats 4091 times>, key = 3}, {

data = "neil", '\0' <repeats 4091 times>, key =4}, {

data = "john", '\0' <repeats 4091 times>, key =2}, {

data = "rick", '\0' <repeats 4091 times>, key =5}, {

data = "alex", '\0' <repeats 4091 times>, key = 1}}

Учтите, что вывод немного подчищен, чтобы его легче было читать. Поскольку это первый проход цикла, массив еще не изменен. Когда вы разрешите программе продолжить выполнение, то увидите последовательные перестройки массива array, происходящие по мере выполнения программы:

(gdb) cont

Continuing.

Breakpoint 1, sort (a=0x8049580, n=4) at debug4.c:21

21 /* 21 */ s = 0;

(gdb) print array[0]@5

$3 = {{data = "bill", '\0' <repeats 4091 times>, key = 3}, {

data = "john", '\0' <repeats 4091 times>, key =2}, {

data = "neil", '\0' <repeats 4091 times>, key = 4}, {

data = "alex", '\0' <repeats 4091 times>, key =1}, {

data = "rick", '\0' <repeats 4091 times>, key =5}}

(gdb)

Можно воспользоваться командой

display
, чтобы задать в gdb автоматическое отображение массива при каждой остановке программы в точке останова:

(gdb) display array[0]@5

1: array[0]@5 = {{data = "bill", '\0' <repeats 4091 times>, key = 3}, {

data = "john", '\0' <repeats 4091 times>, key = 2}, {

data = "neil", '\0' <repeats 4091 times>, key = 4}, {

data = "alex", '\0' <repeats 4091 times>, key = 1}, {

data = "rick", '\0' <repeats 4091 times>, key, = 5}}

Более

того, вы можете изменить точку останова таким образом, что вместо остановки программы она просто отобразит данные, которые вы запросили, и продолжит выполнение. Для этого примените команду
commands
. Она позволит указать, какие команды отладчика выполнять при попадании в точку останова. Поскольку вы уже указали отображение, вам нужно лишь задать команду в точке останова для продолжения выполнения:

(gdb) commands

Type commands for when breakpoint 1 is hit, one per line.

End with a line saying just "end".

> cont

> end

Теперь, когда вы разрешите программе продолжить выполнение, она продолжается до завершения, выводя значение массива каждый раз, когда оказывается вблизи внешнего цикла.

(gdb) cont

Continuing.

Breakpoint 1, sort (a=0x8049684, n=3) at debug4.c:21

21 /* 21 */ s = 0;

1: array[0]@5 = {{data = "john", '\000' <repeats 4091 times>, key = 2}, {

data = "bill", '\000' <repeats 4091 times>, key =3}, {

data = "alex", '\000' <repeats 4091 times>, key =1}, {

data = "neil", '\000' <repeats 4091 times>, key =4}, {

data = "rick", '\000' <repeats 4091 times>, key = 5}}

array[0] = {john, 2}

array[1] = {alex, 1}

array[2] = {bill, 3}

array[3] = {neil, 4}

array[4] = {rick, 5}

Program exited with code 025.

(gdb)

Отладчик gdb сообщает о том, что программа завершается с необычным кодом завершения. Это происходит потому, что программа сама не вызывает

exit
и не возвращает значение из функции
main
. Код завершения в данном случае не имеет смысла, значимый код должен предоставляться вызовом функции
exit
.

Кажется, что программа не выполняет внешний цикл столько раз, сколько ожидалось. Вы можете увидеть, что значение параметра

n
, используемого в условии завершения цикла, уменьшается при каждом достижении точки останова. Это значит, что цикл не будет выполняться нужное число раз. Дело в уменьшении
n
в строке 30.

/* 30 */ n--;

Это попытка оптимизировать программу за счет того, что в конце каждого прохода внешнего цикла наибольший, элемент

array
окажется внизу и поэтому остается меньше элементов для сортировки. Но как видно, это мешает внешнему циклу и создает проблемы. Простейший способ исправления (хотя есть и другие) — удалить ошибочную строку. Давайте проверим, применив отладчик для корректировки, устранило ли такое исправление проблему.

Поделиться с друзьями: