Инструментальные средство UNIX, работающие в среде DOS
   [URL 30] The UWIN Development Tools
    www.gtlinc.com/Products/Uwin/uwin.html
   Фирма Global Technologies, Inc., Old Bridge, NJ
   Пакет UWIN предоставляет библиотеки динамической компоновки (DLL) Windows, которые эмулируют большую часть библиотечного интерфейса уровня Unix С. Используя данный интерфейс, фирма Global Technologies, Inc. перенесла большое число инструментальных средств из командной строки Unix в систему Windows. См. также [URL 31].
   [URL 31] The Cygnus Cygwin Tools
    www.sourceware.cygnus.com/cygwin/
   Фирма Cygnus Solutions, Sunnyvale, CA
   Пакет Cygnus также эмулирует интерфейс библиотеки Unix С и предоставляет большой набор инструментальных средств, работающих в режиме командной строки Unix, при работе в операционной системе Windows.
   [URL 32] Perl Power Tools
    www.perl.com/language/ppt/
   Данный проект посвящен повторной реализации классического набора команд Unix на языке Perl, что дает возможность их использования при работе со всеми платформами, поддерживающими Perl (их довольно много).
   Средствауправленияисходнымтекстомпрограмм
   [URL 33] RCS – Revision Control System
    www.cyclic.com
   Система управления исходным текстом программ GNU для Unix и Windows NT.
   [URL 34] CVS – Concurrent Version System
   www.cyclic.com
   Система управления исходным текстом программ для Unix и Windows NT, распространяемая бесплатно. Расширяет возможности Revision Control System, поддерживая модель «клиент-сервер» и параллельный доступ к файлам.
   [URL 35] Aegis Transaction-Based Configuration Management
    www.canb.auug.org.au/"mil!erp/aegis.html
   Инструментальное средство управления версиями (ориентированное на процесс), которое применяет к ним существующие стандарты проекта (например, проверку прохождения тестов для возвращаемых программных кодов).
   [URL 36] ClearCase
   www.rational.com
   Управление версиями программы, рабочей областью и полной сборкой программы, управление процессом.
   [URL 37] MKS Source Integrity
    www.mks.com
   Управление версиями и конфигурацией. Некоторые версии включают средства, позволяющие удаленным пользователям-разработчикам одновременно работать над одними и теми же файлами.
   [URL 38] PVCS Configuration Management
    www.merant.com
   Система управления исходным текстом программ, очень популярная при работе в Windows.
   [URL 39] Visual SourceSafe
    www.microsoft.com
   Система управления версиями, интегрируемая с инструментами визуальной разработки фирмы Microsoft.
   [URL 40] Perforce
   www.perforce.com
   Менеджер конфигурирования программного обеспечения "клиент-сервер".
   Прочие инструментальные средства
   [URL 41] Winzip – Archive Utility for Windows
    www.winzip.com
   Фирма Nico Mak Computing, Inc., Mansfield, CT
   Утилита архивирования файлов, работающая в среде Windows. Поддерживает форматы zip и tar.
   [URL 42] The Z Shell
    www.sunsite.auc.dk/zsh
   Оболочка, предназначенная для интерактивной работы и содержащая мощный язык сценариев. В оболочку zsh было включено много полезных средств из оболочек bash, ksh и tcsh и добавлен ряд оригинальных элементов.
   [URL 43] A Free SMB Client for Unix Systems
   www.samba.anu.edu.au/pub/samba/
   Дает возможность совместного использования файлов и других ресурсов из операционных систем Unix и Windows. Samba включает в себя:
   • Сервер SMB, предоставляющий средства для работы с файлами и для печати (схожие со средствами, предоставляемыми Windows NT и LAN Manager) клиентам SMB, в роли которых могут выступать Windows 95, Warp Server, smbfs и др.
   • Сервер имен Netbios, обеспечивающий, кроме всего прочего, поддержку функций браузера. По желанию пользователя Samba может быть главным браузером в локальной сети.
   • Клиент SMB (схожий с клиентом ftp), позволяющий получать доступ к ресурсам ПК(дискам и принтерам) из Unix, Netware и других операционных систем.
Статьи и публикации
   [URL 44] The comp.object FAQ
   www.cyberdyne-object-sys.com/oofaq2
   Солидный, четко организованный список часто задаваемых вопросов по группе новостей comp.object.
   [URL 45] extreme Programming
   www.Xprogramming.com
   Цитата с интернет-сайта: "При создании команды, способной быстро создать исключительно надежное, эффективное, четко структурированное программное обеспечение, в ХР используется весьма легковесное сочетание методик. Многие из методик ХР создавались и опробовались в части проекта СЗ фирмы «Крайслер», представляющего собой весьма успешную систему расчета заработной платы, написанную на языке Smalltalk".
   [URL 46] Alistair Cockburn's Home Page
    www.members.aol.com/acockburn
   Стоит посмотреть раздел "Structuring Use Cases with Goals" и так называемые шаблоны сценариев использования.
   [URL 47] Martin Fowler's Home Page
    www.ourworld.compuseimcom/homepages/martinjowler
   Мартин Фаулер является автором книги "Analysis Patterns" и соавтором книг "UML Distilled" и "Refactoring: Improvingthe Design of Existing Code". На домашней странице автора обсуждаются его книги и работа с UML.
   [URL 48] Robert С. Martin's Home Page
    www.objectmentor.com/home
   Неплохое собрание статей ознакомительного плана по объектно-ориентированным методам, включая анализ зависимости и метрики.
   [URL 49] Aspect-Oriented Programming
    www.parc.xerox.com/csl/projects/aop/
   Описывается методика придания функциональности программному коду с позиций ортогональности и описательности.
   [URL 50] JavaSpaces Specifications
    www.java.sun.com/products/javaspaces
   Linda-подобная система для Java, поддерживающая распределенное сохранение состояния объекта и распределенные алгоритмы.
   [URL 51] Netscape Source Code
   www.mozilla.org
   Исходный текст браузера Netscape.
   [URL 52] The Jargon File
    www.jargon.org
   Eric S. Raymond
   Определения дня многих общих (и не очень общих) терминов, применяемых в компьютерной индустрии, сдобренные хорошей дозой соответствующего фольклора.
   [URL 53] Eric S. Raymond's Papers
    www.tuxedo.org/~esr
   Статьи Эрика Раймона "The Cathedral and the Bazaar" и "HomesteadingNoosphere", в которых описаны психосоциальные основы и смысл движения Open Source.
   [URL 54] The К Desktop Environment
    www.kde.org
   Цитата с web-страницы: "KDE представляет собой мощную графическую настольную среду для рабочих станций Unix. KDE является интернет-проектом, открытым в полном смысле этого слова".
   [URL 55] The GNU Image Manipulation Program
    www.gimp.org
   Gimp является бесплатной программой по созданию, композиции и ретушированию изображений.
   [URL 56] The Demeter Project
    www.ccs.neu.edu/research/demeter
   Исследовательский проект, призванный упростить поддержку и развитие программного обеспечения с помощью адаптивного программирования.
Другие источники
   [URL 57] The GNU Project
    www.gnu.org
   Фонд Free Software Foundation, Boston, MA
   Фонд "Free Software Foundation" – это некоммерческая благотворительная организация, осуществляющая сбор средств на проект GNU. Целью проекта GNU является создание завершенной, бесплатной UNIX-подобной операционной системы. Многие из попутно разработанных инструментальных средств уже стали отраслевыми стандартами.
   [URL 58] Web Server Information
    www.netcraft.com/survey/servers.html
   Ссылки на домашние страницы, находящиеся более чем на 50 web-серверах. Часть из них является коммерческими продуктами, другие же распространяются бесплатно.

Библиография

   [Bak72] F.T. Baker. Chief programmer team management of production programming. IBM Systems Journal, 1 1(1 ):56—73, 1972.
   [BBM96] V. Basili, L. Brand, and W.L. Melo. A validation of object-oriented design metrics as quality indicators. IEEE Transactions on Software Engineering, 22(10):751-761, October 1996.
   [Ber96] Albert J. Bernstein. Dinosaur Brains: Dealing with All Those Impossible People at Work. Ballantine Books, New York, NY, 1996.
   [Bra95] Marshall Brain. Win32 System Services. Prentice Hall, Englewood Cliffs, NJ, 1995.
   [Bro95] Frederick P. Brooks, Jr.. The Mythical Man Month: Essays on Software Engineering. Addison -Wesley, Reading, MA, anniversary edition, 1995.
   [CG90] N. Carriero and D. Gelenter. How to Write Parallel Programs: A First Course. MIT Press, Cambridge, MA, 1990.
   [CN91] Brad J. Cox and Andrex J. Novobilski. Object-Oriented Programming, An Evolutionary Approach. Addison-Wesley, Reading, MA, 1991.
   [Coc97a] Alistair Cockburn. Goals and use cases. Journal of Object Oriented Programming, 9(7):35-40, September 1997.
   [Coc97b] Alistair Cockburn. Surviving Object-Oriented Projects: A Manager's Guide. Addison Wesley Longman, Reading, MA, 1997.
   [Cop92] James O. Coplien. Advanced С++ Programming Styles and Idioms. Addison-Wesley, Reading, MA, 1992.
   [DL99] Tom Demarco and Timothy Lister. Peopleware: Productive Projects and Teams, Dorset House, New York, NY, second edition, 1999.
   [FBB+99] Martin Fowler, Kent Beck, John Brant, William Opdyke, and Don Roberts. Refactoring: Improving the Design of Existing Code. Addison Wesley Longman, Reading, MA, 1999
   [Fow96] Martin Fowler. Analysis Patterns: Reusable Object Models. Addison Wesley Longman, Reading, MA, 1996.
   [FS97] Martin Fowler and Kendall Scott. UML Distilled: Applying the Standard Object Modeling Language. Addison Wesley Longman, Reading, MA, 1997.
   [GHJV95] Erich Gamma, Richard Helm, Ralph Johnson, and John Vlissides. Design Patterns: Elements of Reusable Object-Oriented Software. Addison-Wesley, Reading, MA, 1995.
   [Gla99a] Robert L. Glass. Inspections – Some surprising findings. Communications of the ACM, 42(4): 17–19, April 1999.
   [Gla99b] Robert L. Glass. The realities of software technology payoffs. Communications of the ACM, 42(2):74–79, February 1999.
   [Hol78] Michael Holt. Math Puzzles and Games, Dorset Press, New York, NY, 1978.
   [Jac94] Ivar Jacobson. Object-Oriented Software Engineering: A Use-Case Driven Approach. Addison-Wesley, Reading, MA, 1994.
   [KLM+97] Gregor Kiczales, John Lamping, AnuragMendhekar, Chris Maeda, Cris-tina Videira Lopes, Jean-Marc Loingtier, and John Irwin. Aspect-oriented programming. In European Conference on Object-Oriented Programming (ECOOP), volume LNCS 1241. Springer-Verlag, June 1997.
   [Knu97a] Donald Ervin Knuth. The Art of Computer Programming: Fundamental Algorithms, volume 1. Addison Wesley Longman, Reading, MA, third edition, 1997.
   [Knu97b] Donald Ervin Knuth. The Art of Computer Programming: Seminumeri-cal Algorithms, volume 2. Addison Wesley Longman, Reading, MA, third edition, 1997.
   [Knu98] Donald Ervin Knuth. The Art of Computer Programming: Sorting and Searching, volume 3. Addison Wesley Longman, Reading, MA, second edition 1998.
   [KP99] Brian W. Kernighan and Rob Pike. The Practice of Programming. Addison Wesley Longman, Reading, MA, 1999.
   [Kru98] Philippe Kruchten. The Rational Unified Process: An Introduction. Addison Wesley Longman, Reading, MA, 1998.
   [Lak96] John Lakos. Large-Scale С++Software Design. Addison Wesley Longman, Reading, MA, 1996.
   [LH89] Karl J. Lieberherr and Ian Holland. Assuring good style for object-oriented programs. IEEE Software, pages 38–48, September 1989.
   [Lis88] Barbara Liskov. Data abstraction and hierarchy. SIGPLAN Notices, 23(5), May 1988.
   [LMB92] John R. Levine, Tony Mason, and Doug Brown. Lex and Yacc. O'Reilly & Associates, Inc., Sebastopol, CA, second edition, 1992.
   [McC95] Jim McCarthy. Dynamics of Software Development. Microsoft Press, Redmond, WA, 1995.
   [Mey96] Scott Meyers. More Effective С++: 35 New Ways to Improve Your Programs and Designs. Addison-Wesley, Reading, MA, 1996.
   [Mey97a] Scott Meyers. Effective С++: 50 Specific Ways to Improve Your Programs and Designs. Addison Wesley Longman, Reading, MA, 1997.
   [Mey97b] Bertrand Meyer. Object-Oriented Software Construction. Prentice Hall, Englewood Cliffs, NJ, second edition, 1997.
   [Pet 98] Charles Petzold. Programming Windows, The Definitive Guide to the Win32 API. Microsoft Press, Redmond, WA, fifth edition, 1998.
   [Sch95] Bruce Schneier. Applied Cryptography: Protocols, Algorithms, and Source Code in C. John Wiley & Sons, New York, NY, 1995.
   [Sed83] Robert Sedgewick. Algorithms. Addison-Wesley, Reading, MA, 1983.
   [Sed 92] Robert Sedgewick. Algorithms in С++. Addison-Wesley, Reading, MA, 1992.
   [SF96] Robert Sedgewick and Phillipc Flajolet. An Introduction to the Analysis of Algorithms. Addison-Wesley, Reading, MA, 1996.
   [Ste92] W. Richard Stevens. Advanced Programming in the Unix Environment. Addison-Wesley, Reading, MA, 1992.
   [Ste98] W. Richard Stevens. Unix Network Programming, Volume I: Networking APIs: Sockets andXti. Prentice Hall, Englewood Cliffs, NJ, 1998.
   [Ste99] W. Richard Stevens. Unix Network Programming, Volume 2: Interprocess Communications. Prentice Hall, Englewood Cliffs, NJ, second edition, 1999.
   [Str35] James Ridley Stroop. Studies of interference in serial verbal reactions. Journal of Experimental Psychology, 18:643–662, 1935.
   [WK82] James Q. Wilson and George Kelling. The police and neighborhood safety. The Atlantic Monthly, 249(3):29–38, March 1982.
   [YC86] Edward Yourdon and Larry L. Constantine. Structured Design: Fundamentals of a Discipline of Computer Program and Systems Design. Prentice Hall, Englewood Cliffs, NJ, second edition, 1986.
   [You95] Edward Yourdon. Managing projects to produce good-enough software. IEEE Software, March 1995.

Приложение В
Ответы к упражнениям

Упражнение 1 из раздела "Ортогональность"
   Ответ: По нашему разумению, более ортогональным является класс Split2. Он сосредоточен на собственной задаче – расщеплении строк и игнорирует подробности, связанные с источником обрабатываемых им строк. Это не только упрощает разработку программы, но и придает ей большую гибкость. Класс Split2 может расщеплять строки, считываемые из файла, сгенерированные другой программой или передаваемые через операционную среду.
Упражнение 2 из раздела "Ортогональность"
   Ответ: Если все сделано корректно, то, по всей вероятности, немодальное. Система, которая использует немодальные диалоговые окна, испытывает меньшее беспокойство о том, что происходит в любой конкретный момент времени. Скорее всего, она будет обладать лучшей инфрастуктурой взаимодействия между модулями по сравнению с модальной системой, которая может содержать встроенные предположения о состоянии системы – предположения, которые могут привести к большему связыванию и уменьшению ортогональности.
Упражнение 3 из раздела "Ортогональность"
   Ответ: Здесь есть элемент лукавства. Объектная технология может обеспечить наличие более ортогональной системы, но поскольку она имеет больше средств, которые могут эксплуатироваться с нарушением режима, в реальности легче создать неортогональную систему, используя объекты, чем создавать ее при помощи процедурного языка. Ее особенности – множественное наследование, исключительные ситуации, перегрузка операторов и переопределение родительского метода (через механизм подклассов) – предоставляют достаточные возможности для увеличения связанности не столь очевидными способами.
   Применяя объектную технологию и приложив небольшое дополнительное усилие, вы можете добиться наличия более ортогональной системы. И хотя вы всегда можете написать неструктурированную программу на процедурном языке, объектно-ориентированные языки, используемые в малых дозах, могут сделать ее более насыщенной.
Упражнение 4 из раздела "Прототипы и памятные записки"
   Ответ: Для спасения ситуации прибегнем к устаревшим технологиям! Нарисуйте на лекционной доске несколько картинок – автомобиль, телефон и дом – с помощью фломастера. Для этого не нужно быть великим художником, вполне достаточно условных изображений. Поместите на доску памятные записки, описывающие содержимое целевых страниц в активных областях экрана. В ходе встречи вы можете совершенствовать рисунки и менять расположение памятных записок.
Упражнение 5 из раздела "Языки, отражающие специфику предметной области"
   Ответ: Поскольку мы хотим, чтобы язык был расширяемым, сделаем таблицу синтаксического анализатора управляемой. Каждый элемент таблицы содержит символ команды, флаг, говорящий о необходимости аргумента, и имя подпрограммы, вызываемой для обработки этой конкретной команды.
   typedef struct {
   char cmd; /* the command letter */
   int hasArg; /* does it take an argument */
   void (*func)(int, int); /* routine to call */
   } Command;
   static Command cmds[] = {
   {'P', ARG, doSelectPen},
   ('V', NO_ARG, doPenUp},
   {'D', NO_ARG, doPenDown},
   {'N,' ARG, doPenDir},
   {'E', ARG, doPenDir},
   {'S', ARG, doPenDir},
   {'W', ARG, doPenDir}
   };
   Основная программа довольно проста: считать строку, отыскать команду, при необходимости принять аргумент, затем вызвать функцию обработчика.
   while (fgetsfbuff, sizeof(buff), stdin)) {
   Command *cmd = findCommand(*buff);
   if (cmd) {
     int arg = 0;
     if (cmd->hasAr&& !getArg(buff+1, &arg)) {
        fprintf(stderr,"'%с' needs an argument\n", *buff);
        continue;
     }
   cmd->func(*buff, arg);
   }
   }
   Функция, которая ищет команду, исполняет последовательный перебор таблицы, возвращая либо совпадающий элемент, либо NULL.
   Command *findCommand(int cmd) {
   int i;
   for (i = 0; i<ARRAY.SIZE(cmds); i++) {
     if (cmds[i].cmd==cmd)
       return cmds + i;
     }
   fprintf(stderr, "Unknown command %c'\n", cmd);
   return 0;
   }
   И наконец, считывание числового аргумента довольно просто, если использовать подпрограмму sscanf.
   int getArg(const char *buff, int 'result) {
      return sscanf(buff, "%d", result) == 1;
   }
Упражнение 6 из раздела "Языки, отражающие специфику предметной области"
   Ответ 6: При использовании BNF спецификация времени могла бы выглядеть следующим образом:
   <tlme>::= <hour> <ampm> |
           <hour>: <minute> <ampm> |
           <hour>: <minute>
   <ampm>::= am|pm
   <hour>::=<digit> |
           <digit>::=<digit>
   <mlnute>::=<digit><digit>
   <digit>::= 0|1|2|3|4|5|6|7|8|9
Упражнение 7 из раздела "Языки, отражающие специфику предметной области"
   Ответ: В нашем примере мы составили программу, используя генератор bison, который представляет собой GNU-версию генератора уасс. Для ясности здесь показано только тело программы синтаксического анализатора. Полная версия есть на сайте www.pragmaticprogrammmer.com.
   time: spec EOF
   { if ($1>= 24*60) yyerror("Time is too large");
      printf("%d minutes past midnight\n", $1);
      exit(0);
   }
   ;
   spec:  hour ':' minute
       { $$ = $1 + $3;
       }
     | hour ':' minute ampm
   { if ($1>11*60) yyerrorf "Hour out of range");
      $$ = $1 + $3 + $4;
   }
   | hour ampm
     {if ($1>11*60) yyerror("Hour out of range");
       $$ = $1 + $2;
      }
   ;
   hour: hour_num
       {if ($1>23) yyerror("Hour out of range");
        $$ = $1 * 60;
       };
   minute: DIGIT DIGIT
        {$$ = $1*10 + $2;
        if ($$> 59) yyerrorf "minute out of range") ,
        };
   ampm: AM {$$ = AM_MINS;}
       |      PM {$$ = PM_MINS;)
       ;
   hour num:  DIGIT {$$ = $1;)
       | DIGIT DIGIT {$$ = $1*10 + $2;}
       ;
Упражнение 8 из раздела "Языки, отражающие специфику предметной области"
   Ответ:
   $_ = shift;
   /"(\d\d?)(am|pm)$/ && doTime($1, 0, $2, 12);
   /"(\d\d?):(\d\d)(am|pm)$/ && doTime($1, $2, $3, 12);
   /"(\d\d?):(\d\d)$/ && doTime($1, $2, 0, 24);
   die "Invalid time $_\n";
   #
   # doTime(hour, min, ampm, maxHour)
   #
   sub doTime($$$$) {
   my ($hour, $min, $offset, $maxHour) = @_;
   die "Invalid hour: $hour" if ($hour>= $maxHour);
   $hour += 12 if ($offset eq "pm")
   print $hour*60 + $min, " minutes past midnight\n";
   exit(0);
   }
Упражнение 9: из раздела "Оценка"
   Ответ: Ответ должен быть изложен, исходя из нескольких допущений:
   • Лента содержит информацию, которую необходимо передать.
   • Известна скорость ходьбы человека.
   • Известно расстояние между компьютерами.
   • Временем, необходимым для переноса информации на ленту и с ленты, можно пренебречь.
   • Потери данных при хранении на ленте примерно равны их потерям при передаче по каналу связи.
Упражнение 10 из раздела "Оценка"
   Ответ: Учитывая допущения ответа 9: Объем информации, содержащейся на стриммерной кассете (4 Гбайт), составляет 32 х 10^9 бит, так что передача эквивалентного объема по каналу со скоростью 1 Мбайт/с заняла бы около 32000 сек. (примерно 9 ч). Если человек движется с постоянной скоростью 3,5 мили в час, то, для того чтобы канал связи превзошел курьера, два компьютера должны располагаться друг от друга на расстоянии не менее 31 мили. Если это расстояние меньше, то победа остается за человеком.
Упражнение 11 из раздела "Обработка текста"
   Ответ: Ответ к данному упражнению составлен на языке Perl.
   my @consts;
   my $name = <>;
   die "Invalid format – missing name" unless defined($name);
   chomp $name;
   # Read in the rest of the file
   while (<>) {
     chomp;
     s/"\s*//; s/\s*$//;
     die "Invalid line: $_" unless /"(\w+)$/;
     push @consts, $_;
   }
   # Now generate the file
   open(HDR, ">$name.h") or die "Can't open $name.h: $!";
   open(SRC, ">$name.c") or die "Can't open $name.c: $!";
   my $uc_name = uc($name);
   print HDR "/* File generated automatically – do not edit */\n";
   print HDR "extern const char *$ {ucjiame)_name[];";
   print HDR "typedef enum {\n"; print HDR join",\n", @consts;
   print HDR "\n) $uc_name;\n\n";
   print SRC "* File generated automatically – do not edit */\n";
   print SRC "const char *$ {uc name}_name[] = {\n \"";
   print SRC join "\",\n \"", @consts;
   print SRC "\"\n};\n";
   close(SRC);
   close(HDR);
   Используя принцип DRY, мы не будет вырезать и вклеивать этот вновь написанный файл в нашу программу. Вместо этого мы «включим» его – данный плоский файл является главным источником этих констант. Поэтому нам понадобится файл сборки для восстановления заголовка при изменении файла. Следующий фрагмент содержится в системе отладки в исходном дереве (имеется на web-сайте).
   etest.c etest.h: etest.inc enumerated.pl
                  perl enumerated.pl etest.inc
 
Упражнение 12 из раздела "Обработка текста"
   Ответ: Вот ответ, написанный на языке Perl.
   my $dir = shift or die "Missing directory" ,
   for my Sfile (glob(u$dir/*.pr)) {
   open(IP, "$file") or die "Opening $file: $!";
   undef $/; # Turn off input record separator -
   my Scontent = <IP>; # read whole file as one string.
   close(IP);
   if (Scontent Гrusestrict/m) {
   rename Sfile, "$file.bak" or die "Renaming $file: $!"; open(OP, ">$file") or die "Creating $file: $!";
   # Put 'use strict' on first line that
   # doesn't start #
   Scontent =" sr(V.#)/\nuse strict;\n\n/m',
   print OP Scontent; close(OP);
   print "Updated $file\n";
   )
   else {
   print "Sfile already strict\n":
   )
   }
Упражнение 13 из раздела "Генераторы исходных текстов"
   Ответ: Решение реализовано на языке Perl. В программе происходит динамическая загрузка модуля для генерации требуемого языка, так что добавление новых языков не представляет труда. Главная программа загружает внутреннюю часть (основанную на параметре командной строки), затем считывает ее входные данные и вызывает подпрограммы генерации текста, основанные на содержимом каждой из строк. Мы особенно не суетимся, если речь идет об обработке ошибок: если что-то не так, узнаем об этом довольно быстро.
   my lang = shift or die "Missing language";
   $lang .= "_cg.pm";
   require <$lang> or die "Couldn't load $lang";
   # Read and parse the file
   my $name;
   while (<>) {
     chomp;
   if (/^ \s*$/)   {CG::blankLine();)
   elsif ((/^ \#(.*)/)   {CG::comment($1);}
   elsif ((/^M\s*(.+)/)   {CG::startMsg($1); $name = $1;}
   elsif ((/^E/)    {CG::endMsg($name);}
   elsif (/^F\s*(\w+)\s+(\w+)$/) {CG::simpleType($1,$2);}
   elsif (/^F\s*(\w+)\s+(\w+)\[(\d+)\]$/) {CG::arrayType $1,$2,$3);}
   else {
   die "Invalid line: $ ";
   }
   }
   Написание языковой серверной части не составит труда: создайте модуль, который реализует шесть точек входа. Вот генератор текста на языке С:
   #!/usr/bin/perl – w
   package CG;
   use strict;
   # Code generator for 'C' (see cg_base.pl)
   sub blankLine()  {print "\n"; }
   sub comment()  {print "/*$_[0] */\n"; }
   sub startMsg()   {print "typedef struct {\n"; }
   sub endMsg()    {print "} $_[0];\n\n"; }
   sub arrayType() {
     my ($name, $type, $size) = @_;
     print " $type $name\[$size];\n":
   }
   sub simpleType{) {
     my ($name, $type) = @_;
     print " $type $пате;\п";
   }
   1;
 
   А вот генератор текста на языке Pascal:
   #!/usr/bin/perl – w
   package CG;
   use strict;
   # Code generator for 'Pascal' (see cg_base.pl)
   sub blankLine() {print "V;)
   sub comment() {print "{$_[0] }\n";)
   sub startMsg()  {print "$_[0] = packed record\n"; }
   sub endMsg()   {print "end;\n\n"; }
   sub arrayType() {
     my ($name, $type, $size) = @_;
     $size--;
     print" $name: array[0…$size] of $type;\n";
   )
   sub simpleType() {
     my ($name, $type) = @_;
     print" $name: $type;\n7 ',
   }
   1;
 
Упражнение 14 из раздела "Проектирование по контракту"
   Ответ: Этот пример на языке Eiffel удачен. Мы требуем передачи непустых данных и гарантируем, что семантика циклического двунаправленного списка будет соблюдена. Это также способствует нахождению сохраненной строки. Поскольку это некий отложенный класс, то действительный класс, его реализующий, допускает использование любого основного механизма по своему усмотрению. Он может использовать указатели или массив, или что-то еще; пока он соблюдает контракт, мы не беспокоимся.
Упражнение 15 из раздела "Проектирование по контракту"
   Ответ: Это неудачно. Математическое действие в индексном выражении (index-1) не будет работать с граничными условиями, подобными первой точке входа. Постусловие предполагает определенную реализацию; контракты должны быть более абстрактными по сравнению с указанным выше.
Упражнение 16 из раздела "Проектирование по контракту"
   Ответ 16: Это удачный контракт, но неудачная реализация. Здесь высовывает свою уродливую голову ошибка типа "Heisenbug" [URL52]. Вероятно, программист допустил опечатку – набрал pop вместо top. Хотя это простой и надуманный пример, весьма трудно диагностировать побочные эффекты в утверждениях (или в любом, самом неожиданном месте в программе).
Упражнение 17 из раздела "Проектирование по контракту"
   Ответ: Мы продемонстрируем функциональные сигнатуры на языке Java, обозначая предусловия и постусловия в соответствии с iContract. Сначала инвариант для класса:
   /**
   * @invariant getSpeed() >0
   * implies isFull() // Не запускать пустое
   * @invariant getSpeed()>=0 &&
   * getSpeed() <10 // Проверка границ
   */
   Затем предусловия и постусловия:
   /**
   * @pre Math.abs(getSpeed() – х) <= 1 // Единственный выбор
   * @рге х>= 0 &&  х > 10  // Проверка границ
   * @post getSpeed() == х    // Проверка скорости
   */
   public void setSpeed (final int x)
   /**
   * @pre !isFull()  // Heзаполнять дважды
   * @post isFull() // Убедитесь, что было выполнено
   */
   void fill()
   /**
   * @pre isFull()    // He очищатьдважды
   * @post !isFull() // Убедиться, что выполнено
   */
   void empty()
 
Упражнение 18 из раздела "Проектирование по контракту"
   Ответ: В этом ряду содержится 21 число. Если вы ответили «20», то допустили так называемую ошибку "поста охраны".
Упражнение 19 из раздела "Программирование утверждений"
   Ответ:
   1. В сентябре 1752 г. было всего лишь 19 дней. Это было сделано с целью синхронизации при переходе с юлианского на григорианский календарь.
   2. Каталог мог быть удален другим процессом, у вас нет прав доступа на его чтение, выражение &sb может быть недопустимым – вы все уловили.
   3. Мы проявили малодушие, не указав типов а и b. Во время перегрузки операторов могло случиться так, что поведение знаков +, =, или != стало непредсказуемым. Кроме того, а и b могут быть псевдонимами одной и той же переменной, так что при втором присвоении произойдет перезапись значения, сохраненного во время первого.
   4. В неевклидовой геометрии сумма углов треугольника не будет составлять 180°. Подумайте о треугольнике, отображаемом на поверхности сферы.
   5. Минуты, приходящиеся на високосный год, могут состоять из 61 или 62 секунд.
   6. Переполнение может оставить результат операции а+1 отрицательным (это также может произойти в языках С и С++).
Упражнение 20 из раздела "Программирование утверждений"
   Ответ: Мы решили реализовать очень простой класс с единственным статическим методом TEST, который выводит на печать сообщение и след стека, если переданный параметр condition является ложным.
   package com.pragprog.util;
   import java.lang.System; //для exit()
   import java.lang.Thread; //для dumpStack()
   public class Assert {
   /** Write a message, print a stack trace and exit if
      * our parameter is false.
      */
   public static void TEST(boolean condition) {
     if (Icondition) {
       System.out.println("==Assertion Failed==");
       Thread.dumpStack();
        System.exit(1);
      }
   }
   // Testbed. If our argument is 'okay', try an assertion that
   // succeeds, if 'fail' try one that fails
   public static final void main(String args[]) {
   if (args[0].compareTo("okay") == 0) {
      TEST(1 == 1);
   }
   else if (args[0].compareTo("fail") == 0) {
      TEST(1 == 2);
   }
   else {
     throw new RuntimeException("Bad argument") ,
     }
    }
   }
Упражнение 21 из раздела "Случаи, когда используются исключения"
   Ответ: Нехватка памяти является исключительным состоянием, поэтому мы полагаем, что в случае (1) должно возбуждаться исключение.
   Невозможность отыскания точки входа – вполне нормальная ситуация. Приложение, которое вызывает наш класс-набор, может написать программу, которая проверяет наличие точки входа, перед тем как добавить потенциальный дубликат. Мы полагаем, что в случае (2) нужно просто осуществить возврат ошибки.