Programming Basics SQL HTML CSS JavaScript React Python C++ Java JavaFX Swing Problem Solving English English Conversations Computer Fundamentals Linux Learn Typing

C++طريقة إنشاء ثريد و تمرير دالة له

في المثال التالي قمنا بتعريف دالة إسمها func() عند استدعائها تقوم بطباعة الجملة "func is executed.." 5 مرات بواسطة حلقة مع التوقف لمدة ثانية في كل مرة.
في الدالة main() قمنا بإنشاء كائن من الكلاس thread ينفذ الدالة func() بالإضافة إلى أننا قمنا بإنشاء حلقة تقوم بطباعة الجملة "main is executed.." 5 مرات مع التوقف لمدة ثانية في كل مرة أيضاً حتى تلاحظ كيف أنهما سيتنفذان في وقت واحد.

المثال الأول

main.cpp
#include <iostream>
#include <thread>
#include <chrono>

using namespace std;

// خمس مرات مع التوقف لمدة ثانية في كل مرة "func is executed.." عند استدعائها تقوم بطباعة النص func() هنا قمنا بتعريف دالة إسمها
void func()
{
    for(int i=1; i<=5; i++)
    {
        cout << "func is executed.. \n";
        this_thread::sleep_for(chrono::seconds(1));
    }
}

int main()
{
    // مع الإنتقال إلى الكود الموجود بعدها بشكل مباشر func() حتى يقوم بتنفيذ الدالة thread هنا قمنا بإنشاء كائن من الكلاس 
    thread t(func);
    
    // خمس مرات مع التوقف لمدة ثانية في كل مرة "main is executed.." عندما يصل المترجم لهذه الحلقة سيقوم بطباعة الجملة
    for(int i=1; i<=5; i++)
    {
        cout << "main is executed.. \n";
        this_thread::sleep_for(chrono::seconds(1));
    }

    // لجعل المترجم ينتظر تنفيذ أوامره قبل أن يتوقف الثريد الرئيسي في البرنامج حتى لا يسبب مشكلة t من الكائن join() هنا قمنا باستدعاء الدالة
    t.join();
    
    return 0;
}

سنحصل على نتيجة تشبه النتيجة التالية عند التشغيل.

main is executed..
func is executed..
func is executed..
main is executed..
func is executed..
main is executed..
func is executed..
main is executed..
func is executed..
main is executed..


في المثال التالي قمنا بإعادة المثال الأول و لكننا قمنا بتعريف الدالة التي سينفذها الثريد بأسلوب Lambda Expression.

المثال الثاني

main.cpp
#include <iostream>
#include <thread>
#include <chrono>

using namespace std;

int main()
{
    // Lambda Expression هنا قمنا بتعريف و تمرير الدالة التي سينفذها الثريد مباشرةً فيه بأسلوب
    thread t( []{
        for(int i=1; i<=5; i++)
        {
            cout << "lambda is executed.. \n";
            this_thread::sleep_for(chrono::seconds(1));
        }
    });
    
    for(int i=1; i<=5; i++)
    {
        cout << "main is executed.. \n";
        this_thread::sleep_for(chrono::seconds(1));
    }

    t.join();
    
    return 0;
}

سنحصل على نتيجة تشبه النتيجة التالية عند التشغيل.

main is executed..
lambda is executed..
lambda is executed..
main is executed..
lambda is executed..
main is executed..
lambda is executed..
main is executed..
lambda is executed..
main is executed..


في المثال التالي قمنا بإعادة المثال الأول و لكننا قمنا بتمرير دالة للثريد بأسلوب Function Object.
ملاحظة: هنا يجب جعل الدالة التي سينفذها الثريد تحتوي على باراميتر واحد على الأقل لهذا السبب وضعنا فيها الباراميتر n.

المثال الثالث

main.cpp
#include <iostream>
#include <thread>
#include <chrono>

using namespace std;

// هنا قمنا بتعريف كلاس يحتوي على الدالة التي سنمررها لكائن الثريد
class FnObject
{
    public:
        void operator()(int n) {
            for(int i=1; i<=n; i++)
            {
                cout << "FnObject is executed.. \n";
                this_thread::sleep_for(chrono::seconds(1));
            }
        }
};

int main()
{
    // هنا قمنا بتمرير كائن من الكلاس الذي قمنا بتعريفه لكائن الثريد
    // لأنه يتوقع أن نمرر له قيمة n كما أننا قمنا بتمرير القيمة 5 للبارميتر
    thread t(FnObject(), 5);
    
    for(int i=1; i<=5; i++)
    {
        cout << "main is executed.. \n";
        this_thread::sleep_for(chrono::seconds(1));
    }

    t.join();
    
    return 0;
}

سنحصل على نتيجة تشبه النتيجة التالية عند التشغيل.

main is executed..
FnObject is executed..
main is executed..
FnObject is executed..
main is executed..
FnObject is executed..
FnObject is executed..
main is executed..
FnObject is executed..
main is executed..