Форум по Delphi программированию

Delphi Sources



Вернуться   Форум по Delphi программированию > Все о Delphi > [ "Начинающим" ]
Ник
Пароль
Регистрация <<         Правила форума         >> FAQ Пользователи Календарь Поиск Сообщения за сегодня Все разделы прочитаны

Ответ
 
Опции темы Поиск в этой теме Опции просмотра
  #1  
Старый 25.02.2015, 12:11
Аватар для Uniq!
Uniq! Uniq! вне форума
Местный
 
Регистрация: 29.09.2010
Сообщения: 539
Версия Delphi: Delphi XE3
Репутация: 374
По умолчанию Многопоточность и XML

У месть есть большой XML файл, который содержит в себе 20-30 структур строительных зданий (nObj - одна структура).
Без потоков я просто беру параметрический цикл:
Код:
iObjs := XMLDoc.DocumentElement;
for ObjNr := 0 to iObjs.ChildNodes.Count - 1 do
    begin
		nObj := iObjs.ChildNodes[ObjNr];
		txtObjNr := nObj.ChildValues['T_Number'];
		txtObjCity := nObj.ChildValues['City'];
		txtObjAddress := nObj.ChildValues['Street'];

		// внутренний парсинг конкретного nObj

	end

Медленно. Поэтому решил каждый объект отдать на растерзание парсеру в отдельный поток:

Класс потока:
Код:
type
  TObjThread = class(TThread)
  private
    sObj: IXMLNode;
    { Private declarations }
  protected
    procedure Execute;
    constructor Create(nObj: IXMLNode);
  end;

constructor TObjThread.Create(nObj: IXMLNode);
begin
  inherited Create(True);
  self.sObj := nObj;
  Resume;
end;
Далее в вышеуказанном цикле пытаюсь создать экземпляр потока:

Код:
for ObjNr := 0 to iObjs.ChildNodes.Count - 1 do
	begin
		nObj := iObjs.ChildNodes[ObjNr];
		ObjThread := TObjThread.Create(nObj);
	end

Но когда дело доходит до Execute получаю ошибку
Код:
Abstract Error.
Вот код процедуры Execute (Ради пробы пера просто пытаюсь получить значение полей из структуры nObj)

Код:
procedure TObjThread.Execute;
var
  txtObjNr, txtObjCity, txtObjAddress: OleVariant;
begin
  txtObjNr := sObj.ChildValues['T_Number'];
  txtObjCity := sObj.ChildValues['City'];
  txtObjAddress := sObj.ChildValues['Street'];
end;

Что я делаю не так? Чую, что дело в переменной nObj, которая является лишь ссылкой на часть "байт" в структуре iObjs (TXMLDocument),и по факту каждый поток пытается оперировать с одним и тем же TXMLDocument.

Последний раз редактировалось Uniq!, 25.02.2015 в 15:19.
Ответить с цитированием
  #2  
Старый 25.02.2015, 14:52
icWasya icWasya вне форума
Местный
 
Регистрация: 09.11.2010
Сообщения: 499
Репутация: 10
По умолчанию

А почему OnExecute ?
Ответить с цитированием
  #3  
Старый 25.02.2015, 15:19
Аватар для Uniq!
Uniq! Uniq! вне форума
Местный
 
Регистрация: 29.09.2010
Сообщения: 539
Версия Delphi: Delphi XE3
Репутация: 374
По умолчанию

Цитата:
Сообщение от icWasya
А почему OnExecute ?
Не из того файла название процедуры скопировал. Не заметил.

Конструктивно ничего не поменялось
Ответить с цитированием
  #4  
Старый 26.02.2015, 04:22
Аватар для Freeman
Freeman Freeman вне форума
Местный
 
Регистрация: 05.10.2012
Адрес: Санкт-Петербург
Сообщения: 577
Версия Delphi: 6
Репутация: выкл
По умолчанию

Цитата:
Сообщение от Uniq!
У месть есть большой XML файл
Насколько большой? 20-30 МБ или 200-300?
__________________
Не стоит путать форумы с богадельнями. © Bargest
Ответить с цитированием
  #5  
Старый 26.02.2015, 09:43
Аватар для Uniq!
Uniq! Uniq! вне форума
Местный
 
Регистрация: 29.09.2010
Сообщения: 539
Версия Delphi: Delphi XE3
Репутация: 374
По умолчанию

Цитата:
Сообщение от Freeman
Насколько большой? 20-30 МБ или 200-300?

Сейчас в нём 22 объекта, 18 MB.
Планируется обслуживать 150 штук, но есть возможность разделить их по группам, тем самым генерируя несколько файлов вместо одного.
Архитектурно получается всё достаточно гибко.
Ответить с цитированием
  #6  
Старый 27.02.2015, 02:01
Аватар для Freeman
Freeman Freeman вне форума
Местный
 
Регистрация: 05.10.2012
Адрес: Санкт-Петербург
Сообщения: 577
Версия Delphi: 6
Репутация: выкл
По умолчанию

Цитата:
Сообщение от Uniq!
Планируется обслуживать 150 штук
С такими размерами, наверное, стоит перейти на что-то более производительное, вроде SimpleXML (SAX). У него другая идеология: однократно перегнать исходные данные в объекты, а потом делай с ними что хошь. Объекты придется написать самому. Если получателем данных выступает БД, вместо объектов данные крутятся в БД.
__________________
Не стоит путать форумы с богадельнями. © Bargest
Ответить с цитированием
  #7  
Старый 27.02.2015, 20:58
lmikle lmikle вне форума
Модератор
 
Регистрация: 17.04.2008
Сообщения: 8,087
Версия Delphi: 7, XE3, 10.2
Репутация: 49089
По умолчанию

Для начала. В потоке COM-подсистему инициализировал (CoInitialize)? Для приложения (читай: главного потока) это для тебя делает Application, а для дочерних потоков надо делать самому.
Ответить с цитированием
  #8  
Старый 28.02.2015, 09:14
Аватар для Страдалецъ
Страдалецъ Страдалецъ вне форума
Гуру
 
Регистрация: 09.03.2009
Адрес: На курорте, из окна вижу теплое Баренцево море. Бррр.
Сообщения: 4,723
Репутация: 52347
По умолчанию

Код:
type
  TObjThread = class(TThread)
  private
    sObj: IXMLNode;
    { Private declarations }
  protected
    procedure Execute; override;
    constructor Create(nObj: IXMLNode);
  end;
__________________
Жизнь такова какова она есть и больше никакова.
Помогаю за спасибо.
Ответить с цитированием
  #9  
Старый 28.02.2015, 09:22
Аватар для Freeman
Freeman Freeman вне форума
Местный
 
Регистрация: 05.10.2012
Адрес: Санкт-Петербург
Сообщения: 577
Версия Delphi: 6
Репутация: выкл
По умолчанию

У меня еще такой вопрос: медленно -- это насколько медленно?
__________________
Не стоит путать форумы с богадельнями. © Bargest
Ответить с цитированием
  #10  
Старый 05.03.2015, 17:14
Аватар для Uniq!
Uniq! Uniq! вне форума
Местный
 
Регистрация: 29.09.2010
Сообщения: 539
Версия Delphi: Delphi XE3
Репутация: 374
По умолчанию

Работа такая работа, про форум совсем забыл.

Медленно, это около 6ти часов. структура очень жёсткая, много полей и записей. С самой ошибкой я уже разобрался, не хватало
Код:
override;

Но проблему я так и не смог побороть, чтоб один поток обрабатывал определённую часть файла (читай: один объект).

Переводить всё на SAX не имеет смысла, ибо файл всегда можно разделить. Просто из-за хлипкости генерирующего движка приходиться работать по принципу: работает? не трогай! Конкретно сейчас ограничиваюсь имеющимся файлом.

Про
Код:
CoInitialize
надо почитать. Разберусь с этим - отпишусь.
Ответить с цитированием
  #11  
Старый 05.03.2015, 18:16
Аватар для Freeman
Freeman Freeman вне форума
Местный
 
Регистрация: 05.10.2012
Адрес: Санкт-Петербург
Сообщения: 577
Версия Delphi: 6
Репутация: выкл
По умолчанию

Цитата:
Сообщение от Uniq!
Медленно, это около 6ти часов. структура очень жёсткая, много полей и записей.
Чего? Какие 6 часов? Что можно делать 6 часов с 20 МБ данных? Пример кода обработки можно?

А насчет многопоточности -- мож стоит вначале poAsyncLoad в IXMLDocument.ParseOptions установить? Дока говорит, что это якобы должно увеличить производительность.
__________________
Не стоит путать форумы с богадельнями. © Bargest

Последний раз редактировалось Freeman, 05.03.2015 в 18:19.
Ответить с цитированием
  #12  
Старый 07.03.2015, 17:20
Аватар для Uniq!
Uniq! Uniq! вне форума
Местный
 
Регистрация: 29.09.2010
Сообщения: 539
Версия Delphi: Delphi XE3
Репутация: 374
По умолчанию

lmikle, огромное спасибо. Справился.
Собрал многопоточное приложение так, как это было необходимо.

Наткнулся на проблему: нужно ограничить количество потоков.
Прочитал, что нужен менеджер потоков. Массив ограниченной длины, в котором будут жить и умирать потоки.
Код получился таким. Чую, что можно проще. С радостью выслушал бы критику.

Код:
  iObjs := XMLDoc.DocumentElement;
  SetLength(XMLObjects, iObjs.ChildNodes.Count);

  for ObjNr := 0 to iObjs.ChildNodes.Count - 1 do
    XMLObjects[ObjNr] := iObjs.ChildNodes[ObjNr];

  while Length(XMLObjects) <> 0 do
  begin
    for i := 0 to 4 do
    begin
      if (Parser[i] = nil) or (Parser[i].Finished) then
      begin
        Parser[i] := TObjectParser.Create
          (iObjs.ChildNodes[Length(XMLObjects) - 1], i + 1);
        SetLength(XMLObjects, Length(XMLObjects) - 1);
      end;

    end;

    Application.ProcessMessages;
  end;

XMLObjects[ObjNr] - массив задач, которые нужно выполнить *массив объектов
Parser[i] - массив, живущих потоков
Ответить с цитированием
  #13  
Старый 10.03.2015, 11:06
Аватар для Freeman
Freeman Freeman вне форума
Местный
 
Регистрация: 05.10.2012
Адрес: Санкт-Петербург
Сообщения: 577
Версия Delphi: 6
Репутация: выкл
По умолчанию

А я по мотивам этой темы потестировал MSXML на стометровом XML-файле. Программа выжрала порядка 1 ГБ ОЗУ и упала с сообщением о нехватке памяти. А я всего лишь максимальную вложенность тегов посчитать хотел. Теперь раздумываю о написании XML-парсера на основе своей библиотеки для компилятора.
__________________
Не стоит путать форумы с богадельнями. © Bargest
Ответить с цитированием
  #14  
Старый 11.03.2015, 01:31
lmikle lmikle вне форума
Модератор
 
Регистрация: 17.04.2008
Сообщения: 8,087
Версия Delphi: 7, XE3, 10.2
Репутация: 49089
По умолчанию

Цитата:
Сообщение от Uniq!
lmikle, огромное спасибо. Справился.
Собрал многопоточное приложение так, как это было необходимо.

Наткнулся на проблему: нужно ограничить количество потоков.
Прочитал, что нужен менеджер потоков. Массив ограниченной длины, в котором будут жить и умирать потоки.
Код получился таким. Чую, что можно проще. С радостью выслушал бы критику.

Не с того конца идешь.
я бы сделал очередь заданий и массив потоков. Ну это, в принципе, как у тебя и сделано. А вот дальше я просто сделал бы так, что потоки сами берут задания из очереди и выполняют их (не забудь синхронизацию через критическую секцию). А основной код периодически, например, опрашивает состояние потоков. Как только все потоки Idle, то все сделано.
Ответить с цитированием
  #15  
Старый 14.03.2015, 23:27
Аватар для Uniq!
Uniq! Uniq! вне форума
Местный
 
Регистрация: 29.09.2010
Сообщения: 539
Версия Delphi: Delphi XE3
Репутация: 374
По умолчанию

Цитата:
Сообщение от lmikle
Не с того конца идешь.
Т.е. я должен создать TThreadList.
Напичкать его указателями на i-ые объекты
И научить свой класс потока общаться с этим листом.

Я не понимаю что значит "сами берут задания из очереди".

было вот так: я сам передавал ему информацию на обработку.
Код:
Parser := TParser.Create(iObjs.ChildNodes[Length(XMLObjects) - 1]);

А теперь я так понимаю создаётся поток без параметров.
И в методе Execute
1) лочит массив, где лежит информация
2) забирает её к себе на обработку (зануляя при этом ячейку массива
3) разлочит массив.

Так что ли?

Последний раз редактировалось Uniq!, 15.03.2015 в 20:39.
Ответить с цитированием
Ответ


Delphi Sources

Опции темы Поиск в этой теме
Поиск в этой теме:

Расширенный поиск
Опции просмотра

Ваши права в разделе
Вы не можете создавать темы
Вы не можете отвечать на сообщения
Вы не можете прикреплять файлы
Вы не можете редактировать сообщения

BB-коды Вкл.
Смайлы Вкл.
[IMG] код Вкл.
HTML код Выкл.
Быстрый переход


Часовой пояс GMT +3, время: 03:42.


 

Сайт

Форум

FAQ

Соглашения

Прочее

 

Copyright © Форум "Delphi Sources" by BrokenByte Software, 2004-2025