Вход/Регистрация
Разработка приложений в среде Linux. Второе издание
вернуться

Троан Эрик В.

Шрифт:

14.6. Добавление к

ladsh
возможностей работы с каталогами и универсализацией

Продолжим эволюцию

ladsh
, добавив к
ladsh3.с
четыре новых возможности.

1. Встроенная команда

cd
для смены каталогов.

2. Встроенная команда

pwd
для отображения текущего каталога.

3. Универсализация файловых имен.

4. Отображение ряда новых сообщений, позволяющее воспользоваться преимуществами

strsignal
. Эти изменения обсуждались в главе 12.

14.6.1. Добавление встроенных команд

cd
и
pwd

Добавление встроенных команд является прямым применением вызовов

chdir
и
getcwd
. Код соответствует
runProgram
как раз там, где обрабатываются другие встроенные команды. Ниже показан раздел обработки встроенных команд в
ladsh3.с
.

422: if (!strcmp(newJob.progs[0].argv[0], "exit")) {

423: /* здесь должен возвращаться реальный код завершения */

424: exit(0);

425: } else if (!strcmp(newJob.progs[0].argv[0], "pwd")) {

426: len = 50;

427: buf = malloc(len);

428: while (!getcwd(buf, len) && errno == ERANGE) {

429: len += 50;

430: buf = realloc(buf, len);

431: }

432: printf("%s\n", buf);

433: free(buf);

434: return 0;

435: } else if (!strcmp(newJob.progs[0].argv[0], "cd")) {

436: if (!new Job.progs[0].argv[1] == 1)

437: newdir = getenv("HOME");

438: else

439: newdir = newJob.progs[0].argv[1];

440: if (chdir(newdir))

441: printf("сбой при смене текущего каталога: %s\n",

442: strerror(errno));

443: return 0;

444: } else if (!strcmp(newJob.progs[0].argv[0], "jobs")) {

445: for (job = jobList->head; job; job = job->next)

446: printf(JOB_STATUS_FORMAT, job->jobId, "Выполняется",

447: job -> text);

448: return 0;

449: }

14.6.2. Добавление универсализации файловых имен

Универсализацию файловых имен, при которой оболочка разворачивает символы

*
,
[]
и
?
в соответствующие файловые имена, в определенной мере сложно реализовать из-за разнообразных методов применения кавычек. Первая модификация заключается в построении каждого аргумента в виде строки, подходящей для передачи в
glob
. Если символ универсализации помещен в кавычки, принятые в оболочке (например, двойные кавычки), тогда символу универсализации предшествует
\
с целью предотвращения его разворачивания в
glob
. Этот процесс реализуется легко, хотя с первого взгляда может показаться сложным.

Две части синтаксического разбора команд в

parseCommand(
) необходимо слегка изменить. Последовательности
"
и
'
обрабатываются ближе к началу цикла, что обеспечивает разделение командной строки на аргументы. Если во время синтаксического разбора мы находимся в середине строки в кавычках и сталкиваемся с символом универсализации, мы заключаем его в кавычки с предваряющим символом
\
, что выглядит следующим образом.

189: } else if (quote) {

190: if (*src == '\\') {

191: src++;

192: if (!*src) {

193: fprintf(stderr,

194: "после \\ ожидался символ\n");

195: freeJob(job);

196: return 1;

197: }

198:

199: /* в оболочке "\'" должен дать \' */

200: if (* src ! = quote) *buf++ = '\\';

201: } else if (*src = '*' | | *src == '?' || *src == '[' ||

202: *src == ']')

203: *buf++ = '\\';

204: *buf++ = *src;

205: } else if (isspace(*src)) {

В код были добавлены только средний

else if
и оператор присваивания в его теле. Похожий код потребуется предусмотреть для обработки символов
\
, встречающихся вне строк в кавычках. Это реализовано в конце главного цикла
parseCommand
. Ниже приведен измененный код.

329: case '\\':

  • Читать дальше
  • 1
  • ...
  • 105
  • 106
  • 107
  • 108
  • 109
  • 110
  • 111
  • 112
  • 113
  • 114
  • 115
  • ...

Ебукер (ebooker) – онлайн-библиотека на русском языке. Книги доступны онлайн, без утомительной регистрации. Огромный выбор и удобный дизайн, позволяющий читать без проблем. Добавляйте сайт в закладки! Все произведения загружаются пользователями: если считаете, что ваши авторские права нарушены – используйте форму обратной связи.

Полезные ссылки

  • Моя полка

Контакты

  • chitat.ebooker@gmail.com

Подпишитесь на рассылку: