Как правильно разделить проект Delphi на BPL?
Компания, в которой я работаю, разрабатывает систему в Delphi, которая содержит десятки exe-модулей, и каждый из них в определенной степени идентичен, если речь идет об исходном коде. К сожалению, никто никогда не заботился об использовании библиотек для добавления общего кода. Это означает, что каждый раз, когда в коде, который разделяют все эти модули, есть исправление ошибки, программист должен вносить исправления во все из них отдельно! Это всегда занимает так много времени...
Я решил найти способ поместить общий код в библиотеки. Я рассмотрел DLL и BPL. В этом случае BPL казались гораздо более удобными для программиста и намного менее хлопотными, особенно в том, что код используется только в нашем программном обеспечении и только в Delphi.
Я поместил весь код, совместно используемый всеми exe-модулями, в BPL, и все выглядит нормально, но есть некоторые вещи, которые я не понимаю, и был бы благодарен, если бы вы объяснили их мне.
После разделения кода на BPL я ожидал, что будет достаточно развернуть исполняемые файлы с созданными мною BPL. Но оказалось, что им нужны также rtl100.bpl и vcl100.bpl. Почему это так? Я хочу развернуть exes и только мои BPL. Я не хочу предоставлять конечным пользователям целую кучу библиотек, поставляемых Borland и сторонними компаниями:). Я хочу, чтобы они были скомпилированы в exes, как раньше. Возможно ли это сделать?
То, что я сделал до сих пор, было:
- Я поместил все общие единицы pas в BPL. Каждый BPL содержит блоки, принадлежащие к одной и той же категории, поэтому программистам понятно, какой код ожидать в данном BPL.
- Каждый BPL - это библиотека времени выполнения и времени разработки.
- Каждый BPL "перестраивается явно". Два последних являются настройками проекта по умолчанию для BPL.
И если речь идет о проектах exe:
- Я удалил все единицы, которые я ранее поместил в BPL.
- Я установил свои BPL из меню Сервис-> Установить пакет в BDS 2006.
- В моих настройках exe-проекта я установил опцию "строить с пакетами времени выполнения" и перечислил все мои пакеты BPL в поле редактирования ниже (только мои пакеты, так как я очистил все остальные, которые там появились).
Это все, что я сделал. Проекты exe правильно компилируются, но у меня нет доступа к исходному коду BPL (я не могу перейти к этому коду из моих exe-проектов), даже если все BPL хранятся вместе с их файлами исходного кода. Зачем? Это кажется странным для меня.
Я всегда склонен писать длинные описания - извините за это:). Я буду признателен за вашу помощь. Мне просто нужно несколько слов объяснения по поводу упомянутых мной моментов: развертывание exe только с моими BPL, правильность того, что я сделал в целом, и невозможность перехода к исходным кодам BPL. Заранее большое спасибо!
Спасибо всем за обсуждение. Некоторые говорили, что подход, который я выбрал, не был хорошей идеей. Наше программное обеспечение состоит из более чем 100 модулей (большинство из них являются драйверами для разных устройств). Большинство из них используют один и тот же код - в большинстве случаев это классы. Проблема в том, что эти классы не всегда помещаются в отдельные, отдельные единицы. Я имею в виду, что общий код часто помещается в блоки, содержащие код, специфичный для модуля. Это означает, что когда вы исправляете ошибку в совместно используемом классе, недостаточно скопировать модуль pas, в котором он определен, во все программные модули и перекомпилировать их. К сожалению, вы должны скопировать и вставить фиксированные фрагменты кода в каждый модуль, один за другим, в соответствующий модуль и класс. Это занимает много времени, и это то, что я хотел бы устранить, выбрав правильный подход - пожалуйста, помогите мне.
Я думал, что использование BPL было бы хорошим решением, но оно имеет некоторые недостатки, как некоторые из вас упоминали. Худшая проблема заключается в том, что если каждому EXE-файлу требуется несколько BPL, специалистам нашей службы технической поддержки нужно будет знать, каким EXE-файлам нужны BPL-файлы, а затем предоставить конечным пользователям нужные файлы. Пока у нас нет программы обновления программного обеспечения, это будет очень полезно как для наших технических специалистов, так и для конечного пользователя. Они наверняка заблудятся и разозлятся.
Также могут возникнуть проблемы совместимости - если один BPL используется многими EXE-файлами, модификация одного BPL может быть полезна для одного EXE-файла и плоха для некоторых других - @Warren P.
Что мне делать, чтобы исправить ошибки быстрее во многих проектах? Я думаю об одном из следующих подходов. Если у вас есть идеи получше, пожалуйста, дайте мне знать.
- Размещайте общий код в отдельные и автономные модули pas, поэтому, когда в одном из них есть исправление ошибки, достаточно скопировать его во все проекты (перезаписать старые файлы) и перекомпилировать все из них.
Это решение кажется приемлемым, если задний код изменен. Но у нас также есть блоки с функциями и процедурами общего назначения, которые часто не требуют изменений - мы добавляем новые функции там, где это необходимо, но в отдельных проектах. Итак, представьте, что вы пишете новую функцию в одном из 100 модулей и помещаете ее в модуль общего пользования. Через месяц или два вы модифицируете другой модуль и думаете, что вам нужна та же функция, которую вы написали 2 месяца назад. Вы должны найти модуль (это сложно, если вы не помните, какой он был) и скопировать функцию в свой код. И очевидно - единицы общего использования становятся совершенно разными в каждом модуле, если они хранятся в каждом проекте отдельно. И затем, если есть исправление ошибки... вся история повторяется.
- Создайте BPL для всего общего кода, но свяжите их с EXE-файлами, чтобы EXE-файлы были автономными.
Для меня это кажется лучшим решением сейчас, но есть несколько минусов. Если я исправлю ошибку в BPL, каждый программист должен будет обновить BPL на своем компьютере. Что если они забудут? Но все же, я думаю, что это небольшая проблема. Если мы позаботимся о том, чтобы информировать друг друга об изменениях, все должно быть хорошо.
- @CodeInChaos: я не знаю, правильно ли я вас понял. Вы имеете в виду совместное использование файлов pas между проектами? Как это сделать? Мы храним исходники в SVN. Это означает, что нам придется хранить общий код в отдельной папке и заставить все проекты искать этот код там, верно? И скачать из SVN проект и все папки, от которых он зависит...
Пожалуйста, помогите мне выбрать хорошее решение. Я просто не хочу, чтобы компания теряла гораздо больше времени и денег, чем необходимо, на исправления ошибок только из-за глупого подхода к разработке программного обеспечения.
Большое спасибо.
4 ответа
Несмотря на то, что у этого вопроса есть принятый ответ, я собираюсь нанести удар в этом.
Заголовок спрашивает, как разделить проект на bpls, но реальный вопрос, по-видимому, таков: "Какой лучший способ поделиться кодом между проектами?"
Есть несколько способов сделать это:
- Общие единицы
- Dlls
- ВР
Независимо от того, в каком направлении вы идете, вам, скорее всего, придется реструктурировать свои проекты. Из вашего описания звучит так, будто каждый проект разрабатывается в относительной изоляции. Код передается с использованием функции копирования / вставки, что быстро нарушает синхронизацию и приводит к большим дублирующим усилиям. Итак, давайте рассмотрим каждый из методов для обмена кодом.
Общие единицы
Это самый простой подход. Вы создаете общее местоположение и размещаете код, который хотите использовать в своих проектах в этом месте. Модули статически связаны с вашими проектами, поэтому вам не нужно беспокоиться о развертывании дополнительных зависимостей вместе с основными исполняемыми файлами. Статически связанные блоки являются наиболее простыми для устранения неполадок и отладки.
Компилятор должен быть в состоянии найти ваши общие модули. Есть 4 способа указать компилятору, где искать.
- Добавьте их в проект - SHIFT+F11 - Добавляет ссылку на модуль в файлы проекта (dpr, dproj). В среде IDE обычно используются относительные пути, если устройство находится в том же дереве каталогов, что и файлы проекта, в противном случае будут использоваться абсолютные пути, что может быть проблематично, если машины разработчика не настроены одинаково.
- Путь поиска проекта - CTRL+SHIFT+F11 Delphi Compiler> Путь поиска - добавьте каталог, и компилятор будет искать там, чтобы найти модули, упомянутые в предложении использования любого модуля в проекте. Лучше всего использовать относительные пути, если можете. Вы также можете использовать переменные окружения: $(MyPath)
- Глобальный путь поиска - Инструменты> Параметры> Параметры среды> Параметры Delphi> Библиотека - Win32 > Путь к библиотеке - любые пути, перечисленные здесь, доступны для всех проектов на компьютере. Это зависит от машины
- Командная строка - если вы строите из скрипта или инструмента автоматизации сборки, вы можете установить путь поиска, используя dcc32
-U
переключатель или msbuild's/property:UnitSearchPath=
переключатель.
Варианты 1 и 2 будут наиболее полезными.
Что касается вашего репозитория SVN, у вас есть несколько вариантов организации проектов и общих блоков. Простейшим было бы поместить все проекты в одну магистраль вместе с общими блоками:
Projects
trunk
ProjectA
ProjectB
ProjectC
Library (shared units)
Если по какой-либо причине приведенная выше структура не возможна, вы можете попробовать эту альтернативу:
ProjectA
trunk
Library (branch of main library)
ProjectB
trunk
Library (branch of main library)
ProjectC
trunk
Library (branch of main library)
Library
trunk (main library)
В этой конфигурации изменения, внесенные в папку библиотеки каждого проекта, не будут немедленно доступны для других проектов. Каждый проект должен будет регулярно синхронизировать изменения с основным проектом библиотеки. Побочным эффектом этого является то, что изменения, которые нарушают другие проекты, будут отложены до синхронизации других проектов. Считаете ли вы это хорошей или плохой вещью, зависит. С одной стороны, ошибки легче и дешевле исправить, если код, который они используют, еще свеж в сознании разработчика. С другой стороны, если вы не практикуете модульное тестирование (что я настоятельно рекомендую вам делать), или код очень хрупкий, или у вас просто есть разработчики, склонные к безрассудным изменениям, вы можете контролировать, как часто эти изменения проталкиваются в другие проекты.,
Dlls
Dlls позволяют вам делиться кодом, связываясь с ним во время выполнения. Они предоставляют функции, которые можно вызывать из основного исполняемого файла или другой библиотеки DLL.
Хотя библиотеки DLL всегда связаны во время выполнения, вы решаете, будут ли они загружены при запуске приложения или только при необходимости. Загрузка при запуске называется статической загрузкой, а в Delphi выполняется с помощью external
директивы. Подавляющее большинство классов rtl/vcl, которые переносят системные вызовы API, используют статическую загрузку. Динамическая загрузка позволяет отложить загрузку DLL до тех пор, пока она не потребуется. При этом используются функции WinAPI LoadLibrary и GetProcAddress. Соответствующий вызов FreeLibrary разгрузит DLL.
К сожалению, стандартные dll ограничивают типы данных, которые можно передавать. Если вам нужен доступ к DLL из не-Delphi проектов, вам нужно ограничиться использованием типов данных в стиле c. Если вы будете использовать dll только с проектами Delphi, вы можете безопасно использовать строки и динамические массивы Delphi, если вы используете модуль SharedMem в dll и любые проекты, которые его используют.
Вы можете безопасно использовать объекты внутри dll без проблем, но если вы хотите передавать объекты между dll и приложением, вам нужно извлечь данные объекта и передать их как примитивные типы и собрать их в объект на другом конце. Это называется (де) сериализацией или сортировкой, и есть гораздо более простые способы сделать это, чем использовать собственные.
COM (компонентная объектная модель) хорошо поддерживается в Delphi, но имеет некоторую кривую обучения. Потребление объектов COM довольно просто, но разработка одного займет некоторое время, если вы не знакомы с COM. Преимущество COM заключается в том, что он не зависит от языка и поддерживается на большинстве языков, ориентированных на платформу Windows (включая языки, нацеленные на платформу.NET).
ВР
Bpls (также называемые просто "пакетами") - это специально отформатированные dll, которые значительно облегчают работу с объектами. Как и стандартные библиотеки DLL, они связаны во время выполнения и могут быть статически или динамически загружены. Их легче изучать и использовать, чем COM-библиотеки, и они обеспечивают больше интеграции в ваши проекты, чем COM. Пакеты состоят из двух частей: bpl и dcp. Dcp похож на файлы dcu, сгенерированные при компиляции обычного файла модулей, за исключением того, что он содержит целую кучу модулей. Использование класса, скомпилированного в bpl, так же просто, как добавление dcp в список пакетов проекта, а затем добавление модуля к предложению использования одного из модулей проекта.
При развертывании приложения вам также необходимо установить bpl. Как уже отмечали другие, вы должны как минимум включить пакет rtl и, скорее всего, пакет vcl, если вы используете какие-либо формы. Существует способ развернуть поставленные Borland BPL с вашими проектами. Вы можете создать "мини" пакет RTL, который содержит только те единицы, которые нужны вашему проекту. Трудно определить, какие единицы включить.
Резюме
Из описания, которое вы дали, создание библиотеки совместно используемых файлов модулей для статической ссылки может быть наиболее целесообразным путем. Я также предложил бы попробовать программу под названием Simian. Это поможет вам найти дубликаты кода в вашей кодовой базе для включения в общую библиотеку. Он не поддерживает паскаль напрямую, но делает достаточно приличную работу, используя анализатор простого текста с небольшой настройкой его конфигурации.
Также я не могу не подчеркнуть ценность модульного тестирования. Особенно если вы переходите к общим библиотекам. Набор хорошо написанных модульных тестов, выполняемых часто, даст вам мгновенную обратную связь, когда разработчик меняет класс и ломает несвязанный проект.
Представьте, что у вас есть проект с EXE и двумя разными модулями BPL, и где-то в этой кодовой базе есть строка, которая говорит: if MyObject is TStringList then DoSomething;
, is
Оператор работает, проверяя метаданные класса объекта, сохраненные в VMT, а затем следуя цепочке VMT через ClassParent
указатель, чтобы увидеть, соответствует ли какой-либо из них ссылка на класс (также указатель VMT) для TStringList
, Чтобы убедиться, что это будет работать правильно, должен быть один VMT для TStringList, который будет одинаковым во всей вашей программе, независимо от того, на сколько BPL он разделен, что означает, что он должен быть в своем собственном пакете. Вот почему необходимы такие среды выполнения системы, как rtl*.bpl и vcl*.bpl, и с этим ничего не поделаешь. Это часть стоимости использования BPL.
Что касается невозможности отладки, вам необходимо убедиться, что BPL созданы с включенной информацией отладки и что отладчик знает, как найти папку, в которой находится DCP (файл, содержащий информацию отладки для BPL). И вы не сможете отследить системные BPL, потому что DCP с поддержкой отладки не поставлялись с вашей версией. Они были добавлены довольно недавно, я думаю, в XE, но это могло быть в D2010.
Почему я не могу просмотреть свой исходный код? Есть ли способ это исправить?
Вы не можете просматривать исходный код модулей, включенных в пакеты, потому что их нет ни в вашем проекте, ни в вашей библиотеке, ни в пути поиска.
Я решил это путем добавления каталогов в путь поиска проекта. Таким образом, компилятор не знает об этих файлах (и не пытается их перекомпилировать), но в среде IDE вы можете просматривать их содержимое и отлаживать их.
"В моих настройках exe-проекта я выбрал опцию" строить с помощью пакетов времени выполнения "
Вот почему вы не можете выполнить развертывание без BPL и т. Д. - этот параметр сбивает с толку многих разработчиков -"сборка с помощью пакетов времени выполнения" означает, что вам потребуется подарок bpl во время выполнения. Снимите этот флажок, и пакеты будут связаны с вашим exe-файлом в compileTime. (Ваш exe-файл будет увеличиваться в размерах.) Идея "сборки с пакетами времени выполнения" состоит в том, чтобы уменьшить размер exe-файла и позволить нескольким приложениям совместно использовать общие bpl-файлы, потому что они НЕ связаны с exe @ compileTime - в этом и есть преимущество. Недостаток, который вы сейчас испытываете - вы должны раздавать свои bpl вместе с вашим exe.