m
mihenze19 ноября 2017 г. 5:20

Потоки и философы

Здравствуйте, реализовал обедающих философов с помощью семафоров. Но возник вопрос, как это сделать использую QMutex или QSemaphore, а не как я с помощью условий. Очень буду благодарен за помощь.


Ps. И еще такой вопрос, как в моем коде остановить потоки по кнопке стоп, а затем, если нажать старт, симуляция обеда начиналась заново? (Я так понимаю, нужно выставлять флаг цикла while в слоте life() в false, а затем вызывать для потока terminate()? )

mainwindow.h
#ifndef MAINWINDOW_H
#define MAINWINDOW_H

#include <QMainWindow>
#include <QThread>
#include <QStringList>
#include <QTableView>
#include <QStandardItemModel>
#include <QVector>

#include "philosopher.h"
#include "fork.h"

namespace Ui {
class MainWindow;
}

class MainWindow : public QMainWindow
{
    Q_OBJECT

public:
    explicit MainWindow(QWidget *parent = 0);
    ~MainWindow();

public slots:
    void slotStart(); //начало эмуляции
    void slotStop(); //завершение эмуляции
    void slotTextOutput(QString sOut); //вывод текста в textedit

private:
    Ui::MainWindow *ui;

    QThread thread1;
    QThread thread2;
    QThread thread3;
    QThread thread4;
    QThread thread5;
    QVector <Fork> list_fork;
    int listing = 1;
};

#endif // MAINWINDOW_H
mainwindow.cpp
#include "mainwindow.h"
#include "ui_mainwindow.h"

MainWindow::MainWindow(QWidget *parent) :
    QMainWindow(parent),
    ui(new Ui::MainWindow)
{
    ui->setupUi(this);

    connect(ui->action, SIGNAL(triggered(bool)), this, SLOT(slotStart()));
    connect(ui->action_2, SIGNAL(triggered(bool)), this, SLOT(slotStop()));

}

MainWindow::~MainWindow()
{
    delete ui;
}
void MainWindow::slotStart()
{
    for (int i = 0; i < 5; i++)
    {
        Fork fork;
        list_fork.push_back(fork);
    }

    Philosopher *phil1 = new Philosopher("Socrat", &list_fork, 0);
    Philosopher *phil2 = new Philosopher("Arestotel", &list_fork, 1);
    Philosopher *phil3 = new Philosopher("Platon", &list_fork, 2);
    Philosopher *phil4 = new Philosopher("Evklid", &list_fork, 3);
    Philosopher *phil5 = new Philosopher("Diogen", &list_fork, 4);


    connect(phil1, SIGNAL(signalAddText(QString)), this, SLOT(slotTextOutput(QString)));
    connect(phil2, SIGNAL(signalAddText(QString)), this, SLOT(slotTextOutput(QString)));
    connect(phil3, SIGNAL(signalAddText(QString)), this, SLOT(slotTextOutput(QString)));
    connect(phil4, SIGNAL(signalAddText(QString)), this, SLOT(slotTextOutput(QString)));
    connect(phil5, SIGNAL(signalAddText(QString)), this, SLOT(slotTextOutput(QString)));


    connect(&thread1, SIGNAL(started()), phil1, SLOT(life()));
    connect(&thread2, SIGNAL(started()), phil2, SLOT(life()));
    connect(&thread3, SIGNAL(started()), phil3, SLOT(life()));
    connect(&thread4, SIGNAL(started()), phil4, SLOT(life()));
    connect(&thread5, SIGNAL(started()), phil5, SLOT(life()));


    phil1->moveToThread(&thread1);
    phil2->moveToThread(&thread2);
    phil3->moveToThread(&thread3);
    phil4->moveToThread(&thread4);
    phil5->moveToThread(&thread5);


    thread1.start(QThread::NormalPriority);
    thread2.start(QThread::NormalPriority);
    thread3.start(QThread::NormalPriority);
    thread4.start(QThread::NormalPriority);
    thread5.start(QThread::NormalPriority);
}

void MainWindow::slotStop()
{


}

void MainWindow::slotTextOutput(QString sOut)
{
    QString sss = QString::number(listing);
    sss+= ": " + sOut;

    ui->plainTextEdit->appendPlainText(sss);
    listing++;
}
fork.h
#ifndef FORK_H
#define FORK_H


class Fork
{
public:
    Fork();

    bool getIsUsing() const;
    void take();
    void put();

private:
    bool isUsing;
};

#endif // FORK_H
fork.cpp
#include "fork.h"

Fork::Fork()
{
    isUsing = false;
}

bool Fork::getIsUsing() const
{
    return isUsing;
}

void Fork::take()
{

    isUsing = true;
}

void Fork::put()
{

    isUsing = false;
}
philosopher.h
#ifndef PHILOSOPHER_H
#define PHILOSOPHER_H

#include <QObject>
#include <windows.h>
#include <QVector>

#include "fork.h"

class Philosopher : public QObject
{
    Q_OBJECT
public:
    Philosopher(QString s, QVector<Fork> *fork, int id);

    void eat();
    void think();
    void takeForks();
    void putForks();
signals:
    void signalAddText(QString);

public slots:
    void life();

private:
    QString name;
    bool left_fork;
    bool right_fork;
    QVector <Fork> *forkNumber;
    int idPhil;
};

#endif // PHILOSOPHER_H
philosopher.cpp
#include "philosopher.h"

Philosopher::Philosopher(QString s, QVector<Fork> *fork, int id) : name(s)
{
    left_fork = false;
    right_fork = false;
    forkNumber = fork;
    idPhil = id;
}

void Philosopher::eat()
{
    QString temp = name + " кушает.";
    Sleep(1000);
    emit(signalAddText(temp));
}

void Philosopher::think()
{
    QString temp = name + " думает.";
    Sleep(1000);
    emit(signalAddText(temp));
}

void Philosopher::takeForks()
{
    if((!(*forkNumber)[idPhil].getIsUsing())&&(!(*forkNumber)[(idPhil+4)%5].getIsUsing()))
    {
        (*forkNumber)[idPhil].take();
        left_fork = true;
        (*forkNumber)[(idPhil+4)%5].take();
        right_fork = true;
        QString temp = name + " взял вилки.";

        emit(signalAddText(temp));
        Sleep(1000);

    }
}

void Philosopher::putForks()
{
    QString temp = name + " положил вилки.";

    emit(signalAddText(temp));

    (*forkNumber)[idPhil].put();
    left_fork = false;

    (*forkNumber)[(idPhil+4)%5].put();
    right_fork = false;

    Sleep(1000);
}

void Philosopher::life()
{
    while(true)
    {
        think();
        takeForks();
        if (left_fork && right_fork)
        {
            eat();
            putForks();

        }
    }
}
PhilosopherSimulator.rar
Рекомендуем хостинг TIMEWEB
Рекомендуем хостинг TIMEWEB
Стабильный хостинг, на котором располагается социальная сеть EVILEG. Для проектов на Django рекомендуем VDS хостинг.

Вам это нравится? Поделитесь в социальных сетях!

1
Evgenii Legotckoi
  • 22 ноября 2017 г. 13:51
  • (ред.)

Извиняюсь за поздний ответ. У вас вопрос достаточно обширный, как бы "академические" задачки требуют значительно больше времени на размышление, которого лично у меня немного.


Поэтому отвечу на один из ваших вопросов, может быть ещё кто-нибудь на сайте подтянется с ответом.

Для остановки метода, который является условной петлёй алгоритма вам действительно нужно выставлять переменную в false.
Посмотрите вот эту статью по moveToThread , там сделан подобный функционал.

    Комментарии

    Только авторизованные пользователи могут публиковать комментарии.
    Пожалуйста, авторизуйтесь или зарегистрируйтесь
    d
    • dsfs
    • 26 апреля 2024 г. 14:56

    C++ - Тест 004. Указатели, Массивы и Циклы

    • Результат:80баллов,
    • Очки рейтинга4
    d
    • dsfs
    • 26 апреля 2024 г. 14:45

    C++ - Тест 002. Константы

    • Результат:50баллов,
    • Очки рейтинга-4
    d
    • dsfs
    • 26 апреля 2024 г. 14:35

    C++ - Тест 001. Первая программа и типы данных

    • Результат:73баллов,
    • Очки рейтинга1
    Последние комментарии
    k
    kmssr9 февраля 2024 г. 5:43
    Qt Linux - Урок 001. Автозапуск Qt приложения под Linux как сделать автозапуск для флэтпака, который не даёт создавать файлы в ~/.config - вот это вопрос ))
    АК
    Анатолий Кононенко5 февраля 2024 г. 12:50
    Qt WinAPI - Урок 007. Работаем с ICMP Ping в Qt Без строки #include <QRegularExpressionValidator> в заголовочном файле не работает валидатор.
    EVA
    EVA25 декабря 2023 г. 21:30
    Boost - статическая линковка в CMake проекте под Windows Ошибка LNK1104 часто возникает, когда компоновщик не может найти или открыть файл библиотеки. В вашем случае, это файл libboost_locale-vc142-mt-gd-x64-1_74.lib из библиотеки Boost для C+…
    J
    JonnyJo25 декабря 2023 г. 19:38
    Boost - статическая линковка в CMake проекте под Windows Сделал всё по-как у вас, но выдаёт ошибку [build] LINK : fatal error LNK1104: не удается открыть файл "libboost_locale-vc142-mt-gd-x64-1_74.lib" Хоть убей, не могу понять в чём дел…
    G
    Gvozdik19 декабря 2023 г. 8:01
    Qt/C++ - Урок 056. Подключение библиотеки Boost в Qt для компиляторов MinGW и MSVC Для решения твой проблемы добавь в файл .pro строчку "LIBS += -lws2_32" она решит проблему , лично мне помогло.
    Сейчас обсуждают на форуме
    G
    Gar22 апреля 2024 г. 15:46
    Clipboard Как скопировать окно целиком в clipb?
    DA
    Dr Gangil Academics20 апреля 2024 г. 17:45
    Unlock Your Aesthetic Potential: Explore MSC in Facial Aesthetics and Cosmetology in India Embark on a transformative journey with an msc in facial aesthetics and cosmetology in india . Delve into the intricate world of beauty and rejuvenation, guided by expert faculty and …
    a
    a_vlasov14 апреля 2024 г. 16:41
    Мобильное приложение на C++Qt и бэкенд к нему на Django Rest Framework Евгений, добрый день! Такой вопрос. Верно ли следующее утверждение: Любое Android-приложение, написанное на Java/Kotlin чисто теоретически (пусть и с большими трудностями) можно написать и на C+…
    Павел Дорофеев
    Павел Дорофеев14 апреля 2024 г. 12:35
    QTableWidget с 2 заголовками Вот тут есть кастомный QTableView с многорядностью проект поддерживается, обращайтесь
    f
    fastrex4 апреля 2024 г. 14:47
    Вернуть старое поведение QComboBox, не менять индекс при resetModel Добрый день! У нас много проектов в которых используется QComboBox, в версии 5.5.1, когда модель испускает сигнал resetModel, currentIndex не менялся. В версии 5.15 при resetModel происходит try…

    Следите за нами в социальных сетях