-
Notifications
You must be signed in to change notification settings - Fork 3
/
show.py
632 lines (592 loc) · 30.4 KB
/
show.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
import random
import sys
import copy
from PyQt5.QtWidgets import *
from PyQt5 import QtGui
from PyQt5.QtCore import Qt
from process import Process
from subject import Subject
from rr import RR
from fcfs import FCFS
from hrrn import HRRN
from spn import SPN
from srtn import SRTN
from yosa import YOSA
class MyApp(QWidget):
def __init__(self):
super().__init__()
self.change_font()
self.proc_list = []
self.history = []
self.algo_list = ["FCFS", "RR", "SPN", "SRTN", "HRRN", "YOSA"]
self.subject_list = ["C++", "DB 설계", "컴퓨터 구조", "웹 프로그래밍", "자료구조", "자바", "디지털 공학"]
self.column_count = 3
self.cur_algo = "FCFS"
self.process_count = 0
self.subject_count = [0, 0, 0, 0]
self.init_ui()
# 수정 : font
def change_font(self):
font = QtGui.QFont()
# 폰트 없으시면 주석 처리하세요!
font.setFamily("카카오 Regular")
# font.setFamily("Consolas")
font.setPointSize(10)
# font.setBold(True)
self.setFont(font)
def init_ui(self):
self.resize(1400, 900)
self.center()
# 알고리즘 종류를 선택핧 Alg_Select를 콤보박스로 구현
self.alg_select = QComboBox(self)
for algo in self.algo_list:
self.alg_select.addItem(algo)
self.alg_select.activated.connect(self.enable_slot)
# 프로세스 이름을 사용자에게 받을 ProName, AT를 사용자에게 받을 AT, BT를 사용자에게 받을 BT
# AT와 BT같이 사용자에게 숫자로만 받을거라면 스핀박스로 하는게 더 편하다고 함
self.process_name = QLineEdit()
self.process_name.setMaxLength(10)
self.at_label = QLabel("AT")
self.at = QSpinBox()
self.at.setRange(0, 65535)
self.bt = QSpinBox()
self.bt.setRange(1, 65535)
self.student_list = QComboBox(self)
self.student_list.addItem("학생 1")
self.student_list.setDisabled(True)
# 프로세스 목록을 표로 보여줄 Proc_Table선언
self.proc_table = QTableWidget(self)
self.proc_table.setColumnCount(3)
self.proc_table.setHorizontalHeaderLabels(["Process Name", "Arrival Time", "Burst Time"])
self.proc_table.setEditTriggers(QAbstractItemView.NoEditTriggers)
self.proc_table.verticalHeader().setVisible(False)
header = self.proc_table.horizontalHeader()
for column_idx in range(self.column_count):
header.setSectionResizeMode(column_idx, QHeaderView.Stretch)
# CPU의 코어 개수 선택에 사용할 cpu_count를 콤보박스로 선언
self.cpu_count = QComboBox(self)
for cpu_idx in range(1, 5):
self.cpu_count.addItem(str(cpu_idx))
self.cpu_count.activated.connect(self.set_cpu_slot)
self.cpu_label = QLabel("CPU")
# Ready Queue 보여줄 테이블
self.ready_table = QTableWidget(self)
self.ready_table.setRowCount(1)
self.ready_table.verticalHeader().setVisible(False)
self.ready_table.setMaximumHeight(50)
self.ready_table.verticalHeader().setDefaultSectionSize(50)
self.ready_table.horizontalHeader().setVisible(False)
self.ready_table.setEditTriggers(QAbstractItemView.NoEditTriggers)
# CPU 일 목록
self.gantt_table = QTableWidget(self)
self.gantt_table.setRowCount(1)
self.gantt_table.setVerticalHeaderLabels(["CPU 1"])
header = self.gantt_table.verticalHeader()
header.setSectionResizeMode(0, QHeaderView.Stretch)
self.gantt_table.setEditTriggers(QAbstractItemView.NoEditTriggers)
self.real_time_label = QLabel("Real Time = 0 sec")
# quantum을 넣는 스핀박스, 초기는 FCFS이기에 비활성화해둠
self.tq = QSpinBox()
self.tq.setRange(1, 65535)
self.tq.setDisabled(True)
self.tq_label = QLabel("Quantum")
# Gantt Chart를 표로 보여줄 Result_Table선언
self.result_table = QTableWidget(self)
self.result_table.setColumnCount(6)
self.result_table.setHorizontalHeaderLabels(
["Process Name", "Arrival Time", "Burst Time", "Waiting Time", "Turnaround Time", "Normalized TT"]
)
self.result_table.verticalHeader().setVisible(False)
header = self.result_table.horizontalHeader()
for time_table_col in range(6):
header.setSectionResizeMode(time_table_col, QHeaderView.Stretch)
self.result_table.setEditTriggers(QAbstractItemView.NoEditTriggers)
# Proc_List에 프로세스 목록을 추가 및 화면 내용을 리셋하는 버튼
self.add_button = QPushButton("Add", self)
self.add_button.clicked.connect(self.add)
# 수정 : 디버깅용 test를 add 버튼에 일시적으로 연결 (test 누르면 자동으로 값 입력됨)
# self.add_button.clicked.connect(self.test)
reset_button = QPushButton("Reset", self)
reset_button.clicked.connect(self.reset)
# 알고리즘 실행 버튼
self.run_alg = QPushButton("Run", self)
self.run_alg.clicked.connect(self.run_algorithm)
self.run_alg.setDisabled(True)
self.run_alg.setFixedHeight(65)
# 각 히스토리 시간마다 보여주기 위한 슬라이드
self.history_slider = QSlider(Qt.Horizontal, self)
self.history_slider.setDisabled(True)
self.history_slider.valueChanged.connect(self.slider_control)
# 첫째줄, 그리드에 프로세스 이름, AT, BT, 추가버튼, 리셋버튼 추가
grid_line = QGridLayout()
grid_line.addWidget(QLabel("Algorithm"), 0, 0)
grid_line.addWidget(self.at_label, 0, 1)
grid_line.addWidget(QLabel("BT"), 0, 2)
grid_line.addWidget(self.alg_select, 1, 0)
grid_line.addWidget(self.at, 1, 1)
grid_line.addWidget(self.bt, 1, 2)
grid_line.addWidget(self.cpu_label, 0, 3)
grid_line.addWidget(self.cpu_count, 1, 3)
grid_line.addWidget(self.tq_label, 0, 4)
grid_line.addWidget(self.tq, 1, 4)
grid_line.addWidget(QLabel("대상 학생"), 0, 5)
grid_line.addWidget(self.student_list, 1, 5)
grid_line.addWidget(self.add_button, 0, 6)
grid_line.addWidget(reset_button, 1, 6)
grid_line.addWidget(self.run_alg, 0, 7, 2, 1)
# 그냥 이름용
vbox_line2 = QVBoxLayout()
ready_name = QLabel("Ready Queue")
ready_name.setMaximumHeight(25) # 수정 : ready queue 글자가 잘려서 25로 늘렸음
vbox_line2.addWidget(ready_name)
vbox_line2.addWidget(self.ready_table)
hbox_line3 = QHBoxLayout()
hbox_line3.addWidget(QLabel("Gantt Chart"))
hbox_line3.addWidget(self.real_time_label)
# 레이아웃 및 위젯을 통합할 vbox_main을 선언, 메인 레이아웃 지정 후 레이아웃 및 위젯 통합
vbox_main = QVBoxLayout()
self.setLayout(vbox_main)
vbox_main.addLayout(grid_line)
vbox_main.addWidget(self.proc_table)
vbox_main.addWidget(self.history_slider)
vbox_main.addLayout(vbox_line2)
vbox_main.addLayout(hbox_line3)
vbox_main.addWidget(self.gantt_table)
vbox_main.addWidget(self.result_table)
# vbox_main.addStretch(3)
self.setWindowTitle("Process Scheduling Simulator")
# setGeometry가 크기랑 위치 지정하는건데 잘몰르겟음 필요한가, 센터로 화면 가운데 위치지정중임
# self.setGeometry(0, 0, 800, 600)
self.center()
self.show()
def center(self):
# 프로그램이 화면 중앙을 알아서 찾아가는 함수
qr = self.frameGeometry()
cp = QDesktopWidget().availableGeometry().center()
qr.moveCenter(cp)
self.move(qr.topLeft())
def enable_slot(self):
if self.alg_select.currentText() != "YOSA":
# 이전 세팅이 YOSA 였다면
if self.cur_algo == "YOSA":
self.reset()
# 화면을 기본 세팅으로 바꿈
self.default_setting()
if self.alg_select.currentText() == "RR":
self.tq.setEnabled(True)
else:
self.tq.setDisabled(True)
else:
self.reset()
# 화면을 YOSA용 세팅으로 변경
self.yosa_setting()
self.cur_algo = self.alg_select.currentText()
def set_cpu_slot(self):
name_header = []
cpu_select = int(self.cpu_count.currentText())
self.gantt_table.setRowCount(cpu_select)
if self.alg_select.currentText() != "YOSA":
for i in range(1, cpu_select + 1):
name_header.append("CPU " + str(i))
else:
for i in range(1, cpu_select + 1):
name_header.append("학생 " + str(i))
# CPU에 개수에 따라 변동되는 내역들은 전부 여기에 있다고 생각하면 편함
self.result_table.setRowCount(cpu_select)
self.result_table.verticalHeader().setVisible(True)
self.result_table.setVerticalHeaderLabels(name_header)
# 팀원 전체 학점 평균의 셀 병합
if cpu_select > 1:
self.result_table.setSpan(0, 10, cpu_select, 1)
self.result_table.setSpan(0, 12, cpu_select, 1)
header = self.result_table.verticalHeader()
for i in range(len(name_header)):
header.setSectionResizeMode(i, QHeaderView.Stretch)
self.student_list.clear()
for i in range(1, cpu_select + 1):
self.student_list.addItem("학생 " + str(i))
# 간트차트 정렬 하는거
self.gantt_table.setVerticalHeaderLabels(name_header)
header = self.gantt_table.verticalHeader()
for i in range(len(name_header)):
header.setSectionResizeMode(i, QHeaderView.Stretch)
# 프로그램 좀 굴리다 깨달은건데 run 돌리고 CPU 만지고 슬라이더 돌리면 오류생길수있음
self.history_slider.setDisabled(True)
# 수정 : 디버깅용(자동 입력 해줌 - 프로세스 개수, AT, BT까지 자동으로 입력)
def test(self):
self.run_alg.setEnabled(True)
# Proc_List에 프로세스를 저장.======================
for process_id in range(random.randrange(1, 16)):
self.proc_list.append(
Process("p" + str(process_id), random.randrange(0, 10), random.randrange(1, 8), process_id)
)
# ===============================================
# 테스트 값 내가 넣어줄때-----------
self.proc_list = [
Process("p0", 1, 4, 0),
Process("p1", 2, 3, 1),
Process("p2", 2, 3, 2),
Process("p3", 2, 3, 3),
Process("p4", 2, 3, 4),
Process("p5", 0, 2, 5),
Process("p6", 0, 2, 6),
Process("p7", 0, 2, 7),
Process("p8", 1, 4, 8),
Process("p9", 1, 4, 9),
]
# ----------------------------
self.proc_list = [
Subject("알고리즘", 4, 8, 0, 0),
Subject("웹프", 3, 7, 1, 0),
Subject("직능훈", 2, 4, 2, 0),
# Subject("알고리즘", 4, 5, 3, 1),
Subject("C++", 4, 4, 4, 1),
# Subject("웹프", 3, 3, 5, 1),
# Subject("알고리즘", 4, 4, 6, 2),
# Subject("데베설", 3, 5, 7, 2),
# Subject("운영체제", 2, 6, 8, 2),
]
# -------------------------------
print("[self.proc_list]")
print("[", end="")
for idx, process in enumerate(self.proc_list):
print("Process('{0}', {1}, {2}, {3}),".format(process.id, process.at, process.at, idx), end="")
print("]")
# 기본
# self.proc_list.append(Process("p1", 0, 3))
# self.proc_list.append(Process("p2", 1, 7))
# self.proc_list.append(Process("p3", 3, 2))
# self.proc_list.append(Process("p4", 5, 5))
# self.proc_list.append(Process("p5", 6, 3))
# 이후 Proc_Table에 프로세스를 띄우도록 함, 열크기 = Proc_List 크기
self.proc_table.setRowCount(len(self.proc_list))
for i in range(len(self.proc_list)):
self.proc_table.setItem(i, 0, QTableWidgetItem(self.proc_list[i].id))
self.proc_table.item(i, 0).setBackground(
QtGui.QColor(self.proc_list[i].color[0], self.proc_list[i].color[1], self.proc_list[i].color[2])
)
# if self.proc_list[i].color[0] + self.proc_list[i].color[1] + self.proc_list[i].color[2] < 350:
# self.proc_table.item(i, 0).setForeground(QtGui.QBrush(QtGui.QColor(255, 255, 255)))
self.proc_table.setItem(i, 1, QTableWidgetItem(str(self.proc_list[i].at)))
self.proc_table.setItem(i, 2, QTableWidgetItem(str(self.proc_list[i].bt)))
# 초기화 부분
self.process_name.clear()
self.at.setValue(0)
self.bt.setValue(0)
def add(self):
self.run_alg.setEnabled(True)
# Proc_List에 프로세스를 저장.
if self.cur_algo != "YOSA":
proc_name = "P" + str(self.process_count + 1)
# 색상을 현재 프로세스 개수를 기준으로 매겨주고, 만약 프로세스가 15개가 넘어가면 추가를 막음.
self.proc_list.append(Process(proc_name, self.at.value(), self.bt.value(), self.process_count))
self.process_count += 1
if self.process_count == 15:
self.add_button.setDisabled(True)
# YOSA인 경우 해당되는 CPU의 정보도 같이 넣어줌
else:
# select에 선택한 학생 번호를 넣어주고, 이를 Proc_List에 넣을떄 같이 넣어줌
select = int(self.student_list.currentText()[-1])
# 이름 무작위 선정
subject_name = self.subject_list[random.randint(0, len(self.subject_list) - 1)]
check = False
# 겹치면 안되서 겹치지 않도록 하는 것
for subject in self.proc_list:
if subject.id == subject_name and subject.student_id == select - 1:
check = True
break
while check:
subject_name = self.subject_list[random.randint(0, len(self.subject_list) - 1)]
check = False
for subject in self.proc_list:
if subject.id == subject_name and subject.student_id == select - 1:
check = True
break
self.proc_list.append(
Subject(subject_name, self.at.value(), self.bt.value(), self.process_count, select - 1)
)
self.subject_count[select - 1] += 1
# 만약 학생 3이 해야하는 프로세스가 등록이 되었을 때, 이후 학생 수를 1이나 2로 선정하고 run을 돌리면 문제가 될 것.
# 이를 보완하기 위해 학생 3이 해야하는 프로세스가 등록이 되면 학생수 1이랑 2를 비활성화 하함
while int(self.cpu_count.itemText(0)) < select:
self.cpu_count.removeItem(0)
# 만약 해당 학생의 프로세스 개수가 4가 되면 StudentList의 해당 학생을 빼서 더이상 넣지 못하게 함
# 그리고 그런 상황에서 cpu_count 건들면 괜히 이상해질거같으니 cpu_count 비활성화
if self.subject_count[select - 1] == 4:
find = "학생 " + str(select)
delete_index = self.student_list.findText(find)
self.student_list.removeItem(delete_index)
self.cpu_count.setDisabled(True)
# 위에 하면서 StudentList 내용 지우다가 내용 싹다 없어지면 addbutton 비활성화
if self.student_list.count() == 0:
self.add_button.setDisabled(True)
self.process_count += 1
# 이후 Proc_Table에 프로세스를 띄우도록 함, 열크기 = Proc_List 크기
self.proc_table.setRowCount(len(self.proc_list))
for proc_idx, process in enumerate(self.proc_list):
self.proc_table.setItem(proc_idx, 0, QTableWidgetItem(process.id))
self.proc_table.item(proc_idx, 0).setBackground(
QtGui.QColor(process.color[0], process.color[1], process.color[2])
)
self.proc_table.setItem(proc_idx, 2, QTableWidgetItem(str(process.bt)))
if self.cur_algo != "YOSA":
# YOSA가 아닌경우엔 AT를
self.proc_table.setItem(proc_idx, 1, QTableWidgetItem(str(process.at)))
else:
# YOSA인 경우엔 student_id값과 학점을 넣어줌, 위에서 YOSA 구분때 넣을까 했는데 그러면 코드 줄 늘어나서 안했음
self.proc_table.setItem(proc_idx, 1, QTableWidgetItem(str(process.credit)))
self.proc_table.setItem(proc_idx, 3, QTableWidgetItem("학생 " + str(process.student_id + 1)))
# 초기화 부분
self.process_name.clear()
self.at.setValue(0)
self.bt.setValue(1)
def reset(self):
# Proc_Table과 Result_Table 열을 0으로 만들고 Proc_List를 clear
self.proc_table.setRowCount(0)
# 수정 : Ready_Table - reset 안되는 문제 해결
self.ready_table.clear()
self.ready_table.setColumnCount(0)
self.gantt_table.setColumnCount(0)
self.result_table.setRowCount(0)
self.run_alg.setDisabled(True)
self.proc_list.clear()
self.history.clear()
self.history_slider.setDisabled(True)
self.add_button.setEnabled(True)
# YOSA에서 학생 목록 지워버린것도 고려해야해서 set_CPU_slot 넣었음
self.set_cpu_slot()
self.cpu_count.setEnabled(True)
# cpu_count 건드릴 일 있으니 cpu_count 갱신해줘야함.
self.cpu_count.clear()
for cpu_idx in range(1, 5):
self.cpu_count.addItem(str(cpu_idx))
self.subject_count = [0, 0, 0, 0]
self.process_count = 0
self.ready_table.setEnabled(True)
# 알고리즘 실행
def run_algorithm(self):
# 아직 알고리즘 구분 안넣고 RR을 시험삼아 돌림, 그래서 Quantum설정 하려면 RR선택하고 돌려볼것
# 수정 : 알고리즘 선택하는 부분
# 한번 설정한 프로세스 목록을 계속해서 돌릴 상황을 가정해야하기 떄문에 깊은 복사로 proc_copy_list에 가져와서 실행
proc_copy_list = copy.deepcopy(self.proc_list)
if self.cur_algo == "FCFS":
scheduler = FCFS(proc_copy_list, int(self.cpu_count.currentText()))
elif self.cur_algo == "RR":
scheduler = RR(proc_copy_list, int(self.cpu_count.currentText()), self.tq.value())
elif self.cur_algo == "SPN":
scheduler = SPN(proc_copy_list, int(self.cpu_count.currentText()))
elif self.cur_algo == "SRTN":
scheduler = SRTN(proc_copy_list, int(self.cpu_count.currentText()))
elif self.cur_algo == "HRRN":
scheduler = HRRN(proc_copy_list, int(self.cpu_count.currentText()))
elif self.cur_algo == "YOSA":
scheduler = YOSA(proc_copy_list, int(self.cpu_count.currentText()), self.tq.value())
scheduler.run()
self.history_slider.setEnabled(True)
# 실행 이후 히스토리를 self.history에 저장
# 이후 함수들은 0초를 기준으로 화면에 띄우는것, slider_control과 거이 동일하기 때문에 거기에 주석을 달겠습니다.
if self.cur_algo != "YOSA":
self.history = scheduler.history
# len(self.history) = 경과 시간, 경과 시간을 탐색할 수 있게 슬라이더를 활성화하고 슬라이더의 범위 지정, 초기는 0초
self.history_slider.setRange(0, len(self.history) - 1)
self.history_slider.setValue(0)
self.ready_table.setColumnCount(len(self.history[0][0]))
self.gantt_table.setColumnCount(0)
self.result_table.setRowCount(len(self.history[0][2]))
for q_proc_idx, q_process in enumerate(self.history[0][0]):
self.ready_table.setItem(0, q_proc_idx, QTableWidgetItem(q_process.id))
self.ready_table.item(0, q_proc_idx).setBackground(
QtGui.QColor(
q_process.color[0],
q_process.color[1],
q_process.color[2],
)
)
# DEBUG
# print("queue:", self.history[0][0][queue_process_idx].id)
for proc_idx, process in enumerate(self.history[0][2]):
self.result_table.setItem(proc_idx, 0, QTableWidgetItem(process.id))
self.result_table.setItem(proc_idx, 1, QTableWidgetItem(str(process.at)))
self.result_table.setItem(proc_idx, 2, QTableWidgetItem(str(process.bt)))
self.result_table.setItem(proc_idx, 3, QTableWidgetItem(str(process.wt)))
self.result_table.setItem(proc_idx, 4, QTableWidgetItem(str(process.tt)))
self.result_table.setItem(proc_idx, 5, QTableWidgetItem(str(process.ntt)))
self.result_table.item(proc_idx, 0).setBackground(
QtGui.QColor(
process.color[0],
process.color[1],
process.color[2],
)
)
else:
self.history = scheduler.each_student_history_list
self.history_slider.setRange(0, 24)
self.history_slider.setValue(0)
for student_idx, student in enumerate(scheduler.students):
# 공부시간
self.result_table.setItem(student_idx, 0, QTableWidgetItem(str(student.best_solo_total_study_time)))
# 각 과목 당 공부 시간
for subject in range(len(student.best_solo_subjects_grade)):
self.result_table.setItem(
student_idx,
1 + subject * 2,
QTableWidgetItem(str(student.best_solo_subject_study_case[subject])),
)
self.result_table.setItem(
student_idx,
2 + subject * 2,
QTableWidgetItem(str(round(student.best_solo_subjects_grade[subject], 2))),
)
# 개인 투자시간이 0인 경우에 결과창에 과목 당 투자 시간 및 학점이 안적히는 문제 수정
if len(student.best_solo_subjects_grade) == 0:
for subject_idx in range(len(student.subject_list)):
self.result_table.setItem(
student_idx,
1 + subject_idx * 2,
QTableWidgetItem(str(0)),
)
self.result_table.setItem(student_idx, 2 + subject_idx * 2, QTableWidgetItem(str(0)))
self.result_table.setItem(student_idx, 9, QTableWidgetItem(str(student.best_each_team_play_time)))
self.result_table.setItem(student_idx, 11, QTableWidgetItem(str(round(student.best_avg_grade, 2))))
# 팀플 학점과 전체 평균은 병합했기에 한번만 입력하면 될 것이라 생각
self.result_table.setItem(
0, 10, QTableWidgetItem(str(round(scheduler.students[0].best_team_play_grade, 2)))
)
self.result_table.setItem(0, 12, QTableWidgetItem(str(round(scheduler.team_avg_grade, 2))))
self.ready_table.setDisabled(True)
def slider_control(self):
# 원래 repaint 있던 부분들은 1초마다 자동갱신때 쓰던거라 필요없어서 지움
# 초 = 슬라이더의 값
second = self.history_slider.value()
# 표의 너비 지정
if self.cur_algo != "YOSA":
cpu_count = len(self.history[0][1])
self.ready_table.setColumnCount(len(self.history[second][0]))
self.gantt_table.setColumnCount(second)
for q_proc_idx, q_process in enumerate(self.history[second][0]):
self.ready_table.setItem(0, q_proc_idx, QTableWidgetItem(q_process.id))
self.ready_table.item(0, q_proc_idx).setBackground(
QtGui.QColor(
q_process.color[0],
q_process.color[1],
q_process.color[2],
)
)
max_len_cpu = 0
for seconds in range(second):
for cpu in range(cpu_count):
# CPU가 쉬는 도중에는 할당될 프로세스가 없을 가능성 존재
if self.history[seconds + 1][1][cpu]:
max_len_cpu = cpu
self.gantt_table.setItem(cpu, seconds, QTableWidgetItem(self.history[seconds + 1][1][cpu].id))
self.gantt_table.item(cpu, seconds).setBackground(
QtGui.QColor(
self.history[seconds + 1][1][cpu].color[0],
self.history[seconds + 1][1][cpu].color[1],
self.history[seconds + 1][1][cpu].color[2],
)
)
self.gantt_table.scrollToItem(self.gantt_table.item(max_len_cpu, second - 1))
fortext = "Real Time = " + str(second) + " sec"
self.real_time_label.setText(fortext)
# 매초마다 Result_Table값이 바뀌지 않기 떄문에 Result_Table 갱신 내역은 지웠음
else:
max_len_student = 0
student_count = len(self.history)
self.gantt_table.setColumnCount(second)
for seconds in range(second):
for student in range(student_count):
# CPU가 쉬는 도중에는 할당될 프로세스가 없을 가능성 존재
# 강제로 24시간을 잡기 떄문에 SPN / RR을 돌리고 반환하는 시간이 24시간보다 적을 가능성이 존재
if seconds + 1 < len(self.history[student]):
if self.history[student][seconds + 1][1][0]:
max_len_student = student
self.gantt_table.setItem(
student, seconds, QTableWidgetItem(self.history[student][seconds + 1][1][0].id)
)
self.gantt_table.item(student, seconds).setBackground(
QtGui.QColor(
self.history[student][seconds + 1][1][0].color[0],
self.history[student][seconds + 1][1][0].color[1],
self.history[student][seconds + 1][1][0].color[2],
)
)
self.gantt_table.scrollToItem(self.gantt_table.item(max_len_student, second - 1))
fortext = "Real Time = " + str(second) + " hour"
self.real_time_label.setText(fortext)
def default_setting(self):
# AT BT 범위 바꾼거 수정
self.at.setRange(0, 65535)
self.at_label.setText("AT")
self.bt.setRange(1, 65535)
self.cpu_label.setText("CPU")
self.tq.setRange(1, 65535)
self.tq_label.setText("Quantum")
# Proc_Table 기본 세팅
self.proc_table.setColumnCount(3)
self.proc_table.setHorizontalHeaderLabels(["Process Name", "Arrival Time", "Burst Time"])
header = self.proc_table.horizontalHeader()
for column_idx in range(self.column_count):
header.setSectionResizeMode(column_idx, QHeaderView.Stretch)
# Gantt Table 기본 세팅
self.set_cpu_slot()
# Result_Table 기본 세팅
self.result_table.setColumnCount(6)
self.result_table.setHorizontalHeaderLabels(
["Process Name", "Arrival Time", "Burst Time", "Waiting Time", "Turnaround Time", "Normalized TT"]
)
self.result_table.verticalHeader().setVisible(False)
header = self.result_table.horizontalHeader()
self.result_table.horizontalHeader().setFixedHeight(25)
for time_table_col in range(6):
header.setSectionResizeMode(time_table_col, QHeaderView.Stretch)
self.real_time_label.setText("Real Time = 0 sec")
self.student_list.setDisabled(True)
def yosa_setting(self):
# AT BT 범위 바꾼거 수정
self.at.setRange(1, 4)
self.at_label.setText("학점")
self.bt.setRange(1, 24)
self.cpu_label.setText("학생 수")
self.tq.setRange(1, 96)
self.tq_label.setText("팀프 시간")
# Proc_Table 기본 세팅
self.proc_table.setColumnCount(4)
self.proc_table.setHorizontalHeaderLabels(["과목 이름", "학점", "소요 시간", "대상 학생"])
header = self.proc_table.horizontalHeader()
for column_idx in range(4):
header.setSectionResizeMode(column_idx, QHeaderView.Stretch)
# Gantt Table 기본 세팅
# 이건 setCPU 여기서 바꿔야할듯
self.set_cpu_slot()
# Result_Table 기본 세팅
self.result_table.setColumnCount(13)
self.result_table.setHorizontalHeaderLabels(
[
"개인 공부\n투자 시간",
"과목 1\n공부 시간",
"과목 1\n학점",
"과목 2\n공부 시간",
"과목 2\n학점",
"과목 3\n공부 시간",
"과목 3\n학점",
"과목 4\n공부 시간",
"과목 4\n학점",
"팀플\n투자 시간",
"팀플 학점",
"평균 학점",
"팀 전체\n평균 학점",
]
)
self.result_table.horizontalHeader().setFixedHeight(60)
# Result의 Vertical과 StudentList부분은 set_CPU_slot으로 넘김
header = self.result_table.horizontalHeader()
for time_table_col in range(13):
header.setSectionResizeMode(time_table_col, QHeaderView.Stretch)
self.real_time_label.setText("Real Time = 0 hour")
self.student_list.setEnabled(True)
self.tq.setEnabled(True)
if __name__ == "__main__":
app = QApplication(sys.argv)
ex = MyApp()
sys.exit(app.exec_())