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 , там сделан подобный функционал.

    Комментарии

    Только авторизованные пользователи могут публиковать комментарии.
    Пожалуйста, авторизуйтесь или зарегистрируйтесь
    AD

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

    • Результат:50баллов,
    • Очки рейтинга-4
    m
    • molni99
    • 26 октября 2024 г. 1:37

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

    • Результат:80баллов,
    • Очки рейтинга4
    m
    • molni99
    • 26 октября 2024 г. 1:29

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

    • Результат:20баллов,
    • Очки рейтинга-10
    Последние комментарии
    ИМ
    Игорь Максимов22 ноября 2024 г. 11:51
    Django - Урок 017. Кастомизированная страница авторизации на Django Добрый вечер Евгений! Я сделал себе авторизацию аналогичную вашей, все работает, кроме возврата к предидущей странице. Редеректит всегда на главную, хотя в логах сервера вижу запросы на правильн…
    Evgenii Legotckoi
    Evgenii Legotckoi31 октября 2024 г. 14:37
    Django - Урок 064. Как написать расширение для Python Markdown Добрый день. Да, можно. Либо через такие же плагины, либо с постобработкой через python библиотеку Beautiful Soup
    A
    ALO1ZE19 октября 2024 г. 8:19
    Читалка fb3-файлов на Qt Creator Подскажите как это запустить? Я не шарю в программировании и кодинге. Скачал и установаил Qt, но куча ошибок выдается и не запустить. А очень надо fb3 переконвертировать в html
    ИМ
    Игорь Максимов5 октября 2024 г. 7:51
    Django - Урок 064. Как написать расширение для Python Markdown Приветствую Евгений! У меня вопрос. Можно ли вставлять свои классы в разметку редактора markdown? Допустим имея стандартную разметку: <ul> <li></li> <li></l…
    d
    dblas55 июля 2024 г. 11:02
    QML - Урок 016. База данных SQLite и работа с ней в QML Qt Здравствуйте, возникает такая проблема (я новичок): ApplicationWindow неизвестный элемент. (М300) для TextField и Button аналогично. Могу предположить, что из-за более новой верси…
    Сейчас обсуждают на форуме
    Evgenii Legotckoi
    Evgenii Legotckoi24 июня 2024 г. 15:11
    добавить qlineseries в функции Я тут. Работы оень много. Отправил его в бан.
    t
    tonypeachey115 ноября 2024 г. 6:04
    google domain [url=https://google.com/]domain[/url] domain [http://www.example.com link title]
    NSProject
    NSProject4 июня 2022 г. 3:49
    Всё ещё разбираюсь с кешем. В следствии прочтения данной статьи. Я принял для себя решение сделать кеширование свойств менеджера модели LikeDislike. И так как установка evileg_core для меня не была возможна, ибо он писался…
    9
    9Anonim25 октября 2024 г. 9:10
    Машина тьюринга // Начальное состояние 0 0, ,<,1 // Переход в состояние 1 при пустом символе 0,0,>,0 // Остаемся в состоянии 0, двигаясь вправо при встрече 0 0,1,>…

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