Delphi Can

Orjinalini görmek için tıklayınız: Thread Durdurma
Şu anda (Arşiv) modunu görüntülemektesiniz. Orjinal Sürümü Görüntüle internal link
Sayfalar: 1 2
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.
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.
(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
idBig Grinword;
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.
Hocam Event Nesneleri kullanarak mı yapacağız.  Cry
(08-11-2017, Saat: 22:15)ismailkocacan Adlı Kullanıcıdan Alıntı: [ -> ]Hocam Event Nesneleri kullanarak mı yapacağız.  Cry

İ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 Wink

Küçücük bir ip ucu vereyim ki ilerlemek kolay olsun. APC
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;
(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.
Asıl ben teşekkür ederim. Benim için çok öğretici oldu. Açıkçası QueueUserAPC'yi daha önce hiç duymamıştım Sad  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
Smile
(10-11-2017, Saat: 09:51)SimaWB Adlı Kullanıcıdan Alıntı: [ -> ]Asıl ben teşekkür ederim. Benim için çok öğretici oldu. Açıkçası QueueUserAPC'yi daha önce hiç duymamıştım Sad  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
Smile

Rica ederim üstad. Aslında çeşitli zamanlarda çeşitli vesileler ile ilgili API'den biraz bahsetmiştim Smile 2 örnek bulabildim kısaca şimdilik Smile

Örneğin
Örneğin
Bu Access violation bölümü baklava gibi bişi oldu, tadından yenmiyor Smile
Sayfalar: 1 2