Проблема неустойчивости базовых классов и контроль версии
Процедура Sub Main выглядит так:
Sub Main() Dim torn As New EmployeeC'Tom". 50000) tom.Address ="901 Grayson" Console.WriteCtom.TheName & "lives at " & tom.Address) Console. ReadLine() End Sub
Результат показан на рис. 5.7. Программа работает именно так, как предполагалось.
Рис. 5.7. Демонстрация неустойчивости базовых классов (контроль версии отсутствует)
Программа компилируется в исполняемый файл Versiomngl.exe, все идет прекрасно.
Теперь предположим, что класс PayableEntity был разработан независимой фирмой. Гениальные разработчики класса PayableEntity не желают почивать на лаврах! Заботясь о благе пользователей, они включают в свой класс объект с адресом и рассылают новый вариант DLL. Исходный текст они держат в секрете, но мы его приводим ниже.
Imports Microsoft.Vi sualBasic.Control Chars Public Class PayableEntity Private m_Name As String Private m_Address As Address Public Sub New(ByVal theName As String,ByVal theAddress As Address) m_Name = theName m_Address = theAddress End Sub Public Readonly Property TheName()As String Get Return m_Name End Get End Property Public Readonly Property TheAddress() Get Return m_Address.DisplayAddress End Get End Property End Class Public Class Address Private m_Address As String Private m_City As String Private m_State As String Private m_Zip As String Public Sub New(ByVal theAddress As String.ByVal theCity As String. ByVal theState As String.ByVal theZip As String) m_Address = theAddress m_City = theCity m_State = theState m_Zip = theZip End Sub Public Function DisplayAddress() As String Return m_Address & CrLf & m_City & "." & m_State _ &crLF & m_Zip End Function End Class
Перед вами пример редкостной халтуры. В процессе "усовершенствования" авторы умудрились потерять исходный конструктор класса PayableEntity! Конечно, такого быть не должно, но раньше подобные катастрофы все же случались. Старая DLL устанавливалась на жесткий диск пользователя (обычно в каталог Windows\System). Затем выходила новая версия, устанавливалась поверх старой, и вполне благополучная программа Versioningl переставала работать (а как ей работать, если изменился конструктор базового класса?).
Конечно, проектировщики базовых классов так поступать не должны, однако на практике бывало всякое. Но попробуйте воспроизвести этот пример в .NET, и произойдет настоящее чудо: ваша старая программа будет нормально работать, потому что она использует исходную версию PayableEntity из библиотеки, хранящейся в каталоге \bin решения Versioningl.
Примечание
Решение проблемы несовместимости версий в .NET Framework в конечном счете основа-но на том, что ваш класс знает версию DLL, необходимую для его работы, и отказывается работать при отсутствии нужной версии. Успешная работа этого механизма зависит от особых свойств сборок (см. главу 13). Тем не менее.в описанной нами ситуации защита .NET Framework преодолевается простым копированием новой DLL на место старой.