العودية أو الاستدعاء الذاتي في علم الحاسوب هو طريقة التي فيها الحل لمسألة ما يعتمد على حلول مسائل أصغر من نفس النوع.
[1]
بالإمكان تطبيق هذا النهج على أنواع عديدة من المسائل، وهو واحد من الأفكار المركزية بعلم الحاسوب.
[2]
تدعم أغلب لغات البرمجة الاستدعاء الذاتي عن طريق السماح لدالة أن تستدعي نفسها ضمن نص البرنامج نفسه. بعض لغات البرمجة الوظيفة لا تعرّف مبان تكرارية (looping constructs) ولكن تعتمد فقط على الاستدعاء الذاتي لتكرار تنفيذ كود معين. برهنت نظرية الحاسوبية أن اللغات التي تستخدم الاستدعاء الذاتي فقط معادلة رياضياً للغات الحتمية، بمعنى أنه باستطاعتهم حل أي نوع من المسائل دون الحاجة لمباني التحكم النموذجية مثل حلقات "while" أو "for".
مثال تقليدي لدالة الاستدعاء الذاتي هي دالة تستخدم لحساب العامليلعدد صحيح:
Pseudocode
function factorial is: input: integer n such that n >= 0 output: [n × (n-1) × (n-2) × … × 1]
1. if n is 0, return 1
2. otherwise, return [ n × factorial(n-1) ]
end factorial
/** * Recursively calculate the kth Fibonacci number. * * @param k indicates which (positive) Fibonacci number to compute. * @return the kth Fibonacci number. */privatestaticintfib(intk){// Base Cases:// If k == 0 then fib(k) = 0.// If k == 1 then fib(k) = 1.if(k<2){returnk;}// Recursive Case:// If k >= 2 then fib(k) = fib(k-1) + fib(k-2).else{returnfib(k-1)+fib(k-2);}}
function gcd is:
input: integer x, integer y such that x >= y and y >= 0
1. if y is 0, returnx
2. otherwise, return [ gcd(y, (remainder of x/y)) ]
end gcd
علاقة تكرارية للقاسم المشترك الأكبر، بحيث أن يمثل الباقي من قسمة :
للنقاش الكامل وشرح المسألة، راجع المقال التفصيلي. ببساطة احسب المسألة هكذا: بإعطاء ثلاثة أعمدة، واحد مع مجموعة من N أقراص بحجم متزايد، حدد العدد الأصغر من الخطوات لنقل جميع الأقراص من مكانهم البدائي إلى عمود آخر بدون وضع قرص ما فوق قرص أصغر منه.
علاقة تكرارية لبرج هانوي:
Pseudocode
function hanoi is: input: integer n, such that n >= 1
1. if n is 1 then return 1
2. return [2 * [call hanoi(n-1)] + 1]
end hanoi
بحث ثنائي
خوارزمية البحث الثنائي هي طريقة للبحث في مصفوفة مرتبة عن عنصر واحد عن طريق تقسيم المصفوفة للنصف في كل مرور. الخدعة هي اختيار نقطة وسطية قريبة عن مركز المصفوفة، قارن القيمة في النقطة مع القيم المراد البحث عنها ومن ثم اتباع واحد من ثلاث الخيارات التالية: القيمة وجدت في النفطة الوسطية، قيمة النقطة الوسطية أكبر من القيمة المراد البحث عنها، أو أن قيمة النقطة الوسطية أصغير من القيمة المراد البحث عنها.
يتم استخدام الاستدعاء الذاتي لأن بكل مرور يتم إنشاء مصفوفة جديدة عن طريق تقسيم القديمة للنصف. ويتم استدعاء دالة البحث ذاتياً، هذه المرة على مصفوفة جديدة (وأصغر).
/* Call binary_search with proper initial conditions. INPUT: data is an array of integers SORTED in ASCENDING order, toFind is the integer to search for, count is the total number of elements in the array OUTPUT: result of binary_search */intsearch(int*data,inttoFind,intcount){// Start = 0 (beginning index)// End = count - 1 (top index)returnbinary_search(data,toFind,0,count-1);}/* Binary Search Algorithm. INPUT: data is a array of integers SORTED in ASCENDING order, toFind is the integer to search for, start is the minimum array index, end is the maximum array index OUTPUT: position of the integer toFind within array data, -1 if not found */intbinary_search(int*data,inttoFind,intstart,intend){//Get the midpoint.intmid=start+(end-start)/2;//Integer division//Stop condition.if(start>end)return-1;elseif(data[mid]==toFind)//Found?returnmid;elseif(data[mid]>toFind)//Data is greater than toFind, search lower halfreturnbinary_search(data,toFind,start,mid-1);else//Data is less than toFind, search upper halfreturnbinary_search(data,toFind,mid+1,end);}
^Graham، Ronald (1990). Concrete Mathematics. Chapter 1: Recurrent Problems. مؤرشف من الأصل في 2019-09-15. {{استشهاد بكتاب}}: الوسيط author-name-list parameters تكرر أكثر من مرة (مساعدة) والوسيط غير المعروف |nopp= تم تجاهله يقترح استخدام |no-pp= (مساعدة)