|
|
Регистрация | << Правила форума >> | FAQ | Пользователи | Календарь | Поиск | Сообщения за сегодня | Все разделы прочитаны |
|
Опции темы | Поиск в этой теме | Опции просмотра |
#1
|
|||
|
|||
Перебор данных из бд в несколько потоков
Пытаюсь перебирать так ничего не выходит скорость никак не меняется. Подскажите пожалуйста в чем может быть причина
Код:
...... type TMyThread = class(TThread) private protected procedure Execute; override; procedure memoAdd; public msg:string; N:integer; end; ....... private MyT:array of TMyThread; public o:integer; CounArray:integer; end; ......... procedure TMyThread.Execute; var //nw :string; b,A:integer; begin dm.AdUpdatePOsitions.ExecSQL; b:= N * form5.CounArray; //Присваеваем единицу A := b - form5.CounArray; while A <> b do begin inc(A); msg:= inttostr(A) + ' Поток номер '+ inttostr(N) +' запущен! '+inttostr(Form5.o)+' '+dm.AdUpdatePOsitions.FieldByName('Words').AsString; Synchronize(memoAdd); dm.AdUpdatePOsitions.Next; end; end; procedure TMyThread.memoAdd; begin Form5.Memo1.Lines.Add(msg); end; procedure TForm5.BitBtn1Click(Sender: TObject); var i,nn:integer; begin dm.AdUpdatePOsitions.SQL.Clear; dm.AdUpdatePOsitions.SQL.Add('SELECT * FROM Position'); dm.AdUpdatePOsitions.Active:=True; dm.AdUpdatePOsitions.first; nn := dm.AdUpdatePOsitions.RecordCount; //подсчет количества данных в массиве CounArray := Trunc ( nn / SpinEdit1.value ); //округлим в меньшую сторону //создание потоков for i := 1 to SpinEdit1.value do begin SetLength(MyT,i+1); MyT[i]:=TMyThread.Create(True); MyT[i].Priority:= tpLowest; MyT[i].N:=i; MyT[i].FreeOnTerminate:=True; MyT[i].Start; end; end; |
#2
|
||||
|
||||
Если потоков много, а датасет один, то будет ли быстрее?
По идее надо создать несколько датасетов, по числу потоков, чтоб каждый поток работал только со "своим" датасетом. |
#3
|
|||
|
|||
Guaho, в общем правильно, но можно сделать и с одним источником, просто использовать его как очередь заданий. Имеет смысл если только обработка одной записи занимает некоторое время, иначе все потоки будут "толпиться" на доступе к этому компоненту.
Что бы такое сделать, надо использоывть объекты синхронизации для блокировки доступа к ДатаСету, т.к. каждый поток при получении данных сначала блокирукт CriticalSection, потом делает ДатаСету Next, читает данные из записи, снимает блокировку и уже потом начинает обрабатывать данные внутри себя. |
#4
|
|||
|
|||
Обработка данных будет занимать около 5 секунд в одном потоке, потом будет переход на следующую строку в БД. Что можете порекомендовать в этом случае?
|
#5
|
||||
|
||||
Хотелось бы уточнить, а как именно Вы обрабатываете запись? В запросе Вы выбираете все поля, но если их много и не все требуются, то может имеет смысл выбрать только нужные. А ещё я бы порекомендовал в циклах не использовать FieldByName, а обращаться к полю по его индексу через Fields[]. Про это даже в справке написано.
Всегда пишите код так, будто сопровождать его будет склонный к насилию психопат, который знает, где вы живете. |
#6
|
||||
|
||||
Цитата:
|
#7
|
|||
|
|||
Цитата:
Crypto-checker https://chrome.google.com/webstore/detail/crypto-checker/bahhmhhmnedfhndhdagobnknhcflhhkp?hl=ru - расширение для начинающих трейдеров, которое может помочь им в выборе криптовалютной биржи, обойти стороной мошенников и проводить криптоарбитраж безопасно Последний раз редактировалось OnlyTheProg, 17.01.2022 в 18:26. |
#8
|
|||
|
|||
В общем в потоке все перебирает отлично, но вот делаю запись в бд Access через синхронизацию
использую вот эту функцию RecordPositions. Создается большая очередь и программа работает не очень быстро из-за этого как можно произвести запись прямо в потоке? (например без синхронизации) Код:
function RecordPositions(a,b,c:integer; d,e,r,t:string):string; begin dm.AdQ_Pos1.Active:=true; dm.AdQ_Pos1.Append; dm.AdQ_Pos1.FieldByName('t').AsString := t; dm.AdQ_Pos1.FieldByName('b').AsInteger := b; dm.AdQ_Pos1.FieldByName('c').AsInteger := c; dm.AdQ_Pos1.FieldByName('d').AsString := d; dm.AdQ_Pos1.FieldByName('e').AsString := e; dm.AdQ_Pos1.FieldByName('r').AsString := r; dm.AdQ_Pos1.FieldByName('a').AsInteger := a; dm.AdQ_Pos1.POST; dm.AdQ_Pos1.Active:=false; Exit; end; |