Linux программирование в примерах
Шрифт:
Рис. 12.1. Перекрывающиеся копии
Целью является скопировать четыре экземпляра
struct xyz
от data[0]
до data[3]
в участок от data[3]
до data[6]
. Здесь проблемой является data[3]
, побайтовое копирование с перемещением в памяти из data[0]
затрет data[3]
до того, как он будет
data[6]
! (Может возникнуть также сценарий, когда копирование в памяти в обратном направлении разрушит перекрывающиеся данные.) Функция
memcpy
была первоначальной функцией в System V API для копирования блоков памяти; ее поведение для перекрывающихся блоков памяти не была подробно определена тем или иным способом. Для стандарта С 1989 г. комитет почувствовал, что это отсутствие определенности является проблемой, поэтому они придумали memmove
. Для обратной совместимости memcpy
была оставлена, причем поведение для перекрывающейся памяти было специально отмечено как неопределенное, а в качестве процедуры, корректно разрешающей проблемные случаи, была предложена memmove
. Какую из них использовать в своем коде? Для библиотечной функции, которая не знает, какие области памяти ей передаются, следует использовать
memmove
. Таким способом вы гарантируете, что не будет проблем с перекрывающимися областями. Для кода приложения, который «знает», что две области не перекрываются, можно безопасно использовать memcpy
. Как для
memcpy
, так и для memmove
(как и для strcpy
) буфер назначения является первым аргументом, а источник — вторым. Чтобы запомнить это, обратите внимание на порядок, который тот же самый, как в операторе присваивания:
dest = src;
(Справочные страницы во многих системах не помогают, предлагая прототип в виде '
void *memcpy(void *buf1, void *buf2, size_t n)
' и полагаясь на то, что текст объяснит, что есть что. К счастью, справочная страница GNU/Linux использует более осмысленные имена.) 12.2.3. Сравнение блоков памяти:
memcmp
Функция
memcmp
сравнивает count
байтов из двух произвольных буферов данных. Возвращаемое ею значение подобно strcmp
: отрицательное, нулевое или положительное, если первый буфер меньше, равен или больше второго. Вы можете поинтересоваться: «Почему бы не использовать для такого сравнения
strcmp
?» Разница между двумя функциями в том, что memcmp
не принимает во внимание нулевые байты (завершающий строку '\0
'.) Таким образом, memcmp
является функцией, которая используется, когда вам нужно сравнить произвольные двоичные данные. Другим преимуществом
memcmp
является то, что она быстрее типичной реализации на C:
/* memcmp --- пример реализации на С, НЕ для реального использования */
int memcmp(const void *buf1, const void *buf2, size_t count) {
const unsigned char *cp1 = (const unsigned char*)buf1;
const unsigned char *cp2 = (const unsigned char*)buf2;
int diff;
while (count-- != 0) {
diff = *cp1++ - *cp2++;
if (diff != 0)
return diff;
}
return 0;
}
Скорость
По этим причинам всегда следует использовать вашу библиотечную версию
memcmp
вместо прокручивания своей собственной. Велика вероятность, что автор библиотеки знает машину лучше вас 12.2.4. Поиск байта с данным значением:
memchr
Функция
memchr
сходна с функцией strchr
: она возвращает местоположение определенного значения внутри произвольного буфера. Как и в случае memcmp
против strcmp
, основной причиной для использования memchr
является использование произвольных двоичных данных. GNU
wc
использует memchr
при подсчете лишь строк и байтов [124] , и это позволяет wс
быть быстрой. Из wc.c
в GNU Coreutils:124
См. wс(1).
wc
подсчитывает строки, слова и символы — Примеч. автора.
257 else if (!count_chars && !count_complicated)
258 {
259 /* Использует отдельный цикл при подсчете лишь строк или строк и байтов -
260 но не символов или слов. */
261 while ((bytes_read = safe_read(fd, buf, BUFFER_SIZE)) > 0)
262 {
263 register char *p = buf;
264
265 if (bytes_read == SAFE_READ_ERROR)
266 {
267 error(0, errno, "%s", file);
268 exit_status = 1;
269 break;
270 }
271
272 while ((p = memchr(p, '\n', (buf + bytes_read) - p)))
273 {
274 ++p;
275 ++lines;
276 }
277 bytes += bytes_read;
278 }
Поделиться:
Популярные книги
Возвышение Меркурия. Книга 7
7. Меркурий
Фантастика:
героическая фантастика
попаданцы
аниме
5.00
рейтинг книги
Жена со скидкой, или Случайный брак
Любовные романы:
любовно-фантастические романы
8.15
рейтинг книги
Совок 2
2. Совок
Фантастика:
альтернативная история
7.61
рейтинг книги
Прометей: каменный век
1. Прометей
Фантастика:
альтернативная история
6.82
рейтинг книги
Ты предал нашу семью
2. Предатели
Любовные романы:
современные любовные романы
5.00
рейтинг книги
Уязвимость
Любовные романы:
современные любовные романы
7.44
рейтинг книги
Венецианский купец
1. Венецианский купец
Фантастика:
фэнтези
героическая фантастика
альтернативная история
7.31
рейтинг книги
Попаданка для Дракона, или Жена любой ценой
Любовные романы:
любовно-фантастические романы
7.17
рейтинг книги
Любимая учительница
1. совершенная любовь
Любовные романы:
современные любовные романы
эро литература
8.73
рейтинг книги
Воевода
5. Помещик
Фантастика:
альтернативная история
5.00
рейтинг книги
Деспот
Любовные романы:
современные любовные романы
эро литература
5.00
рейтинг книги
Мастер...
1. Мастер
Фантастика:
героическая фантастика
попаданцы
аниме
6.50
рейтинг книги
Не кровный Брат
Любовные романы:
эро литература
6.83
рейтинг книги
Не грози Дубровскому! Том III
3. РОС: Не грози Дубровскому!
Фантастика:
фэнтези
попаданцы
аниме
5.00