Форум программистов, компьютерный форум, киберфорум
Lisp
Войти
Регистрация
Восстановить пароль
Блоги Сообщество Поиск Заказать работу  
 
Рейтинг 4.82/11: Рейтинг темы: голосов - 11, средняя оценка - 4.82
 Аватар для Кровавая Рука
28 / 4 / 1
Регистрация: 08.09.2012
Сообщений: 76

Почему на выходе nil?

23.09.2013, 20:12. Показов 2047. Ответов 7
Метки нет (Все метки)

Студворк — интернет-сервис помощи студентам
Хочу сделать ввод в список с клавиатуры до того, как будет введён nil. Подскажите пожалуйста, что я недопонимаю и почему в итоге список пустой?

Lisp
1
2
3
4
5
6
7
8
9
 
(progn 
  (setq li (list ())) 
  (print "введите числа, для окончания ввода: nil") 
  (do ((i))
    ((eq (read) nil) li)
    (cons '(read) li)
  )
)
0
Лучшие ответы (1)
Programming
Эксперт
39485 / 9562 / 3019
Регистрация: 12.04.2006
Сообщений: 41,671
Блог
23.09.2013, 20:12
Ответы с готовыми решениями:

Enum - почему на выходе получаю 0, а не 1 ?
начал писать на шарпе, вот появился вопрос есть перечисление enum days { one=1, two, ...

Почему на выходе всегда NaN
import static java.lang.Math.pow; import static java.lang.Math.sin; import static java.lang.Math.cos; import static...

Почему запрос postgres выдает на выходе массив?
Добрый день, Есть 3 связанных последовательно таблички: categories ( id, name) products ( id, category_id, name ) tarifs (id,...

7
4528 / 3522 / 358
Регистрация: 12.03.2013
Сообщений: 6,038
23.09.2013, 20:18
Поставьте push вместо cons и продолжайте удивляться.

Цитата Сообщение от Кровавая Рука Посмотреть сообщение
Подскажите пожалуйста, что я недопонимаю
По-моему, всё.
0
12 / 12 / 2
Регистрация: 02.09.2012
Сообщений: 40
23.09.2013, 23:01
Лучший ответ Сообщение было отмечено как решение

Решение

Так, на скорую руку накарябал.. Функция будет возвращать список, полученный путём ввода с клавиатуры. Если его нужно будет вывести в консоль, например, просто добавишь туда format или ещё чего-нибудь.
Code
1
2
3
4
5
6
7
(defun f ()
  (format t "Введите числа, для окончания ввода: nil~%")
  (let ((lst nil))
    (do* ((elt (read) (read)))
         ((eql elt nil))
      (push elt lst))
    lst))
З.Ы: Одинарная кавычка перед списком запрещает его работу как функции (не умею грамотно объяснять, короче))) .. когда ты пишешь '(read), то функция read тут даже и не вызывается вообще.. тут получается просто список из одного элемента. Это во-первых.
Во-вторых вот тут ((eq (read) nil) li) ты теряешь то, что вводится с клавиатуры, потому что результат работы read ничему не присваивается.
В-третьих, вообще какой-то странный do у тебя получился.. да и весь код в целом)) Ты на чём писал до того как решил взяться за Lisp? Скобки так никто не оставляет в конце.. все скобки ставятся подряд на одной строке. И не надо ля-ля про то, что так читать удобней)))
3
 Аватар для Кровавая Рука
28 / 4 / 1
Регистрация: 08.09.2012
Сообщений: 76
24.09.2013, 10:12  [ТС]
Цитата Сообщение от helter Посмотреть сообщение
По-моему, всё.
Это ж самоочевидно, кэп.. иначе не спрашивал бы..

Цитата Сообщение от filimonix Посмотреть сообщение
Ты на чём писал до того как решил взяться за Lisp?
pascal, c++, assembler, исполнитель муравей.

спасибо)
0
4528 / 3522 / 358
Регистрация: 12.03.2013
Сообщений: 6,038
24.09.2013, 10:30
Переменные нужно объявлять, и желательно, не с помощью setq.

Обычно используют переменные локальные лексические и глобальные динамические (если глобальные на самом деле нужны; в данном случае - нет). Лексические переменные чаще всего объявляются через let, глобальные - через defvar и defparameter.

Для построения списков с нуля часто используется идиома push/nreverse. Функция push добавляет элемент в голову списка и присваивает получившийся список переменной. То есть эта функция имеет побочный эффект (изменение байндинга), что плохо с точки зрения функционального программирования, но нормально с точки зрения CL. Функция nreverse разворачивает список, модифицируя его конс-ячейки. Ею можно пользоваться, только если понимаешь, что делаешь. При построении списка внутри функции с нуля это, как правило, безопасно: конс-ячейки построены у нас на глазах, больше никем не используются и могут быть модифицированы.

Кстати, надо учитывать, что (read) и (read) - разные вещи (то есть возвращают разные значения). Если однажды полученное значение не используется, оно, естественно, потеряется. Таковы они, побочные эффекты.
1
Супер-модератор
Эксперт функциональных языков программированияЭксперт Python
 Аватар для Catstail
38173 / 21108 / 4307
Регистрация: 12.02.2012
Сообщений: 34,705
Записей в блоге: 14
24.09.2013, 11:12
Почему в итоге список пустой?

Давай взглянем на твой код:

Lisp
1
2
3
4
5
6
7
8
9
10
11
(progn 
  (setq li (list ()))  ;; переменая li будет глобальной... 
                         ;; это не слишком хорошо, но дело не в том.
  (print "введите числа, для окончания ввода: nil") 
  (do ((i))
    ((eq (read) nil) li) ;; Читаешь с консоли элемент данных и сравниваешь с Nil
                            ;; предположим, я ввел "abc" условие не сработает, а "abc" пропадет
    (cons '(read) li)   ;; здесь ты приделываешь к li первым элементом список (read)
                            ;; (из-за апострофа). Кстати, результат тоже пропадает, т.к. ничему не присваивается
  )
)
Я бы поступил так:

Lisp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
(progn 
 (let ((li nil) (v nil))
     (loop 
        (princ "Enter number. End - nil ")
        (setq v (read))
        (when (null v) (return (reverse li)))
        (push v li))))
 
Enter number. End - nil 1
Enter number. End - nil 2
Enter number. End - nil 3
Enter number. End - nil 4
Enter number. End - nil 5
Enter number. End - nil nil
(1 2 3 4 5)
А лучше - вот так (полезно отвыкать от явного присвоения типа setq, setf и.т.п.)

Lisp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
(progn 
 (let ((li nil) (v nil))
     (loop (princ "Enter number. End - nil ")
        (push (read) v) ;; результат ввода - в стек v
        (when (null (car v)) (return (reverse li))) ;; если вершина стека - Nil - выход
        (push (pop v) li)))) ;; вершину стека v -> в стек li
 
Enter number. End - nil 1
Enter number. End - nil 2
Enter number. End - nil 3
Enter number. End - nil 4
Enter number. End - nil 5
Enter number. End - nil 6
Enter number. End - nil 7
Enter number. End - nil nil
 
(1 2 3 4 5 6 7)
1
Супер-модератор
Эксперт функциональных языков программированияЭксперт Python
 Аватар для Catstail
38173 / 21108 / 4307
Регистрация: 12.02.2012
Сообщений: 34,705
Записей в блоге: 14
24.09.2013, 11:21
А можно - гораздо проще. Например, вот так (HomeLisp):

Lisp
1
2
3
4
5
6
7
(defun get-num-list ()
  (input (strCat "(" (ask "Введите несколько чисел через пробел") ")")))
 
==> get-num-list
(get-num-list)
 
==> (1 2 3 4 5 6)
Миниатюры
Почему на выходе nil?  
2
 Аватар для Кровавая Рука
28 / 4 / 1
Регистрация: 08.09.2012
Сообщений: 76
24.09.2013, 21:21  [ТС]
акей. тут вроде понял, спасибо.
пойду в следующую тему..
0
Надоела реклама? Зарегистрируйтесь и она исчезнет полностью.
inter-admin
Эксперт
29715 / 6470 / 2152
Регистрация: 06.03.2009
Сообщений: 28,500
Блог
24.09.2013, 21:21
Помогаю со студенческими работами здесь

Почему работает программа при выходе за пределы массива?
Добрый день, возникла проблема. Я задал одномерный массив из 10 элементов, однако, когда в я цикле пытаюсь присвоить значения большему...

Глобальный вектор. Почему на выходе этот массив пуст?
юзаю либу вектор. почему на выходе этот массив пуст? std::vector<int> qw;//global int main() { int i=1; qw.push_back(i); ...

На выходе печатается одна строка с дополнительным третьим классом.Почему?
Вариант 1 – на выходе печатается одна строка с дополнительным третьим классом. class Trt{ int passengers; int price; } ...

Почему при выходе из цикла while переменная счетчик увеличивается на единицу?
Есть такой код: #include <iostream> using namespace std; int main() { int x = 30; cout<<"Внешнее значение:...

Почему массив путей файлов не обнуляется при выходе из программы
Foreach считывает файлы по циклу в папке, но т.к. их бывает много то сделана сортировка по дате. Суть проблемы следующая: файлов много, и...


Искать еще темы с ответами

Или воспользуйтесь поиском по форуму:
8
Ответ Создать тему
Новые блоги и статьи
SDL3 для Web (WebAssembly): Основы отладки веб-приложений на SDL3 по USB и Wi-Fi, запущенных в браузере мобильных устройств
8Observer8 07.02.2026
Содержание блога Браузер Chrome имеет средства для отладки мобильных веб-приложений по USB. В этой пошаговой инструкции ограничимся работой с консолью. Вывод в консоль - это часть процесса. . .
SDL3 для Web (WebAssembly): Обработчик клика мыши в браузере ПК и касания экрана в браузере на мобильном устройстве
8Observer8 02.02.2026
Содержание блога Для начала пошагово создадим рабочий пример для подготовки к экспериментам в браузере ПК и в браузере мобильного устройства. Потом напишем обработчик клика мыши и обработчик. . .
Философия технологии
iceja 01.02.2026
На мой взгляд у человека в технических проектах остается роль генерального директора. Все остальное нейронки делают уже лучше человека. Они не могут нести предпринимательские риски, не могут. . .
SDL3 для Web (WebAssembly): Вывод текста со шрифтом TTF с помощью SDL3_ttf
8Observer8 01.02.2026
Содержание блога В этой пошаговой инструкции создадим с нуля веб-приложение, которое выводит текст в окне браузера. Запустим на Android на локальном сервере. Загрузим Release на бесплатный. . .
SDL3 для Web (WebAssembly): Сборка C/C++ проекта из консоли
8Observer8 30.01.2026
Содержание блога Если вы откроете примеры для начинающих на официальном репозитории SDL3 в папке: examples, то вы увидите, что все примеры используют следующие четыре обязательные функции, а. . .
SDL3 для Web (WebAssembly): Установка Emscripten SDK (emsdk) и CMake для сборки C и C++ приложений в Wasm
8Observer8 30.01.2026
Содержание блога Для того чтобы скачать Emscripten SDK (emsdk) необходимо сначало скачать и уставить Git: Install for Windows. Следуйте стандартной процедуре установки Git через установщик. . . .
SDL3 для Android: Подключение Box2D v3, физика и отрисовка коллайдеров
8Observer8 29.01.2026
Содержание блога Box2D - это библиотека для 2D физики для анимаций и игр. С её помощью можно определять были ли коллизии между конкретными объектами. Версия v3 была полностью переписана на Си, в. . .
Инструменты COM: Сохранение данный из VARIANT в файл и загрузка из файла в VARIANT
bedvit 28.01.2026
Сохранение базовых типов COM и массивов (одномерных или двухмерных) любой вложенности (деревья) в файл, с возможностью выбора алгоритмов сжатия и шифрования. Часть библиотеки BedvitCOM Использованы. . .
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin
Copyright ©2000 - 2026, CyberForum.ru