Задача об обедающих философах python

Задача обедающих философов

недавно начал знакомиться с многопоточным программированием и попалась задачка про обедающих философов:

Пять безмолвных философов сидят вокруг круглого стола, перед каждым философом стоит тарелка спагетти. Вилки лежат на столе между каждой парой ближайших философов.

Каждый философ может либо есть, либо размышлять. Приём пищи не ограничен количеством оставшихся спагетти — подразумевается бесконечный запас. Тем не менее, философ может есть только тогда, когда держит две вилки — взятую справа и слева (альтернативная формулировка проблемы подразумевает миски с рисом и палочки для еды вместо тарелок со спагетти и вилок).

Каждый философ может взять ближайшую вилку (если она доступна) или положить — если он уже держит её. Взятие каждой вилки и возвращение её на стол являются раздельными действиями, которые должны выполняться одно за другим.

Вопрос задачи заключается в том, чтобы разработать модель поведения (параллельный алгоритм), при котором ни один из философов не будет голодать, то есть будет вечно чередовать приём пищи и размышления.

Алгоритм достаточно простой — на основе иерархии ресурсов, взят из википедии. Вкратце:

Пусть ресурсы (вилки) будут пронумерованы от 1 до 5, и каждая рабочая единица (философ) всегда берёт сначала вилку с наименьшим номером, а потом вилку с наибольшим номером из двух доступных. Далее, философ кладёт сначала вилку с бо́льшим номером, потом — с меньшим. Когда он закончит использовать вилки, он в первую очередь положит на стол вилку с бо́льшим номером, потом — с меньшим, тем самым позволив другому философу взять недостающую вилку и приступить к еде.

import threading from time import sleep import random forks = 5 philosophers = 5 forks_lock = [threading.Lock() for n in range(forks)] def philosophers_dinner(right_fork, left_fork, philosopher): while True: first_fork = min(right_fork, left_fork) second_fork = max(right_fork, left_fork) forks_lock[first_fork].acquire() forks_lock[second_fork].acquire() print(f'Философ ест') sleep(random.randint(1, 5)) forks_lock[second_fork].release() forks_lock[first_fork].release() for philosopher in range(philosophers): right_fork = philosopher left_fork = (philosopher+1) % philosophers threading.Thread(target=philosophers_dinner, args=(right_fork, left_fork, philosopher)).start() 

Подскажите, верно ли я решил задачу? Дедлоков вроде нет, филисофы меняются, но не хватает экспертного мнения.

Читайте также:  Saving text to file java

Источник

Saved searches

Use saved searches to filter your results more quickly

You signed in with another tab or window. Reload to refresh your session. You signed out in another tab or window. Reload to refresh your session. You switched accounts on another tab or window. Reload to refresh your session.

philosopher with threads/mutex and processes/semaphore (ecole 42).

luta-wolf/philosophers

This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.

Name already in use

A tag already exists with the provided branch name. Many Git commands accept both tag and branch names, so creating this branch may cause unexpected behavior. Are you sure you want to create this branch?

Sign In Required

Please sign in to use Codespaces.

Launching GitHub Desktop

If nothing happens, download GitHub Desktop and try again.

Launching GitHub Desktop

If nothing happens, download GitHub Desktop and try again.

Launching Xcode

If nothing happens, download Xcode and try again.

Launching Visual Studio Code

Your codespace will open once ready.

There was a problem preparing your codespace, please try again.

Latest commit

Git stats

Files

Failed to load latest commit information.

README.md

image

В этом проекте вы изучите основы многопоточности процесса.Вы увидите, как создавать потоки, и откроете для себя мьютексы.

Описание задачи обедающий философов

  • Один или несколько философов сидят за круглым столом. Посреди стола стоит большая миска со спагетти.
  • Философы альтернативно едят, думают или спят.
    • во время еды они не думают и не спят;
    • во время мышления они не едят и не спят;
    • и, конечно, во время сна они не едят и не думают.

    Правила для основной части:

    • Каждый философ должен быть потоком.
    • Между каждой парой философов есть одна развилка. Поэтому, если есть несколько философов, то у каждого философа есть вилка с левой стороны и вилка с правой. Если есть только один философ, то на столе должна быть только одна вилка.
    • Чтобы философы не дублировали вилки, вы должны защитить состояние вилки с помощью мьютекса для каждой из них.

    Правила для бонусной части:

    • Все вилки кладутся на середину стола.
    • У них нет состояний в памяти, но количество доступных вилок представлено семафором.
    • Каждый философ должен быть процессом. Но основной процесс не должен бытьфилософ.
    • Глобальные переменные запрещены!
    • Ваша программа должна принимать следующие аргументы:
      • number_of_philosophers (количество философов, а также количествовилок.)
      • time_to_die ((в миллисекундах): Если философ не начал есть time_to_die миллисекунды с начала своего последнего приема пищи или начала симуляции, он умирает.)
      • time_to_eat ((в миллисекундах): время, необходимое философу, чтобы поесть. За это время им нужно будет держать две вилки.)
      • time_to_sleep ((в миллисекундах): время, которое философ проведет во сне.)
      • [number_of_times_each_philosopher_must_eat] ((необязательный аргумент): Если все философы съели хотя бы number_of_times_each_philosopher_must_eat раз, симуляция прекращается. Если не указано иное, моделирование прекращается, когда философ умирает.)

      О журналах вашей программы:

      • Любое изменение состояния философа должно быть отформатировано следующим образом:
        • timestamp_in_ms X has taken a fork
        • timestamp_in_ms X is thinking
        • timestamp_in_ms X is sleeping
        • timestamp_in_ms X is thinking
        • timestamp_in_ms X is died

        (Замените timestamp_in_ms текущей меткой времени в миллисекундах, а X — числом философа.)

        • Отображаемое сообщение о состоянии не должно смешиваться с другим сообщением.
        • Сообщение о смерти философа должно отображаться не более чем через 10 мс после его фактической смерти.
        • Опять же, философы должны избегать смерти!

        Важное замечание: в вашей программе не должно быть гонки данных.

        image

        image

        для проверки гонки данных

        image

        1.Клонируем этот репозиторий рядом с каталогом нашего проекта.

         git clone https://github.com/nesvoboda/socrates 
         cd socrates python3.7 socrates.py ../philosophers () 

        image

        image

        image

        image

        • Разбор обедающих философов в лекции от marmand, еще лекция.
        • Цикл лекций по потокам и процессам Unix Threads in C от CodeVault.
        • У Столярова «Программирование: введение в профессию» том 2 глава 7, страница 516 и «Введение в операционные системы» лекция 13, страница 162
        • Статьи по дедлоку, по потокам, по структуре процессов.
        • Про гонку данных
        50 210 100 100 199 2000 600 60

        на школьных Маках с одновременно работающей конференцией в Зуме.

          Проверяем сколько раз философы поели:

        ./philo 5 800 200 200 7 | grep eat | wc -l 

        About

        philosopher with threads/mutex and processes/semaphore (ecole 42).

        Источник

        Русские Блоги

        Python решает проблему обеда философа (проблема тупика)

        #Philosopher Dining (Решить проблему тупика) import threading,time rlock1 = threading.RLock() rlock2 = threading.RLock() rlock3 = threading.RLock() rlock4 = threading.RLock() rlock5 = threading.RLock() class Zhexuejia(): def __init__(self,left,right): self.left = left self.right = right z1 = Zhexuejia(rlock5,rlock1) z2 = Zhexuejia(rlock1,rlock2) z3 = Zhexuejia(rlock2,rlock3) z4 = Zhexuejia(rlock3,rlock4) z5 = Zhexuejia(rlock4,rlock5) def run(z,name): f=z.left.acquire() if f: print (name, «Достань левые палочки для еды») ff = z.right.acquire() if ff: print (name, «Получить палочки для еды») print («Философ начинает есть», имя) time.sleep(1) z.right.release() z.left.release() t1 = threading.Thread(target=run,args=(z1,"z1")) t2 = threading.Thread(target=run,args=(z2,"z2")) t3 = threading.Thread(target=run,args=(z3,"z3")) t4 = threading.Thread(target=run,args=(z4,"z4")) t5 = threading.Thread(target=run,args=(z5,"z5")) t1.start() t2.start() t3.start() t4.start() t5.start() 
        z1 Возьмите палочки для еды слева z1 Получите палочки для еды Философ начинает есть z1 z3 Возьмите палочки для еды слева z3 Получите палочки для еды Философ начинает есть z3 z5 Возьмите палочки для еды слева z5 получить палочки для еды z2 Возьмите палочки для еды слева Философ начинает есть z5 z2 Получите палочки для еды z4 Возьмите палочки для еды слева Философ начинает есть z2 z4 Получите палочки для еды Философ начинает есть z4 Process finished with exit code 0 

        Источник

        Saved searches

        Use saved searches to filter your results more quickly

        You signed in with another tab or window. Reload to refresh your session. You signed out in another tab or window. Reload to refresh your session. You switched accounts on another tab or window. Reload to refresh your session.

        Задача об обедающих философах

        License

        dbond762/dining_philosophers

        This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.

        Name already in use

        A tag already exists with the provided branch name. Many Git commands accept both tag and branch names, so creating this branch may cause unexpected behavior. Are you sure you want to create this branch?

        Sign In Required

        Please sign in to use Codespaces.

        Launching GitHub Desktop

        If nothing happens, download GitHub Desktop and try again.

        Launching GitHub Desktop

        If nothing happens, download GitHub Desktop and try again.

        Launching Xcode

        If nothing happens, download Xcode and try again.

        Launching Visual Studio Code

        Your codespace will open once ready.

        There was a problem preparing your codespace, please try again.

        Latest commit

        Git stats

        Files

        Failed to load latest commit information.

        README.md

        Пять безмолвных философов сидят вокруг круглого стола, перед каждым философом стоит тарелка спагетти. Вилки лежат на столе между каждой парой ближайших философов.

        Каждый философ может либо есть, либо размышлять. Приём пищи не ограничен количеством оставшихся спагетти — подразумевается бесконечный запас. Тем не менее, философ может есть только тогда, когда держит две вилки — взятую справа и слева.

        Каждый философ может взять ближайшую вилку (если она доступна), или положить — если он уже держит её. Взятие каждой вилки и возвращение её на стол являются раздельными действиями, которые должны выполняться одно за другим.

        Суть проблемы заключается в том, чтобы разработать модель поведения (параллельный алгоритм), при котором ни один из философов не будет голодать, то есть будет вечно чередовать приём пищи и размышления.

        git clone https://github.com/dbond762/dining_philosophers.git cd dining_philosophers git build ./dining_philosophers

        Источник

Оцените статью