Как использовать файлы модулей в Delphi
Я просто пытаюсь освоить отдельные модули, чтобы сделать мой код более инкапсулированным. Я пытаюсь отсортировать публичные / частные объявления моих методов, чтобы я мог вызывать их из других модулей, которые используют testunit
, В этом примере я хочу сделать hellofromotherunit
публично, но stickletters
частный.
unit testunit;
interface
uses
Windows, Messages, Dialogs;
implementation
function stickletters(a,b:string):string;
begin
result:=a+b;
end;
procedure hellofromotherunit();
begin
showmessage(stickletters('h','i'));
end;
end.
Я не могу скопировать частную / публичную структуру из других модулей, как в:
Type
private
function stickletters(a,b:inter):integer;
public
procedure hellofromotherunit();
end
4 ответа
Структура модуля выглядит как открытые / открытые разделы объектов, можно сказать, что это их предшественник. Но синтаксис другой.
Вам нужно только объявить заголовок метода в разделе интерфейса, как в:
interface
procedure hellofromotherunit();
implementation
procedure hellofromotherunit(); begin .. end;
Разрешен только один из каждого раздела.
Частные и общедоступные относятся только к занятиям.
Что вы хотите сделать, это поместить копию объявления hellofromotherunit в разделе интерфейса. Однако не помещайте туда копию декларации наклеек.
Все, что появляется в разделе интерфейса, является общедоступным. Все, что только в реализации, является частным.
К тому же,
Каждый блок состоит из двух отдельных частей. Интерфейс и реализация.
Раздел интерфейса содержит все открытые определения (типы, заголовки процедур, константы). Раздел реализации содержит все детали реализации.
Когда вы используете модуль, (с помощью условия использования) вы получаете доступ к общедоступным определениям этого модуля. Этот доступ не является рекурсивным, поэтому, если интерфейс блока A использует блок B, а блок C использует блок A, вы не получите доступ к блоку B, если не используете его явно.
Раздел реализации имеет доступ к интерфейсу, к модулю, используемому в обоих разделах использования (интерфейс и реализация).
Интерфейсы используемых модулей компилируются в первую очередь, прежде чем он продолжает компилировать остальные. Это имеет то преимущество, что вы можете иметь циклические зависимости внутри реализации:
unit A;
interface
uses B;
unit B;
interface
implementation
uses A;
Который компилирует:
- попробуй интерфейс А, не нужен В
- попробуйте интерфейс B, хорошо!
- попробуй интерфейс А, ок!
- попробуйте реализацию A, хорошо!
- попробуйте реализацию B, хорошо!
Каждый блок также имеет раздел инициализации (и если он имеет раздел инициализации, он также может иметь раздел финализации.) Раздел инициализации используется для инициализации переменных модуля. Разделы финализации используются для очистки. Когда вы используете их, не стоит рассчитывать на инициализацию других модулей. Просто держите их простыми и короткими.
Единицей также являются пространства имен. Рассмотрим следующее:
unit A;
interface
const foo = 1;
unit B;
interface
const foo = 2;
unit C;
interface
uses A, B;
const
f1 = foo;
f2 = A.foo;
f3 = B.foo;
Если идентификатор определен в нескольких использованных единицах, берется последняя возможная единица в списке использований. Таким образом, f1 = 2. Но вы можете поставить перед ним имя устройства (пространства имен), чтобы решить эту проблему.
С введением.net разрешены многокомпонентные пространства имен, что создает другие приятные проблемы:
unit foo;
interface
type
rec1 = record
baz : Boolean;
end;
var
bar : rec1;
unit foo.bar;
interface
var
baz : Integer;
uses
foo, foo.bar;
begin
foo.bar.baz := true;
foo.bar.baz := 1;
end.
// 1. Which these lines gives an error and why?
// 2. Does the result change if you write uses foo.bar, foo?
В этом случае у вас конфликт. Но это решается путем придания именам пространства имен более высокого приоритета. Так что первая строка не проходит.
Просто не объявляйте метод в разделе интерфейса, и он останется закрытым.
unit Unit2;
interface
function MyPublicFunction():Boolean;
implementation
function MyPrivateFunction():Boolean;
begin
// blah blah
end;
function MyPublicFunction():Boolean;
begin
// blah blah
end;
end.