Категории
Самые читаемые
PochitayKnigi » Компьютеры и Интернет » Интернет » Linux программирование в примерах - Роббинс Арнольд

Linux программирование в примерах - Роббинс Арнольд

Читать онлайн Linux программирование в примерах - Роббинс Арнольд

Шрифт:

-
+

Интервал:

-
+

Закладка:

Сделать
1 ... 187 188 189 190 191 192 193 194 195 ... 253
Перейти на страницу:

26    setlocale(LC_ALL, locale);

27    strcpy(curloc, locale);

28   }

29

30   printf("%s: strcmp("%s", "%s") is %dn", curloc, left,

31    right, strcmp(left, right));

32   printf("%s: strcoll("%s", "%s") is %dn", curloc, left,

33    right, strcoll(left, right));

34

35   printf("n--> "); fflush(stdout);

36  }

37

38  exit(0);

39 }

Программа читает входные строки, состоящие из двух сравниваемых слов и необязательной локали, использующейся для сравнения. Если локаль дана, она становится локалью для последующих элементов. Программа начинает с любой локалью, которая установлена в окружении.

Массив curloc сохраняет текущую локаль для вывода результатов; left и right являются левым и правым сравниваемыми словами (строки 10–11). Основную часть программы составляет цикл (строки 19–36), который читает строки и выполняет работу. Строки 20–23 разделяют входную строку, locale инициализируется пустой строкой, если третья строка не предусмотрена.

Строки 25–28 устанавливают новую локаль, если она приведена. Строки 30–33 выводят результаты сравнения, а строка 35 приглашает для дальнейшего ввода. Вот демонстрация:

$ <b>ch13-compare</b> /* Запуск программы */

--&gt; <b>ABC abc</b> /* Ввести два слова */

С: strcmp(&quot;ABC&quot;, &quot;abc&quot;) is -1 /* Программа началась в локали &quot;С&quot; */

С: strcoll(&quot;ABC&quot;, &quot;abc&quot;) is -1 /* В локали &quot;С&quot; идентичные рез-ты */

--&gt; <b>ABC abc en_US</b> /* Слова те же, локаль &quot;en_US&quot; */

en_US: strcmp(&quot;ABC&quot;, &quot;abc&quot;) is -1 /* strcmp() без изменений */

en_US: strcoll(&quot;ABC&quot;, &quot;abc&quot;) is 2 /* рез-ты strcoll() изменились' */

--&gt; <b>ABC abc en_US.UTF-8</b> /* Слова те же, локаль &quot;en_US.UTF-8&quot; */

en_US.UTF-8: strcmp(&quot;ABC&quot;, &quot;abc&quot;) is -1

en_US. UTF-8: strcoll(&quot;ABC&quot;, &quot;abc&quot;) is 6

 /* Другое значение, все еще положительное */

--&gt; <b>junk JUNK</b> /* Новые слова */

en_US.UTF-8: strcmp(&quot;junk&quot;, &quot;JUNK&quot;) is 1 /* предыдущая локаль */

en_US.UTF-8: strcoll(&quot;junk&quot;, &quot;JUNK&quot;) is -6

Эта программа ясно показывает различие между strcmp() и strcoll(). Поскольку strcmp() работает в соответствии с числовыми значениями символов, она всегда возвращает тот же самый результат, strcoll() понимает проблемы сортировки, и ее результат меняется в соответствии с локалью. Мы видим, что в обеих локалях en_US заглавные буквы идут после строчных.

ЗАМЕЧАНИЕ. Специфическая для локали сортировка строк является проблемой также и для сопоставления регулярных выражений. Регулярные выражения допускают диапазоны символов внутри выражений со скобками, такие, как '[a-z]' или '[&quot;-/]'. Точное значение такой конструкции (символы, численно располагающиеся между начальной и конечной точками включительно) определено лишь для локалей «С» и «POSIX»

Для локалей, не являющихся ASCII, такие диапазоны как '[a-z]' могут соответствовать также и заглавным буквам, а не только строчным! Диапазон '[&quot;-/]' действителен в ASCII, но не в "en_US.UTF-8".

Долговременным наиболее переносимым решением является использование классов символов POSIX, таких, как '[[:lower:]]' и '[[:punct:]]'. Если вам кажется, что нужно использовать выражения с диапазонами на системах, использующих локали, и на более старых системах, не использующих их, без изменения своей программы, решение заключается в применении грубой силы и индивидуальном перечислении каждого символа внутри скобок. Это неприятно, но это работает.

Основанная на локалях сортировка потенциально дорогостоящая. Если вы ожидаете большого числа сравнений, где по крайней мере одна из строк не будет изменяться или где значения строк будут сравниваться друг с другом по несколько раз (как при сортировке списка), следует рассмотреть использование функции strxfrm() для преобразования своих строк для использования с strcmp(). Функция strxfrm() объявлена следующим образом:

#include &lt;string.h&gt; /* ISO С */

size_t strxfrm(char *dest, const char *src, size_t n);

Идея в том, что strxfrm() преобразует первые n символов src, помещая их в dest. Возвращаемое значение является числом символов, необходимых для сохранения преобразованных символов. Если она превышает n, содержимое dest «неопределенно».

Стандарт POSIX явным образом разрешает устанавливать в n ноль, а в dest NULL. В этом случае strxfrm() возвращает размер массива, необходимого для сохранения преобразованной версии src (не включая завершающий символ ''). Предполагается, что это значение впоследствии будет использовано с malloc() для создания массива dest или для проверки размера предопределенных границ массива (При этом, очевидно, src должен иметь завершающий нулевой байт.) Этот фрагмент иллюстрирует использование strxfrm():

1 ... 187 188 189 190 191 192 193 194 195 ... 253
Перейти на страницу:
Тут вы можете бесплатно читать книгу Linux программирование в примерах - Роббинс Арнольд.
Комментарии