Обеденный философский код Операционная система
Программа:
import threading
import time
import random
global spoons
spoons = [1, 1, 1, 1, 1]
class mythread(threading.Thread):
running = True
def __init__(self, name, number):
threading.Thread.__init__(self)
self.name = name
self.number = number
def run(self):
while self.running:
time.sleep( random.uniform(5,10))
print ('{} is hungry.'.format(self.name))
self.check()
def eat(self):
global spoons
z=self.number
print("{} is eating\n".format(self.name))
time.sleep(random.uniform(1, 5))
print("{} has finished eating\n".format(self.name))
if(z == 1):
spoons[z] = spoons[4] = 1
else:
spoons[z] = spoons[z - 1] = 1
def wait(self):
time.sleep(1)
self.check()
def check(self):
global spoons
z = self.number
if(z % 2 != 0):
if(spoons[z - 1] == 1):
spoons[z - 1] = 0
if(z == 1):
if(spoons[4] == 1):
spoons[4] = 0
self.eat()
else:
self.wait()
else:
if(spoons[z - 2] == 1):
spoons[z - 2] = 0
self.eat()
else:
self.wait()
else:
self.wait()
else:
if(spoons[z - 2] == 1):
spoons[z - 2] = 0
if(spoons[z - 1] == 1):
spoons[z - 1] = 0
self.eat()
else:
self.wait()
else:
self.wait()
Name=['p1', 'p2', 'p3', 'p4', 'p5']
philosophers= [mythread(name = Name[i % 5], number = (i + 1) % 5)
for i in range(5)]
for p in philosophers:
p.start()
time.sleep(100)
mythread.running = False
print("Now we're finishing.")
выход:
Exception in thread p3:
Traceback (most recent call last):
File "C:\Users\devda\AppData\Local\Programs\Python\Python36\lib\threading.py", line 916, in _bootstrap_inner
self.run()
File "D:/ANIKET/osproject/dining_philosophers_probelm.py", line 22, in run
def eat(self):
File "D:/ANIKET/osproject/dining_philosophers_probelm.py", line 56, in check
self.wait()
File "D:/ANIKET/osproject/dining_philosophers_probelm.py", line 37, in wait
def check(self):
File "D:/ANIKET/osproject/dining_philosophers_probelm.py", line 58, in check
if(spoons[z - 2] == 1):
File "D:/ANIKET/osproject/dining_philosophers_probelm.py", line 37, in wait
def check(self):
File "D:/ANIKET/osproject/dining_philosophers_probelm.py", line 58, in check
if(spoons[z - 2] == 1):
File "D:/ANIKET/osproject/dining_philosophers_probelm.py", line 37, in wait
def check(self):
File "D:/ANIKET/osproject/dining_philosophers_probelm.py", line 58, in check
if(spoons[z - 2] == 1):
File "D:/ANIKET/osproject/dining_philosophers_probelm.py", line 37, in wait
def check(self):
File "D:/ANIKET/osproject/dining_philosophers_probelm.py", line 58, in check
if(spoons[z - 2] == 1):
File "D:/ANIKET/osproject/dining_philosophers_probelm.py", line 37, in wait
def check(self):
File "D:/ANIKET/osproject/dining_philosophers_probelm.py", line 58, in check
if(spoons[z - 2] == 1):
File "D:/ANIKET/osproject/dining_philosophers_probelm.py", line 37, in wait
def check(self):
File "D:/ANIKET/osproject/dining_philosophers_probelm.py", line 58, in check
if(spoons[z - 2] == 1):
File "D:/ANIKET/osproject/dining_philosophers_probelm.py", line 37, in wait
def check(self):
File "D:/ANIKET/osproject/dining_philosophers_probelm.py", line 58, in check
Это работает в течение некоторого времени, а затем показывает ошибки. Я думаю, что это логическая ошибка. Но я не могу понять, что.
Логика, которую я использовал, состоит в том, что философ с четными номерами сначала берет правую развилку, а философ с нечетными числами сначала берет правую развилку.
Эта логика используется в функции check() внутри моего кода.
Кроме того, поскольку я использовал много рекурсивных вызовов, возникает ли эта ошибка из-за исчерпания рекурсивного стека?
1 ответ
Решение
Код:
import threading
import random
import time
class Philosopher(threading.Thread):
running = True
def __init__(self, xname, forkOnLeft, forkOnRight):
threading.Thread.__init__(self)
self.name = xname
self.forkOnLeft = forkOnLeft
self.forkOnRight = forkOnRight
def run(self):
while(self.running):
time.sleep( random.uniform(3,13))
print ('%s is hungry.' % self.name)
self.dine()
def dine(self):
fork1, fork2 = self.forkOnLeft, self.forkOnRight
while self.running:
fork1.acquire(True)
locked = fork2.acquire(False)
if locked: break
fork1.release()
print ('%s swaps forks' % self.name)
fork1, fork2 = fork2, fork1
else:
return
self.dining()
fork2.release()
fork1.release()
def dining(self):
print ('%s starts eating '% self.name)
time.sleep(random.uniform(1,10))
print ('%s finishes eating and leaves to think.' % self.name)
def DiningPhilosophers():
forks = [threading.Lock() for n in range(5)]
philosopherNames = ('1','2','3','4', '5')
philosophers= [Philosopher(philosopherNames[i], forks[i%5], forks[(i+1)%5])
for i in range(5)]
Philosopher.running = True
for p in philosophers:
p.start()
time.sleep(30)
Philosopher.running = False
DiningPhilosophers()
Попробуйте эту проверку, если это работает:)