Обновление QProgressbar из другого потока
Я разработал свой собственный гибридный потоковый шифр, и для графического интерфейса я использую Qt. Изначально я писал его в одном потоке, но это был потоковый шифр, который делал GUI дисфункциональным при работе с большими файлами. Поэтому я переключил шифрование / дешифрование на отдельный Qthread. Кроме того, чтобы показать прогресс, я включил стандартные QProgressbar на графический интерфейс. Но когда я запускаю файл ввода-вывода, шифрование / дешифрование работает отлично, но индикатор выполнения не обновляется должным образом. После всей операции завершается, индикатор выполнения внезапно переходит от 0% до 100%, показывая, что он не получил возможности обновить во время операции. Для кода я отправил завершенный процент из FileCrypto в основной поток GUI на слот setValue(int) QProgressbar. Поскольку это не сработало, я также попытался отправить int poitner в поток FileCrypto, обновляя указатель с процентом и используя QTimer в потоке GUI, чтобы проверить значение значения int локально и обновить прогресс-бар, но все равно я получил тот же результат.
Вот мой код:
Класс FileCrypto:
#include <QThread>
#include <QFile>
#include <PolyVernam.h> //my algo header
class FileCrypto : public QThread
{
Q_OBJECT
public:
FileCrypto(QString, QString, int);
bool stopIt;
protected:
void run();
signals:
void completed(int);
void msg(QString);
void pathMsg1(QString);
void pathMsg2(QString);
void keyMsg(QString);
private:
QFile src, dest;
QString tag;
int mode;
qint64 length;
PolyVernam pv;
};
Код:
#include <FileCrypto.h>
FileCrypto::FileCrypto(QString input, QString keyFile, int mode)
{
stopIt = false;
this->mode = mode;
src.setFileName(input);
if(mode == 1)
{
emit msg("Current Encryption/Decryption status: Encrypting file... :D:D");
tag = "-encrypted";
pv.setMode("encrypt", "");
}
else
{
emit msg("Current Encryption/Decryption status: Decrypting file... :D:D");
tag = "-decrypted";
pv.setMode("decrypt", keyFile);
}
dest.setFileName(QFileInfo(src).absolutePath() + "/" + QFileInfo(src).baseName()
+ tag + "." + QFileInfo(src).completeSuffix());
length = src.bytesAvailable();
}
void FileCrypto::run()
{
qint64 done = 0;
quint8 r, outChar;
char ch;
QDataStream in(&src);
in.setVersion(QDataStream::Qt_4_7);
src.open(QIODevice::ReadOnly);
QDataStream out(&dest);
out.setVersion(QDataStream::Qt_4_7);
dest.open(QIODevice::WriteOnly);
while(!in.atEnd() && !stopIt)
{
done++;
in >> r;
ch = char(r);
if(mode == 1)
outChar = pv.encrypt(QString(ch)).at(0).toAscii();
else
outChar = pv.decrypt(QString(ch)).at(0).toAscii();
out << outChar;
emit completed(int((done / length) * 100));
}
src.close();
dest.close();
if(stopIt)
this->exit(0);
if(mode == 1)
{
emit pathMsg1(QFileInfo(src).absoluteFilePath());
emit pathMsg2(QFileInfo(dest).absoluteFilePath());
}
else
{
emit pathMsg1(QFileInfo(dest).absoluteFilePath());
emit pathMsg2(QFileInfo(src).absoluteFilePath());
}
emit keyMsg(pv.keyFilePath);
emit msg("Current Encryption/Decryption status: Idle... :'(");
}
Вот как я создаю поток и подключаю его к основному потоку GUI:
FileCrypto *fc = new FileCrypto(ui->lineEdit_4->text(), "", 1);
connect(fc, SIGNAL(completed(int)), ui->progressBar, SLOT(setValue(int)));
connect(fc, SIGNAL(msg(QString)), ui->statusBar, SLOT(showMessage(QString)));
connect(fc, SIGNAL(pathMsg1(QString)), ui->lineEdit_4, SLOT(setText(QString)));
connect(fc, SIGNAL(pathMsg2(QString)), ui->lineEdit_5, SLOT(setText(QString)));
connect(fc, SIGNAL(keyMsg(QString)), ui->lineEdit_2, SLOT(setText(QString)));
connect(fc, SIGNAL(keyMsg(QString)), this, SLOT(done()));
Если я не обновляю индикатор выполнения, то есть не выделяю процент, процесс происходит намного быстрее. Я также попытался напечатать процент. Это замедляет его, как ад, но значения в порядке. Также вы можете предложить способ изменить его на буферизованный ИО....
Здесь очень ценится любая помощь.......
1 ответ:
Проблема не заключается в том, что вы вызываете из другого потока. Он расположен в:
emit completed(int((done / length) * 100));
Так как
done
иlength
являются типами int, иdone <= length
,done/length == 0
. Поэтому измените его на:emit completed(100 * done / length);
(если может привести к арифметическому переполнению).