Yorumları: 1.499
Konuları: 83
Kayıt Tarihi: 05-08-2016
Aktif Kullandığınız Delphi Sürümü:
Rep Puanı: 12.314 Üstad
Amaç: Her 15 dakikada bir belirlenen bir işi yapması istenen bir thread'in Sleep / SleepEx API'leri vasıtası ile kodlanması ve uygulamanın ana thread'inden istenilen herhangi bir zamanda thread'den çıkılmasını sağlamak. (Elbette TerminateThread kullanmadan)
Not: 20 puan değerindedir.
Mal sahibi, mülk sahibi
Hani bunun ilk sahibi ?
Mal da yalan mülk de yalan
Var biraz da sen oyalan...
Yorumları: 357
Konuları: 30
Kayıt Tarihi: 10-07-2017
Aktif Kullandığınız Delphi Sürümü:
- Delphi 10.3
- Lazarus / FPC
Rep Puanı: 4.215 Uzman
08-11-2017, Saat: 17:23
(Son Düzenleme: 08-11-2017, Saat: 17:40, Düzenleyen: savasabd.)
unit Unit4;
interface
uses
Winapi.Windows, Winapi.Messages, System.SysUtils, System.Variants, System.Classes, Vcl.Graphics,
Vcl.Controls, Vcl.Forms, Vcl.Dialogs, Vcl.StdCtrls;
type
TForm4 = class(TForm)
Label1: TLabel;
Button1: TButton;
Button2: TButton;
procedure Button2Click(Sender: TObject);
procedure Button1Click(Sender: TObject);
private
{ Private declarations }
public
{ Public declarations }
end;
var
Form4: TForm4;
cikis:bool;
implementation
{$R *.dfm}
function saydir(param: Pointer) : DWORD;
var
i:integer;
begin
;
for I := 0 to 1000 do
begin
Form4.Label1.Caption:=IntTostr(i); //labelde 1den bine kadar yazdırıyoruz
if(cikis=true)then break;
Sleep(10);
end;
end;
procedure TForm4.Button1Click(Sender: TObject);
var
id : Dword;
begin
cikis := false;
CreateThread(nil, 0, @saydir, 0, 0, id);
button1.Enabled:=false;
end;
procedure TForm4.Button2Click(Sender: TObject);
begin
cikis := true;
button1.Enabled :=true;
end;
procedure TForm4.Timer1Timer(Sender: TObject);//timer'in intervali 900000
var
id : Dword;
begin
cikis :=false;
CreateThread(nil, 0, @saydir, Label1, 0, id);
button1.Enabled:=false;
end;
end.
Yorumları: 1.499
Konuları: 83
Kayıt Tarihi: 05-08-2016
Aktif Kullandığınız Delphi Sürümü:
Rep Puanı: 12.314 Üstad
(08-11-2017, Saat: 17:23)savasabd Adlı Kullanıcıdan Alıntı: unit Unit4;
interface
uses
Winapi.Windows, Winapi.Messages, System.SysUtils, System.Variants, System.Classes, Vcl.Graphics,
Vcl.Controls, Vcl.Forms, Vcl.Dialogs, Vcl.StdCtrls;
type
TForm4 = class(TForm)
Label1: TLabel;
Button1: TButton;
Button2: TButton;
procedure Button2Click(Sender: TObject);
procedure Button1Click(Sender: TObject);
private
{ Private declarations }
public
{ Public declarations }
end;
var
Form4: TForm4;
cikis:bool;
implementation
{$R *.dfm}
function saydir(param: Pointer) : DWORD;
var
i:integer;
begin
;
for I := 0 to 1000 do
begin
Form4.Label1.Caption:=IntTostr(i); //labelde 1den bine kadar yazdırıyoruz
if(cikis=true)then break;
Sleep(10);
end;
end;
procedure TForm4.Button1Click(Sender: TObject);
var
id word;
begin
cikis := false;
CreateThread(nil, 0, @saydir, 0, 0, id);
button1.Enabled:=false;
end;
procedure TForm4.Button2Click(Sender: TObject);
begin
cikis := true;
button1.Enabled :=true;
end;
end.
Uyanıkça bir çözüm yoluna gitmişsiniz. Ama istediğim bu değil. Sleep ya da SleepEx ile toplam 15 dakika bekleyeceksiniz. Parça parça değil. Sorunun amacı buna bina edilmiş zaten.
Ayrıca bir thread içinden bir başka thread(main thread) içindeki paylaşımlı alanlara erişir iken dikkatli olmakta fayda var. Bu amaçla, TThread.Queue, TThread.Synchronize metodlarını kullanmak ya da başka senkronizasyon nesneleri ile paylaşımlı kodu sarmalamak gerekir.
Mal sahibi, mülk sahibi
Hani bunun ilk sahibi ?
Mal da yalan mülk de yalan
Var biraz da sen oyalan...
Yorumları: 253
Konuları: 16
Kayıt Tarihi: 26-09-2016
Aktif Kullandığınız Delphi Sürümü:
Rep Puanı: 3.279 Uzman
Hocam Event Nesneleri kullanarak mı yapacağız.
Yorumları: 1.499
Konuları: 83
Kayıt Tarihi: 05-08-2016
Aktif Kullandığınız Delphi Sürümü:
Rep Puanı: 12.314 Üstad
(08-11-2017, Saat: 22:15)ismailkocacan Adlı Kullanıcıdan Alıntı: Hocam Event Nesneleri kullanarak mı yapacağız. 
İstediğiniz kadar Event kullanabilirsiniz. Ama bekleme kodunu Event üzerinden WaitForSingleObject/WaitForMultipleObjects ya da Event.WaitFor ile yapmayacaksınız. Sleep ya da SleepEx dememin önemli bir nedeni var. Açıklar isem, soru ve sorunun kazandırması gerekenleri yani amacı yitiririz.
Düşünüldüğü kadar kolay bir soru değil. Ama o kadar da zor değil
Küçücük bir ip ucu vereyim ki ilerlemek kolay olsun. APC
Mal sahibi, mülk sahibi
Hani bunun ilk sahibi ?
Mal da yalan mülk de yalan
Var biraz da sen oyalan...
Yorumları: 1.572
Konuları: 88
Kayıt Tarihi: 09-08-2016
Rep Puanı: 13.841 Üstad
10-11-2017, Saat: 00:19
(Son Düzenleme: 10-11-2017, Saat: 00:20, Düzenleyen: SimaWB.)
Thread'imiz içinde CreateWaitableTimer ile Timer oluşturmalıyız.
SetWaitableTimer ile bu timer'ın ayarlarını yapmalıyız.
Sonrasında SleepEx ile beklemeye geçmeliyiz.
SleepEx'i istediğimi zaman sonlandırabilmek için 2. parametresini True yapmalıyız.
SleepEx'ten çıkmak için QueueUserAPC ile thread'imize tanımladığımız bir APC'yi (Asynchronous Procedure Call) gönderebiliriz.
procedure TTimerThread.Execute;
var
timerHandle: THandle;
larg_int : LARGE_INTEGER;
begin
timerHandle := CreateWaitableTimer(nil, False, 'WaitableTimer');
larg_int.QuadPart := -9000000000;
SetWaitableTimer(timerHandle, Int64(larg_int), 900000, @Callback, nil, False);
while not Terminated do
SleepEx(INFINITE, True);
CloseHandle(timerHandle);
end;
procedure Callback(P: Pointer; H, L: LongWord); stdcall;
begin
// Burada periyodik olarak yapılacak işlemler yazılacak
end;
SetWaitableTimer'ın 2. parametresi Timer'ın ne zaman başlayacağını belirtir ve 100 NANO saniye hassasiyettedir. Dilediğiniz bir tarih-saati belirtebileceğiniz gibi bizim örneğimizde olduğu gibi eksi değer kullanılırsa fonksiyon çağrıldıktan ne kadar zaman sonra Timer'ın tetikleneceği belirtilmiş olur. (Bizden istenen 15 dakika olduğu için yukarıdaki değer girilmiştir)
SetWaitableTimer'ın 3. parametresi ise periyottur. Bu parametre ise saniye cinsindendir ve 15 dakikaya göre ayarlanmıştır. 4. parametre ise Timer tetiklendiğinde çağrılacak fonksiyonu belirtir. Örnek Callback fonksiyonu yukarıda verilmiştir.
Threadimizin kullanımına örnek için bir form, üzerinde 2 buton ile deneme yapabiliriz:
type
TForm1 = class(TForm)
btnStart: TButton;
btnStop: TButton;
Memo1: TMemo;
procedure btnStartClick(Sender: TObject);
procedure btnStopClick(Sender: TObject);
private
TT: TTimerThread;
public
{ Public declarations }
end;
var
Form1: TForm1;
implementation
{$R *.dfm}
procedure APC(Param: Cardinal); stdcall;
begin
//
end;
procedure TForm1.btnStartClick(Sender: TObject);
begin
TT := TTimerThread.Create(True);
TT.FreeOnTerminate := True;
TT.Start;
end;
procedure TForm1.btnStopClick(Sender: TObject);
begin
QueueUserAPC(@APC, TT.Handle, 0);
TT.Terminate;
end;
There's no place like 127.0.0.1
Yorumları: 1.499
Konuları: 83
Kayıt Tarihi: 05-08-2016
Aktif Kullandığınız Delphi Sürümü:
Rep Puanı: 12.314 Üstad
(10-11-2017, Saat: 00:19)SimaWB Adlı Kullanıcıdan Alıntı: Thread'imiz içinde CreateWaitableTimer ile Timer oluşturmalıyız.
SetWaitableTimer ile bu timer'ın ayarlarını yapmalıyız.
Sonrasında SleepEx ile beklemeye geçmeliyiz.
SleepEx'i istediğimi zaman sonlandırabilmek için 2. parametresini True yapmalıyız.
SleepEx'ten çıkmak için QueueUserAPC ile thread'imize tanımladığımız bir APC'yi (Asynchronous Procedure Call) gönderebiliriz.
procedure TTimerThread.Execute;
var
timerHandle: THandle;
larg_int : LARGE_INTEGER;
begin
timerHandle := CreateWaitableTimer(nil, False, 'WaitableTimer');
larg_int.QuadPart := -9000000000;
SetWaitableTimer(timerHandle, Int64(larg_int), 900000, @Callback, nil, False);
while not Terminated do
SleepEx(INFINITE, True);
CloseHandle(timerHandle);
end;
procedure Callback(P: Pointer; H, L: LongWord); stdcall;
begin
// Burada periyodik olarak yapılacak işlemler yazılacak
end;
SetWaitableTimer'ın 2. parametresi Timer'ın ne zaman başlayacağını belirtir ve 100 NANO saniye hassasiyettedir. Dilediğiniz bir tarih-saati belirtebileceğiniz gibi bizim örneğimizde olduğu gibi eksi değer kullanılırsa fonksiyon çağrıldıktan ne kadar zaman sonra Timer'ın tetikleneceği belirtilmiş olur. (Bizden istenen 15 dakika olduğu için yukarıdaki değer girilmiştir)
SetWaitableTimer'ın 3. parametresi ise periyottur. Bu parametre ise saniye cinsindendir ve 15 dakikaya göre ayarlanmıştır. 4. parametre ise Timer tetiklendiğinde çağrılacak fonksiyonu belirtir. Örnek Callback fonksiyonu yukarıda verilmiştir.
Threadimizin kullanımına örnek için bir form, üzerinde 2 buton ile deneme yapabiliriz:
type
TForm1 = class(TForm)
btnStart: TButton;
btnStop: TButton;
Memo1: TMemo;
procedure btnStartClick(Sender: TObject);
procedure btnStopClick(Sender: TObject);
private
TT: TTimerThread;
public
{ Public declarations }
end;
var
Form1: TForm1;
implementation
{$R *.dfm}
procedure APC(Param: Cardinal); stdcall;
begin
//
end;
procedure TForm1.btnStartClick(Sender: TObject);
begin
TT := TTimerThread.Create(True);
TT.FreeOnTerminate := True;
TT.Start;
end;
procedure TForm1.btnStopClick(Sender: TObject);
begin
QueueUserAPC(@APC, TT.Handle, 0);
TT.Terminate;
end;
Ellerinize sağlık. Puanınızı verdim sevgili üstadım. Umarım bu soru, araştırmaya ve APC'yi öğrenmeye; yumuşak ve kontrollü bir şekilde thread durdurmayayı öğrenmeye vesile olabilmiştir. Hedeflenen, APC kavramı ve QueueUserAPC apisi idi.
Mal sahibi, mülk sahibi
Hani bunun ilk sahibi ?
Mal da yalan mülk de yalan
Var biraz da sen oyalan...
Yorumları: 1.572
Konuları: 88
Kayıt Tarihi: 09-08-2016
Rep Puanı: 13.841 Üstad
Asıl ben teşekkür ederim. Benim için çok öğretici oldu. Açıkçası QueueUserAPC'yi daha önce hiç duymamıştım  Bu vesile ile öğrendim.
Bu arada; bu soruya cevap vermemde bana en çok yardımcı olan sayfayı belirtmeden geçemeyeceğim:
http://www.tugrulhelvaci.com/?p=528
There's no place like 127.0.0.1
Yorumları: 1.499
Konuları: 83
Kayıt Tarihi: 05-08-2016
Aktif Kullandığınız Delphi Sürümü:
Rep Puanı: 12.314 Üstad
10-11-2017, Saat: 10:12
(Son Düzenleme: 10-11-2017, Saat: 10:13, Düzenleyen: Tuğrul HELVACI.)
Mal sahibi, mülk sahibi
Hani bunun ilk sahibi ?
Mal da yalan mülk de yalan
Var biraz da sen oyalan...
Yorumları: 884
Konuları: 35
Kayıt Tarihi: 12-08-2016
Aktif Kullandığınız Delphi Sürümü:
Rep Puanı: 3.853 Uzman
Bu Access violation bölümü baklava gibi bişi oldu, tadından yenmiyor
|