|
|
|||||||||||||||||||||||||||||
|
Программирование на языке AutoLISPИсточник: intuit intuit
Программирование - это создание оптимального алгоритма и кода для решения поставленной задачи. В процедурах языка AutoLISP основными структурными данными являются списки. Процедуры сохраняются в файлах. Лекция посвящена особенностям программирования на языке AutoLISP. ПеременныеПеременные - это символьные имена, которыми можно оперировать в программе. Переменным можно присваивать значения непосредственно в командной строке:
При использовании переменной в параметрах команды перед ней нужно поставить восклицательный знак (!pt) . Задание 4.1 Создать переменные pt1, pt2 координат двух точек и с их помощью построить отрезок. Создать строковую переменную color с обозначением цвета "Red" и закрасить отрезок. Приведем протокол операций выполнения этого задания. Command: (setq pt1 "2,2")↵ "2,2" Command: (setq pt2 "12,2")↵ "12,2" Command: (setq color "Red")↵ "Red" Command: _line Specify first point: !pt1↵ "2,2" Specify next point or [Undo]: !pt2↵ "12,2" Specify next point or [Undo]: ↵*Cancel* Command: chprop↵ Select objects: 1 found Select objects: ↵ Enter property to change [Color/LAyer/LType/ltScale/LWeight/Thickness/Material]: c↵ New color [Truecolor/COlorbook] <BYLAYER>: !color↵ "Red" По применимости в процедурах переменные разделяются на два типа: локальные и глобальные. Локальные объявляются после косой черты. Они могут принимать определенные значения только во время выполнения той процедуры, в которой они объявлены. После завершения процедуры ячейки памяти, занимаемые этими переменными, освобождаются. Глобальные переменные доступны всем функциям AutoLISP, которые загружены в чертеж. Их значения доступны в проекте и после завершения той программы, в которой они объявлены. Любая переменная, которая специально не объявлена локальной, является глобальной. В глобальной переменной можно хранить константы, промежуточные значения вычислений, результирующие данные. Однако трудно уследить, какое значение ей присвоено в данной функции и не изменено ли оно другой функцией. Поэтому следует избегать введения глобальных переменных. Рекомендуется имена глобальных переменных отмечать справа и слева звездочками: *Global1* и документировать их. Задание 4.2 Создать и протестировать локальные и глобальные переменные.
Переменные характеризуются временем жизни и областью видимости. Локальные переменные определяются внутри функции путем задания слеша с пробелом перед ними. Время жизни локальной переменной совпадает с временем выполнения процедуры. Область видимости локальной переменной ограничена только процедурой (функцией), в которой она задана. Вне этой процедуры текущие значения локальной переменной недоступны. При объявлении глобальных переменных слеш не ставится. Время жизни глобальной переменной совпадает с временем открытия чертежа. Областью ее видимости является весь проект.
Работа с системными переменнымиСреда AutoCAD содержит много системных переменных, которые управляют процессом черчения. Текущие значения системных переменных можно читать и изменять. Не путайте понятия переменной и системной переменной. Переменная - это величина, которая создается для использования в процедуре. Системная переменная - это параметр AutoCAD, который служит для настройки режима работы с чертежом. Для изменения текущего значения системной переменной нужно набрать ее имя в командной строке. Пример 1 Системная переменная FILEDIA включает (значение равно 1) и выключает (значение равно 0) вывод на экран диалоговых окон выбора файлов по команде File - Open. Проверьте это. Затем верните старое значение: Command: filedia Enter new value for FILEDIA <1>: 0 Пример 2 При запуске функции COMMAND процедур AutoLISP все сообщения команд AutoCAD отображались в командной строке. Системная переменная CMDECHOопределяет, будут ли отображаться в командной строке приглашения и параметры при работе функции COMMAND. По умолчанию режим отображения включен - текущее значение системной переменной равно 1. Если изменить это значение на 0, то режим отображения отключится. Для работы с системными переменными в среде AutoLISP используются операторы SETVAR и GETVAR. Оператор SETVAR (SET VARiable - установить переменную) изменяет текущее значение системной переменной. Новое значение помещается после имени переменной, заключенной в двойные кавычки: (setvar "cmdecho" 0) . Оператор GETVAR (GET VARiable - получить переменную) возвращает значение системной переменной. Его можно присвоить другой переменной или системной переменной. Часто так возвращают системной переменной старое значение, которое изменялось в процессе работы процедуры. Синтаксис операции: (getvar "cmdecho"). Задание 4.3 Работа в среде AutoLISP с системными переменными Создайте новый чертеж. Откройте редактор Visul LISP. Откройте новый файл и наберите в нем следующие процедуры: (defun chred (selected_object) (command "_chprop" selected_object "" "_color" "red" "") ) (defun c:chgcolor (/ selected old_cmdecho) ;описываем две локальные переменные (setq old_cmdecho (getvar "cmdecho")) ;присваиваем old_cmdecho значение 1 (setvar "cmdecho" 0) ;присваиваем cmdecho новое значение 0 (terpri) ;в командной строке выводится начало пустой строки (setq selected (entsel "Выделите объект для изменения цвета на красный:")) (chred selected) (setvar "cmdecho" old_cmdecho)) Процедура работает следующим образом.
После составления процедуры выполните следующие действия:
Функции AutoLISPОпределение функции всегда начинается с оператора DEFUN. При этом можно выделить три типа функций:
Задание 4.4 Создание файла инициализации Пусть вы хотите увидеть рамку с приветствием на экране при открытии каждого чертежа. Создадим процедуру и сохраним ее в файле AutoCAD 2010\Support\acaddoc.lsp: (defun s::startup () (command "rectang" "_width" "5" "0,0" "50,35") (command "_chprop" "_last" "" "_color" "t" "255,0,0" "") (command "_text" "10,15" "" "Hello!") (command "_chprop" "_last" "" "_color" "t" "255,0,0" "") (command "_zoom" "_extents" "") ) Откройте новый чертеж. Возможно, что придется выполнить отладку размеров текста и рамки для него. Легче изменить размеры рамки, оставив текущий стиль текста без изменений. Попробуйте также поэкспериментировать с цветом. В среде AutoLISP можно создавать функции с аргументами. Аргумент передает свое значение функции. Функция использует значение аргумента в процессе отработки операций. Аргументы располагают в круглых скобках до косой черты (слеша). Если у функции нет локальных переменных, то слеш не ставится. Пример объявления функции с одним аргументом приведен в задании 4.3: (defun chred (selected_object) … Функцию с аргументом можно вызвать и выполнить с помощью оператора вызова функции. В операторе задается имя функции и фактические переменные, которые заменяют в теле функции формальные аргументы. В задании 4.3 формальным аргументом является переменная selected_object. Фактическим параметром, передаваемым функции chred в качестве аргумента является переменная selected. При каждом обращении к функцииchred из какой-либо процедуры необходимо задавать значение аргумента. В качестве аргумента можно использовать:
Вернемся к заданию 4.3 и опишем работу функций. В первой строке определена функция chred и ее аргумент selected_object. При вводе в командную строку команды chgcolor начинает выполняться функция c: chgcolor. В предпоследнем операторе функции - (chred selected) - управление передается функции chred с аргументом. Значение переменной selected извлекается из предыдущего этапа как результат операции entsel. Переменная selected - это аргумент, который передается функции chred. Функция подставляет при осуществлении своих действий вместо аргумента selected_object значение переменной selected. Прорабатывая свой код, функция chred применяет команду chprop для изменения цвета объекта на красный.
Функции условия, циклы и работа со спискамиСинтаксис условной функции имеет стандартный вид: (if условное_выражение Если_True Если_False) Значение Если_False можно опустить. Тогда интерпретатор исполнит выражение Если_True, когда условное выражение истинно. Если же условное выражение ложно, то управление перейдет к следующей функции. Задание 4.5 Работа с функцией IF (defun c:compare (/ num) (setq num (getint "\nВведите число:")) (if (< num 3) (princ "\nВведенное число меньше 3") (if (= num 3) (princ "\nВведенное число равно 3") (princ "\nВведенное число больше 3"))) (princ)) Префикс \n аналогичен функции terpri: каждое новое приглашение будет выводиться с новой строки. Выполните действия и оцените результат:
Функции цикла обеспечивают выполнение определенных функций несколько раз подряд. При использовании циклической функции WHILE количество повторов неопределенно. Цикл выполняется, пока истинно условное выражение. Иначе управление передается следующей функции: (WHILE условное_выражение тело_цикла) Чаще нужно выполнить цикл определенное число раз (итераций). Для этого создается специальная переменная - счетчик. Условием выхода из цикла является заданное число итераций. Пример (defun c:cycle (/ counter) (setq counter 1) (while (< counter 6) (princ "Number ") (princ Counter)) (terpri) (setq counter (+1 counter)) )) Счетчиком является переменная counter. В начале процедуры функция cycle присваивает счетчику counter значение, равное 1. Затем операторwhile печатает слова Number Counter и увеличивает значение счетчика на единицу. Условием работы оператора while является то, что значение счетчика меньше 6. В результате программа выдаст следующий список: Command: cycle Number 1 Number 2 Number 3 Number 4 Number 5 6 В последней строке напечатано возвращаемое значение функции. Обычно им является значение последнего вычисленного в теле функции оператора. Число 6 - это значение переменной, над которой выполнял действие последний оператор функции. При зацикливании можно прервать выполнение процедуры, нажав ESC, CTRL+BREAK или выбрав команду в меню Visual LISP: Debug - Abort Evaluation (Отладка - Прервать вычисления). Если необходимо объединить несколько операторов в один блок, то используется функция PROGN. Все, что включено в конструкцию PROGN, рассматривается интерпретатором как один оператор. (defun c:compare1 (/ num) (setq num (getint "\nВведите число")) (if (< num 3) (princ "\nВведенное число меньше 3") (if (= num 3) (progn (princ "\nВведенное число равно 3") (terpri) (princ "\nЭто и есть результат эксперимента") ) (princ "\nВведенное число больше 3") )) (princ)) Списки - это основные структуры данных, используемые в процедурах на языке AutoLISP. В программе AutoCADданные об объекте формируются в виде списка, который содержит списки меньшего размера. В AutoLISP список заключается в круглые скобки. Нумерация элементов в списке начинается с нуля. Примером списка является задание координат точки в прямоугольной системе координат X,Y,Z: (1.2 3.4 5.6) . Стандартной операцией во всех языках является извлечение из списка одного или нескольких элементов. Основные функции AutoLISP извлечения из списка приведены в таблице:
Функции NTH нужно задать два параметра: номер элемента и имя списка. Список создается функцией LIST. Если элементы списка являются константами, а не переменными, то для его создания можно использовать функциюQUOTE. Вместо имени QUOTE часто ставят одинарную кавычку. Функции для работы со списками описаны в разделе AutoLISP Developer's Guide - Appendixes - Basic Functions - List Manipulation Functionsсправочной системы AutoCAD. Пример (defun c:funlist (/ corner) (setq corner (list 1.2 3.4 5.6)) (nth 0 corner) ) При запуске этой процедуры функция возвращает значение нулевого члена списка.
Оформление процедурыПри создании процедуры особое внимание следует обращать на обработку ошибок и комментарии. Например, в предыдущем задании требовалось выделить текстовый объект. Если пользователь выделил не текст, то должно быть выдано сообщение: "Выделите текст". Чаще всего такие проверки осуществляют с помощью оператора IF. Для выполнения оператора IF требуется проверка логического выражения на истинность. Если пользователь не выделит объект нужного типа, то логическое выражение примет значение nil. В эту часть условного оператора и нужно поместить сообщение об ошибке. Процедура должна начинаться с заголовка-комментария. Здесь должно быть описание функционального назначения процедуры, дата составления, особенности применения. Это поможет другим пользователям Вашей процедуры сразу оценить возможности ее применения в своих задачах. Кроме того, как правило, через несколько месяцев и составителю трудно вспомнить все те мелкие задачи, на которые сегодня потрачено немало труда. Поэтому комментарии желательно помещать и в тело процедуры. Не надо жалеть времени на комментарии. Среда AutoLISP поддерживает несколько типов комментариев, различающихся по их выравниванию относительно левого края рабочей области. ;;; - три точки с запятой. Такой комментарий выравнивается по левому краю. Он используется в начале процедуры для описания ее назначения и функционирования. ;; - две точки с запятой. Этот комментарий сдвигается в процессе форматирования на уровень следующих скобок. Такой тип комментариев используется для пояснения следующих за ним одной или нескольких строк кода. ; - точка с запятой. Комментарий, помеченный таким символом, сдвигается вправо на 40 символов. Это значение можно изменить в диалоговом окнеFormat options, открываемом по команде Tools - Environment Options -Visual LISP Format Options. Данный тип комментария обычно используется для текущих пояснений в правой части кода. Записи хорошо выделяются на фоне кода. ;/ /; - внутренний комментарий. Его можно поместить внутри любой строки кода. Текст комментария ограничивается точкой с запятой и вертикальной чертой. ;_ - комментарий в конце строки. Используется в конце любой строки. Чаще всего в нем объясняют, какой оператор закрывает данная скобка. Это особенно касается условных операторов, когда закрывающая скобка расположена через несколько строк после открывающей скобки.
Отладка кодаСпециальное окно Visual LISP Console заменяет пользователю в среде Visual LISP командную строку системы AutoCAD. С консоли вводят выражения на языке AutoLISP, загружают и выполняют функции. На консоль выводятся сообщения Visual LISP и AutoCAD. В окне консоли в качестве приглашения пользователя к вводу информации применяются символы "_$". Например, если пользователь загрузил файлval.lsp с одноименной функцией, то для выполнения программы нужно набрать в консоли (с:val) . Visual LISP выполнит программу и выведет результат на консоль. С консоли можно читать значения глобальных переменных. Для этого нужно в ответ на приглашение ввести в строке консоли имя переменной. Visual LISP выведет ее текущее значение. Другой способ узнать значение переменной - двойным щелчком выделить ее в тексте программы и нажать кнопку"Load Selection". При отладке программы удобно следить за значениями выбранных глобальных переменных. Для этого используются пункты Inspect, Symbol Service, Watch Window. Вызов соответствующих диалоговых окон показывает текущие значения выбранных переменных или выражений. Особенно часто встречаются ошибки в типах аргументов, передаваемых функциям. Такие сообщения выводятся в окно консоли, а при работе в окнеAutoCAD - в командной строке. Все сообщения о неправильности типа аргумента выглядят следующим образом: Bad argument type: (ключевое слово - нужный тип): фактическое значение. В таблице приведен список ключевых слов и соответствующих им типов данных.
Отладка (debugging) является необходимым этапом создания программы. В процессе отладки нужно найти и устранить ошибки. Visual LISPпредоставляет свои средства отладки. Для начала загрузите функцию, заведомо содержащую ошибку. Введите ее в редактор Visual LISP и загрузите. ;;;При попытке выполнения этой функции возникает ошибка, так как функция ;;;strcat предназначена для объединения строк (defun Error-Prone-Code () (strcat ""Эта функция никогда не напечатает число:" 1) ) Выберите команду Debug - Break on Error - тогда редактор автоматически перейдет к строке кода с ошибкой. Проверим работу отладчика. Введите в окне Console выражение (Error-Prone-Code) ↵ Visual LISP возвратит сообщение об ошибке: ; error: bad argument type: stringp 1 Получив сообщение, щелкните на кнопке Last Break (последнее прерывание). Вы перейдете в строку кода, которая содержит ошибку. Задание 4.6 Поиск ошибки В новом чертеже в редакторе Visual LISP введите следующий код: ;;;Функция add-3-numbers выполняется правильно только в том случае, ;;;когда все три ее параметра являются числами (defun add-3-numbers (num1 num2 num3) (+ num1 num2 num3) ) Загрузите содержимое активного окна. В окне Console введите (add-3-numbers 1 2 3) ↵ Среда вернет 6. В окне Console введите (add-3-numbers 1 "a" 3) ↵ Среда вернет сообщение об ошибке: error: bad argument type: numberp: "a" . Выберите команду Debug - Break on Error. Щелкните на кнопке Last Break панели инструментов Debug. Вы автоматически перейдете в строку, где найдена ошибка. Чтобы продолжить работу в режиме Break on Error щелкните на кнопке Reset панели инструментов Debug. Иногда в режиме Last Break выделенной оказывается слишком большая часть кода, что затрудняет поиск ошибки. В этом случае всю программу разделяют на небольшие фрагменты точками прерывания. В этих точках приостанавливается выполнение программы - AutoLISP ожидает дальнейших инструкций от пользователя. Чтобы установить точку прерывания, установите курсор справа от закрывающей скобки и нажмите клавишу F9. Строка будет отмечена красной меткой. После разметки необходимо повторно загрузить код, чтобы сообщить Visual LISP об их появлении. Код программы будет выполняться до первой точки прерывания. При разметке программы становятся доступны кнопки панели Debug. Три первые кнопки этой панели предоставляют три разных способа перемещения в пределах кода:
Чтобы удалить точки прерывания, выберите команду Debug - Breakpoints Window. Активизируется диалоговое окно для просмотра и удаления точек прерывания. Упражнения
;;; to the console in a more human readable format. (defun Vertlist (aList) ;; this will print every item in the list (princ (chr 40)) ; an opening parenthesis (princ (car aList)) ; the first item in the list (foreach item (cdr aList) ; the rest of the list except the beginning (terpri) ; place a new line (princ " ") ; print a space (princ item 1) ; and print the item in the list ) (terpri) (princ (chr 41)) ; closing parenthesis (terpri) ; a new line T ; returns T ) Эта процедура создает вертикальный список чисел. Функция Foreach осуществляет переход от одного элемента списка к другому. Выберите команду Debug - Break on Error. Введите в окне Console следующую строку: (Vertlist '(1 2 3 4 5)) Интерпретатор возвратит Вам сообщение об ошибке. Щелкните по кнопке Last Break, чтобы перейти к строке кода, содержащей ошибку.
Ключевые терминыПеременные - символьные имена, которыми можно оперировать в программе. Системные переменные- параметр AutoCAD, который служит для настройки режима работы с чертежом. Файлы инициализации - lsp-функции в файлах acad.lsp и acaddoc.lsp. Списки - набор чисел, литералов и точечных пар. Консоль - специальное окно отладки в среде Visual LISP.
Краткие итогиПроцедура представляет собой команды для выполнения чертежных операций. Составление и отладка процедуры производятся в среде Visual Lisp. Структурными единицами процедуры являются функции, переменные и константы. Отладку процедуры удобно проводить в консоли среды Visual LISP. Ссылки по теме
|
|
|