Uygulamanın Karanlık Yüzü. ProcessMessages

Makale Marcus Junglas tarafından sunulmuştur

Delphi'de bir olay işleyiciyi programlarken ( Tıklamada TButton olması durumunda), uygulamanızın bir süre meşgul olması gerektiği zaman gelir; kodun büyük bir dosya yazması veya bazı verileri sıkıştırması gerekir.

Bunu yaparsanız, uygulamanız kilitli görünüyor. Formunuz artık taşınamıyor ve düğmeler hayat belirtisi göstermiyor. Görünüşe göre çöktü.

Bunun nedeni, bir Delpi uygulamasının tek iş parçacıklı olmasıdır. Yazdığınız kod, bir olay gerçekleştiğinde Delphi'nin ana iş parçacığı tarafından çağrılan bir grup yordamı temsil eder. Ana iş parçacığının geri kalanı sistem mesajlarını ve form ve bileşen taşıma işlevleri gibi diğer şeyleri işlemektedir.

Bu nedenle, uzun bir iş yaparak olay işlemeyi bitirmezseniz, uygulamanın bu iletileri işlemesini önlersiniz.

Bu tür sorunlar için ortak bir çözüm "Uygulama" dır. ProcessMessages". "Uygulama", TApplication sınıfının global bir nesnesidir.

Uygulama. Processmessage, pencere hareketleri, düğme tıklamaları vb. Gibi bekleyen tüm mesajları işler. Genellikle uygulamanızın "çalışmasını" sağlamak için basit bir çözüm olarak kullanılır.

instagram viewer

Ne yazık ki, "ProcessMessages" arkasındaki mekanizma büyük karışıklığa neden olabilir kendi özellikleri vardır!

ProcessMessage ne işe yarar?

PprocessMessages uygulama ileti kuyruğundaki tüm bekleyen sistem iletilerini işler. Windows iletileri çalışan tüm uygulamalarla "konuşmak" için kullanır. Kullanıcı etkileşimi, iletiler aracılığıyla forma getirilir ve "ProcessMessages" tarafından yönetilir.

Fare bir TButton üzerinde ilerliyorsa, ProgressMessages bu olayda olması gerekeni yapar düğmeyi "basılı" bir duruma yeniden boyayın ve elbette, atadıysanız OnClick () işleme prosedürüne bir çağrı yapın.

Sorun budur: ProcessMessage'lara yapılan herhangi bir çağrı, herhangi bir olay işleyicisine tekrarlanan çağrı içerebilir. İşte bir örnek:

Bir düğmenin OnClick çift işleyicisi ("iş") için aşağıdaki kodu kullanın. For-deyimi, arada bir ProcessMessages çağrıları ile uzun bir işleme işi simüle eder.

Bu, daha iyi okunabilirlik için basitleştirilmiştir:

{MyForm'da:}
WorkLevel: tamsayı;
{OnCreate:}
Çalışma Seviyesi: = 0;
prosedür TForm1.WorkBtnClick (Gönderen: TObject);
var
döngü: tamsayı;
başla
inc (WorkLevel);
için döngü: = 1 için 5 yapmak
başla
Memo1.Lines. Ekle ('- İş' + IntToStr (WorkLevel) + ', Döngü' + IntToStr (döngü);
Uygulama. ProcessMessages;
uyku (1000); // veya başka bir iş
son;
Memo1.Lines. Ekle ('Work' + IntToStr (WorkLevel) + 'sona erdi.');
dec (WorkLevel);
son;

"ProcessMessage" OLMADAN Düğmeye kısa bir süre TWICE tuşuna basıldıysa, nota aşağıdaki satırlar yazılır:

 - İş 1, Döngü 1
- İş 1, Döngü 2
- İş 1, Döngü 3
- İş 1, Döngü 4
- İş 1, Döngü 5
Çalışma 1 sona erdi.
- İş 1, Döngü 1
- İş 1, Döngü 2
- İş 1, Döngü 3
- İş 1, Döngü 4
- İş 1, Döngü 5
Çalışma 1 sona erdi.

Yordam meşgulken, form herhangi bir tepki göstermez, ancak ikinci tıklama Windows tarafından ileti kuyruğuna yerleştirilir. "OnClick" bittikten hemen sonra tekrar çağrılacaktır.

"ProcessMessages" DAHİL, çıktı çok farklı olabilir:

 - İş 1, Döngü 1
- İş 1, Döngü 2
- İş 1, Döngü 3
- İş 2, Döngü 1
- İş 2, Döngü 2
- İş 2, Çevrim 3
- İş 2, Döngü 4
- İş 2, Döngü 5
İş 2 sona erdi.
- İş 1, Döngü 4
- İş 1, Döngü 5
Çalışma 1 sona erdi.

Bu kez form tekrar çalışıyor gibi görünüyor ve herhangi bir kullanıcı etkileşimini kabul ediyor. Böylece düğmeye, anında işlenecek olan ilk "işçi" işlevi AGAIN sırasında yarıya kadar basılır. Gelen tüm olaylar diğer işlev çağrıları gibi işlenir.

Teorik olarak, her "ProgressMessages" çağrısı sırasında HERHANGİ bir miktar tıklama ve kullanıcı mesajı "yerinde" olabilir.

Bu yüzden kodunuza dikkat edin!

Farklı örnek (basit sözde kodda!):

prosedür OnClickFileWrite ();
var myfile: = TFileStream;
başla
myfile: = TFileStream.create ('myOutput.txt');
Deneyin
süre BytesReady> 0 yapmak
başla
dosyam. Yazma (DataBlock);
dec (BytesReady, sizeof (DataBlock));
DataBlock [2]: = # 13; {test satırı 1}
Uygulama. ProcessMessages;
DataBlock [2]: = # 13; {test satırı 2}
son;
en sonunda
myfile.free;
son;
son;

Bu işlev büyük miktarda veri yazar ve her veri bloğu yazıldığında "ProcessMessages" kullanarak uygulamanın "kilidini" açmaya çalışır.

Kullanıcı düğmeyi tekrar tıklarsa, dosya hala yazılırken aynı kod yürütülür. Böylece dosya 2 kez açılamaz ve prosedür başarısız olur.

Belki uygulamanız arabelleği boşaltmak gibi bazı hata kurtarma işlemleri yapar.

Olası bir sonuç olarak "Datablock" serbest bırakılacak ve ilk kod erişildiğinde "aniden" bir "Erişim İhlali" oluşturacaktır. Bu durumda: test hattı 1 çalışacak, test hattı 2 çökecektir.

Daha iyi bir yol:

Bunu kolaylaştırmak için, tüm kullanıcı girişini engelleyen ancak bunu kullanıcıya göstermeyen "etkin: = false" Formunun tamamını ayarlayabilirsiniz (tüm Düğmeler gri değildir).

Daha iyi bir yol, tüm düğmeleri "devre dışı" olarak ayarlamak olabilir, ancak örneğin bir "İptal" düğmesini tutmak istiyorsanız bu karmaşık olabilir. Ayrıca, devre dışı bırakmak için tüm bileşenleri gözden geçirmeniz gerekir ve tekrar etkinleştirildiklerinde, devre dışı durumda kalanların olup olmadığını kontrol etmeniz gerekir.

Yapabilirdiniz Enabled özelliği değiştiğinde kapsayıcı alt denetimlerini devre dışı bırakma.

"TNotifyEvent" sınıf adından da anlaşılacağı gibi, yalnızca olaya kısa süreli reaksiyonlar için kullanılmalıdır. Zaman alan kod için en iyi yol IMHO'nun tüm "yavaş" kodu kendi iş parçacığına koymasıdır.

"PrecessMessages" ve / veya bileşenlerin etkinleştirilmesi ve devre dışı bırakılması ile ilgili sorunlar için, ikinci iplik çok karmaşık görünmüyor.

Basit ve hızlı kod satırlarının bile saniyeler sürebileceğini unutmayın, ör. disk sürücüsünde bir dosya açmak sürücünün dönmesi bitene kadar beklemek zorunda kalabilir. Sürücünüz çok yavaş olduğu için uygulamanız çöküyorsa çok iyi görünmüyor.

Bu kadar. Bir dahaki sefere "Uygulama" yı eklediğinizde. ProcessMessage ", iki kere düşünün;)