![]() |
1. Заголовок темы должен быть информативным. В противном случае тема удаляется ...
2. Все тексты программ должны помещаться в теги [code=pas] ... [/code].
3. Прежде чем задавать вопрос, см. "FAQ", если там не нашли ответа, воспользуйтесь ПОИСКОМ, возможно такую задачу уже решали!
4. Не предлагайте свои решения на других языках, кроме Паскаля (исключение - только с согласия модератора).
5. НЕ используйте форум для личного общения, все что не относится к обсуждению темы - на PM!
6. Одна тема - один вопрос (задача)
7. Проверяйте программы перед тем, как разместить их на форуме!!!
8. Спрашивайте и отвечайте четко и по существу!!!
![]() ![]() |
![]() |
volvo |
![]()
Сообщение
#1
|
Гость ![]() |
Итак, в этой теме я бы хотел поговорить об объектах, то есть об ООП (Объектно-Ориентированном Программировании). Я не буду опять описывать всю теорию, называть 3 основных принципа ООП, и т.д., это все можно прочитать вот тут: ООП. Объектно-ориентированное программирование , да и в любой книжке по языку программирования. Любому языку программирования.
Здесь я бы хотел поговорить о другом. О решении конкретной задачи с использованием объектов. Причем хотелось бы, чтобы это был не монолог, а диалог. Если вам (а я обращаюсь не только к тем, кто начинает программировать вообще, как выяснилось, даже люди, программирующие достаточно давно, не считают ООП "своей стезёй", эта тема и для них тоже) интересно - я буду продолжать. Если неинтересно - скажите, я продолжать не буду. Я уже когда-то начинал подобную тему, но особого отклика это не получило. Задавайте вопросы, не бойтесь, что они покажутся странными, простыми и т.д. Здесь важно разобраться во всех мелочах. Для начала я бы хотел взять вот такую задачу (это - реальное условие, в которое я внес небольшие изменения): "Написать программу (работающую в диалоговом режиме), которая осуществляет взаимодействие пользователя с одной из трех ДСД - динамических структур данных - Список, Стек, Бинарное дерево, и позволяет: 1) добавлять элементы в структуру 2) удалять элементы из структуры 3) отображать текущее содержимое выбранной структуры данных на экране". Ну, для начала, "а почему именно ООП?" спросите вы. И отчасти будете правы, поскольку эту программу (как и все остальные, впрочем) можно написать, используя процедурное программирование. Отвечаю ![]() 1) Если б я хотел, чтоб это было написано без ООП, я бы не начинал эту тему (или не включил бы сюда эту задачу), в оригинальном задании так и было сказано: "... объектно-ориентированную программу..." 2) решить задачу с использованием только процедур/функций, конечно, можно, но такое решение будет а) более объемным, чем с использованием объектов; б) менее расширяемым Есть еще третья причина по которой я бы советовал сделать "это" с использованием объектов: в результате вы получите типы "Стек", "Список", "Дерево", которые будут максимально независимы от окружения, от остальной части программы, и их можно будет с успехом применять в других проектах. Итак, задача поставлена, можно начинать. Для начала - о главном, о том, с чего начинается проектирование программы: о ее структуре, о том, какие объектные типы (и как именно) будут описаны в программе. Так называемые "заглушки", дающие представление об общей структуре, но пока не касающиеся реализации... Многие из начинающих быстро сделают так: typeОткуда, собственно, первый вопрос: А хорошо ли делать так, как я показал выше? Какие у этого способа вы видите недостатки, какие преимущества? Как бы вы посоветовали мне сделать это по-другому? (Учтите, написание хорошей программы - это сложный процесс, иногда приходится переписывать все почти с нуля, если сразу ошибся при проектировании, поэтому важно с самого начала спроектировать правильно, и только потом приступать к реализации, а не так, как делают очень многие - сначала нарисуют окошечки-рюшечки, а потом туда пытаются втиснуть, собственно, основную часть задания...) P.S. Если это задание вам кажется неподходящим, предлагайте свои варианты, можете взять любое реальное задание с форума со ссылкой на него (только не надо давать здесь свое задание, в надежде, что вам тут его решат полностью. Может, и решат, но времени это займет достаточно много ![]() P.P.S. Свои ответы скрывайте тегами [SPОILER][/SPОILER] (все буквы в названии тегов должны быть латинские), чтоб другие тоже могли подумать самостоятельно, не видя вашего ответа... |
Archon |
![]()
Сообщение
#2
|
![]() Профи ![]() ![]() ![]() ![]() Группа: Пользователи Сообщений: 618 Пол: Мужской Репутация: ![]() ![]() ![]() |
О, хорошая тема. Советы от гуру =)
Спойлер (Показать/Скрыть)
-------------------- Close the World...txeN eht nepO
|
volvo |
![]()
Сообщение
#3
|
Гость ![]() |
Archon, можешь показать (псевдо)кодом первый и второй путь, и что именно тебе мешает в первом случае использовать полиморфизм, а во втором - повторно использовать код? Я например таких проблем не вижу...
Цитата Возможно, имеет смысл... Возможно, я поэтому и спросил, какие недостатки есть у вышеприведенной схемы, чего нельзя (или сложно) сделать с её помощью... |
Archon |
![]()
Сообщение
#4
|
![]() Профи ![]() ![]() ![]() ![]() Группа: Пользователи Сообщений: 618 Пол: Мужской Репутация: ![]() ![]() ![]() |
Вот что я имел ввиду:
![]() В первом случае непонятно, как написать функцию, способную принимать стек (к примеру) независимо от его реализации. Во втором случае реализация пишется отдельно для стека, очереди и списка. Спойлер (Показать/Скрыть)
-------------------- Close the World...txeN eht nepO
|
volvo |
![]()
Сообщение
#5
|
Гость ![]() |
Цитата Во втором случае реализация пишется отдельно для стека, очереди и списка. Естественно, если ты хочешь написать ее отдельно (причем с нуля) для очереди и для стека - то ты и будешь ее писать с нуля. Я в таких случаях делаю так:Спойлер (Показать/Скрыть)
P.S. Спойлером - чтоб не сбивать своим решением никого, может еще кто-нибудь что-то предложит... |
Archon |
![]()
Сообщение
#6
|
![]() Профи ![]() ![]() ![]() ![]() Группа: Пользователи Сообщений: 618 Пол: Мужской Репутация: ![]() ![]() ![]() |
Цитата Я в таких случаях делаю так: Ага, я что-то такое себе представлял ![]() 1 Дерево интерфейсов для собственно интерфейсов (черт, почему это не 2 разных слова ![]() 2 Дерево классов для реализации (как в моем способе 1). При этом классы реализуют интерфейсы, через которые и идет полиморфизм. -------------------- Close the World...txeN eht nepO
|
volvo |
![]()
Сообщение
#7
|
Гость ![]() |
Цитата А если добавить интерфейсы (имеется ввиду interface, как конструкция языка) Это не даст никакого выигрыша, вот множественное наследование (в стиле С++) дало бы, а с интерфейсами - нет... Ну, разве что... для того, чтобы гарантировать присутствие нужного тебе метода в классе. |
Client |
![]()
Сообщение
#8
|
Профи ![]() ![]() ![]() ![]() Группа: Пользователи Сообщений: 865 Пол: Мужской Реальное имя: Вячеслав Репутация: ![]() ![]() ![]() |
typeА зачем нужны пустые объекты? И обязательно в private надо добавлять? Сообщение отредактировано: Client - 12.06.2009 17:48 |
volvo |
![]()
Сообщение
#9
|
Гость ![]() |
Цитата А зачем нужны пустые объекты? Это не полное описание объектов... Только иерархия. Естественно, что:tstack = object(tbase), теперь для реализации полиморфной процедуры, работающей со стеком, достаточно сделать: procedure P(var stack: tstack);, и передавать в нее можно будет любого наследника TStack, то есть, стек с любой реализацией... Цитата И обязательно в private надо добавлять? Желательно. Чем меньше возможностей извне добраться до внутреннего представления данных - тем лучше. Инкапсуляция. |
Archon |
![]()
Сообщение
#10
|
![]() Профи ![]() ![]() ![]() ![]() Группа: Пользователи Сообщений: 618 Пол: Мужской Репутация: ![]() ![]() ![]() |
Цитата Это не даст никакого выигрыша, вот множественное наследование (в стиле С++) дало бы, а с интерфейсами - нет... Это потому что полиморфизм через интерфейсы в делфи работают медленно?-------------------- Close the World...txeN eht nepO
|
volvo |
![]()
Сообщение
#11
|
Гость ![]() |
Это потому, что работа с Interface-ами ЗАМЕНЯЕТ, а не реализует multiple inheritance... Вот тебе код:
{$mode delphi}, он запускается и работает. А теперь попробуй переписать его так, чтобы не было необходимости дублировать прототипы методов в классе Tmystack, чтоб метод, раз описанный в Imystack (да, да, меня не интересует реализация в моем классе, это убивает смысл того, что было задумано, меня интересует реализация нужного мне метода в одном из предков так, чтобы мой класс наследовал все, что было во всех предках), можно было когда нужно унаследовать и использовать, как это делается в С++ (С++ный код привести?). Функциональность не должна пострадать... Вот если тебе удастся это сделать - продолжим разговор о такой реализации... |
Archon |
![]()
Сообщение
#12
|
![]() Профи ![]() ![]() ![]() ![]() Группа: Пользователи Сообщений: 618 Пол: Мужской Репутация: ![]() ![]() ![]() |
Не зачем наследовать функциональность от всех предков. Достаточно обеспечить полиморфизм для классов из разных ветвей дерева наследования (или вовсе разных деревьев). Вот схема:
![]() Использовать так: varПолучаем следующие преимущества: 1 Можем наследовать реализации как угодно и от чего угодно - пользователю классов это до лампочки, он работает с интерфейсами. В итоге если функция ждет на вход IStack можем пихать туда любую реализацию, лишь бы поддерживался соответствующий интерфейс. 2 Нет такого вот: procedure tstack_first.push(value: T);3 Обычно, когда пользователь смотрит на класс, он видит не только то, что его интересует (public-методы и свойства), но и информацию, касающуюся реализации: секции private/protected, "read ... write ..." в свойствах, а в некоторых языках (C#, Java) - код реализации всех методов. Конечно, не бог весть какая неприятность, но если показывать юзеру класса интерфейс - он видит только то, что нужно для его использования. Это как-то эстетичнее, что-ли. PS Вышесказанное - даже не мнение, а только попытка его сформировать. -------------------- Close the World...txeN eht nepO
|
volvo |
![]()
Сообщение
#13
|
Гость ![]() |
Цитата Вот схема: Бла-бла-бла... На уровне схемы все выглядит прекрасно. Я дал тебе код, приведи и ты РАБОТАЮЩИЙ код, построенный по твоей схеме... А схемы - как полететь на Луну на чайнике - я тоже могу составлять, и все будет выглядеть идеально...Цитата Не зачем наследовать функциональность от всех предков. Угу, угу... А зачем, извини, в таком случае вообще наследоваться от чего-то, если функциональность не наследуется?Кроме всего прочего: попробуй твоей схемой "на лету" изменить реализацию, я хочу это видеть. У меня это делается на ура, просто сам объект типа dds_first меняется на "указатель на корень дерева наследования", и когда надо - убивается старый p_inner, и инициализируется новый, нужного типа. Только не надо про то, что оно не используется, и не надо. Не используется - потому что не реализуете так, все в интерфейсы лезете... Мной - очень даже используется... |
Client |
![]()
Сообщение
#14
|
Профи ![]() ![]() ![]() ![]() Группа: Пользователи Сообщений: 865 Пол: Мужской Реальное имя: Вячеслав Репутация: ![]() ![]() ![]() |
Только не закрывайте тему ПЛИИИИЗ
щас время маловато, экзамен сдам по ООП (теорию повторю, чтоб не задавать глупых вопросов) и присоединюсь по плотному ![]() Сообщение отредактировано: Client - 12.06.2009 21:41 |
Archon |
![]()
Сообщение
#15
|
![]() Профи ![]() ![]() ![]() ![]() Группа: Пользователи Сообщений: 618 Пол: Мужской Репутация: ![]() ![]() ![]() |
Цитата Я дал тебе код, приведи и ты РАБОТАЮЩИЙ код, построенный по твоей схеме...
{$mode objfpc}Да, насчет пункта 2 я погорячился ![]() Цитата Угу, угу... А зачем, извини, в таком случае вообще наследоваться от чего-то, если функциональность не наследуется? Я уже много раз упоминал слово полиморфизм. Ради него же.Цитата Кроме всего прочего: попробуй твоей схемой "на лету" изменить реализацию, я хочу это видеть. Никак. Действительно, недостаток. Не приходило в голову, спасибо. Хотя вот это:
Stack := TStack1.Create;можно назвать сменой реализации? Если нужно при этом сохранить какие-то данные, то { . . . }Как думаешь, ересь? ![]() Сообщение отредактировано: Archon - 12.06.2009 21:55 -------------------- Close the World...txeN eht nepO
|
volvo |
![]()
Сообщение
#16
|
Гость ![]() |
Цитата Как думаешь, ересь? Я б так делать не стал, это все, что я могу тебе сказать. Ересь или нет - это ты решай сам, я не могу тебе говорить, что и как делать. НО... То, что у тебя начиная с какого-то момента времени существует 2 экземпляра Стека - уже наводит на некоторые размышления. Их не должно быть, потому что все, что я хотел - это сменить реализацию, а это достигается с помощью "pImpl idiom" - того способа, который я тебе показал... Все делается внутри одного экземпляра.Ну и то, что эта идиома работает начиная с TP версии 5.5, а не с Object Pascal-я - тоже склоняет чашу весов в ее сторону (для меня, по крайней мере). Вот так выглядит основная часть программы, написанной с использованием моего способа: var Как видишь, никаких лишних экземпляров, ничего подозрительного... Кстати, еще один минус твоего кода: если тебе понадобится сменить "реализацию" с TImplementation2 на TImplementation1 - будешь добавлять еще один метод? А если реализаций не 2, а 4? ![]() ![]() |
Archon |
![]()
Сообщение
#17
|
![]() Профи ![]() ![]() ![]() ![]() Группа: Пользователи Сообщений: 618 Пол: Мужской Репутация: ![]() ![]() ![]() |
Цитата То, что у тебя начиная с какого-то момента времени существует 2 экземпляра Стека - уже наводит на некоторые размышления. Их не должно быть, потому что все, что я хотел - это сменить реализацию, а это достигается с помощью "pImpl idiom" - того способа, который я тебе показал... Все делается внутри одного экземпляра. Однако внутри экземпляра происходит тоже самое - удаление старого экземпляра PImpl и создание нового. Хотя то, что это происходит внутри - уже плюс.Цитата Кстати, еще один минус твоего кода: если тебе понадобится сменить "реализацию" с TImplementation2 на TImplementation1 - будешь добавлять еще один метод? А если реализаций не 2, а 4? В моем случае - хоть заизменяйся, как только появилась единственная реализация change_impl, можно менять все что угодно, в любую сторону... В твоем случае можно свободно менять реализации только если интерфейсы у них одинаковые и наследуются от общего предка. У меня даже способы работы с реализациями различны (я специально писал код, чтобы показать такую возможность). Правда не представляю пока где это может понадобиться, так что вряд ли это действительно "фича". Пожалуй, твой способ все же правильней. Спасибо, что помог разобраться. ![]() -------------------- Close the World...txeN eht nepO
|
volvo |
![]()
Сообщение
#18
|
Гость ![]() |
Цитата Правда не представляю пока где это может понадобиться, так что вряд ли это действительно "фича". Если ты внимательно читал мои ответы в теме про Doomed Game (или это было в ЛС, не помню, но где-то я уже говорил об этом), я приводил пример, когда вот такое вот изменение реализации "на лету" может значительно облегчить задачу... Если что, оно относилось к AI (к задаче, решаемой юнитом в данный момент: движется себе модуль, движется, вдруг бац, атака на него. Что делать? А ничего. Реализация "Дозор" удаляется, вместо нее инициализируется реализация "Оборона", и начинаем битву. Закончили обороняться - возвращаемся к предыдущему состоянию. Просто и эффективно...). Опять "идиома pImpl", то есть, она таки имеет право на существование ![]() Но что-то мне подсказывает, что новички все-же не очень хотят присоединяться, обсуждение ушло опять далеко от начального уровня. ![]() Кстати, Archon, проект все-таки заглох окончательно? |
Archon |
![]()
Сообщение
#19
|
![]() Профи ![]() ![]() ![]() ![]() Группа: Пользователи Сообщений: 618 Пол: Мужской Репутация: ![]() ![]() ![]() |
Да, без тебя заглох =). Я было пытался продолжать в одиночку, но энтузиазм угас. Главное, он не прошел бесследно. Все, что я знаю об объектах, я знаю благодаря этому проекту. Кроме того, он помог окончательно перейти на 32-битные компиляторы и научиться работать с OpenGL и DirectX. Еще навыки проектирования структуры приложений получил какие-никакие. Есть у меня пара тестов старых. Там структура проекта прослеживается (с кучей заглушек) и немного графики. Могу показать, конечно, но до самого интересного (ИИ) дело не дошло даже близко.
Сообщение отредактировано: Archon - 15.06.2009 21:17 -------------------- Close the World...txeN eht nepO
|
Client |
![]()
Сообщение
#20
|
Профи ![]() ![]() ![]() ![]() Группа: Пользователи Сообщений: 865 Пол: Мужской Реальное имя: Вячеслав Репутация: ![]() ![]() ![]() |
Тему несколько раз читал, но не очень понял, поэтому начну с начала
![]() Цитата А хорошо ли делать так, как я показал выше? Какие у этого способа вы видите недостатки, какие преимущества? Хм, даже не знаю, описал объект и его обработку и работаешь с ним-ничто не мешает. Т.е. ответа я не знаю ![]() Допустим есть объект с полем х и его потомок. Могу ли я переопределить поле х в потомке (метод можно, а поле)? |
![]() ![]() |
![]() |
Текстовая версия | 20.07.2025 2:30 |