Главная страница
    Top.Mail.Ru    Яндекс.Метрика
Текущий архив: 2005.02.20;
Скачать: CL | DM;

Вниз

помогите сделать временную копию объекта   Найти похожие ветки 

 
newid   (2005-01-30 23:14) [0]

может, вопрос совсем несерьезный, но не получается что-то...
нужно создать временную копию объекта ("резервную копию", из которой можно будет восстановить состояние исходного объекта при необходимости), при этом объект может быть экземпляром одного из нескольких наследников общего дедушки, т.е. точно класс объекта не знаем.
такое:
obj2 := obj1;
try
DoSome(obj1);
except
obj1 := obj2;
end;
само собой не катит, потому что с сылками работаем.
как вообще оперировать объектами в Object Pascal?


 
P.N.P. ©   (2005-01-30 23:18) [1]

Obj2.Assign(Obj1);
...
Obj1.Assign(Obj2);


 
newid   (2005-01-30 23:43) [2]


> [1] P.N.P.


ну... это если объекты -- экземпляры класса-наследника TPersistent.
и, кстати, у меня почему-то первая операция:
Obj2.Assign(Obj1);
дает такую ошибку:
---------------------------
map
---------------------------
Cannot assign a TSomeObject to a TSomeObject.
---------------------------
OK  
---------------------------
вот мой код:

var
 o1: TSomeObject;
 o2: TSomeObject;
begin
 o1 := TSomeObject.Create;
 o2 := TSomeObject.Create;
 o1.StringProp := "test1";
 ShowMessage(o1.StringProp);
 o2.Assign(o1);  //Cannot assign a TSomeObject to a TSomeObject.
 o1.StringProp := "test2";
 ShowMessage(o1.StringProp);
 o1.Assign(o2);
 ShowMessage(o1.StringProp);  //по задумке должно показать "test1"
 o1.Free;
 o2.Free;
end;


 
jack128 ©   (2005-01-31 00:00) [3]

newid   (30.01.05 23:43) [2]
дает такую ошибку:
---------------------------
map
---------------------------
Cannot assign a TSomeObject to a TSomeObject.

потому что этот метод нужно реализововать самому.


 
Юрий Зотов ©   (2005-01-31 01:49) [4]

> как вообще оперировать объектами в Object Pascal?

Хороший вопрос. Но не знаю, как ответить. Писать книгу в online почему-то не готов.


 
хм ©   (2005-01-31 02:19) [5]

можно использовать CopyMemory, если в объекте нет указателей. Т.е, нужно много думать.
Но лучше не надо.(делать, а не думать :)


 
Просто Джо ©   (2005-01-31 03:29) [6]

Проще всего, неверное, унаследовать все классы от TPersistent. Далее, все свойства, которые нужно сохранять/восстанавливать сделать Pusblished. После этого использовать стандартные возможности стриминга для ТПерсистент.
Впрочем, в этом решении, как в остальных также много подводных камней. Но - лиха беда начало! :)


 
Просто Джо ©   (2005-01-31 03:35) [7]


> [6] Просто Джо ©   (31.01.05 03:29)


Блин, заработался. Имелся в виду TComponent.


 
newid   (2005-01-31 07:22) [8]


>  [6] Просто Джо ©   (31.01.05 03:29)
> Проще всего, неверное, унаследовать все классы от TPersistent.

можно. тут еще проблема: одно или несколько свойств объекта сами могут быть объектами. как такие свойства буду сохраняться?
мне кажется, нужно с памятью работать. что-то типа:

sz := o1.InstanceSize;
p2 := AllocMem(sz);
CopyMemory(p2, o1, sz);
DoSome(o1);
CopyMemory(o1, p2, sz);
FreeMem(p2, sz);

только это не работает. потому что неправильно написано. научил бы кто-нибудь правильно такое делать


 
ЮЮ ©   (2005-01-31 07:58) [9]

А не лучше вместо
obj2 := obj1;
try
 DoSome(obj1);
except
 obj1 := obj2;
end;

DoSome сделать виртуальным методом и уже в нем, реально зная, что можно "попортить", и делать необходимый "сохранения" и, собственно, откат в случае неудачи?


 
jack128 ©   (2005-01-31 07:59) [10]

newid   (31.01.05 7:22) [8]
мне кажется, нужно с памятью работать. что-то типа:

Нет. Вот как раз в этом случае так делать не надо. см
хм ©   (31.01.05 2:19) [5] можно использовать CopyMemory, если в объекте нет указателей объекты - это тоже указатели.


 
Erik1 ©   (2005-01-31 11:23) [11]

Лучше логику ререработать, чем создовать механизм сохранения объектов. Небывает таких методов которые все даные сразу должны портить.


 
pasha_golub ©   (2005-01-31 11:37) [12]

А можно ситуацию подробней описать. Имею ввиду, зачем нужен резервный экземпляр? При каких таких раскладах? У меня даже мысли накакой нету.


 
newid   (2005-01-31 15:48) [13]


>  [12] pasha_golub ©   (31.01.05 11:37)
> А можно ситуацию подробней описать

можно. резервный экземляр нужен, если пользователь работает с объектом, может менять его данные. а сам объект (его состояние) хранится постоянно где-то. ну начал юзер редактировать данные, вместо подтверждения изменений -- передумал, отменил изменения. объект (вернее его отображение) должен вернуться в прежнее состояние. можно, конечно, восстановить из места постоянного храниения, как это делается при инициализации, но ведь лишние операции ввода-вывода...
да кучу всяких подобных ситуаций можно придумать. мне кажется, что это вообще рабочая ситуация, с которой постоянно приходится сталкиваться.
вообще-то я полагал, что этот вопрос только от моего недостатка знаний, а получается, что он решается не так уж просто. или вообще не решается?...


 
jack128 ©   (2005-01-31 16:02) [14]

pasha_golub ©   (31.01.05 11:37) [12]
А можно ситуацию подробней описать. Имею ввиду, зачем нужен резервный экземпляр? При каких таких раскладах? У меня даже мысли накакой нету.


Ну, например, для реализации механизма транзакций..
newid   (31.01.05 15:48) [13]
а получается, что он решается не так уж просто

В общем случае он не решается. В часных случаях возможны варианты. Например, для TPersistent можно сохранить состояние в поток, а потом, если нужно, восстановить его..
newid   (31.01.05 15:48) [13]
как это делается при инициализации, но ведь лишние операции ввода-вывода...

Эти лишние операции существенно снижают производительность твоей программы?


 
Gloomer ©   (2005-01-31 16:12) [15]

>newid   (30.01.05 23:43) [2]

Если я не ошибаюсь, то при определении
var
o1: TSomeObject;
o2: TSomeObject;

о1 и о2 не относятся к одному типу. Попробуй лучше
var o1, o2: TSomeObject;


 
У   (2005-01-31 16:15) [16]

>Gloomer ©   (31.01.05 16:12) [15]
>Если я не ошибаюсь,
ты ошибаешься


 
jack128 ©   (2005-01-31 16:36) [17]

Gloomer ©   (31.01.05 16:12) [15]
о1 и о2 не относятся к одному типу.

ошибаешся.
А вот при таком
arr1: array of Integer;
arr2: array of Integer; действительно к разным типам


 
newid   (2005-01-31 16:38) [18]


> [14] jack128 ©   (31.01.05 16:02)
> Эти лишние операции существенно снижают производительность
> твоей программы?

нет, существенно не снижают. но ведь разумнее на "клиенте" эти вещи делать, согласись.
ладно. ткните меня носом в какие-нибудь исходники, где подобные механизмы реализованы


 
GuAV ©   (2005-01-31 19:06) [19]


> ткните меня носом в какие-нибудь исходники, где
> подобные механизмы реализованы

TStream.WriteComponent;


 
Просто Джо ©   (2005-01-31 19:12) [20]

http://www.rsdn.ru/article/delphi/serialization.xml


 
newid   (2005-01-31 19:33) [21]


> [19] GuAV ©


>  [20] Просто Джо ©

получается, сохранять в потоке -- единственный более-менее разумный подход...


 
Просто Джо ©   (2005-02-01 03:07) [22]


> единственный более-менее разумный подход...


Мне лично он кажется наиболее "системным", универсальным что-ли.


 
newid   (2005-02-02 13:53) [23]

да, кстати, а каких-нибудь готовых отлаженных библиотек на эту тему нет в природе?


 
Amoeba ©   (2005-02-02 14:17) [24]


> newid   (02.02.05 13:53) [23]
> да, кстати, а каких-нибудь готовых отлаженных библиотек
> на эту тему нет в природе?

IMHO таких нет.


 
New11   (2005-02-02 14:25) [25]

реализуй для своих классов метод Assign, как и было сказано. Самый внятный подход. А c использованием TypInfo получишь кучу глюков, тем паче что он не документирован.



Страницы: 1 вся ветка

Текущий архив: 2005.02.20;
Скачать: CL | DM;

Наверх




Память: 0.51 MB
Время: 0.038 c
1-1107418107
Arbiter
2005-02-03 11:08
2005.02.20
TComponent, а как "детей" определить?


14-1107155585
TUser
2005-01-31 10:13
2005.02.20
Бейсик


1-1107867893
Neznaika
2005-02-08 16:04
2005.02.20
Load в TImage при MouseMove


3-1106207417
АМБ
2005-01-20 10:50
2005.02.20
DBGrid. Работа скролинга.


1-1107769900
Guest1
2005-02-07 12:51
2005.02.20
Cannot load package ibxpress70





Afrikaans Albanian Arabic Armenian Azerbaijani Basque Belarusian Bulgarian Catalan Chinese (Simplified) Chinese (Traditional) Croatian Czech Danish Dutch English Estonian Filipino Finnish French
Galician Georgian German Greek Haitian Creole Hebrew Hindi Hungarian Icelandic Indonesian Irish Italian Japanese Korean Latvian Lithuanian Macedonian Malay Maltese Norwegian
Persian Polish Portuguese Romanian Russian Serbian Slovak Slovenian Spanish Swahili Swedish Thai Turkish Ukrainian Urdu Vietnamese Welsh Yiddish Bengali Bosnian
Cebuano Esperanto Gujarati Hausa Hmong Igbo Javanese Kannada Khmer Lao Latin Maori Marathi Mongolian Nepali Punjabi Somali Tamil Telugu Yoruba
Zulu
Английский Французский Немецкий Итальянский Португальский Русский Испанский