-
Notifications
You must be signed in to change notification settings - Fork 19
/
Copy pathChapter_IV-2.tex
480 lines (388 loc) · 20.9 KB
/
Chapter_IV-2.tex
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
\chapter{المكدّسات والطوابير (\textenglish{Stacks and Queues})}
لقد اكتشفنا مع القوائم المتسلسلة طريقة جديدة أكثر مرونة من الجداول لتخزين البيانات. هذه القوائم مرنة بشكل خاص لأنه يمكننا أن نُدرج فيها ونحذف منها بيانات من أي مكان أردنا وفي أية لحظة.
المكدّسات والطوابير التي سنكتشفها هنا هما شكلان خاصّان نوعًا ما من القوائم المتسلسلة. فهما تسمحان بالتحكّم بالطريقة التي تُضاف بها العناصر الجديدة إليها. هذه المرة لن نقوم بإضافة عناصر جديدة في وسط القائمة، بل فقط في البداية أو النهاية.
المكدّسات والطوابير تعتبران مفيدتان للغاية من أجل البرامج التي تحلل المعطيات الّتي تصل بالتدريج. سنرى بالتفصيل كيف تعملان في هذا الفصل.
تتشابه المكدّسات والطوابير كثيرًا، لكنهما تختلفان اختلافًا بسيطًا ستتعرف عليه بسرعة. سنكتشف أولًا المكدّسات والتي ستذكّرك بالقوائم المتّصلة بشكل كبير.
بشكل عام، سيكون هذا الفصل بسيطًا إذا كنت قد فهمت جيّدًا كيفية عمل القوائم المتسلسلة. إن لم تكن هذه حالتك، فأعد قراءة الفصل السابق لأننا سنحتاج إليه.
\section{المكدّسات (\textenglish{Stacks})}
تخيّل مكدّسًا للقطع النقدية (الصورة التالية). يمكنك إضافة قطع أخرى واحدة تلو الأخرى في أعلى المكدّس، ويمكنك أيضًا نزع القطع من أعلى المكدّس. بالمقابل، لا يمكن نزع قطعة من أسفل المكدّس. إن أردت التجريب، أتمنى لك حظًا موفّقًا!
\begin{figure}[H]
\centering
\includegraphics[height=0.2\textheight]{Chapter_IV-2_Money-stack}
\end{figure}
\subsection{كيفية عمل المكدّسات}
مبدأ عمل المكدّسات في البرمجة ينصّ على تخزين البيانات مع وصولها على التوالي واحدة فوق الأخرى لكي نستطيع استرجاعها فيما بعد. مثلا، تخيّل مكدّسًا للأعداد الصحيحة من نوع
\InlineCode{int}
(الصورة الموالية). لو أضيف عنصرًا (نتكلّم عن
\textbf{التكديس})،
فستتم إضافته في أعلى المكدّس، تمامًا كما في لعبة
\textenglish{Tetris}:
\begin{figure}[H]
\centering
\includegraphics[height=0.15\textheight]{Chapter_IV-2_Stack}
\end{figure}
\begin{figure}[H]
\centering
\includegraphics[height=0.2\textheight]{Chapter_IV-2_Stack-add}
\end{figure}
الأكثر أهمية هو وجود عملية تقوم باستخراج الأعداد من المكدّس. نحن نتكلّم عن
\textbf{إلغاء التكديس}.
نسترجع القيم واحدة تلو الأخرى، بدءً من الأخيرة الموضوعة أعلى المكدّس (الصورة الموالية). ننزع البيانات على التوالي حتى نصل إلى قاع المكدّس.
\begin{figure}[H]
\centering
\includegraphics[height=0.2\textheight]{Chapter_IV-2_Stack-remove}
\end{figure}
نسمي هذا بخوارزمية
\textbf{\textenglish{LIFO}}،
و التي تعني
"\textenglish{Last In First Out}".
الترجمة: "آخر عُنصر تمت إضافته، هو أول عنصر يخرج".
عناصر المكدّس مرتبطة ببعضها بطريقة القوائم المتسلسلة. فهي تحمل مؤشّرًا نحو العنصر الموالي ولا تُوضع بالضرورة بجنب بعضها في الذاكرة. يجب على آخر عُنصر (في أقصى أسفل المكدّس) أن يؤشّر نحو
\InlineCode{NULL}
لكي يقول أنّنا\dots لمسنا القاع:
\begin{figure}[H]
\centering
\includegraphics[height=0.15\textheight]{Chapter_IV-2_Stack-numbers}
\end{figure}
\begin{question}
فيما ينفع كلّ هذا، واقعيّا؟
\end{question}
توجد برامج تحتاج فيها إلى تخزين البيانات مؤقّتًا لإخراجها اعتمادًا على ترتيب محدد: يجب على آخر عُنصر أدخلته أن يخرج هو الأول.
لأعطي مثالًا واقعيًا، يستعمل نظام التشغيل في حاسوبك هذا النوع من الخوارزميات لكي يتذكر الترتيب الذي تم استدعاء الدوال فيه. تخيّل مثالا:
\begin{enumerate}
\item يبدأ البرنامج بالدالة
\InlineCode{main}
(مثل كل مرّة).
\item تستدعي فيها الدالة
\InlineCode{play}.
\item تقوم هذه الدالة
\InlineCode{play}
بدورها باستدعاء الدالة
\InlineCode{load}.
\item ما إن تنتهي الدالة
\InlineCode{load}،
نعود إلى الدالة
\InlineCode{play}.
\item ما إن تنتهي الدالة
\InlineCode{play}،
نعود إلى الدالة
\InlineCode{main}.
\item أخيرًا، ما إن تنتهي الدالة
\InlineCode{main}.
لا تبقى أية دالة تحتاج إلى الاستدعاء، ينتهي البرنامج.
\end{enumerate}
لكي "يتذكّر" الترتيب الذي تم فيه استدعاء الدوال، يُنشئ الحاسوب مكدّسًا لهذه الدوال على التوالي:
\begin{figure}[H]
\centering
\includegraphics[width=0.5\textwidth]{Chapter_IV-2_Stack-history}
\end{figure}
هذا مثال واقعيّ عن استعمال المكدّسات. وبفضل هذه التقنية يعرف الجهاز الآن إلى أي دالة يجب عليه أن يعود. يمكنه أن يكدّس 100 استدعاء للدوال إن وجب الأمر، لكنّه سيرجع ليجد الدالة الرئيسية في أسفل المكدّس!
\subsection{إنشاء نظام مكدّس}
و الآن بما أننا فهمنا مبدأ عمل المكدّسات، فلنحاول بناء واحد. تماما مثل القوائم المتسلسلة، لا يوجد نظام مكدّس مضمّن في لغة
\textenglish{C}.
يجب أن ننشئه بأنفسنا.
سيكون لكل عنصر من المكدّس هيكل مماثل للهيكل الخاص بالقائمة المتسلسلة:
\begin{Csource}
typedef struct Element Element;
struct Element
{
int number;
Element *next;
};
\end{Csource}
يحتوي هيكل التحكّم على عنوان أول عنصر من المكدس، أي العنصر المتواجد في الأعلى:
\begin{Csource}
typedef struct Stack Stack;
struct Stack
{
Element *first;
};
\end{Csource}
سنحتاج ككل إلى الدوال التالية:
\begin{itemize}
\item تكديس عنصر.
\item إلغاء تكديس عنصر.
\end{itemize}
ستلاحظ أنه، على خلاف القوائم المتسلسلة، لا نتكلّم لا عن الإضافة ولا عن الحذف. نتكلّم عن التكديس وإلغاء التكديس. لأن هاتين العمليتين محدودتان على عنصر واحد محدد، كما رأينا. بهذا، يمكننا إضافة ونزع عنصر من المكدّس من الأعلى فقط.
يمكننا أيضًا كتابة دالة لإظهار محتوى المكدّس، أمر عمليّ للتأكد من أن البرنامج يعمل بشكل صحيح.
هيا بنا!
\subsubsection{التكديس (\textenglish{stacking})}
يجدر بدالتنا
\InlineCode{stack}
أن تأخذ كمعاملين هيكل التحكّم في المكدّس (من نوع
\InlineCode{Stack})
و العدد الجديد لتخزينه.\\
أذكّرك بأننا نخزّن هنا أعدادًا صحيحة
\InlineCode{int}،
لكن لا شيء يمنعنا من تعديل هذه الأمثلة لأنواع أخرى من البيانات. يمكننا تخزين أي شيء: أعداد
\InlineCode{double}،
محارف
\InlineCode{char}،
سلاسل محارف، جداول وحتى هياكل أخرى!
\begin{Csource}
void stack(Stack *stk, int newNumber)
{
Element *new = malloc(sizeof(*new));
if (stk == NULL || new == NULL)
{
exit(EXIT_FAILURE);
}
new->number = newNumber;
new->next = stk->first;
stk->first = new;
}
\end{Csource}
تتم الإضافة في بداية المكدّس لأنه، كما رأينا، يستحيل القيام بذلك في المنتصف. هذا مبدأ عمل المكدّسات، نضيف دائما من الأعلى. \\
بهذا، على عكس القوائم المتسلسلة، لا يجب أن ننشئ دالة لإدراج عنصر في منتصف المكدّس. يجب أن تكون الدالة
\InlineCode{stack}
هي الوحيدة الّتي يمكنها إضافة عناصر جديدة للمكدس.
\subsubsection{إلغاء التكديس (\textenglish{unstacking})}
دور دالة إلغاء التكديس هو حذف العنصر المتواجد في أعلى المكدّس، قد تشك في ذلك. لكن يجب على هذه الدالة أيضا أن تُرجع إلينا العنصر الذي حذفته. في حالتنا، هو العدد الّذي كان موجودا في أعلى المكدّس.
هذه هي الطريقة التي نصل بها إلى عناصر المكدّس: ننزعها واحدًا تلو الآخر. نحن لا نتقدّم فيها باحثين عن الوصول إلى ثاني وثالث عنصر. بل نطلب دائمًا استرجاع على أول عنصر.
دالتنا
\InlineCode{unstack}
ستُرجع إذا
\InlineCode{int}
يوافق العدد المتواجد في رأس المكدّس:
\begin{Csource}
int unstack(Stack *stack)
{
if (stack == NULL)
{
exit(EXIT_FAILURE);
}
int unstackedNumber = 0;
Element *unstackedElement = stack->first;
if (stack != NULL§\footnotemark§ && stack->first != NULL)
{
unstackedNumber = unstackedElement->number;
stack->first = unstackedElement->next;
free(unstackedElement);
}
return unstackedNumber;
}
\end{Csource}
\begin{tcolorbox}[title={\footnotemark[1]ملاحظة مُرَاجِع الكتاب}, colback=orange!20, colframe=orange!70, fontupper=\small, coltitle=white, fonttitle=\normalsize, attach title]
يبدو لي أن مؤلّف الكتاب قد أضاف جزءً عديم الفائدة في الشرط الثاني:
\InlineCode{stack != NULL}،
فَبِوُصول البرنامج إلى ذلك السطر سيكون هذا الجزء من الشرط خاطئا بكلّ تأكيد لأنّنا قد تحقّقنا من عكسه في الشرط الأوّل
(\InlineCode{if(stack == NULL)}).
و حتى عند عدم وجود الشرط الأوّل، فالتعليمة
\InlineCode{stack->first}
كانت ستعطّل البرنامج لأنّ مؤشّرًا
\InlineCode{NULL}
لا يملك أيّة مركّبات!
\end{tcolorbox}
نسترجع العدد الذي في رأس المكدّس لنبعثه في نهاية الدالة. نعدّل عنوان أول عنصر من المكدّس بما أن هذا الأخير يتغير.
أخيرًا، نحذف بالتأكيد رأس المكدّس القديم باستعمال
\InlineCode{free}.
\subsubsection{إظهار محتوى المكدّس}
بالرغم من أن هذه الدالة غير ضروريّة (الدالتان
\InlineCode{stack}
و
\InlineCode{unstack}
كافيتان للتحكّم في المكدّس!)، ستكون مهمّة لاختبار عمل المكدّس وخاصّة "لمعاينة" النتيجة:
\begin{Csource}
void displayStack(Stack *stack)
{
if (stack == NULL)
{
exit(EXIT_FAILURE);
}
Element *current = stack->first;
while (current != NULL)
{
printf("%d\n", current->number);
current = current ->next;
}
printf("\n");
}
\end{Csource}
بما أن هذه الدالة بسيطة بسخافة، فهي لا تحتاج شرحًا.
بالمقابل، حان الآن إذًا لكتابة الدالة الرئيسية لاختبار سلوك مكدّسنا:
\begin{Csource}
int main()
{
Stack *myStack = initialization(); // See the previous chapter
stack(myStack, 4);
stack(myStack, 8);
stack(myStack, 15);
stack(myStack, 16);
stack(myStack, 23);
stack(myStack, 42);
printf("\nStack's state :\n");
displayStack(myStack);
printf("I unstack %d\n", unstack(myStack));
printf("I unstack %d\n", stack(myStack));
printf("\nStack's state :\n");
displayStack(myStack);
return 0;
}
\end{Csource}
نُظهر حالة المكدّس بعد الكثير من التكديس ومرة أخرى بعد كثير من إلغاء التكديس. نُظهر أيضًا العدد الذي قُمنا بحذفه في كلّ مرة نقوم بإلغاء التكديس. النتيجة في الكونسول هي التالية:
\begin{Console}
Stack's state :
42
23
16
15
8
4
I unstack 42
I unstack 23
Stack's state :
16
15
8
4
\end{Console}
تأكّد أنك تفهم جيّدًا ما يحصل في البرنامج. إذا فهمت هذا، فقد فهمت كيفية عمل المكدّسات!
يمكنك تنزيل مشروع المكدّسات كاملًا لو أردت:
\url{http://www.siteduzero.com/uploads/fr/ftp/mateo21/c/piles.zip}
\section{الطوابير (\textenglish{Queues})}
تشبه الطوابير المكدّسات كثيرًا، إلا أنها تعمل بالاتجاه المعاكس!
\subsection{كيفية عمل الطوابير}
في هذا النظام، يتم وضع العناصر الواحد بعدَ الآخر. أوّل عنصر يخرج من الطابور هو أول عنصر يدخل إليه. نتكلّم هنا عن خوارزمية
\textenglish{FIFO} (\textenglish{First In First Out})،
و هذا يعني: "أول من يصل هو أول من يخرج".
تسهل المشابهة بالحياة اليوميّة. حينما تشتري تذكرة لمشاهدة السينما، تقف في طابور شبّاك التذاكر (الصورة الموالية). باستثناء إن كنت أحد إخوة بائع التذاكر، يجدر بك الوقوف في الطابور وانتظار دورك مثل كلّ الآخرين. أول الواصلين هو أول من تتم خدمته.
\begin{figure}[H]
\centering
\includegraphics[width=0.6\textwidth]{Chapter_IV-2_Humans-queue}
\end{figure}
في البرمجة، تفيد الطوابير في إيقاف مؤقّت للمعلومات حسب الترتيب الذي وصلت به. مثلًا، في برنامج مُحادثة، لو تتلقى ثلاثة رسائل يفصلها فارق زمني قصير جدًا، يتم صفُّها في طابور واحدة تلو الأخرى في الذاكرة. ثم يتم إظهار أوّل رسالة وصلت ثم الثانية وهكذا.
يتم تخزين الأحداث التي تبعثها المكتبة
\textenglish{SDL}
التي قُمنا بدراستها أيضا في طابور. إذا حرّكت الفأرة، يتم توليد حدث من أجل كلّ بيكسل تحرّك فوقه مؤشّر الفأرة. تخزن
\textenglish{SDL}
الأحداث في طابور ثم تبعثها لنا واحدا واحدا في كلّ مرة نستدعي فيها الدالة
\InlineCode{SDL\_PollEvent}
(أو
\InlineCode{SDL\_WaitEvent}).
في لغة
\textenglish{C}،
الطابور هو قائمة متسلسلة أين يقوم كلّ عنصر فيها بالتأشير على العنصر الموالي، تمامًا مثل المكدّسات. آخر عنصر من الطابور يؤشّر نحو
\InlineCode{NULL}:
\begin{figure}[H]
\centering
\includegraphics[width=0.4\textwidth]{Chapter_IV-2_Queue}
\end{figure}
\subsection{إنشاء نظام طابور}
نظام الطابور يشبه ذلك الخاص بالمكدّسات. يوجد اختلاف بسيط في كون أن العناصر تخرج في الاتجاه المعاكس، لا يوجد شيء صعب إن كنت قد فهمت المكدّسات.
سننشئ هيكل
\InlineCode{Element}
و هيكل تحكّم
\InlineCode{Queue}:
\begin{Csource}
typedef struct Element Element;
struct Element
{
int number;
Element *next;
};
typedef struct Queue Queue;
struct Queue
{
Element *first;
};
\end{Csource}
تماما كالمكدّسات، كل عنصر من الطابور سيكون من نوع
\InlineCode{Element}.
بالاستعانة بالمؤشّر
\InlineCode{first}
سنتوفّر دائمًا على العنصر الأول ويمكننا من خلاله الصعود إلى آخر عنصر.
\subsubsection{إضافة عنصر إلى الطابور (\textenglish{enqueuing})}
الدالة التي تضيف عُنصُرًا إلى الطابور تسمّى دالة "الإلحاق"
(\textenglish{enqueuing}).
توجد حالتان:
\begin{itemize}
\item إما أن الطابور فارغ، في هذه الحالة يجب أن ننشئ الطابور بجعل المؤشّر
\InlineCode{first}
يؤشّر نحو العنصر الجديد الذي نحن بصدد إنشائه.
\item إما أن الطابور غير فارغ، في هذه الحالة يجب أن نتقدّم في الطابور انطلاقًا من العنصر الأول حتى نصل إلى آخر عنصر. نضيف العنصر الجديد بعد آخر عنصر.
\end{itemize}
إليك ما يمكننا فعله عمليًا:
\begin{Csource}
void enqueue(Queue *q, int newNumber)
{
Element *new = malloc(sizeof(*new));
if (q == NULL || new == NULL)
{
exit(EXIT_FAILURE);
}
new->number = newNumber;
new->next = NULL;
if (q->first != NULL) // The queue is not empty
{
// We move to the end of the queue
Element *currentElement = q->first;
while (currentElement->next != NULL)
{
currentElement = currentElement->next;
}
currentElement->next = new;
}
else /* The queue is empty, it's our first element */
{
q->first = new;
}
}
\end{Csource}
ترى في هذه الشفرة المصدرية تحليل كلتا الحالتين، كلّ منهما يجب أن تتم دراستها على حدى. الاختلاف مقارنةً بالمكدّسات، والذي يضيف لمسة صعوبة صغيرة، هو أنه يجب التموضع في نهاية الطابور لوضع العنصر الجديد. لكن لابأس فحلقة
\InlineCode{while}
كافية للقيام باللازم، هذا ما يمكنك ملاحظته.
\subsubsection{إزالة عنصر من الطابور (\textenglish{dequeuing})}
عملية إلغاء الإلحاق
(\textenglish{dequeuing})
تشابه كثيرًا عملية إلغاء التكديس. بامتلاكنا مؤشّرا نحو أول عنصر من الطابور، يكفي أن ننزعه وأن نُرجع قيمته.
\begin{Csource}
int dequeue(Queue *queue)
{
if (queue == NULL)
{
exit(EXIT_FAILURE);
}
int dequeuedNumber = 0;
// We verify if there's something to dequeue
if (queue->first != NULL)
{
Element *dequeuedElement = queue->first;
dequeuedNumber = dequeuedElement->number;
queue->first = dequeuedElement->next;
free(dequeuedElement);
}
return dequeuedNumber;
}
\end{Csource}
\subsubsection{حان دورك!}
تبقى دالة إظهار محتوى الطابور
\InlineCode{displayQueue}
و عملها مشابه لما قمنا به مع المكدّسات. سيسمح لك هذا بالتأكد من سلامة عمل الطابور.
قم بعد ذلك بكتابة
\InlineCode{main}
من أجل تجريب البرنامج. يجدر بنتيجة البرنامج أن تشبه هذه:
\begin{Console}
Queue's state :
4 8 15 16 23 42
I dequeue 4
I dequeue 8
Queue's state :
15 16 23 42
\end{Console}
يجدر أن تكون قادرًا على إنشاء مكتبة الطوابير الخاصة بك، بملفات
\InlineCode{queue.h}
و
\InlineCode{queue.c}
مثلًا.
أقترح عليك تنزيل مشروع التحكّم في الطوابير كاملًا إن أردت. إنه يتضمّن الدالة
\InlineCode{displayQueue}:
\url{http://www.siteduzero.com/uploads/fr/ftp/mateo21/c/files.zip}
\section*{ملخّص}
\begin{itemize}
\item تسمح المكدّسات والطوابير بتنظيم معطيات في الذاكرة عند وصولها بالتوالي.
\item تستعمل المكدّسات والطوابير نظام قوائم متسلسلة لتجميع العناصر.
\item في حالة المكدّسات، تتم إضافة المعطيات الواحدة فوق الأخرى. وإن أردنا استخراج بيانة، فسنستخرج آخر واحدة والتي كنا بصدد إضافتها (الأحدث). نتكّلم هنا عن خوارزمية
\textenglish{LIFO} (\textenglish{Last In First Out}).
\item في حالة الطوابير، تتم إضافة المعطيات الواحدة بعد الأخرى. نقوم باستخراج البيانة الأولى والتي قمنا بإضافتها أولا للطابور (الأقدم). نتكلّم عن خوارزمية
\textenglish{FIFO} (\textenglish{First In First Out}).
\end{itemize}