Download presentation
Presentation is loading. Please wait.
1
بسم الله الرحمن الرحيم
2
دانشكده فناوري اطلاعات
دانشگاه پيام نور دانشكده فناوري اطلاعات
3
تهيه كننده: دكتر احمد فراهي
برنامه سازي پيشرفته تهيه كننده: دكتر احمد فراهي
4
مقدمه: زبان C يک زبان همه منظوره است. دستورالعملهاي اين زبان بسيار شبيه عبارات جبري و نحو آن شبيه جملات انگليسي مي باشد. اين امر سبب ميشود که C يک زبان سطح بالا باشد که برنامهنويسي در آن آسان است ›››
5
++C که از نسل C است، تمام ويژگيهاي C را به ارث برده است
++C که از نسل C است، تمام ويژگيهاي C را به ارث برده است. اما برتري فني ديگري هم دارد: C++ اکنون «شيگرا» است. ميتوان با استفاده از اين خاصيت، برنامههاي شيگرا توليد نمود. برنامههاي شيگرا منظم و ساختيافتهاند، قابل روزآمد کردناند، به سهولت تغيير و بهبود مييابند و قابليت اطمينان و پايداري بيشتري دارند.
6
اهم مطالب اين كتاب : جلسه اول: «مقدمات برنامهنويسي با C++»
جلسه دوم: «انواع اصلي» جلسه سوم: «انتخاب» جلسه چهارم: ‹‹تكرار» جلسه پنجم: «توابع» جلسه ششم: « آرايهها»
7
جلسه هفتم: «اشارهگرها و ارجاعها»
جلسه هشتم: «رشتههاي كاراكتري و فايلها در ++Cاستاندارد» جلسه نهم: «شيئگرايي» جلسه دهم: «سربارگذاري عملگرها» جلسه يازدهم: «تركيب و وراثت»
8
مقدمات برنامهنويسي با C++
جلسه اول مقدمات برنامهنويسي با C++
9
آنچه در اين جلسه مي خوانيد:
1- چرا C++ ؟ 2- تاريخچۀ C++ 3- آمادهسازي مقدمات 4- شروع کار با C++ 5- عملگر خروجي 6- ليترالها و کاراکترها 7- متغيرها و تعريف آنها 8- مقداردهي اوليه به متغيرها 9- ثابتها 10- عملگر ورودي
10
هدف کلي: آشنايي با تاريخچه و مزاياي زبان برنامهنويسي C++ و بيان مفاهيم بنيادي شيگرايي و عناصر مهم برنامههاي C++
11
هدفهاي رفتاري: انتظار ميرود پس از پايان اين جلسه بتوانيد:
- مزاياي زبان C++ را بر زبانهاي مشابه ذکر کرده و تفاوت آن را با زبان C بيان کنيد. - شرح مختصري از روند پيشرفت زبانهاي برنامهنويسي را بيان کرده و مشکلات هر دوره را به اختصار شرح دهيد. - مزاياي شيگرايي در توليد نرمافزار را برشماريد. - اصول سهگانۀ شيگرايي را نام برده و هر يک را به اختصار شرح دهيد. >>
12
- قالب کلي برنامههاي C++ را بشناسيد و بتوانيد برنامههاي کوچک را نوشته و آزمايش کنيد.
- نحوۀ اعلان متغيرها و شيوۀ مقداردهي به آنها را بدانيد. - سه موجوديت «ليترال»، «کاراکتر» و «عدد» را شناخته و فرق بين آنها را شرح دهيد. - علت و شيوههاي افزودن توضيح به کد برنامه را شرح دهيد. - علت و شيوۀ معرفي ثابتها در برنامه را شرح دهيد.
13
مقدمه در دهه 1970 در آزمايشگاههاي بل زباني به نام C ايجاد شد. انحصار اين زبان در اختيار شرکت بل بود تا اين که در سال 1978 توسط Kernighan و Richie شرح کاملي از اين زبان منتشر شد و به سرعت نظر برنامهنويسان حرفهاي را جلب نمود. هنگامي که بحث شيگرايي و مزاياي آن در جهان نرمافزار رونق يافت، زبان C که قابليت شيگرايي نداشت ناقص به نظر ميرسيد تا اين که در اوايل دهۀ 1980 دوباره شرکت بل دست به کار شد و Bjarne Stroustrup زبان C++ را طراحي نمود
14
C++ ترکيبي از دو زبان C و Simula بود و قابليتهاي شيگرايي نيز داشت
C++ ترکيبي از دو زبان C و Simula بود و قابليتهاي شيگرايي نيز داشت. از آن زمان به بعد شرکتهاي زيادي کامپايلرهايي براي C++ طراحي کردند. اين امر سبب شد تفاوتهايي بين نسخههاي مختلف اين زبان به وجود بيايد و از قابليت سازگاري و انتقال آن کاسته شود. به همين دليل در سال 1998 زبان C++ توسط موسسۀ استانداردهاي ملي آمريکا (ANSI) به شکل استاندارد و يکپارچه درآمد.
15
1- چرا C++ ؟ زبان C يک زبان همه منظوره است
در اين زبان عملگرهايي تعبيه شده که برنامهنويسي سطح پايين و به زبان ماشين را نيز امکانپذير ميسازد چون C عملگرهاي فراواني دارد، کد منبع برنامهها در اين زبان بسيار کوتاه است
16
- زبان C براي اجراي بسياري از دستوراتش از توابع کتابخانهاي استفاده ميکند و بيشتر خصوصيات وابسته به سختافزار را به اين توابع واگذار مينمايد. برنامۀ مقصدي که توسط کامپايلرهاي C ساخته ميشود بسيار فشردهتر و کمحجمتر از برنامههاي مشابه در ساير زبانها است. C++ که از نسل C است، تمام ويژگيهاي جذاب C را به ارث برده است . و سرانجام آخرين دليل استفاده از C++ ورود به دنياي C# است.
17
2- تاريخچۀ C++ در دهه 1970 در آزمايشگاههاي بل زباني به نام C ايجاد شد. انحصار اين زبان در اختيار شرکت بل بود تا اين که در سال 1978 توسط Kernighan و Richie شرح کاملي از اين زبان منتشر شد و به سرعت نظر برنامهنويسان حرفهاي را جلب نمود. هنگامي که بحث شيگرايي و مزاياي آن در جهان نرمافزار رونق يافت، زبان C که قابليت شيگرايي نداشت ناقص به نظر ميرسيد تا اين که در اوايل دهۀ 1980 دوباره شرکت بل دست به کار شد و Bjarne Stroustrup زبان C++ را طراحي نمود.
18
C++ ترکيبي از دو زبان C و Simula بود و قابليتهاي شيگرايي نيز داشت از آن زمان به بعد شرکتهاي زيادي کامپايلرهايي براي C++ طراحي کردند. اين امر سبب شد تفاوتهايي بين نسخههاي مختلف اين زبان به وجود بيايد و از قابليت سازگاري و انتقال آن کاسته شود. به همين دليل در سال 1998 زبان C++ توسط موسسۀ استانداردهاي ملي آمريکا (ANSI) به شکل استاندارد و يکپارچه درآمد. کامپايلرهاي کنوني به اين استاندارد پايبندند. کتاب حاضر نيز بر مبناي همين استاندارد نگارش يافته است.
19
3- آمادهسازي مقدمات يک «برنامه» دستورالعملهاي متوالي است که ميتواند توسط يک رايانه اجرا شود. براي نوشتن و اجراي هر برنامه به يک «ويرايشگر متن» و يک «کامپايلر» احتياج داريم. بستۀ Visual C++ محصول شرکت ميکروسافت و بستۀ C++ Builder محصول شرکت بورلند نمونههاي جالبي از محيط مجتمع توليد براي زبان C++ به شمار ميروند.
20
4- شروع کار با C++ #include <iostream> int main()
C++ نسبت به حروف «حساس به حالت» است يعني A و a را يکي نميداند مثال : اولين برنامه اولين برنامهاي که مينويسيم به محض تولد، به شما سلام ميکند و عبارت "Hello, my programmer!" را نمايش ميدهد: #include <iostream> int main() { std::cout << "Hello, my programmer!\n" ; return 0; }
21
اولين خط از کد بالا يک «راهنماي پيشپردازنده» است
اولين خط از کد بالا يک «راهنماي پيشپردازنده» است. راهنماي پيشپردازنده شامل اجزاي زير است: 1- کاراکتر # که نشان ميدهد اين خط، يک راهنماي پيشپردازنده است. اين کاراکتر بايد در ابتداي همۀ خطوط راهنماي پيشپردازنده باشد. 2- عبارت include 3- نام يک «فايل کتابخانهاي» که ميان دو علامت <> محصور شده است.
22
خط دوم برنامه نيز بايد در همه برنامههاي C++ وجود داشته باشد.
اين خط به کامپايلر ميگويد که «بدنۀ اصلي برنامه» از کجا شروع ميشود. اين خط داراي اجزاي زير است: 1 – عبارت int که يک نوع عددي در C++ است. 2 – عبارت main که به آن «تابع اصلي» در C++ ميگويند. 3 – دو پرانتز () که نشان ميدهد عبارت main يک «تابع» است. هر برنامه فقط بايد يک تابع main() داشته باشد .
23
سه خط آخر برنامه، «بدنۀ اصلي برنامه» را تشکيل ميدهند.
دستورات برنامه از خط سوم شروع شده است. دستور خط سوم با علامت سميکولن ; پايان يافته است.
24
توضيح توضيح، متني است که به منظور راهنمايي و درک بهتر به برنامه اضافه ميشود و تاثيري در اجراي برنامه ندارد. . کامپايلر توضيحات برنامه را قبل از اجرا حذف ميکند. استفاده از توضيح سبب ميشود که ساير افراد کد برنامۀ شما را راحتتر درک کنند.
25
به دو صورت ميتوانيم به برنامههاي C++ توضيحات اضافه کنيم:
1 – با استفاده از دو علامت اسلش // : هر متني که بعد از دو علامت اسلش بيايد تا پايان همان سطر يک توضيح تلقي ميشود . 2 – با استفاده از حالت C : هر متني که با علامت /* شروع شود و با علامت */ پايان يابد يک توضيح تلقي ميشود.
26
5- عملگر خروجي علامت << عملگر خروجي در C++ نام دارد (به آن عملگر درج نيز ميگويند). يک «عملگر» چيزي است که عملياتي را روي يک يا چند شي انجام ميدهد. عملگر خروجي، مقادير موجود در سمت راستش را به خروجي سمت چپش ميفرستد. به اين ترتيب دستور cout<< 66 ; مقدار 66 را به خروجي cout ميفرستد که cout معمولا به صفحهنمايش اشاره دارد. در نتيجه مقدار 66 روي صفحه نمايش درج ميشود.
27
6 -ليترالها و کاراکترها
يک «ليترال» رشتهاي از حروف، ارقام يا علايم چاپي است که ميان دو علامت نقل قول " " محصور شده باشد. يک «کاراکتر» يک حرف، رقم يا علامت قابل چاپ است که ميان دونشانۀ ' ' محصور شده باشد. پس 'w' و '!' و '1' هر کدام يک کاراکتر است. به تفاوت سه موجوديت «عدد» و «کاراکتر» و «ليترال رشتهاي» دقت کنيد: 6 يک عدد است، '6' يک کاراکتر است و "6" يک ليترال رشتهاي است.
28
7 - متغيرها و تعريف آنها:
«متغير» مکاني در حافظه است که چهار مشخصه دارد: نام، نوع، مقدار، آدرس. وقتي متغيري را تعريف ميکنيم، ابتدا با توجه به نوع متغير، آدرسي از حافظه در نظر گرفته ميشود، سپس به آن آدرس يک نام تعلق ميگيرد.
29
نحو اعلان يک متغير type name initializer
در C++ قبل از اين که بتوانيم از متغيري استفاده کنيم، بايد آن را اعلان نماييم. نحو اعلان يک متغير type name initializer عبارت type نوع متغير را مشخص ميکند. نوع متغير به کامپايلر اطلاع ميدهد که اين متغير چه مقاديري ميتواند داشته باشد و چه اعمالي ميتوان روي آن انجام داد.
30
name \ initializer مقداردهي اوليه
دستور زير تعريف يک متغير صحيح را نشان ميدهد: int n = 50;
31
8 - مقداردهي اوليه به متغيرها
در بسياري از موارد بهتر است متغيرها را در همان محلي که اعلان ميشوند مقداردهي کنيم. استفاده از متغيرهاي مقداردهي نشده ممکن است باعث ايجاد دردسرهايي شود. دردسر متغيرهاي مقداردهي نشده وقتي بزرگتر ميشود که سعي کنيم متغير مقداردهي نشده را در يک محاسبه به کار ببريم. مثلا اگر x را که مقداردهي نشده در عبارت y = x + 5; به کار ببريم، حاصل y غير قابل پيشبيني خواهد بود. براي اجتناب از چنين مشکلاتي عاقلانه است که متغيرها را هميشه هنگام تعريف، مقداردهي کنيم. مثال: int x=45; int y=0;
32
9- ثابتها در بعضي از برنامهها از متغيري استفاده ميکنيم که فقط يک بار لازم است آن را مقداردهي کنيم و سپس مقدار آن متغير در سراسر برنامه بدون تغيير باقي ميماند. مثلا در يک برنامۀ محاسبات رياضي، متغيري به نام PI تعريف ميکنيم و آن را با 3.14 مقداردهي ميکنيم و ميخواهيم که مقدار اين متغير در سراسر برنامه ثابت بماند. در چنين حالاتي از «ثابتها» استفاده ميکنيم. يک ثابت، يک نوع متغير است که فقط يک بار مقداردهي ميشود و سپس تغيير دادن مقدار آن در ادامۀ برنامه ممکن نيست. تعريف ثابتها مانند تعريف متغيرهاست با اين تفاوت که کلمه کليدي const به ابتداي تعريف اضافه ميشود.
33
{ // defines constants; has no output: const char BEEP ='\b';
مثال تعريف ثابتها: int main() { // defines constants; has no output: const char BEEP ='\b'; const int MAXINT= ; const float DEGREE=23.53; const double PI= return 0; } برنامه فوق خروجي ندارد:
34
10 - عملگر ورودي براي اين که بتوانيم هنگام اجراي برنامه مقاديري را وارد کنيم از عملگر ورودي >> استفاده ميکنيم. استفاده از دستور ورودي به شکل زير است: cin >> variable; variable نام يک متغير است.
35
{ // reads an integer from input: int m;
مثال 10 – 1 استفاده از عملگر ورودي برنامۀ زير يک عدد از کاربر گرفته و همان عدد را دوباره در خروجي نمايش ميدهد: int main() { // reads an integer from input: int m; cout << "Enter a number: "; cin >> m; cout << "your number is: " << m << endl; return 0; } Enter a number: 52 your number is: 52
36
cin >> x >> y >> z;
عملگر ورودي نيز مانند عملگر خروجي به شکل جرياني رفتار ميکند. يعني همان طور که در عملگر خروجي ميتوانستيم چند عبارت را با استفاده از چند عملگر << به صورت پشت سر هم چاپ کنيم، در عملگر ورودي نيز ميتوانيم با استفاده از چند عملگر >> چند مقدار را به صورت پشت سر هم دريافت کنيم. مثلا با استفاده از دستور: cin >> x >> y >> z; سه مقدار x و y و z به ترتيب از ورودي دريافت ميشوند. براي اين کار بايد بين هر ورودي يک فضاي خالي (space) بگذاريد و پس از تايپ کردن همۀ وروديها، کليد enter را بفشاريد. آخرين مثال جلسه، اين موضوع را بهتر نشان ميدهد.
37
مثال 11 – 1 چند ورودي روي يک خط
برنامۀ زير مانند مثال 10 – 2 است با اين تفاوت که سه عدد را از ورودي گرفته و همان اعداد را دوباره در خروجي نمايش ميدهد: int main() { // reads 3 integers from input: int q, r, s; cout << "Enter three numbers: "; cin >> q >> r >> s; cout << "your numbers are: << q << ", " << r << ", " << s << endl; return 0; } Enter three numbers: your numbers are: 35, 70, 9
38
پايان جلسه اول
39
جلسه دوم «انواع اصلي»
40
آنچه در اين جلسه مي خوانيد:
1- انواع دادۀ عددي 2- متغير عدد صحيح 3- محاسبات اعداد صحيح 4- عملگرهاي افزايشي و کاهشي 5- عملگرهاي مقدارگذاري مرکب 6- انواع مميز شناور ›››
41
››› 7- تعريف متغير مميز شناور 8 - شکل علمي مقادير مميز شناور
9- نوع بولين bool 10- نوع کاراکتري char 11- نوع شمارشي enum 12- تبديل نوع، گسترش نوع ›››
42
13- برخي از خطاهاي برنامهنويسي
14 - سرريزي عددي 15- خطاي گرد کردن 16- حوزۀ متغيرها
43
هدف کلي: معرفي انواع متغييرها و نحوۀ بهکارگيري آنها در برنامههاي C++ هدفهاي رفتاري: انتظار ميرود پس از پايان اين جلسه بتوانيد: - انواع عددي صحيح در C++ را نام ببريد و متغيرهايي از اين نوعها را در برنامهها به کار ببريد. - انواع عددي مميز شناور در C++ را نام ببريد و متغيرهايي از اين نوعها را در برنامهها به کار ببريد. - نوع بولين را تعريف کرده و متغيرهايي از اين نوع را در برنامهها به کار ببريد. >>>
44
- نوع شمارشي را شناخته و متغيرهايي از اين نوع را در برنامهها به کار ببريد.
- مفاهيم «تبديل نوع» و «گسترش نوع» را شناخته و انواع مختلف را به يکديگر تبديل نماييد. - علت خطاهاي «سرريزي عددي» و «گردکردن» را دانسته و بتوانيد محل وقوع آنها را کشف کنيد. - عملگرهاي حسابي و افزايشي و کاهشي و مقدارگذاري مرکب را در برنامهها به کار ببريد.
45
مقدمه ما در زندگي روزمره از دادههاي مختلفي استفاده ميکنيم: اعداد ، تصاوير، نوشتهها يا حروف الفبا، صداها، بوها و با پردازش اين دادهها ميتوانيم تصميماتي اتخاذ کنيم، عکسالعملهايي نشان دهيم و مسالهاي را حل کنيم. رايانهها نيز قرار است همين کار را انجام دهند. يعني دادههايي را بگيرند، آنها را به شکلي که ما تعيين ميکنيم پردازش کنند و در نتيجه اطلاعات مورد نيازمان را استخراج کنند.
46
1- انواع دادۀ عددي در C++ دو نوع اصلي داده وجود دارد: «نوع صحيح» و «نوع مميز شناور». همۀ انواع ديگر از روي اين دو ساخته ميشوند (به شکل زير دقت کنيد).
47
نوع صحيح نوع صحيح براي نگهداري اعداد صحيح (اعداد 0 و 1 و 2 و ...) استفاده ميشود. اين اعداد بيشتر براي شمارش به کار ميروند و دامنه محدودي دارند.
49
نوع مميز شناور براي نگهداري اعداد اعشاري استفاده ميشود
نوع مميز شناور براي نگهداري اعداد اعشاري استفاده ميشود. اعداد اعشاري بيشتر براي اندازهگيري دقيق به کار ميروند و دامنۀ بزرگتري دارند. يک عدد اعشاري مثل 352/187 را ميتوان به شکل 10×7352/18 يا 102×87352/1 يا1-10×52/1873يا2-10×2/18735 و يا ... نوشت. به اين ترتيب با کم و زياد کردن توان عدد 10 مميز عدد نيز جابهجا ميشود. به همين دليل است که به اعداد اعشاري «اعداد مميز شناور» ميگويند.
50
2- متغير عدد صحيح C++ شش نوع متغير عدد صحيح دارد تفاوت اين شش نوع مربوط به ميزان حافظۀ مورد استفاده و محدودۀ مقاديري است که هر کدام ميتوانند داشته باشند. اين ميزان حافظۀ مورد استفاده و محدودۀ مقادير، بستگي زيادي به سختافزار و همچنين سيستم عامل دارد. يعني ممکن است روي يک رايانه، نوع int دو بايت از حافظه را اشغال کند در حالي که روي رايانهاي از نوع ديگر نوع int به چهار بايت حافظه نياز داشته باشد.
51
حداكثر مقدار قابل پذيرش
نوع متغيير حداقل مقدار قابل پذيرش حداكثر مقدار قابل پذيرش short -32768 32767 unsigned short 65535 int unsigned int long unsigned long وقتي برنامهاي مينويسيد، توجه داشته باشيد که از نوع صحيح مناسب استفاده کنيد تا هم برنامه دچار خطا نشود و هم حافظۀ سيستم را هدر ندهيد.
52
3 -محاسبات اعداد صحيح C++ مانند اغلب زبانهاي برنامهنويسي براي محاسبات از عملگرهاي جمع (+) ، تفريق (-) ، ضرب (*) ، تقسيم (/) و باقيمانده (%) استفاده ميکند.
53
4 - عملگرهاي افزايشي و کاهشي
C++ براي دستکاري مقدار متغيرهاي صحيح، دو عملگر جالب ديگر دارد: عملگر ++ : مقدار يک متغير را يک واحد افزايش ميدهد. عملگر -- : مقدار يک متغير را يک واحد کاهش ميدهد. اما هر کدام از اين عملگرها دو شکل متفاوت دارند: شکل «پيشوندي» و شکل «پسوندي».
54
در شکل پيشوندي ابتدا متغير، متناسب با عملگر، افزايش يا کاهش مييابد و پس از آن مقدار متغير براي محاسبات ديگر استفاده ميشود. در شکل پسوندي ابتدا مقدار متغير در محاسبات به کار ميرود و پس از آن مقدار متغير يک واحد افزايش يا کاهش مييابد. در شکل پيشوندي، عملگر قبل از نام متغير ميآيد مثل ++m يا --n . در شکل پسوندي، عملگر بعد از نام متغير ميآيد مثل m++ يا n-- .
55
5 – عملگرهاي مقدارگذاري مرکب
5 – عملگرهاي مقدارگذاري مرکب C++ عملگرهاي ديگري دارد که مقدارگذاري در متغيرها را تسهيل مينمايند. مثلا با استفاده از عملگر += ميتوانيم هشت واحد به m اضافه کنيم اما با دستور کوتاهتر: m += 8; دستور بالا معادل دستور m = m + 8; است با اين تفاوت که کوتاهتر است. به عملگر += «عملگر مرکب» ميگويند زيرا ترکيبي از عملگرهاي + و = ميباشد
56
5- عملگرهاي مقدارگذاري مرکب
قبلا از عملگر = براي مقدارگذاري در متغيرها استفاده کرديم. C++ عملگرهاي ديگري دارد که مقدارگذاري در متغيرها را تسهيل مينمايند. عملگر مرکب در C++ عبارتند از: += و -= و *= و /= و =%
57
نحوۀ عمل اين عملگرها به شکل زير است:
m += 8; → m = m + 8; m -= 8; → m = m - 8; m *= 8; → m = m * 8; m /= 8; →m = m / 8; m %= 8; →m = m % 8;
58
6 – انواع مميز شناور عدد مميز شناور به بيان ساده همان عدد اعشاري است. عددي مثل يک عدد اعشاري است. براي اين که مقدار اين عدد در رايانه ذخيره شود، ابتدا بايد به شکل دودويي تبديل شود: = اکنون براي مشخص نمودن محل اعشار در عدد، تمام رقمها را به سمت راست مميز منتقل ميکنيم. البته با هر جابجايي مميز، عدد حاصل بايد در تواني از 2 ضرب شود: = × 27 به مقدار «مانتيس عدد» و به 7 که توان روي دو است، «نماي عدد» گفته ميشود.
59
درC++ سه نوع مميز شناور وجود دارد:
نوع double از هشت بايت براي نگهداري عدد استفاده ميکند. نوع long double از هشت يا ده يا دوازده يا شانزده بايت براي نگهداري عدد استفاده ميکند. معمولا نوع float از چهار بايت براي نگهداري عدد استفاده ميکند.
60
23 8 1 52 11 مانتيس نما علامت عدد float 32 بيتي double 64 بيتي
جدول تخصيص حافظه براي متغيير هاي مميز شناور نوع متغير تعداد بيت براي ذخيرهسازيِ مانتيس نما علامت عدد float 32 بيتي 23 8 1 double 64 بيتي 52 11
61
7 – تعريف متغير مميز شناور
7 – تعريف متغير مميز شناور تعريف متغير مميز شناور مانند تعريف متغير صحيح است. با اين تفاوت که از کلمۀ کليدي float يا double براي مشخص نمودن نوع متغير استفاده ميکنيم. مثال: float x; double x,y=0; تفاوت نوع float با نوع double در اين است که نوع double دو برابر float از حافظه استفاده ميکند. پس نوع double دقتي بسيار بيشتر از float دارد. به همين دليل محاسبات double وقتگيرتر از محاسبات float است.
62
8- شکل علمي مقادير مميز شناور
اعداد مميز شناور به دو صورت در ورودي و خروجي نشان داده ميشوند: به شکل «ساده» و به شکل «علمي». 2- علمي ×104 1- ساده مشخص است که شکل علمي براي نشان دادن اعداد خيلي کوچک و همچنين اعداد خيلي بزرگ، کارآيي بيشتري دارد.
63
9 – نوع بولين bool نوع bool يک نوع صحيح است که متغيرهاي اين نوع فقط ميتوانند مقدار true يا false داشته باشند. true به معني درست و false به معني نادرست است. اما اين مقادير در اصل به صورت 1 و 0 درون رايانه ذخيره ميشوند: 1 براي true و 0 براي false.
64
10- نوع کاراکتري char يک کاراکتر يک حرف، رقم يا نشانه است که يک شمارۀ منحصر به فرد دارد. به عبارت عاميانه، هر کليدي که روي صفحهکليد خود ميبينيد يک کاراکتر را نشان ميدهد. مثلا هر يک از حروف 'A' تا 'Z' و 'a' تا 'z' و هر يک از اعداد '0' تا '9' و يا نشانههاي '~' تا '+' روي صفحهکليد را يک کاراکتر مينامند.
65
براي تعريف متغيري از نوع کاراکتر از کلمه کليدي char استفاده ميکنيم
براي تعريف متغيري از نوع کاراکتر از کلمه کليدي char استفاده ميکنيم. يک کاراکتر بايد درون دو علامت آپستروف (') محصور شده باشد. پس 'A' يک کاراکتر است؛ همچنين'8' يک کاراکتر است اما 8 يک کاراکتر نيست بلکه يک عدد صحيح است . مثال: char c ='A';
66
11 – نوع شمارشي enum enum typename{enumerator-list}
يک نوع شمارشي يک نوع صحيح است که توسط کاربر مشخص ميشود. نحو تعريف يک نوع شمارشي به شکل زير است: enum typename{enumerator-list} که enum کلمهاي کليدي است، typename نام نوع جديد است که کاربر مشخص ميکند و enumerator-list مجموعه مقاديري است که اين نوع جديد ميتواند داشته باشد.
67
به عنوان مثال به تعريف زير دقت کنيد:
enum Day{SAT,SUN,MON,TUE,WED,THU,FRI} حالا Day يک نوع جديد است و متغيرهايي که از اين نوع تعريف ميشوند ميتوانند يکي از مقادير SAT و SUN و MON و TUE و WED و THU و FRI را داشته باشند: Day day1,day2; day1 = MON; day2 = THU; وقتي نوع جديد Day و محدودۀ مقاديرش را تعيين کرديم، ميتوانيم متغيرهايي از اين نوع جديد بسازيم. در کد بالا متغيرهاي day1 و day2 از نوع Day تعريف شدهاند. آنگاه day1 با مقدار MON و day2 با مقدار THU مقداردهي شده است.
68
enum Answer{NO=0,FALSE=0,YES=1,TRUE=1,OK=1}
مقادير SAT و SUN و هر چند که به همين شکل به کار ميروند اما در رايانه به شکل اعداد صحيح 0 و 1 و 2 و ... ذخيره ميشوند. به همين دليل است که به هر يک از مقادير SAT و SUN و ... يک شمارشگر ميگويند. ميتوان مقادير صحيح دلخواهي را به شمارشگرها نسبت داد: enum Day{SAT=1,SUN=2,MON=4,TUE=8,WED=16,THU=32,FRI=64} اگر فقط بعضي از شمارشگرها مقداردهي شوند، آنگاه ساير شمارشگرها که مقداردهي نشدهاند مقادير متوالي بعدي را خواهند گرفت: enum Day{SAT=1,SUN,MON,TUE,WED,THU,FRI} دستور بالا مقادير 1 تا 7 را به ترتيب به روزهاي هفته تخصيص خواهد داد. همچنين دو يا چند شمارشگر در يک فهرست ميتوانند مقادير يکساني داشته باشند: enum Answer{NO=0,FALSE=0,YES=1,TRUE=1,OK=1}
69
نام شمارشگر بايد معتبر باشد: يعني: 1- کلمۀ کليدي نباشد.
نحوۀ انتخاب نامشمارشگرها آزاد است اما بيشتر برنامهنويسان از توافق زير در برنامههايشان استفاده ميکنند: 1 – براي نام ثابتها از حروف بزرگ استفاده کنيد 2 – اولين حرف از نام نوع شمارشي را با حرف بزرگ بنويسيد. 3 – در هر جاي ديگر از حروف کوچک استفاده کنيد. نام شمارشگر بايد معتبر باشد: يعني: 1- کلمۀ کليدي نباشد. 2- با عدد شروع نشود. 3- نشانههاي رياضي نيز نداشته باشد.
70
enum Score{A,B,C,D} float B; char c;
آخر اين که نام شمارشگرها نبايد به عنوان نام متغيرهاي ديگر در جاهاي ديگر برنامه استفاده شود. مثلا: enum Score{A,B,C,D} float B; char c; در تعريفهاي بالا B و C را نبايد به عنوان نام متغيرهاي ديگر به کار برد زيرا اين نامها در نوع شمارشي Score به کار رفته است . شمارشگرهاي همنام نبايد در محدودههاي مشترک استفاده شوند. براي مثال تعريفهاي زير را در نظر بگيريد: enum Score{A,B,C,D} enum Group{AB,B,BC} دو تعريف بالا غيرمجاز است زيرا شمارشگر B در هر دو تعريف Score و Group آمده است.
71
انواع شمارشي براي توليد کد «خود مستند» به کار ميروند، يعني کدي که به راحتي درک شود و نياز به توضيحات اضافي نداشته باشد. مثلا تعاريف زير خودمستند هستند زيرا به راحتي نام و نوع کاربرد و محدودۀ مقاديرشان درک ميشود: enum Color{RED,GREEN,BLUE,BLACK,ORANGE} enum Time{SECOND,MINUTE,HOUR} enum Date{DAY,MONTH,YEAR} enum Language{C,DELPHI,JAVA,PERL} enum Gender{MALE,FEMALE}
72
12 – تبديل نوع، گسترش نوع در محاسباتي که چند نوع متغير وجود دارد، جواب هميشه به شکل متغيري است که دقت بالاتري دارد. يعني اگر يک عدد صحيح را با يک عدد مميز شناور جمع ببنديم، پاسخ به شکل مميز شناور است به اين عمل گسترش نوع ميگويند. براي اين که مقدار يک متغير از نوع مميز شناور را به نوع صحيح تبديل کنيم از عبارت int() استفاده ميکنيم به اين عمل تبديل نوع گفته مي شود
73
مثالهاي زير تبديل نوع و گسترش نوع را نشان ميدهند.
مثال گسترش نوع برنامۀ زير يک عدد صحيح را با يک عدد مميز شناور جمع ميکند: int main() { // adds an int value with a double value: int n = 22; double p = ; p += n; cout << "p = " << p << ", n = " << n << endl; return 0; } مثال تبديل نوع: اين برنامه، يک نوع double را به نوع int تبديل ميکند: int main() { // casts a double value as an int: double v= ; int n; n = int(v); cout << "v = " << v << ", n = " << n << endl; return 0; }
74
13 – برخي از خطاهاي برنامهنويسي
13 – برخي از خطاهاي برنامهنويسي «خطاي زمان کامپايل» اين قبيل خطاها که اغلب خطاهاي نحوي هستند ، توسط کامپايلر کشف ميشوند و به راحتي ميتوان آنها را رفع نمود. «خطاي زمان اجرا» کشف اينگونه خطاها به راحتي ممکن نيست و کامپايلر نيز چيزي راجع به آن نميداند. برخي از خطاهاي زمان اجرا سبب ميشوند که برنامه به طور کامل متوقف شود و از کار بيفتد.
75
14- سرريزي عددي يک متغير هر قدر هم که گنجايش داشته باشد، بالاخره مقداري هست که از گنجايش آن متغير بيشتر باشد. اگر سعي کنيم در يک متغير مقداري قرار دهيم که از گنجايش آن متغير فراتر باشد، متغير «سرريز» ميشود،در چنين حالتي ميگوييم که خطاي سرريزي رخ داده است.
76
مثال 12 – 2 سرريزي عدد صحيح اين برنامه به طور مكرر n را در 1000 ضرب ميكند تا سرانجام سرريز شود: int main() { //prints n until it overflows: int n =1000; cout << "n = " << n << endl; n *= 1000; // multiplies n by 1000 cout << " n = " << n << endl; return 0; } وقتي يک عدد صحيح سرريز شود، عدد سرريز شده به يک مقدار منفي «گردانيده» ميشود اما وقتي يک عدد مميز شناور سرريز شود، نماد inf به معناي بينهايت را به دست ميدهد.
77
15 – خطاي گرد کردن خطاي گرد كردن نوع ديگري از خطاست كه اغلب وقتي رايانهها روي اعداد حقيقي محاسبه ميكنند، رخ ميدهد. براي مثال عدد 1/3ممكن است به صورت ذخيره شود كه دقيقا معادل 1/3 نيست. اين خطا از آنجا ناشي ميشود که اعدادي مثل 1/3 مقدار دقيق ندارند و رايانه نميتواند اين مقدار را پيدا کند، پس نزديکترين عدد قابل محاسبه را به جاي چنين اعدادي منظور ميکند. «هيچگاه از متغير مميز شناور براي مقايسه برابري استفاده نکنيد» زيرا در متغيرهاي مميز شناور خطاي گرد کردن سبب ميشود که پاسخ با آن چه مورد نظر شماست متفاوت باشد.
78
16 – حوزۀ متغيرها اصطلاح «بلوک» در C++ واژه مناسبي است که ميتوان به وسيلۀ آن حوزۀ متغير را مشخص نمود. يک بلوک برنامه، قسمتي از برنامه است که درون يک جفت علامت کروشه { } محدود شده است. انتخاب نامهاي نامفهوم يا ناقص سبب کاهش خوانايي برنامه و افزايش خطاهاي برنامهنويسي ميشود. استفاده از متغيرها در حوزۀ نامناسب هم سبب بروز خطاهايي ميشود. «حوزه متغير» محدودهاي است که يک متغير خاص اجازه دارد در آن محدوده به کار رود يا فراخواني شود.
79
حوزۀ يک متغير از محل اعلان آن شروع ميشود و تا پايان همان بلوک ادامه مييابد. خارج از آن بلوک نميتوان به متغير دسترسي داشت. همچنين قبل از اين که متغير اعلان شود نميتوان آن را استفاده نمود. ميتوانيم در يک برنامه، چند متغير متفاوت با يک نام داشته باشيم به شرطي که در حوزههاي مشترک نباشند.
80
پايان جلسه دوم
81
جلسه سوم «انتخاب»
82
آنچه در اين جلسه مي خوانيد:
1- دستور if 2- دستور if..else 3- عملگرهاي مقايسهاي 4- بلوكهاي دستورالعمل 5- شرطهاي مركب 6- ارزيابي ميانبري ›››
83
7- عبارات منطقي 8 - دستورهاي انتخاب تودرتو 9- ساختار else if 10- دستورالعمل switch 11- عملگر عبارت شرطي 12- كلمات كليدي
84
شناخت انواع دستورالعملهاي انتخاب و شيوۀ بهکارگيري هر يک
هدفهاي رفتاري: انتظار ميرود پس از پايان اين جلسه بتوانيد: - نحو دستور if را شناخته و آن را در برنامهها به کار ببريد. - نحو دستور if..else را شناخته و آن را در برنامهها به کار ببريد. - از ساختار else..if در تصميمگيريهاي پيچيده استفاده کنيد. - نحو دستور switch را شناخته و خطاي «تلۀ سقوط» را تشخيص دهيد. - بلوک دستورالعمل را تعريف کنيد. - عملگرهاي مقايسهاي و عملگر عبارت شرطي را در دستورات شرطي به کار ببريد. - از شرطهاي مرکب استفاده کرده و ارزيابي ميانبري را شرح دهيد. - «کلمۀ کليدي» را تعريف کنيد. هدف کلي: شناخت انواع دستورالعملهاي انتخاب و شيوۀ بهکارگيري هر يک >>>
85
مقدمه همۀ برنامههايي که در دو جلسه اول بيان شد، به شکل ترتيبي اجرا ميشوند، يعني دستورات برنامه به ترتيب از بالا به پايين و هر کدام دقيقا يک بار اجرا ميشوند. در اين جلسه نشان داده ميشود چگونه از دستورالعملهاي انتخاب1 جهت انعطافپذيري بيشتر برنامه استفاده کنيم. همچنين در اين جلسه انواع صحيح كه در C++ وجود دارد بيشتر بررسي ميگردد.
86
دستور if If (condition) statement;
Condition که شرط ناميده ميشود يك عبارت صحيح است (عبارتي که با يک مقدار صحيح برآورد ميشود) و statement ميتواند هر فرمان قابل اجرا باشد. Statement وقتي اجرا خواهد شد كه condition مقدار غير صفر داشته باشد. دقت كنيد كه شرط بايد درون پرانتز قرار داده شود.
87
2- دستور if..else دستور if..else موجب ميشود بسته به اين که شرط درست باشد يا خير، يكي از دو دستورالعمل فرعي اجرا گردد. نحو اين دستور به شکل زير است: if (condition) statement1; else statement2; condition همان شرط مساله است که يك عبارت صحيح ميباشد و statement1 و statement2 فرمانهاي قابل اجرا هستند. اگر مقدار شرط، غير صفر باشد، statement1 اجرا خواهد شد وگرنه statement2 اجرا ميشود.
88
مثال يک آزمون قابليت تقسيم
مثال يک آزمون قابليت تقسيم int main() { int n, d; cout << " Enter two positive integers: "; cin >> n >> d; if (n%d) cout << n << " is not divisible by " << d << endl; else cout << n << " is divisible by " << d << endl; }
89
4- عملگرهاي مقايسهاي در C++ شش عملگر مقايسهاي وجود دارد: < و > و <= و >= و == و != . هر يک از اين شش عملگر به شکل زير به کار ميروند: x < y // است y کوچکتر از x x > y // است y بزرگتر از x x <= y // است y کوچکتر يا مساوي x x >= y // است y بزرگتر يا مساوي x x == y // است y مساوي با x x != y // نيست y مساوي با x
90
اينها ميتوانند براي مقايسۀ مقدار عبارات با هر نوع ترتيبي استفاده شوند. عبارت حاصل به عنوان يك شرط تفسير ميشود. مقدار اين شرط صفر است اگر شرط نادرست باشد و غير صفر است اگر شرط درست باشد. براي نمونه، عبارت 7*8<6*5 برابر با صفر ارزيابي ميشود، به اين معني كه اين شرط نادرست است.
91
2- متغير عدد صحيح C++ شش نوع متغير عدد صحيح دارد تفاوت اين شش نوع مربوط به ميزان حافظۀ مورد استفاده و محدودۀ مقاديري است که هر کدام ميتوانند داشته باشند. اين ميزان حافظۀ مورد استفاده و محدودۀ مقادير، بستگي زيادي به سختافزار و همچنين سيستم عامل دارد. يعني ممکن است روي يک رايانه، نوع int دو بايت از حافظه را اشغال کند در حالي که روي رايانهاي از نوع ديگر نوع int به چهار بايت حافظه نياز داشته باشد.
92
دقت کنيد كه در ++C عملگر جايگزيني با عملگر برابري فرق دارد
مثلا دستور x = 33; مقدار 33 را در x قرار ميدهد ولي دستور x == 33; بررسي ميکند که آيا مقدار x با 33 برابر است يا خير. درک اين تفاوت اهميت زيادي دارد.
93
4- بلوكهاي دستورالعمل يك بلوك دستورالعمل زنجيرهاي از دستورالعملهاست كه درون براكت {} محصور شده، مانند : { int temp=x; x = y; y = temp; } در برنامههاي ++C يک بلوک دستورالعمل مانند يک دستورالعمل تکي است.
94
int main() { int x, y; cout << "Enter two integers: "; cin >> x >> y; if (x > y) { int temp = x; x = y; y = temp; } //swap x and y cout << x << " <= " << y << endl; } مثال : يك بلوك دستورالعمل درون يك دستور if اين برنامه دو عدد صحيح را گرفته و به ترتيب بزرگتري، آنها را چاپ ميكند:
95
{ int n=44; int main() cout << "n = " << n << endl;
cout << "Enter an integer: "; cin >> n; cout << "n = " << n << endl; } { cout << " n = " << n << endl; }
96
5 – شرطهاي مركب شرطهايي مانند n%d و x>=y ميتوانند به صورت يك شرط مركب با هم تركيب شوند. اين كار با استفاده ازعملگرهاي منطقي && (and) و || (or) و ! (not) صورت ميپذيرد. اين عملگرها به شکل زير تعريف ميشوند: p && q درست است اگر و تنها اگر هم p و هم q هر دو درست باشند p || q نادرست است اگر و تنها اگر هم p و هم q هر دو نادرست باشند !pدرست است اگر و تنها اگر p نادرست باشد براي مثال(n%d || x>=y) نادرست است اگر و تنها اگر n%d برابر صفر و x كوچكتر از y باشد.
97
P&&q q p T F P||q q p T F P F T !P
سه عملگر منطقي && (and) و || (or) و ! (not) معمولا با استفاده از جداول درستي به گونۀ زير بيان ميشوند: P&&q q p T F P||q q p T F !P P F T طبق جدولهاي فوق اگر p درست و q نادرست باشد، عبارت p&&q نادرست و عبارت p||q درست است.
98
6- ارزيابي ميانبري عملگرهاي && و || به دو عملوند نياز دارندتا مقايسه را روي آن دو انجام دهند. جداول درستي نشان ميدهد که p&&q نادرست است اگر p نادرست باشد. در اين حالت ديگر نيازي نيست که q بررسي شود. همچنين p||q درست است اگر p درست باشد و در اين حالت هم نيازي نيست که q بررسي شود. در هر دو حالت گفته شده، با ارزيابي عملوند اول به سرعت نتيجه معلوم ميشود. اين كار ارزيابي ميانبري ناميده ميشود. شرطهاي مركب كه از && و || استفاده ميكنند عملوند دوم را بررسي نميكنند مگر اين كه لازم باشد.
99
7- عبارات منطقي يك عبارت منطقي شرطي است كه يا درست است يا نادرست. قبلا ديديم که عبارات منطقي با مقادير صحيح ارزيابي ميشوند. مقدار صفر به معناي نادرست و هر مقدار غير صفر به معناي درست است. به عبارات منطقي «عبارات بولي» هم ميگويند.
100
چون همۀ مقادير صحيح ناصفر به معناي درست تفسير ميشوند، عبارات منطقي اغلب تغيير قيافه ميدهند. براي مثال دستور if (n) cout << "n is not zero"; وقتي n غير صفر است عبارت n is not zero را چاپ ميكند زيرا عبارت منطقي (n) وقتي مقدار n غير صفر است به عنوان درست تفسير ميگردد.
101
کد زير را نگاه کنيد: if (n%d) cout << "n is not a multiple of d"; دستور خروجي فقط وقتي كه n%d ناصفر است اجرا ميگردد و n%d وقتي ناصفر است که n بر d بخشپذير نباشد. گاهي ممکن است فراموش کنيم که عبارات منطقي مقادير صحيح دارند و اين فراموشي باعث ايجاد نتايج غير منتظره و نامتعارف شود.
102
cout << "Enter three integers: ";
يك خطاي منطقي ديگر، اين برنامه خطادار است: int main() { int n1, n2, n3; cout << "Enter three integers: "; cin >> n1 >> n2 >> n3; if (n1 >= n2 >= n3) cout << "max = " << n1; } منشأ خطا در برنامۀ بالا اين اصل است كه عبارات منطقي مقدارهاي عددي دارند.
103
8- دستورهاي انتخاب تودرتو
دستورهاي انتخاب ميتوانند مانند دستورالعملهاي مركب به كار روند. به اين صورت که يك دستور انتخاب ميتواند درون دستور انتخاب ديگر استفاده شود. به اين روش، جملات تودرتو ميگويند.
104
مثال 12-3 دستورهاي انتخاب تودرتو
اين برنامه همان اثر مثال 10-3 را دارد: int main() { int n, d; cout << "Enter two positive integers: "; cin >> n >> d; if (d != 0) if (n%d = = 0) cout << d << " divides " << n << endl; else cout << d << " does not divide " << n << endl; } در برنامۀ بالا، دستور if..else دوم درون دستور if..else اول قرار گرفته است. وقتي دستور if..else به شکل تو در تو به کار ميرود، كامپايلر از قانون زير جهت تجزيه اين دستورالعمل مركب استفاده ميكند: « هر else با آخرين if تنها جفت ميشود.»
105
9- ساختار else if دستور if..else تودرتو، اغلب براي بررسي مجموعهاي از حالتهاي متناوب يا موازي به كار ميرود. در اين حالات فقط عبارت else شامل دستور if بعدي خواهد بود. اين قبيل کدها را معمولا با ساختار else ifميسازند.
106
استفاده از ساختار else if براي مشخص کردن محدودۀ نمره
برنامۀ زير يك نمرۀ امتحان را به درجۀ حرفي معادل تبديل ميكند: int main() { int score; cout << "Enter your test score: "; cin >> score; if (score > 100) cout << "Error: that score is out of range."; else if (score >= 90) cout << "Your grade is an A." << endl; else if (score >= 80) cout << "Your grade is a B." << endl; else if (score >= 70) cout << "Your grade is a C." << endl; else if (score >= 60) cout << "Your grade is a D." << endl; else if (score >= 0) cout << "Your grade is an F." << endl; else cout << "Error: that score is out of range."; }
107
10- دستورالعمل switch دستور switch ميتواند به جاي ساختار else if براي بررسي مجموعهاي از حالتهاي متناوب و موازي به كار رود. نحو دستور switch به شکل زير است: switch (expression) { case constant1: statementlist1; case constant2: statementlist2; case constant3: statementlist3; : case constantN: statementlistN; default: statementlist0; }
108
اين دستور ابتدا expression را برآورد ميكند و سپس ميان ثابتهاي case به دنبال مقدار آن ميگردد. اگر مقدار مربوطه از ميان ثابتهاي فهرستشده يافت شد، دستور statementlist مقابل آن case اجرا ميشود. اگر مقدار مورد نظر ميان caseها يافت نشد و عبارت default وجود داشت، دستور statementlist مقابل آن اجرا ميشود. عبارتdefault يک عبارت اختياري است. يعني ميتوانيم در دستور switch آن را قيد نکنيم. expression بايد به شکل يك نوع صحيح ارزيابي شود و constantها بايد ثابتهاي صحيح باشند.
109
لازم است در انتهاي هر case دستور break قرار بگيرد
لازم است در انتهاي هر case دستور break قرار بگيرد. بدون اين دستور، اجراي برنامه پس از اين كه case مربوطه را اجرا کرد از دستور switch خارج نميشود، بلکه همۀ caseهاي زيرين را هم خط به خط ميپيمايد و دستورات مقابل آنها را اجرا ميکند. به اين اتفاق، تلۀ سقوط ميگويند. case constant1: statementlist1;break;
110
11- عملگر عبارت شرطي عملگر عبارت شرطي يکي از امکاناتي است که جهت اختصار در کدنويسي تدارک ديده شده است. اين عملگر را ميتوانيم به جاي دستور if..else به کار ببريم. اين عملگر از نشانههاي ? و : به شکل زير استفاده ميكند: condition ? expression1 : expression2; در اين عملگر ابتدا شرط condition بررسي ميشود. اگر اين شرط درست بود، حاصل کل عبارت برابر با expression1 ميشود و اگر شرط نادرست بود، حاصل کل عبارت برابر با expression2 ميشود.
111
مثلا در دستور انتساب زير:
min = ( x<y ? x : y ); اگر x<y باشد مقدار x را درون min قرار ميدهد و اگر x<y نباشد مقدار y را درون min قرار ميدهد. يعني به همين سادگي و اختصار، مقدار کمينۀ x و y درون متغير min قرار ميگيرد.
112
12- كلمات كليدي اکنون با کلماتي مثل if و case و float آشنا شديم. دانستيم که اين کلمات براي C++ معاني خاصي دارند. از اين کلمات نميتوان به عنوان نام يک متغير يا هر منظور ديگري استفاده کرد و فقط بايد براي انجام همان کار خاص استفاده شوند. مثلا کلمۀ float فقط بايد براي معرفي يک نوع اعشاري به کار رود. يك كلمۀ كليدي در يك زبان برنامهنويسي كلمهاي است كه از قبل تعريف شده و براي هدف مشخصي منظور شده است.
113
C++ استاندارد اكنون شامل 74 كلمۀ كليدي است:
asm and_eq and Bitor bitand auto case break bool class char catch const_cast const compl delete default continue else dynamic_cast dodouble export explicit enum float dfalse extern goto friend for C++ استاندارد اكنون شامل 74 كلمۀ كليدي است:
114
int inline if namespace mutable long not_eq not new or_eq or operator public eprotected privat return reinterpret_cast register sizeof signed short struct static_cast static this template swich try TRUE throw
115
typename typoid typedef unsigned union using volatile void virtual xor while wchar_t xor_eq
116
دو نوع كلمۀ كليدي وجود دارد:
1- كلمههاي رزرو شده 2- شناسههاي استاندارد. يك كلمۀ رزرو شده كلمهاي است که يک دستور خاص از آن زبان را نشان ميدهد. كلمۀ كليدي if و else كلمات رزرو شده هستند. يك شناسۀ استاندارد كلمهاي است كه يك نوع دادۀ استاندارد از زبان را مشخص ميكند. كلمات كليدي bool و int شناسههاي استاندارد هستند
117
پايان جلسه سوم
118
جلسه چهارم «تكرار»
119
آنچه در اين جلسه مي خوانيد:
1- دستور while 2- خاتمه دادن به يك حلقه 3- دستور do..while 4- دستور for 5- دستور break 6- دستور continue 7- دستور goto 8- توليد اعداد شبه تصادفي
120
شناخت انواع ساختارهاي تکرار و نحو آنها و تبديل آنها به يکديگر.
هدفهاي رفتاري: انتظار ميرود پس از مطالعۀ اين جلسه بتوانيد: - نحو دستورwhile را شناخته و از آن براي ايجاد حلقه استفاده کنيد. - نحو دستور do..while را شناخته و تفاوت آن با دستور while را بيان کنيد. - نحو دستور for را شناخته و با استفاده از آن حلقههاي گوناگون بسازيد. - حلقههاي فوق را به يکديگر تبديل کنيد. - علت استفاده از «دستورات پرش» را ذکر کرده و تفاوت سه دستور break و continue و goto را بيان کنيد. - اهميت اعداد تصادفي را بيان کرده و نحوۀ توليد «اعداد شبه تصادفي» را بدانيد. هدف کلي: شناخت انواع ساختارهاي تکرار و نحو آنها و تبديل آنها به يکديگر.
121
مقدمه تكرار، اجراي پي در پي يك دستور يا بلوكي از دستورالعملها در يك برنامه است. با استفاده از تکرار ميتوانيم کنترل برنامه را مجبور کنيم تا به خطوط قبلي برگردد و آنها را دوباره اجرا نمايد. C++ داراي سه دستور تكرار است: دستور while، دستور do_while و دستور for. دستورهاي تکرار به علت طبيعت چرخهمانندشان، حلقه نيز ناميده ميشوند.
122
1- دستور while نحو دستور while به شکل زير است:
while (condition) statement; به جاي condition، يك شرط قرار ميگيرد و به جاي statement دستوري که بايد تکرار شود قرار ميگيرد. اگر مقدار شرط، صفر(يعني نادرست) باشد، statement ناديده گرفته ميشود و برنامه به اولين دستور بعد از while پرش ميكند. اگر مقدار شرط ناصفر(يعني درست) باشد، statement اجرا شده و دوباره مقدار شرط بررسي ميشود. اين تکرار آن قدر ادامه مييابد تا اين که مقدار شرط صفر شود.
123
مثال 1-4 محاسبۀ حاصل جمع اعداد صحيح متوالي با حلقۀ while
اين برنامه مقدار … + n را براي عدد ورودي n محاسبه ميكند: int main() { int n, i=1; cout << "Enter a positive integer: "; cin >> n; long sum=0; while (i <= n) sum += i++; cout << "The sum of the first " << n << " integers is " << sum; }
124
2- خاتمه دادن به يك حلقه int main() { int n, i=1; cout << "Enter a positive integer: "; cin >> n; long sum=0; while (true) { if (i > n) break; sum += i++; } cout << "The sum of the first " << n << " integers is " << sum; قبلا ديديم كه چگونه دستور break براي كنترل دستورالعمل switch استفاده ميشود (به مثال 17-4 نگاه كنيد). از دستور break براي پايان دادن به حلقهها نيز ميتوان استفاده کرد. يكي از مزيتهاي دستور break اين است كه فورا حلقه را خاتمه ميدهد بدون اين که مابقي دستورهاي درون حلقه اجرا شوند.
125
* مثال 4-4 اعداد فيبوناچي
اعداد فيبوناچي F0, F1, F2, F3, … به شکل بازگشتي توسط معادلههاي زير تعريف ميشوند: F0 = 0 , F1 = 1 , Fn = Fn-1 + Fn-2 مثلا براي n=2 داريم: F2 = F2-1 + F2-2 = F1 + F0 = = 1 يا براي n=3 داريم: F3 = F3-1 + F3-2 = F2 + F1 = = 2 و براي n=4 داريم: F4 = F4-1 + F4-2 = F3 + F2 = = 3
126
cout << "Enter a positive integer: "; cin >> bound;
برنامۀ زير، همۀ اعداد فيبوناچي را تا يك محدودۀ مشخص که از ورودي دريافت ميشود، محاسبه و چاپ ميكند: int main() { long bound; cout << "Enter a positive integer: "; cin >> bound; cout << "Fibonacci numbers < " << bound << ":\n0, 1"; long f0=0, f1=1; while (true) { long f2 = f0 + f1; if (f2 > bound) break; cout << ", " << f2; f0 = f1; f1 = f2;} } Enter a positive integer: 1000 Fibonacci numbers < 1000: 0, 1, 1, 2, 3, 5, 8, 13, 21, 34, 55, 89, 144, 233, 377, 610, 987
127
مثال5-4 استفاده از تابع exit(0)
int main() { long bound; cout << "Enter a positive integer: "; cin >> bound; cout << "Fibonacci numbers < " << bound << ":\n0, 1"; long f0=0, f1=1; while (true) { long f2 = f0 + f1; if (f2 > bound) exit(0); cout << ", " << f2; f0 = f1; f1 = f2; } } برنامهنويسان ترجيح ميدهند از break براي خاتمه دادن به حلقههاي نامتناهي استفاده کنند زيرا قابليت انعطاف بيشتري دارد.
128
متوقف کردن يك حلقۀ نامتناهي :
با فشردن کليدهاي Ctrl+C سيستم عامل يک برنامه را به اجبار خاتمه ميدهد. كليد Ctrl را پايين نگه داشته و كليد C روي صفحهكليد خود را فشار دهيد تا برنامۀ فعلي خاتمه پيدا کند.
129
3- دستور do..while do statement while (condition);
به جاي condition يك شرط قرار ميگيرد و به جاي statement دستور يا بلوکي قرار ميگيرد که قرار است تکرار شود. اين دستور ابتدا statement را اجرا ميكند و سپس شرط condition را بررسي ميكند. اگر شرط درست بود حلقه دوباره تکرار ميشود وگرنه حلقه پايان مييابد.
130
دستور do. while مانند دستور while است
يعني هر متغير كنترلي به جاي اين كه قبل از شروع حلقه تنظيم شود، ميتواند درون آن تنظيم گردد. نتيجۀ ديگر اين است كه حلقۀ do..while هميشه بدون توجه به مقدار شرط كنترل، لااقل يك بار اجرا ميشود اما حلقۀ while ميتواند اصلا اجرا نشود.
131
مثال 7-4 محاسبۀ حاصل جمع اعداد صحيح متوالي با حلقۀ do..while
اين برنامه همان تأثير مثال 1-5 را دارد: int main() { int n, i=0; cout << "Enter a positive integer: "; cin >> n; long sum=0; do sum += i++; while (i <= n); cout << "The sum of the first " << n << " integers is " << sum; }
132
0! = 1 , n! = n(n-1)! 1! = 1((1-1)!) = 1(0!) = 1(1) = 1
* مثال 8-4 اعداد فاكتوريال اعداد فاكتوريال 0! و 1! و 2! و 3! و … با استفاده از رابطههاي بازگشتي زير تعريف ميشوند: 0! = 1 , n! = n(n-1)! براي مثال، به ازاي n = 1 در معادلۀ دوم داريم: 1! = 1((1-1)!) = 1(0!) = 1(1) = 1 همچنين براي n = 2 داريم: 2! = 2((2-1)!) = 2(1!) = 2(1) = 2 و به ازاي n = 3 داريم: 3! = 3((3-1)!) = 3(2!) = 3(2) = 6
133
برنامۀ زير همۀ اعداد فاكتوريال را که از عدد داده شده کوچکترند، چاپ ميکند:
int main() { long bound; cout << "Enter a positive integer: "; cin >> bound; cout << "Factorial numbers < " << bound << ":\n1"; long f=1, i=1; do { cout << ", " << f; f *= ++i; } while (f < bound);
134
4 - دستور for نحو دستورالعمل for به صورت زير است:
for (initialization; condition; update) statement; سه قسمت داخل پرانتز، حلقه را کنترل ميکنند. عبارت initialization براي اعلان يا مقداردهي اوليه به متغير کنترل حلقه استفاده ميشود.اين عبارت اولين عبارتي است که ارزيابي ميشود پيش از اين که نوبت به تکرارها برسد. عبارت updateبراي پيشبردن متغير کنترل حلقه به کار ميرود. اين عبارت پس از اجراي statement ارزيابي ميگردد. عبارت condition براي تعيين اين که آيا حلقه بايد تکرار شود يا خير به کار ميرود. يعني اين عبارت، شرط کنترل حلقه است. اگر اين شرط درست باشد دستور statement اجرا ميشود.
135
بنابراين زنجيرۀ وقايعي که تکرار را ايجاد ميکنند عبارتند از:
1 – ارزيابي عبارت initialization 2 – بررسي شرط condition . اگر نادرست باشد، حلقه خاتمه مييابد. 3 – اجراي statement 4 – ارزيابي عبارت update 5 – تکرار گامهاي 2 تا 4 عبارتهاي initialization و condition و updateعبارتهاي اختياري هستند. يعني ميتوانيم آنها را در حلقه ذکر نکنيم.
136
cout << "Enter a positive integer: "; cin >> n;
مثال 9-4 استفاده از حلقۀ for براي محاسبۀ مجموع اعداد صحيح متوالي اين برنامه همان تأثير مثال 1-5 را دارد: int main() { int n; cout << "Enter a positive integer: "; cin >> n; long sum=0; for (int i=1; i <= n; i++) sum += I; cout << "The sum of the first " << n << " integers is " << sum; } در C++ استاندارد وقتي يك متغير كنترل درون يك حلقۀ for اعلان ميشود (مانند i در مثال بالا) حوزۀ آن متغير به همان حلقۀ for محدود ميگردد. يعني آن متغير نميتواند بيرون از آن حلقه استفاده شود. نتيجۀ ديگر اين است که ميتوان از نام مشابهي در خارج از حلقۀ for براي يك متغير ديگر استفاده نمود.
137
{ for (int i=10; i > 0; i--) cout << " " << i; }
برنامۀ زير ده عدد صحيح مثبت را به ترتيب نزولي چاپ ميكند: int main() { for (int i=10; i > 0; i--) cout << " " << i; }
138
مثال 15-4 بيشتر از يك متغير كنترل در حلقۀ for
int main() { for (int m=95, n=11, m%n > 0; m -= 3, n++) cout << m << "%" << n << " = " << m%n << endl; }
139
#include <iomanip> #include <iostream> int main()
مثال 16-4 حلقههاي for تودرتو برنامۀ زير يك جدول ضرب چاپ ميكند: #include <iomanip> #include <iostream> int main() { for (int x=1; x <= 10; x++) { for (int y=1; y <= 10; y++) cout << setw(4) << x*y; cout << endl; }
140
5- دستور break وقتي دستور break درون حلقههاي تودرتو استفاده شود، فقط روي حلقهاي که مستقيما درون آن قرار گرفته تاثير ميگذارد. حلقههاي بيروني بدون هيچ تغييري ادامه مييابند. دستور break يک دستور آشناست. قبلا از آن براي خاتمه دادن به دستور switch و همچنين حلقههاي while و do..while استفاده کردهايم. از اين دستور براي خاتمه دادن به حلقۀ for نيز ميتوانيم استفاده کنيم. دستور break در هر جايي درون حلقه ميتواند جا بگيرد و در همان جا حلقه را خاتمه دهد.
141
6- دستور continue دستور break بقيۀ دستورهاي درون بلوك حلقه را ناديده گرفته و به اولين دستور بيرون حلقه پرش ميكند. دستور continue نيز شبيه همين است اما به جاي اين که حلقه را خاتمه دهد، اجرا را به تكرار بعدي حلقه منتقل ميكند. اين دستور، ادامۀ چرخۀ فعلي را لغو کرده و اجراي دور بعدي حلقه را آغاز ميکند.
142
{ cout << "\nLoop no: " << n << endl;
مثال 19-4 استفاده از دستورهاي break و continue اين برنامۀ كوچك، دستورهاي break و continue را شرح ميدهد: int main() { int n = 1; char c; for( ; ;n++ ) { cout << "\nLoop no: " << n << endl; cout << "Continue? <y|n> "; cin >> c; if (c = = 'y') continue; break; } cout << "\nTotal of loops: " << n;
143
7- دستور goto دستورgoto نوع ديگري از دستورهاي پرش است. مقصد اين پرش توسط يك برچسب معين ميشود. برچسب شناسهاي است كه جلوي آن علامت كولن( : ) ميآيد و جلوي يك دستور ديگر قرار ميگيرد. يک مزيت دستور goto اين است که با استفاده از آن ميتوان از همۀ حلقههاي تودرتو خارج شد و به مکان دلخواهي در برنامه پرش نمود.
144
for (int i=0; i<N; i++) { for (int j=0; j<N; j++)
مثال 20-4 استفاده از دستور goto براي خارج شدن از حلقههاي تودرتو int main() { const int N=5; for (int i=0; i<N; i++) { for (int j=0; j<N; j++) { for (int k=0; k<N; k++) if (i+j+k>N) goto esc; else cout << i+j+k << " "; cout << "* "; } esc: cout << "." << endl;
145
8- توليد اعداد شبه تصادفي
يكي از كاربردهاي بسيار مهم رايانهها، «شبيهسازي» سيستمهاي دنياي واقعي است. تحقيقات و توسعههاي بسيار پيشرفته به اين راهکار خيلي وابسته است. به وسيلۀ شبيهسازي ميتوانيم رفتار سيستمهاي مختلف را مطالعه کنيم بدون اين که لازم باشد واقعا آنها را پيادهسازي نماييم. در شبيهسازي نياز است «اعداد تصادفي» توسط رايانهها توليد شود تا نادانستههاي دنياي واقعي مدلسازي شود.
146
رايانهها «ثابتکار» هستند يعني با دادن دادههاي مشابه به رايانههاي مشابه، هميشه خروجي يکسان توليد ميشود. با وجود اين ميتوان اعدادي توليد کرد که به ظاهر تصادفي هستند؛ اعدادي که به طور يکنواخت در يک محدودۀ خاص گستردهاند و براي هيچکدام الگوي مشخصي وجود ندارد. چنين اعدادي را «اعداد شبهتصادفي» ميناميم.
147
#include<cstdlib>//defines the rand() and RAND_MAX
مثال 22-4 توليد اعداد شبه تصادفي اين برنامه از تابع rand() براي توليد اعداد شبهتصادفي استفاده ميكند: #include<cstdlib>//defines the rand() and RAND_MAX #include <iostream> int main() { // prints pseudo-random numbers: for (int i = 0; i < 8; i++) cout << rand() << endl; cout << "RAND_MAX = " << RAND_MAX << endl; } هر بار که برنامۀ بالا اجرا شود، رايانه هشت عدد صحيح unsigned توليد ميکند که به طور يکنواخت در فاصلۀ 0 تا RAND_MAX گسترده شدهاند. RAND_MAX در اين رايانه برابر با 2,147,483,647 است.
148
هر عدد شبهتصادفي از روي عدد قبلي خود ساخته ميشود.
اولين عدد شبهتصادفي از روي يك مقدار داخلي که «هسته» گفته ميشود ايجاد ميگردد. هر دفعه که برنامه اجرا شود، هسته با يک مقدار پيشفرض بارگذاري ميشود. براي حذف اين اثر نامطلوب که از تصادفي بودن اعداد ميکاهد، ميتوانيم با استفاده از تابع ()srand خودمان مقدار هسته را انتخاب کنيم.
149
مثال 23-4 كارگذاري هسته به طور محاورهاي
#include <cstdlib> // defines the rand() and srand() #include <iostream> int main() { // prints pseudo-random numbers: unsigned seed; cout << "Enter seed: "; cin >> seed; srand(seed); // initializes the seed for (int i = 0; i < 8; i++) cout << rand() << endl; } مثال 23-4 كارگذاري هسته به طور محاورهاي اين برنامه مانند برنامۀ مثال 22-4 است بجز اين كه ميتوان هستۀ توليدکنندۀ اعداد تصادفي را به شکل محاورهاي وارد نمود:
150
پايان جلسه چهارم
151
جلسه پنجم « توابع»
152
آنچه در اين جلسه مي خوانيد:
1- توابع كتابخانهاي C++ استاندارد 2- توابع ساخت كاربر 3- برنامۀ آزمون 4- اعلانها و تعاريف تابع 5- كامپايل جداگانۀ توابع 6- متغيرهاي محلي، توابع محلي ›››
153
››› 7- تابع void 8 - توابع بولي 9- توابع ورودي/خروجي (I/O)
10- ارسال به طريق ارجاع (آدرس) 11- ارسال از طريق ارجاع ثابت 12-توابع بيواسطه ›››
154
13- چندشکلي توابع 14- تابع main() 15- آرگومانهاي پيشفرض
155
››› هدف کلي: شناخت و معرفي توابع و مزاياي استفاده از تابع در برنامهها
هدفهاي رفتاري: انتظار ميرود پس از پايان اين جلسه بتوانيد: - اهميت توابع و مزيت استفاده از آنها را بيان کنيد. - «اعلان» و «تعريف» تابع را بدانيد و خودتان توابعي را ايجاد کنيد. - «برنامۀ آزمون» را تعريف کرده و دليل استفاده از آن را بيان نماييد. - مفهوم «آرگومان» را بدانيد. - تفاوت ارسال به طريق «ارجاع» و ارسال به طريق «مقدار» و ارسال به طريق «ارجاع ثابت» را بيان کنيد و شکل استفاده از هر يک را بدانيد. هدف کلي: شناخت و معرفي توابع و مزاياي استفاده از تابع در برنامهها ›››
156
- «تابع بيواسطه» را شناخته و نحوۀ معرفي آن را بدانيد.
- چندشکلي توابع را تعريف کنيد و شيوۀ آن را بدانيد. - طريقۀ بهکارگيري آرگومانهاي پيشفرض را بدانيد. - فرق بين تابع void با ساير توابع را بدانيد.
157
1-مقدمه برنامههاي واقعي و تجاري بسيار بزرگتر از برنامههايي هستند که تاکنون بررسي کرديم. براي اين که برنامههاي بزرگ قابل مديريت باشند، برنامهنويسان اين برنامهها را به زيربرنامههايي بخشبندي ميکنند. اين زيربرنامهها «تابع» ناميده ميشوند. توابع را ميتوان به طور جداگانه کامپايل و آزمايش نمود و در برنامههاي مختلف دوباره از آنها استفاده کرد.
158
2- توابع كتابخانهاي C++ استاندارد
قبلا برخي از آنها را استفاده كردهايم: ثابت INT_MAX که در <climits> تعريف شده ، تابع ()sqrt که در <cmath> تعريف شده است و... .
159
تابع جذر sqrt() ريشۀ دوم يك عدد مثبت، جذر آن عدد است. تابع مانند يک برنامۀ کامل، داراي روند ورودي - پردازش - خروجي است هرچند که پردازش، مرحلهاي پنهان است. يعني نميدانيم که تابع روي عدد 2 چه اعمالي انجام ميدهد که 41421/1 حاصل ميشود.
160
برنامۀ سادۀ زير، تابع از پيش تعريف شدۀ جذر را به کار ميگيرد:
#include <cmath> // defines the sqrt() function #include <iostream> // defines the cout object using namespace std; int main() { //tests the sqrt() function: for (int x=0; x < 6; x++) cout << "\t" << x << "\t" << sqrt(x) << endl; } براي اجراي يك تابع مانند تابع sqrt() کافي است نام آن تابع به صورت يک متغير در دستورالعمل مورد نظر استفاده شود، مانند زير: y=sqrt(x);
161
اين کار «فراخواني تابع» يا «احضار تابع» گفته ميشود
اين کار «فراخواني تابع» يا «احضار تابع» گفته ميشود. بنابراين وقتي كد sqrt(x) اجرا شود، تابع sqrt() فراخواني ميگردد. عبارت x درون پرانتز «آرگومان» يا «پارامتر واقعي» فراخواني ناميده ميشود. در چنين حالتي ميگوييم كه x توسط «مقدار» به تابع فرستاده ميشود. لذا وقتي x=3 است، با اجراي کد sqrt(x) تابع sqrt() فراخواني شده و مقدار 3 به آن فرستاده ميشود. تابع مذکور نيز حاصل را به عنوان پاسخ برميگرداند…
162
Main() … اين فرايند در نمودار زير نشان داده شده. Sqrt() int 3 x 3 y double متغيرهاي x و y در تابع main() تعريف شدهاند. مقدار x که برابر با 3 است به تابع sqrt() فرستاده ميشود و اين تابع مقدار را به تابع main() برميگرداند. جعبهاي كه تابع sqrt() را نشان ميدهد به رنگ تيره است، به اين معنا كه فرايند داخلي و نحوۀ کار آن قابل رويت نيست.
163
{ for (float x=0; x < 2; x += 0.2)
مثال 2-5 آزمايش يك رابطۀ مثلثاتي اين برنامه هم از سرفايل <cmath> استفاده ميكند. هدف اين است که صحت رابطۀ Sin2x=2SinxCosx به شکل تجربي بررسي شود. int main() { for (float x=0; x < 2; x += 0.2) cout << x << "\t\t" << sin(2*x) << "\t“ << 2*sin(x)*cos(x) << endl; }
164
خروجي برنامه: برنامۀ مقدار x را در ستون اول، مقدار Sin2x را در ستون دوم و مقدار 2SinxCosx را در ستون سوم چاپ ميكند. خروجي نشان ميدهد که براي هر مقدار آزمايشي x، مقدار Sin2x با مقدار 2SinxCosx برابر است.
165
بيشتر توابع معروف رياضي كه در ماشينحسابها هم وجود دارد در سرفايل <cmath> تعريف شده است. بعضي از اين توابع در جدول زير نشان داده شده: مثال شرح تابع acos(0.2) مقدار را برميگرداند کسينوس معکوسx (به راديان) acos(x) asin(0.2) مقدار را برميگرداند سينوس معکوس x (به راديان) asin(x) atan(0.2) مقدار را برميگرداند تانژانت معکوس x (به راديان) atan(x) ceil( ) مقدار 4.0 را برميگرداند مقدار سقف x (گرد شده) ceil(x) cos(2) مقدار را برميگرداند کسينوس x (به راديان) cos(x) exp(2) مقدار را برميگرداند تابع نمايي x (در پايه e) exp(x) fabs(-2) مقدار 2.0 را برميگرداند قدر مطلق x fabs(x)
166
floor(3.141593) مقدار 3.0 را برميگرداند
مقدار کف x (گرد شده) floor(x) log(2) مقدار را برميگرداند لگاريتم طبيعي x (در پايه e) log(x) log10(2) مقدار را برميگرداند لگاريتم عمومي x (در پايه 10) log10(x) pow(2,3) مقدار 8.0 را برميگرداند x به توان p pow(x,p) sin(2) مقدار را برميگرداند سينوس x (به راديان) sin(x) sqrt(2) مقدار را برميگرداند جذر x sqrt(x) tan(2) مقدار را برميگرداند تانژانت x (به راديان) tan(x)
167
توجه داشته باشيد که هر تابع رياضي يک مقدار از نوع double را برميگرداند. اگر يك نوع صحيح به تابع فرستاده شود، قبل از اين كه تابع آن را پردازش کند، مقدارش را به نوع double ارتقا ميدهد.
168
#include <cstdlib>
اين سرفايلها از كتابخانۀ C استاندارد گرفته شدهاند. استفاده از آنها شبيه استفاده از سرفايلهاي C++ استاندارد (مانند <iostream> ) است. براي مثال اگر بخواهيم تابع اعداد تصادفي rand() را از سرفايل <cstdlib> به كار ببريم، بايد دستور پيشپردازندۀ زير را به ابتداي فايل برنامۀ اصلي اضافه کنيم: #include <cstdlib> بعضي از سرفايلهاي كتابخانۀ C++ استاندارد که کاربرد بيشتري دارند در جدول زير آمده است: شرح سرفايل تابع <assert> را تعريف ميکند <assert> توابعي را براي بررسي کاراکترها تعريف ميکند <ctype> ثابتهاي مربوط به اعداد مميز شناور را تعريف ميکند <cfloat> محدودۀ اعداد صحيح را روي سيستم موجود تعريف ميکند <climits> توابع رياضي را تعريف ميکند <cmath> توابعي را براي ورودي و خروجي استاندارد تعريف ميکند <cstdio> توابع کاربردي را تعريف ميکند <cstdlib> توابعي را براي پردازش رشتهها تعريف ميکند <cstring> توابع تاريخ و ساعت را تعريف ميکند <ctime>
169
3- توابع ساخت كاربر گرچه توابع بسيار متنوعي در کتابخانۀ C++ استاندارد وجود دارد ولي اين توابع براي بيشتر وظايف برنامهنويسي كافي نيستند. علاوه بر اين برنامهنويسان دوست دارند خودشان بتوانند توابعي را بسازند و استفاده نمايند.
170
مثال 3-5 تابع cube() int cube(int x) { // returns cube of x:
يك مثال ساده از توابع ساخت كاربر: int cube(int x) { // returns cube of x: return x*x*x; } اين تابع، مكعب يك عدد صحيح ارسالي به آن را برميگرداند. بنابراين فراخواني cube(2) مقدار 8 را برميگرداند.
171
يك تابع ساخت كاربر دو قسمت دارد:
1-عنوان بدنه. عنوان يك تابع به صورت زير است: (فهرست پارامترها) نام نوع بازگشتي مثال: int cube(int x) { … بدنه تابع } نوع بازگشتي تابع cube() که در بالا تعريف شد، int است. نام آن cube ميباشد و يک پارامتر از نوع int به نام x دارد. يعني تابع cube() يک مقدار از نوع int ميگيرد و پاسخي از نوع int تحويل ميدهد. بدنۀ تابع، يك بلوك كد است كه در ادامۀ عنوان آن ميآيد. بدنه شامل دستوراتي است كه بايد انجام شود تا نتيجۀ مورد نظر به دست آيد. بدنه شامل دستور return است كه پاسخ نهايي را به مكان فراخواني تابع برميگرداند.
172
دستور return دو وظيفۀ عمده دارد
دستور return دو وظيفۀ عمده دارد. اول اين که اجراي تابع را خاتمه ميدهد و دوم اين که مقدار نهايي را به برنامۀ فراخوان باز ميگرداند. دستور return به شکل زير استفاده ميشود: return expression; به جاي expression هر عبارتي قرار ميگيرد که بتوان مقدار آن را به يک متغير تخصيص داد. نوع آن عبارت بايد با نوع بازگشتي تابع يکي باشد. عبارت int main() که در همۀ برنامهها استفاده کردهايم يک تابع به نام «تابع اصلي» را تعريف ميکند. نوع بازگشتي اين تابع از نوع int است. نام آن main است و فهرست پارامترهاي آن خالي است؛ يعني هيچ پارامتري ندارد.
173
تنها هدف اين برنامه، امتحان کردن تابع و بررسي صحت کار آن است.
4- برنامۀ آزمون تنها هدف اين برنامه، امتحان کردن تابع و بررسي صحت کار آن است. وقتي يک تابع مورد نياز را ايجاد کرديد، فورا بايد آن تابع را با يک برنامۀ ساده امتحان کنيد. چنين برنامهاي برنامۀ آزمون ناميده ميشود. برنامۀ آزمون يک برنامۀ موقتي است که بايد «سريع و کثيف» باشد؛ يعني: لازم نيست در آن تمام ظرافتهاي برنامهنويسي – مثل پيغامهاي خروجي، برچسبها و راهنماهاي خوانا – را لحاظ کنيد.
174
مثال 4-5 يك برنامۀ آزمون براي تابع cube()
int cube(int x) { // returns cube of x: return x*x*x; } int main() { // tests the cube() function: int n=1; while (n != 0) { cin >> n; cout << "\tcube(" << n << ") = " << cube(n) << endl; }} برنامۀ حاضر اعداد صحيح را از ورودي ميگيرد و مكعب آنها را چاپ ميكند تا اين كه كاربر مقدار 0 را وارد كند.
175
ميتوان رابطۀ بين تابع main() و تابع cube() را شبيه اين شکل تصور نمود:
هر عدد صحيحي که خوانده ميشود، با استفاده از کد cube(n) به تابع cube() فرستاده ميشود. مقدار بازگشتي از تابع، جايگزين عبارت cube(n) گشته و با استفاده از cout در خروجي چاپ ميشود. دقت كنيد كه تابع cube() در بالاي تابع main() تعريف شده زيرا قبل از اين كه تابعcube() در تابع main() به كار رود، كامپايلر C++ بايد در بارۀ آن اطلاع حاصل كند. 5 n int 125 cube() main() x ميتوان رابطۀ بين تابع main() و تابع cube() را شبيه اين شکل تصور نمود:
176
مثال 5-5 يك برنامۀ آزمون براي تابع max()
تابع زير دو پارامتر دارد. اين تابع از دو مقدار فرستاده شده به آن، مقدار بزرگتر را برميگرداند: int max(int x, int y) { // returns larger of the two given integers: int z; z = (x > y) ? x : y ; return z; } int main() { int m, n; do { cin >> m >> n; cout << "\tmax(" << m << "," << n << ") = " << max(m,n) << endl; } while (m != 0);}
177
{ // returns larger of the two given integers: if (x < y) return y;
توابع ميتوانند بيش از يک دستور return داشته باشند. مثلا تابع max() را مانند اين نيز ميتوانستيم بنويسيم: int max(int x, int y) { // returns larger of the two given integers: if (x < y) return y; else return x; } دستور return نوعي دستور پرش است (شبيه دستور break ) زيرا اجرا را به بيرون از تابع هدايت ميکند. اگرچه معمولا return در انتهاي تابع قرار ميگيرد، ميتوان آن را در هر نقطۀ ديگري از تابع قرار داد. در اين کد هر دستور return که زودتر اجرا شود مقدار مربوطهاش را بازگشت داده و تابع را خاتمه ميدهد.
178
5- اعلانها و تعاريف تابع
به دو روش ميتوان توابع را تعريف نمود: 1-توابع قبل از تابع main() به طور كامل با بدنه مربوطه آورده شوند. 2-راه ديگري که بيشتر رواج دارد اين گونه است که ابتدا تابع اعلان شود، سپس متن برنامۀ اصليmain() بيايد، پس از آن تعريف کامل تابع قرار بگيرد.
179
اعلان تابع شبيه اعلان متغيرهاست.
اعلان تابع با تعريف تابع تفاوت دارد. اعلان تابع، فقط عنوان تابع است که يک سميکولن در انتهاي آن قرار دارد. تعريف تابع، متن کامل تابع است که هم شامل عنوان است و هم شامل بدنه. اعلان تابع شبيه اعلان متغيرهاست. يک متغير قبل از اين که به کار گرفته شود بايد اعلان شود. تابع هم همين طور است با اين فرق که متغير را در هر جايي از برنامه ميتوان اعلان کرد اما تابع را بايد قبل از برنامۀ اصلي اعلان نمود.
180
در اعلان تابع فقط بيان ميشود که نوع بازگشتي تابع چيست، نام تابع چيست و نوع پارامترهاي تابع چيست.
همينها براي کامپايلر کافي است تا بتواند کامپايل برنامه را آغاز کند. سپس در زمان اجرا به تعريف بدنۀ تابع نيز احتياج ميشود که اين بدنه در انتهاي برنامه و پس از تابع main() قرار ميگيرد.
181
فرق بين «آرگومان» و «پارامتر» :
پارامترها متغيرهايي هستند که در فهرست پارامتر يک تابع نام برده ميشوند. پارامترها متغيرهاي محلي براي تابع محسوب ميشوند؛ يعني فقط در طول اجراي تابع وجود دارند. آرگومانها متغيرهايي هستند که از برنامۀ اصلي به تابع فرستاده ميشوند.
182
int max(int,int); int main() { int m, n; do { cin >> m >> n; cout << "\tmax(" << m << "," << n << ") = " << max(m,n) << endl; } while (m != 0);} int max(int x, int y) { if (x < y) return y; else return x;} مثال 6-5 تابعmax() با اعلان جدا از تعريف آن اين برنامه همان برنامۀ آزمون تابع max() در مثال 5-6 است. اما اينجا اعلان تابع بالاي تابع اصلي ظاهر شده و تعريف تابع بعد از برنامۀ اصلي آمده است: توجه كنيد كه پارامترهاي x و y در بخش عنوان تعريف تابع آمدهاند (طبق معمول) ولي در اعلان تابع وجود ندارند.
183
6- كامپايل جداگانۀ توابع
اغلب اين طور است که تعريف و بدنۀ توابع در فايلهاي جداگانهاي قرار ميگيرد. اين فايلها به طور مستقل کامپايل1 ميشوند و سپس به برنامۀ اصلي که آن توابع را به کار ميگيرد الصاق2 ميشوند. توابع کتابخانۀ C++ استاندارد به همين شکل پيادهسازي شدهاند و هنگامي که يکي از آن توابع را در برنامههايتان به کار ميبريد بايد با دستور راهنماي پيشپردازنده، فايل آن توابع را به برنامهتان ضميمه کنيد. اين کار چند مزيت دارد:
184
1- اولين مزيت «مخفيسازي اطلاعات» است.
2-مزيت ديگر اين است که توابع مورد نياز را ميتوان قبل از اين که برنامۀ اصلي نوشته شود، جداگانه آزمايش نمود. 3-سومين مزيت اين است که در هر زماني به راحتي ميتوان تعريف توابع را عوض کرد بدون اين که لازم باشد برنامۀ اصلي تغيير يابد. 4-چهارمين مزيت هم اين است که ميتوانيد يک بار يک تابع را کامپايل و ذخيره کنيد و از آن پس در برنامههاي مختلفي از همان تابع استفاده ببريد.
185
تابع max() را به خاطر بياوريد
تابع max() را به خاطر بياوريد. براي اين که اين تابع را در فايل جداگانهاي قرار دهيم، تعريف آن را در فايلي به نام max.cpp ذخيره ميکنيم. فايل max.cpp شامل کد زير است: int max(int x, int y) { if (x < y) return y; else return x; } max.cpp
186
حال كافي است عبارت:#include <test
حال كافي است عبارت:#include <test.cpp> را به اول برنامه اصلي وقبل ازmain() اضافه كنيم: #include <test.cpp> int main() { // tests the max() function: int m, n; do { cin >> m >> n; cout << "\tmax(" << m << "," << n << ") = " << max(m,n) << endl; } while (m != 0);}
187
نحوۀ کامپايل کردن فايلها و الصاق آنها به يکديگر به نوع سيستم عامل و نوع کامپايلر بستگي دارد. در سيستم عامل ويندوز معمولا توابع را در فايلهايي از نوع DLL کامپايل و ذخيره ميکنند و سپس اين فايل را در برنامۀ اصلي احضار مينمايند. فايلهاي DLL را به دو طريق ايستا و پويا ميتوان مورد استفاده قرار داد. براي آشنايي بيشتر با فايلهاي DLL به مرجع ويندوز و کامپايلرهاي C++ مراجعه کنيد.
188
6- متغيرهاي محلي، توابع محلي
متغير محلي، متغيري است که در داخل يک بلوک اعلان گردد. اين گونه متغيرها فقط در داخل همان بلوکي که اعلان ميشوند قابل دستيابي هستند. چون بدنۀ تابع، خودش يک بلوک است پس متغيرهاي اعلان شده در يک تابع متغيرهاي محلي براي آن تابع هستند. اين متغيرها فقط تا وقتي که تابع در حال کار است وجود دارند. پارامترهاي تابع نيز متغيرهاي محلي محسوب ميشوند.
189
* مثال 7-5 تابع فاكتوريل اعداد فاكتوريل را در مثال 8-5 ديديم. فاكتوريل عدد صحيح n برابر است با: n! = n(n-1)(n-2)..(3)(2)(1) تابع زير، فاکتوريل عدد n را محاسبه ميکند: long fact(int n) { //returns n! = n*(n-1)*(n-2)*...*(2)*(1) if (n < 0) return 0; int f = 1; while (n > 1) f *= n--; return f; } اين تابع دو متغير محلي دارد: n و f پارامتر n يک متغير محلي است زيرا در فهرست پارامترهاي تابع اعلان شده و متغير f نيز محلي است زيرا درون بدنۀ تابع اعلان شده است.
190
همان گونه که متغيرها ميتوانند محلي باشند، توابع نيز ميتوانند محلي باشند.
يک تابع محلي تابعي است که درون يک تابع ديگر به کار رود. با استفاده از چند تابع ساده و ترکيب آنها ميتوان توابع پيچيدهتري ساخت. به مثال زير نگاه کنيد. تابع محلي در رياضيات، تابع جايگشت را با p(n,k) نشان ميدهند. اين تابع بيان ميکند که به چند طريق ميتوان k عنصر دلخواه از يک مجموعۀ n عنصري را کنار يکديگر قرار داد. براي اين محاسبه از رابطۀ زير استفاده ميشود:
191
اين تابع، خود از تابع ديگري که همان تابع فاکتوريل است استفاده کرده است.
شرط به کار رفته در دستور if براي محدود کردن حالتهاي غير ممکن استفاده شده است. در اين حالتها، تابع مقدار 0 را برميگرداند تا نشان دهد که يک ورودي اشتباه وجود داشته است. پس به 12 طريق ميتوانيم دو عنصر دلخواه از يک مجموعۀ چهار عنصري را کنار هم بچينيم. براي دو عنصر از مجموعۀ {1, 2, 3, 4} حالتهاي ممکن عبارت است از: 12, 13, 14, 21, 23, 24, 31, 32, 34, 41, 42, 43 كد زير تابع جايگشت را پيادهسازي ميكند: long perm(int n, int k) {// returns P(n,k), the number of the permutations of k from n: if (n < 0) || k < 0 || k > n) return 0; return fact(n)/fact(n-k); }
192
برنامۀ آزمون براي تابع perm() در ادامه آمده است: long perm(int,int);
// returns P(n,k), the number of permutations of k from n: int main() { // tests the perm() function: for (int i = -1; i < 8; i++) { for (int j= -1; j <= i+1; j++) cout << " " << perm(i,j); cout << endl; } } 0 0 0 1 0
193
7- تابع void لازم نيست يك تابع حتما مقداري را برگرداند. در C++ براي مشخص کردن چنين توابعي از کلمۀ کليدي void به عنوان نوع بازگشتي تابع استفاده ميکنند يک تابع void تابعي است که هيچ مقدار بازگشتي ندارد. از آنجا كه يك تابع void مقداري را برنميگرداند، نيازي به دستور return نيست ولي اگر قرار باشد اين دستور را در تابع void قرار دهيم، بايد آن را به شکل تنها استفاده کنيم بدون اين که بعد از کلمۀ return هيچ چيز ديگري بيايد: return; در اين حالت دستور return فقط تابع را خاتمه ميدهد.
194
توابع بولي فقط دو مقدار را برميگردانند: true يا false .
8- توابع بولي در بسياري از اوقات لازم است در برنامه، شرطي بررسي شود. اگر بررسي اين شرط به دستورات زيادي نياز داشته باشد، بهتر است که يک تابع اين بررسي را انجام دهد. اين کار مخصوصا هنگامي که از حلقهها استفاده ميشود بسيار مفيد است. توابع بولي فقط دو مقدار را برميگردانند: true يا false . اسم توابع بولي را معمولا به شکل سوالي انتخاب ميکنند زيرا توابع بولي هميشه به يک سوال مفروض پاسخ بلي يا خير ميدهند.
195
{ // returns true if n is prime, false otherwise:
مثال 10-5 تابعي كه اول بودن اعداد را بررسي ميكند کد زير يك تابع بولي است كه تشخيص ميدهد آيا عدد صحيح ارسال شده به آن، اول است يا خير: bool isPrime(int n) { // returns true if n is prime, false otherwise: float sqrtn = sqrt(n); if (n < 2) return false; // 0 and 1 are not primes if (n < 4) return true; // 2 and 3 are the first primes if (n%2 == 0) return false; // 2 is the only even prime for (int d=3; d <= sqrtn; d += 2) if (n%d == 0) return false; // n has a nontrivial divisor return true; // n has no nontrivial divisors }
196
9- توابع ورودي/خروجي (I/O)
بخشهايي از برنامه که به جزييات دست و پا گير ميپردازد و خيلي به هدف اصلي برنامه مربوط نيست را ميتوان به توابع سپرد. در چنين شرايطي سودمندي توابع محسوستر ميشود. فرض کنيد نرمافزاري براي سيستم آموزشي دانشگاه طراحي کردهايد که سوابق تحصيلي دانشجويان را نگه ميدارد. در اين نرمافزار لازم است که سن دانشجو به عنوان يکي از اطلاعات پروندۀ دانشجو وارد شود. اگر وظيفۀ دريافت سن را به عهدۀ يک تابع بگذاريد، ميتوانيد جزيياتي از قبيل کنترل ورودي معتبر، يافتن سن از روي تاريخ تولد و ... را در اين تابع پيادهسازي کنيد بدون اين که از مسير برنامۀ اصلي منحرف شويد.
197
مثال بعد يک تابع ورودي را نشان ميدهد.
قبلا نمونهاي از توابع خروجي را ديديم. تابع PrintDate() در مثال 9-5 هيچ چيزي به برنامۀ اصلي برنميگرداند و فقط براي چاپ نتايج به کار ميرود. اين تابع نمونهاي از توابع خروجي است؛ يعني توابعي که فقط براي چاپ نتايج به کار ميروند و هيچ مقدار بازگشتي ندارند. توابع ورودي نيز به همين روش کار ميکنند اما در جهت معکوس. يعني توابع ورودي فقط براي دريافت ورودي و ارسال آن به برنامۀ اصلي به کار ميروند و هيچ پارامتري ندارند. مثال بعد يک تابع ورودي را نشان ميدهد.
198
مثال 11-5 تابعي براي دريافت سن كاربر
تابع سادۀ زير، سن کاربر را درخواست ميکند و مقدار دريافت شده را به برنامۀ اصلي ميفرستد. اين تابع تقريبا هوشمند است و هر عدد صحيح ورودي غير منطقي را رد ميکند و به طور مکرر درخواست ورودي معتبر ميکند تا اين که يک عدد صحيح در محدودۀ 7 تا 120 دريافت دارد: int age() { // prompts the user to input his/her age and returns that value: int n; while (true) { cout << "How old are you: "; cin >> n; if (n < 0) cout << "\a\tYour age could not be negative."; else if (n > 120) cout << "\a\tYou could not be over 120."; else return n; cout << "\n\tTry again.\n"; }
199
{ // tests the age() function: int a = age();
يك برنامۀ آزمون و خروجي حاصل از آن در ادامه آمده است: int age() int main() { // tests the age() function: int a = age(); cout << "\nYou are " << a << " years old.\n"; } How old are you? 125 You could not be over 120 Try again. How old are you? -3 Your age could not be negative How old are you? 99 You are 99 years old.
200
تا اين لحظه تمام پارامترهايي كه در توابع ديديم به طريق مقدار ارسال شدهاند. يعني ابتدا مقدار متغيري که در فراخواني تابع ذکر شده برآورد ميشود و سپس اين مقدار به پارامترهاي محلي تابع فرستاده ميشود. مثلا در فراخواني cube(x) ابتدا مقدار x برآورد شده و سپس اين مقدار به متغير محلي n در تابع فرستاده ميشود و پس از آن تابع کار خويش را آغاز ميکند. در طي اجراي تابع ممکن است مقدار n تغيير کند اما چون n محلي است هيچ تغييري روي مقدار x نميگذارد.
201
پس خود x به تابع نميرود بلکه مقدار آن درون تابع کپي ميشود.
تغيير دادن اين مقدار کپي شده درون تابع هيچ تاثيري بر x اصلي ندارد. به اين ترتيب تابع ميتواند مقدار x را بخواند اما نميتواند مقدار x را تغيير دهد. به همين دليل به x يک پارامتر «فقط خواندني» ميگويند. وقتي ارسال به وسيلۀ مقدار باشد، هنگام فراخواني تابع ميتوان از عبارات استفاده کرد. مثلا تابع cube() را ميتوان به صورتcube(2*x-3) فراخواني کرد يا به شکل cube(2*sqrt(x)-cube(3)) فراخواني نمود. در هر يک از اين حالات، عبارت درون پرانتز به شکل يک مقدار تکي برآورد شده و حاصل آن مقدار به تابع فرستاده ميشود.
202
10- ارسال به طريق ارجاع (آدرس)
ارسال به طريق مقدار باعث ميشود که متغيرهاي برنامۀ اصلي از تغييرات ناخواسته در توابع مصون بمانند. اما گاهي اوقات عمدا ميخواهيم اين اتفاق رخ دهد. يعني ميخواهيم که تابع بتواند محتويات متغير فرستاده شده به آن را دستکاري کند. در اين حالت از ارسال به طريق ارجاع استفاده ميکنيم.
203
براي اين که مشخص کنيم يک پارامتر به طريق ارجاع ارسال ميشود، علامت را به نوع پارامتر در فهرست پارامترهاي تابع اضافه ميکنيم. اين باعث ميشود که تابع به جاي اين که يک کپي محلي از آن آرگومان ايجاد کند، خود آرگومان محلي را به کار بگيرد. به اين ترتيب تابع هم ميتواند مقدار آرگومان فرستاده شده را بخواند و هم ميتواند مقدار آن را تغيير دهد. در اين حالت آن پارامتر يک پارامتر «خواندني-نوشتني» خواهد بود. &
204
هر تغييري که روي پارامتر خواندني-نوشتني در تابع صورت گيرد به طور مستقيم روي متغير برنامۀ اصلي اعمال ميشود. به مثال زير نگاه کنيد. * مثال 12-5 تابع swap() تابع كوچك زير در مرتب کردن دادهها کاربرد فراوان دارد: void swap(float& x, float& y) { // exchanges the values of x and y: float temp = x; x = y; y = temp; } هدف اين تابع جابجا کردن دو عنصري است که به آن فرستاده ميشوند. براي اين منظور پارامترهاي x و y به صورت پارامترهاي ارجاع تعريف شدهاند: float& x, float& y
205
void swap(float&, float&) // exchanges the values of x and y:
عملگر ارجاع & موجب ميشود كه به جاي x و y آرگومانهاي ارسالي قرار بگيرند. برنامۀ آزمون و اجراي آزمايشي آن در زير آمده است: void swap(float&, float&) // exchanges the values of x and y: int main() { // tests the swap() function: float a = 55.5, b = 88.8; cout << "a = " << a << ", b = " << b << endl; swap(a,b); } a = 55.5, b = 88.8 a = 88.8, b = 55.5
206
هنگام فراخواني تابع swap(a,b)
وقتي فراخواني swap(a,b) اجرا ميشود، x به a اشاره ميکند و y به b. سپس متغير محلي temp اعلان ميشود و مقدار x (که همان a است) درون آن قرار ميگيرد. پس از آن مقدار y (که همان b است) درون x (يعني a) قرار ميگيرد و آنگاه مقدار temp درون y (يعني b) قرار داده ميشود. نتيجۀ نهايي اين است که مقادير a و b با يکديگر جابجا مي شوند. شکل مقابل نشان ميدهد که چطور اين جابجايي رخ ميدهد: هنگام فراخواني تابع swap(a,b) 55.5 a float main() x float& 88.8 b y swap() 88.8 a float swap() main() x float& 55.5 b y temp بعد از بازگشت
207
به اعلان تابع swap() دقت کنيد:
void swap(float&, float&) اين اعلان شامل عملگر ارجاع & براي هر پارامتر است. برنامهنويسان c عادت دارند که عملگر ارجاع & را به عنوان پيشوند نام متغير استفاده کنند (مثلfloat &x) در C++ فرض ميکنيم عملگر ارجاع & پسوند نوع است (مثل float& x) به هر حال کامپايلر هيچ فرقي بين اين دو اعلان نميگذارد و شکل نوشتن عملگر ارجاع کاملا اختياري و سليقهاي است.
208
مثال 13-5 ارسال به طريق مقدار و ارسال به طريق ارجاع
اين برنامه، تفاوت بين ارسال به طريق مقدار و ارسال به طريق ارجاع را نشان ميدهد: void f(int,int&); int main() { int a = 22, b = 44; cout << "a = " << a << ", b = " << b << endl; f(a,b); f(2*a-3,b); cout << "a = " << a << ", b = " << b << endl;} void f(int x , int& y) { x = 88; y = 99;} تابع f() دو پارامتر دارد که اولي به طريق مقدار و دومي به طريق ارجاع ارسال ميشود. فراخواني f(a,b) باعث ميشود که a از طريق مقدار به x ارسال شود و b از طريق ارجاع به y فرستاده شود. a = 22, b = 44 a = 22, b = 99
209
شکل زير نحوۀ کار تابع f() را نشان ميدهد.
22 a int main() x 44 b y int& f() هنگام فراخواني تابع f(a,b) 22 a int main() 88 xx 99 b y int& f() بعد از بازگشت
210
در جدول زير خلاصۀ تفاوتهاي بين ارسال از طريق مقدار و ارسال از طريق ارجاع آمده است.
int& x; int x; پارامتر x يک ارجاع است پارامتر x يک متغير محلي است x مترادف با آرگومان است x يک کپي از آرگومان است ميتواند محتويات آرگومان را تغيير دهد تغيير محتويات آرگومان ممکن نيست آرگومان ارسال شده از طريق ارجاع فقط بايد يک متغير باشد آرگومان ارسال شده از طريق مقدار ميتواند يک ثابت، يک متغير يا يک عبارت باشد آرگومان خواندني-نوشتني است آرگومان فقط خواندني است
211
يكي از مواقعي كه پارامترهاي ارجاع مورد نياز هستند جايي است كه تابع بايد بيش از يك مقدار را بازگرداند. دستور return فقط ميتواند يك مقدار را برگرداند. بنابراين اگر بايد بيش از يك مقدار برگشت داده شود، اين كار را پارامترهاي ارجاع انجام ميدهند.
212
* مثال 14-5 بازگشت بيشتر از يك مقدار
تابع زير از طريق دو پارامتر ارجاع، دو مقدار را بازميگرداند: area و circumference (محيط و مساحت) براي دايرهاي که شعاع آن عدد مفروض r است: void ComputeCircle(double& area, double& circumference, double r) { // returns the area and circumference of a circle with radius r: const double PI = ; area = PI*r*r; circumference = 2*PI*r; }
213
برنامۀ آزمون تابع فوق و يک اجراي آزمايشي آن در شکل زير نشان داده شده است:
void ComputerCircle(double&, double&, double); // returns the area and circumference of a circle with radius r; int main() { // tests the ComputeCircle() function: double r, a, c; cout << "Enter radius: "; cin >> r; ComputeCircle(a, c, r); cout << "area = " << a << ", circumference = " << c << endl;}
214
12- ارسال از طريق ارجاع ثابت
ارسال پارامترها به طريق ارجاع دو خاصيت مهم دارد: اول اين که تابع ميتواند روي آرگومان واقعي تغييراتي بدهد دوم اين که از اشغال بيمورد حافظه جلوگيري ميشود. روش ديگري نيز براي ارسال آرگومان وجود دارد: ارسال از طريق ارجاع ثابت. اين روش مانند ارسال از طريق ارجاع است با اين فرق که تابع نميتواند محتويات پارامتر ارجاع را دستکاري نمايد و فقط اجازۀ خواندن آن را دارد. براي اين که پارامتري را از نوع ارجاع ثابت اعلان کنيم بايد عبارت const را به ابتداي اعلان آن اضافه نماييم.
215
مثال 15-5 ارسال از طريق ارجاع ثابت
سه طريقه ارسال پارامتر در تابع زير به کار رفته است: void f(int x, int& y, const int& z) { x += z; y += z; cout << "x = " << x << ", y = " << y << ", z = " << z << endl; } در تابع فوق اولين پارامتر يعني x از طريق مقدار ارسال ميشود، دومين پارامتر يعني y از طريق ارجاع و سومين پارامتر نيز از طريق ارجاع ثابت.
216
برنامۀ آزمون و يک اجراي آزمايشي از مثال قبل:
void f(int, int&, const int&); int main() { // tests the f() function: int a = 22, b = 33, c = 44; cout << "a = " << a << ", b = " << b << ", c = " << c << endl; f(a,b,c); f(2*a-3,b,c); } a = 22, b = 33, c = 44 x = 66, y = 77, z = 44 a = 22, b = 77, c = 44 x = 85, y = 121, z = 44 a = 22, b = 121, c = 44 تابع فوق پارامترهاي x و y را ميتواند تغيير دهد ولي قادر نيست پارامتر z را تغيير دهد. تغييراتي که روي x صورت ميگيرد اثري روي آرگومان a نخواهد داشت زيرا a از طريق مقدار به تابع ارسال شده. تغييراتي که روي y صورت ميگيرد روي آرگومان b هم تاثير ميگذارد زيرا b از طريق ارجاع به تابع فرستاده شده.
217
ارسال به طريق ارجاع ثابت بيشتر براي توابعي استفاده ميشود که عناصر بزرگ را ويرايش ميکنند مثل آرايهها يا نمونۀ کلاسها که در جلسههاي بعدي توضيح آنها آمده است. عناصري که از انواع اصلي هستند (مثل int يا float) به طريق مقدار ارسال ميشوند به شرطي که قرار نباشد تابع محتويات آنها را دستکاري کند.
218
13- توابع بيواسطه تابعي که به شکل بيواسطه تعريف ميشود، ظاهري شبيه به توابع معمولي دارد با اين فرق که عبارت inline در اعلان و تعريف آن قيد شده است. مثال 16-5 تابع cube() به شکل بيواسطه اين همان تابع cube() مثال 3-5 است: inline int cube(int x) { // returns cube of x: return x*x*x; } تنها تفاوت اين است كه كلمۀ كليدي inline در ابتداي عنوان تابع ذکر شده. اين عبارت به كامپايلر ميگويد كه در برنامه به جاي cube(n) کد واقعي (n)*(n)*(n) را قرار دهد.
219
. به برنامۀ آزمون زير نگاه کنيد:
int main() { // tests the cube() function: cout << cube(4) << endl; int x, y; cin >> x; y = cube(2*x-3);} اين برنامه هنگام کامپايل به شکل زير درميآيد، گويي اصلا تابعي وجود نداشته: cout << (4) * (4) * (4) << endl; y = (2*x-3) * (2*x-3) * (2*x-3);} احتياط: استفاده از توابع بيواسطه ميتواند اثرات منفي داشته باشد. مثلا اگر يک تابع بيواسطه داراي 40 خط کد باشد و اين تابع در 26 نقطه مختلف از برنامۀ اصلي فراخواني شود، هنگام کامپايل بيش از هزار خط کد به برنامۀ اصلي افزوده ميشود. همچنين تابع بيواسطه ميتواند قابليت انتقال برنامۀ شما را روي سيستمهاي مختلف کاهش دهد. وقتي كامپايلر کد واقعي تابع را جايگزين فراخواني آن ميکند، ميگوييم که تابع بيواسطه، باز ميشود.
220
14- چندشکلي توابع در C++ ميتوانيم چند تابع داشته باشيم که همگي يک نام دارند. در اين حالت ميگوييم که تابع مذکور، چندشکلي دارد. شرط اين کار آن است که فهرست پارامترهاي اين توابع با يکديگر تفاوت داشته باشد. يعني تعداد پارامترها متفاوت باشد يا دست کم يکي از پارامترهاي متناظر هم نوع نباشند.
221
مثال 17-5 چندشکلي تابع max()
int max(int, int); int max(int, int, int); int max(double, double); int main() { cout << max(99,77) << " " << max(55,66,33) << " " << max(44.4,88.8); }
222
int max(int x, int y) { // returns the maximum of the two given integers: return (x > y ? x : y); } int max(int x, int y, int z) { // returns the maximum of the three given integers: int m = (x > y ? x : y); // m = max(x , y) return ( z > m ? z : m); int max(double x, double y) { // return the maximum of the two given doubles: return (x>y ? x : y);
223
در اين برنامه سه تابع با نام max() تعريف شده است.
وقتي تابع max() در جايي از برنامه فراخواني ميشود، کامپايلر فهرست آرگومان آن را بررسي ميکند تا بفهمد که کدام نسخه از max بايد احضار شود. مثلا در اولين فراخواني تابع max() دو آرگومان int ارسال شده، پس نسخهاي که دو پارامتر int در فهرست پارامترهايش دارد فراخواني ميشود. اگر اين نسخه وجود نداشته باشد، کامپايلر intها را به double ارتقا ميدهد و سپس نسخهاي که دو پارامتر double دارد را فرا ميخواند.
224
14- تابع main() برنامههايي که تا کنون نوشتيم همه داراي تابعي به نام main() هستند. منطق C++ اين طور است که هر برنامه بايد داراي تابعي به نام main() باشد. در حقيقت هر برنامه کامل، از يک تابع main() به همراه توابع ديگر تشکيل شده است که هر يک از اين توابع به شکل مستقيم يا غير مستقيم از درون تابع main() فراخواني ميشوند.
225
خود برنامه با فراخواني تابع main() شروع ميشود.
چون اين تابع يک نوع بازگشتي int دارد، منطقي است که بلوک تابع main() شامل دستور return 0; باشد هرچند که در برخي از کامپايلرهاي C++ اين خط اجباري نيست و ميتوان آن را ذکر نکرد. مقدار صحيحي که با دستور return به سيستم عامل برميگردد بايد تعداد خطاها را شمارش کند. مقدار پيشفرض آن 0 است به اين معنا که برنامه بدون خطا پايان گرفته است. با استفاده از دستور return ميتوانيم برنامه را به طور غيرمعمول خاتمه دهيم.
226
مثال 18-5 استفاده از دستور return براي پايان دادن به يك برنامه
int main() { // prints the quotient of two input integers: int n, d; cout << "Enter two integers: "; cin >> n >> d; if (d = = 0) return 0; cout << n << "/" << d << " = " << n/d << endl; } دستور return تابع فعلي را خاتمه ميدهد و کنترل را به فراخواننده بازميگرداند. به همين دليل است که اجراي دستور return در تابع main() کل برنامه را خاتمه ميدهد. Enter two integers: 99 17 99/17 = 5
227
1 - استفاده از دستور return 2 - فراخواني تابع exit()
چهار روش وجود دارد که بتوانيم برنامه را به شکل غيرمعمول (يعني قبل از اين که اجرا به پايان بلوک اصلي برسد) خاتمه دهيم: 1 - استفاده از دستور return 2 - فراخواني تابع exit() 3 - فراخواني تابع abort() 4 – ايجاد يک حالت استثنا اين تابع در سرفايل <cstdlib> تعريف شده است. تابع exit() براي خاتمه دادن به کل برنامه در هر تابعي غير از تابع main() مفيد است.
228
مثال 19-5 استفاده از تابع exit() براي پايان دادن به برنامه
#include <cstdlib> // defines the exit() function #include <iostream> // defines thi cin and cout objects using namespace std; double reciprocal(double x); int main() { double x; cin >> x; cout << reciprocal(x); } double reciprocal(double x)1 – Exception { // returns the reciprocal of x: if (x = = 0) exit(1); // terminate the program return 1.0/x; } دراين برنامۀ اگر كاربر عدد 0 را وارد کند، تابع reciprocal() خاتمه مييابد و برنامه بدون هيچ مقدار چاپي به پايان ميرسد
229
15- آرگومانهاي پيشفرض
در C++ ميتوان تعداد آرگومانهاي يک تابع را در زمان اجرا به دلخواه تغيير داد. اين امر با استفاده از آرگومانهاي اختياري و مقادير پيشفرض امکانپذير است. براي اين که به يک پارامتر مقدار پيشفرض بدهيم بايد آن مقدار را در فهرست پارامترهاي تابع و جلوي پارامتر مربوطه به همراه علامت مساوي درج کنيم. به اين ترتيب اگر هنگام فراخواني تابع، آن آرگومان را ذکر نکنيم، مقدار پيشفرض آن در محاسبات تابع استفاده ميشود. به همين خاطر به اين گونه آرگومانها، آرگومان اختياري ميگويند.
230
مثال 20-5 آرگومانهاي پيشفرض
double p(double, double, double=0, double=0, double=0); int main() { // tests the p() function: double x = ; cout << "p(x,7) = " << p(x,7) << endl; cout << "p(x,7,6) = " << p(x,7,6) << endl; cout << "p(x,7,6,5) = " << p(x,7,6,5) << endl; cout << "p(x,7,6,5,4) = " << p(x,7,6,5,4) << endl; } double p(double x, double a0, double a1=0, double a2=0, double a3=0) { // returns a0 + a1*x + a2*x^2 + a3*x^3: return a0 + (a1 + (a2 + a3*x)*x)*x; مثال 20-5 آرگومانهاي پيشفرض برنامۀ زير حاصل چند جملهاي درجه سوم را پيدا ميکند. براي محاسبۀ اين مقدار از الگوريتم هورنر استفاده شده. به اين شکل که براي کارايي بيشتر، محاسبه به صورت دستهبندي ميشود: p(x,7) = 7 p(x,7,6) = p(x,7,6,5) = – Default p(x,7,6,5,4) =
231
دقت کنيد که پارامترهايي که مقدار پيشفرض دارند بايد در فهرست پارامترهاي تابع بعد از همۀ پارامترهاي اجباري قيد شوند مثل: void f( int a, int b, int c=4, int d=7, int e=3); // OK void g(int a, int b=2, int c=4, int d, int e=3); // ERROR همچنين هنگام فراخواني تابع، آرگومانهاي ذکر شده به ترتيب از چپ به راست تخصيص مييابند و پارامترهاي بعدي با مقدار پيشفرض پر ميشوند. مثلا در تابع p() که در بالا قيد شد، فراخواني p(8.0,7,6) باعث ميشود که پارامتر x مقدار 8.0 را بگيرد سپس پارامتر a0 مقدار 7 را بگيرد و سپس پارامتر a1 مقدار 6 را بگيرد. پارامترهاي a2 و a3 مقدار پيشفرضشان را خواهند داشت. اين ترتيب را نميتوانيم به هم بزنيم. مثلا نميتوانيم تابع را طوري فرا بخوانيم که پارامترهاي x و a0 و a3 مستقيما مقدار بگيرند ولي پارامترهاي a1 و a2 مقدار پيشفرضشان را داشته باشند.
232
پايان جلسه پنجم
233
جلسه ششم «آرايهها»
234
آنچه در اين جلسه مي خوانيد:
1- پردازش آرايهها 2- مقداردهي آرايهها 3- ايندكس بيرون از حدود آرايه 4- ارسال آرايه به تابع 5- الگوريتم جستجوي خطي 6- مرتبسازي حبابي 7- الگوريتم جستجوي دودويي ›››
235
8- استفاده از انواع شمارشي در آرايه
9- تعريف انواع 10 -آرايههاي چند بعدي
236
شناخت و معرفي آرايهها و مزيت و طريقۀ بهکارگيري آنها
هدف کلي: شناخت و معرفي آرايهها و مزيت و طريقۀ بهکارگيري آنها هدفهاي رفتاري: انتظار ميرود پس از پايان اين جلسه بتوانيد: - علت استفاده از آرايهها را بدانيد و بتوانيد آنها را در برنامهها به کار ببريد. - آرايههاي «يکبعدي» و «چندبعدي» را تعريف کنيد. - مفهوم «ايندکس» را بدانيد و خطاي «اثر همسايگي» را تعريف و شناسايي کنيد. - طريقۀ ارسال آرايه به توابع را بدانيد. - «جستجوي خطي» و «جستجوي دودويي» را به اختصار شرح دهيد. - «مرتبسازي حبابي» را به اختصار شرح دهيد.
237
مقدمه: در برنامههايي که دادههاي فراواني را پردازش ميکنند استفاده از متغيرهاي معمولي کار عاقلانهاي نيست زيرا در بسياري از اين برنامهها «پردازش دستهاي» صورت ميگيرد به اين معني که مجموعهاي از دادههاي مرتبط با هم در حافظه قرار داده ميشود و پس از پردازش، کل اين مجموعه از حافظه خارج ميشود و مجموعۀ بعدي در حافظه بارگذاري ميشود. اگر قرار باشد براي اين کار از متغيرهاي معمولي استفاده شود بيشتر وقت برنامهنويس صرف پر و خالي کردن انبوهي از متغيرها ميشود. به همين دليل در بيشتر زبانهاي برنامهنويسي «آرايهها» تدارک ديده شدهاند. آرايه را ميتوان متغيري تصور کرد که يک نام دارد ولي چندين مقدار را به طور همزمان نگهداري مينمايد.
238
يک آرايه، يك زنجيره از متغيرهايي است كه همه از يك نوع هستند.
به اين متغيرها «اعضاي آرايه» ميگويند. هر عضو آرايه با يک شماره مشخص ميشود که به اين شماره «ايندکس» يا «زيرنويس» ميگويند عناصر يک آرايه در خانههاي پشت سر هم در حافظه ذخيره ميشوند. به اين ترتيب آرايه را ميتوان بخشي از حافظه تصور کرد که اين بخش خود به قسمتهاي مساوي تقسيم شده و هر قسمت به يک عنصر تعلق دارد.
239
شکل مقابل آرايۀ a که پنج عنصر دارد را نشان ميدهد.
عنصر a[0] حاوي مقدار 17.5 و عنصر a[1] حاوي 19.0 و عنصر a[4] حاوي مقدار 18.0 است. 17.50 1 19.00 2 16.75 3 15.00 4 18.00
240
2- پردازش آرايهها مثال 1-6 دستيابي مستقيم به عناصر آرايه برنامۀ سادۀ زير يک آرايۀ سه عنصري را تعريف ميکند و سپس مقاديري را در آن قرار داده و سرانجام اين مقادير را چاپ ميکند: int main() { int a[3]; a[2] = 55; a[0] = 11; a[1] = 33; cout << "a[0] = " << a[0] << endl; cout << "a[1] = " << a[1] << andl; cout << "a[2] = " << a[2] << endl; } آرايهها را ميتوان مثل متغيرهاي معمولي تعريف و استفاده کرد. با اين تفاوت که آرايه يک متغير مرکب است و براي دستيابي به هر يک از خانههاي آن بايد از ايندکس استفاده نمود. a[0] = 11 a[1] = 33 a[2] = 55
241
عبارت type نوع عناصر آرايه را مشخص ميکند. array_name نام آرايه است .
نحو کلي براي اعلان آرايه به شکل زير است: type array_name[array_size]; عبارت type نوع عناصر آرايه را مشخص ميکند. array_name نام آرايه است . array_size تعداد عناصر آرايه را نشان ميدهد. اين مقدار بايد يک عدد ثابت صحيح باشد و حتما بايد داخل کروشه [] قرار بگيرد.
242
3- مقداردهي آرايهها float a[] = {22.2,44.4,66.6};
در C++ ميتوانيم يک آرايه را با استفاده از فهرست مقداردهي، اعلان و مقدارگذاري کنيم: float a[] = {22.2,44.4,66.6}; به اين ترتيب مقادير داخل فهرست به همان ترتيبي که چيده شدهاند درون عناصر آرايه قرار ميگيرند. اندازه آرايه نيز برابر با تعداد عناصر موجود در فهرست خواهد بود. پس همين خط مختصر، آرايهاي از نوع float و با نام a و با تعداد سه عنصر اعلان کرده و هر سه عنصر را با مقدارهاي درون فهرست، مقداردهي ميکند. a 22.2 1 44.4 2 66.6
243
int size = sizeof(a)/sizeof(float); for (int i=0; i<size; i++)
مثال 3-6 مقداردهي آرايه با استفاده از فهرست مقداردهي برنامۀ زير، آرايۀ a را مقداردهي کرده و سپس مقدار هر عنصر را چاپ ميكند: int main() { float a[] = { 22.2, 44.4, 66.6 }; int size = sizeof(a)/sizeof(float); for (int i=0; i<size; i++) cout << "\ta[" << i << "] = " << a[i] << endl; } a[0] = 22.2 a[1] = 44.4 a[2] = 66.6
244
هنگام استفاده از فهرست مقداردهي براي اعلان آرايه، ميتوانيم تعداد عناصر آرايه را هم به طور صريح ذکر کنيم. در اين صورت اگر تعداد عناصر ذکر شده از تعداد عناصر موجود در فهرست مقداردهي بيشتر باشد، خانههاي بعدي با مقدار صفر پر ميشوند: float a[7] = { 55.5, 66.6, 77.7 }; a 55.5 1 66.6 2 77.7 3 0.0 4 5 6 دقت کنيد که تعداد مقادير موجود در فهرست مقداردهي نبايد از تعداد عناصر آرايه بيشتر باشد: float a[3] = { 22.2, 44.4, 66.6, 88.8 }; // ERROR: too many values!
245
يك آرايه را ميتوانيم به طور کامل با صفر مقداردهي اوليه کنيم
يك آرايه را ميتوانيم به طور کامل با صفر مقداردهي اوليه کنيم. براي مثال سه اعلان زير با هم برابرند: float a[ ] = { 0, 0, 0, 0, 0, 0, 0, 0, 0 }; float a[9] = { 0, 0 }; float a[9] = { 0, 0, 0, 0, 0, 0, 0, 0, 0 }; اما مطلب فوق اصلا به اين معني نيست که از فهرست مقداردهي استفاده نشود. درست مثل يک متغير معمولي، اگر يک آرايه مقداردهي اوليه نشود، عناصر آن حاوي مقادير زباله خواهد بود.
246
مثال 5-6 يك آرايۀ مقداردهي نشده
برنامۀ زير، آرايۀ a را اعلان ميکند ولي مقداردهي نميكند. با وجود اين، مقادير موجود در آن را چاپ ميكند: int main() { const int SIZE=4; // defines the size N for 4 elements float a[SIZE]; // declares the array's elements as float for (int i=0; i<SIZE; i++) cout << "\ta[" << i << "] = " << a[i] << endl; } a[0] = e-39 a[1] = e-39 a[2] = e-39 a[3] = 0
247
b = a; // ERROR: arrays cannot be assigned!
آرايهها را ميتوان با استفاده از عملگر جايگزيني مقداردهي کرد اما نميتوان مقدار آنها را به يکديگر تخصيص داد: float a[7] = { 22.2, 44.4, 66.6 }; float b[7] = { 33.3, 55.5, 77.7 }; b = a; // ERROR: arrays cannot be assigned! همچنين نميتوانيم يك آرايه را به طور مستقيم براي مقداردهي به آرايۀ ديگر استفاده كنيم: float b[7] = a; // ERROR: arrays cannot be used as nitializers!
248
4- ايندكس بيرون از حدود آرايه
در بعضي از زبانهاي برنامهنويسي، ايندکس آرايه نميتواند از محدودۀ تعريف شده براي آن بيشتر باشد. براي مثال در پاسکال اگر آرايۀ a با تعداد پنج عنصر تعريف شده باشد و آنگاه a[7] دستيابي شود، برنامه از کار ميافتد. اين سيستم حفاظتي در C++ وجود ندارد. مثال بعدي نشان ميدهد که ايندکس يک آرايه هنگام دستيابي ميتواند بيشتر از عناصر تعريف شده براي آن باشد و باز هم بدون اين که خطايي گرفته شود، برنامه ادامه يابد.
249
for (int i=0; i<7; i++) //ERROR: index is out of bounds!
مثال 6-6 تجاوز ايندکس آرايه از محدودۀ تعريف شده براي آن برنامۀ زير يک خطاي زمان اجرا دارد؛ به بخشي از حافظه دستيابي ميکند که از محدودۀ آرايه بيرون است: in main() { const int SIZE=4; float a[SIZE} = { 33.3, 44.4, 55.5, 66.6 }; for (int i=0; i<7; i++) //ERROR: index is out of bounds! cout << "\ta[" << i << "] = " << a[i] << endl; } آرايهاي که در اين برنامه تعريف شده، چهار عنصر دارد ولي تلاش ميشود به هفت عنصر دستيابي شود. سه مقدار آخر واقعا جزو آرايه نيستند و فقط سلولهايي از حافظهاند که دقيقا بعد از عنصر چهارم آرايه قرار گرفتهاند. اين سلولها داراي مقدار زباله هستند. a[0] = 33.3 a[1] = 44.4 a[2] = 55.5 a[3] = 66.6 a[4] = e-45 a[5] = e-39 a[6] = e-39
250
cout << "x = " << x << endl;
* مثال 7-6 اثر همسايگي برنامۀ زير از ايندکس خارج از محدوده استفاده ميکند و اين باعث ميشود که مقدار يک متغير به طور ناخواسته تغيير کند: int main() { const int SIZE=4; float a[] = { 22.2, 44.4, 66.6 }; float x=11.1; cout << "x = " << x << endl; a[3] = 88.8; // ERROR: index is out of bounds! } x = 88.8
251
متغير x بعد از آرايۀ a اعلان شده، پس يک سلول چهاربايتي بلافاصله بعد از دوازده بايت آرايه به آن تخصيص مييابد. بنابراين وقتي برنامه تلاش ميکند مقدار 88.8 را در a[3] قرار دهد (که جزو آرايه نيست) اين مقدار به شکل ناخواسته در x قرار ميگيرد. شکل مقابل نشان ميدهد چطور اين اتفاق در حافظه رخ ميدهد. مثال بعدي نوع ديگري از خطاي زمان اجرا را نشان ميدهد: وقتي ايندکس آرايه بيش از حد بزرگ باشد. اين خطا يکي از وحشتناکترين خطاهاي زمان اجراست زيرا ممکن است اصلا نتوانيم منبع خطا را کشف کنيم. حتي ممکن است به اين روش دادههاي برنامههاي ديگري که در حال کارند را خراب کنيم و اين باعث ايجاد اختلال در کل سيستم شود. به اين خطا «اثر همسايگي» ميگويند. اين وظيفۀ برنامهنويس است که تضمين کند ايندکس آرايه هيچگاه از محدودۀ آن خارج نشود. a x 88.8 22.2 44.4 66.6 88.8
252
cout << "x = " << x << endl;
مثال 8-6 ايجاد استثناي مديريت نشده برنامۀ زير از كار ميافتد زيرا ايندكس آرايه خيلي بزرگ است: int main() { const int SIZE=4; float a[] = { 22.2, 44.4, 66.6 }; float x=11.1; cout << "x = " << x << endl; a[3333] =88.8;//ERROR: index is out of bounds! }
253
وقتي اين برنامه روي رايانهاي با سيستم عامل ويندوز اجرا شود، يک صفحۀ هشدار که در شکل نشان داده شده روي صفحه ظاهر ميشود. اين پنجره بيان ميکند که برنامه تلاش دارد به نشاني e از حافظه دستيابي کند. اين مکان خارج از حافظۀ تخصيصي است که براي اين برنامه منظور شده، بنابراين سيستم عامل برنامه را متوقف ميکند.
254
پردازشگر استثنا خطايي که در مثال 8-6 بيان شده يک «استثناي مديريت نشده» ناميده ميشود زيرا کدي وجود ندارد که به اين استثنا پاسخ دهد. در C++ ميتوانيم کدهايي به برنامه اضافه کنيم که هنگام رخ دادن حالتهاي استثنا، از توقف برنامه جلوگيري کند. به اين کدها «پردازشگر استثنا» ميگويند.
255
5- ارسال آرايه به تابع كد float a[]; كه آرايه a را اعلان ميكند دو چيز را به كامپايلر ميگويد: 1- اين که نام آرايه a است 2- عناصر آرايه از نوع float هستند. سمبل a نشاني حافظۀ آرايه را ذخيره ميکند. لازم نيست تعداد عناصر آرايه به کامپايلر گفته شود زيرا از روي نشاني موجود در a ميتوان عناصر را بازيابي نمود. به همين طريق ميتوان يک آرايه را به تابع ارسال کرد. يعني فقط نوع آرايه و نشاني حافظۀ آن به عنوان پارامتر به تابع فرستاده ميشود.
256
مثال 9-6 ارسال آرايه به تابعي كه مجموع عناصر آرايه را برميگرداند
int sum(int[],int); int main() { int a[] = { 11, 33, 55, 77 }; int size = sizeof(a)/sizeof(int); cout << "sum(a,size) = " << sum(a,size) << endl;} int sum(int a[], int n) { int sum=0; for (int i=0; i<n; i++) sum += a[i]; return sum; } فهرست پارامتر تابع فوق به شکل (int a[], int n) است به اين معنا که اين تابع يک آرايه از نوع int و يک متغير از نوع int دريافت ميکند. به اعلان اين تابع در بالاي تابع main() نگاه کنيد. نام پارامترها حذف شده است.
257
هنگام فراخواني تابع نيز از عبارت sum(a,size) استفاده شده که فقط نام آرايه به تابع ارسال شده.
تابع از اين نشاني براي دستيابي به عناصر آرايه استفاده ميکند. همچنين تابع ميتواند با استفاده از اين نشاني، محتويات عناصر آرايه را دستکاري کند. پس ارسال آرايه به تابع شبيه ارسال متغير به طريق ارجاع است. به مثال بعدي دقت کنيد.
258
مثال 10-6 توابع ورودي و خروجي براي يک آرايه در اين برنامه از تابع read() استفاده ميشود تا مقاديري به داخل آرايه وارد شود. سپس با استفاده از تابع print() مقادير داخل آرايه چاپ ميشوند: void read(int[],int&;) void print(int[],int); int main() { const int MAXSIZE=100; int a[MAXSIZE]={0}, size; read(a,size); cout << "The array has " << size << " elements: "; print(a,size); } Enter integers. Terminate with 0: a[0]: 11 a[1]: 22 a[2]: 33 a[3]: 44 a[4]: 0 The array has 4 elements:
259
void read(int a[], int& n)
{ cout << "Enter integers. Terminate with 0:\n"; n = 0; do { cout << "a[" << n << "]: "; cin >> a[n]; { while (a[n++] !=0 && n < MAXSIZE); --n; // don't count the 0 }
260
void print(int a[], int n)
{ for (int i=0; i<n; i++) cout << a[i] << " "; } چون n يك متغير است، براي اين که تابع read() بتواند مقدار آن را تغيير دهد اين متغير بايد به شکل ارجاع ارسال شود. همچنين براي اين که تابع مذکور بتواند مقادير داخل آرايه a را تغيير دهد، آرايه نيز بايد به طريق ارجاع ارسال شود، اما ارجاع آرايهها کمي متفاوت است.
261
1 – آدرس اولين خانۀ آرايه 2 – تعداد عناصر آرايه 3 – نوع عناصر آرايه
در C++ توابع قادر نيستند تعداد عناصر آرايۀ ارسالي را تشخيص دهند. بنابراين به منظور ارسال آرايهها به تابع از سه مشخصه استفاده ميشود: 1 – آدرس اولين خانۀ آرايه 2 – تعداد عناصر آرايه 3 – نوع عناصر آرايه تابع با استفاده از اين سه عنصر ميتواند به تک تک اعضاي آرايه دستيابي کند.
262
آدرس اولين خانۀ آرايه، همان نام آرايه است.
پس وقتي نام آرايه را به تابع بفرستيم آدرس اولين خانه را به تابع فرستادهايم. نوع آرايه نيز در تعريف تابع اعلان ميشود. بنابراين با اين دو مقدار، تابع ميتواند به آرايه دسترسي داشته باشد.
263
مثال 11-6 آدرس اولين خانۀ آرايه و مقدار درون آن
برنامۀ زير، آدرس ذخيره شده در نام آرايه و مقدار موجود در آن خانه را چاپ ميکند: int main() { int a[] = { 22, 44, 66, 88 }; cout << "a = " << a << endl; // the address of a[0] cout << "a[0] = " << a[0]; // the value of a[0] } a = 0x0064fdec a[0] = 22 اين برنامه تلاش ميکند که به طور مستقيم مقدار a را چاپ کند. نتيجۀ چاپ a اين است که يک آدرس به شکل شانزده دهي چاپ ميشود. اين همان آدرس اولين خانۀ آرايه است. يعني درون نام a آدرس اولين عنصر آرايه قرار گرفته. خروجي نيز نشان ميدهد که a آدرس اولين عنصر را دارد و a[0] مقدار اولين عنصر را.
264
6- الگوريتم جستجوي خطي آرايهها بيشتر براي پردازش يک زنجيره از دادهها به کار ميروند. اغلب لازم است که بررسي شود آيا يک مقدار خاص درون يک آرايه موجود است يا خير. سادهترين راه اين است که از اولين عنصر آرايه شروع کنيم و يکي يکي همۀ عناصر آرايه را جستجو نماييم تا بفهميم که مقدار مورد نظر در کدام عنصر قرار گرفته. به اين روش «جستجوي خطي» ميگويند.
265
مثال 12-6 جستجوي خطي برنامۀ زير تابعي را آزمايش ميکند که در اين تابع از روش جستجوي خطي براي يافتن يک مقدار خاص استفاده شده: int index(int,int[],int); int main() { int a[] = { 22, 44, 66, 88, 44, 66, 55}; cout << "index(44,a,7) = " << index(44,a,7) << endl; cout << "index(50,a,7) = " << index(50,a,7) << endl; } int index(int x, int a[], int n) { for (int i=0; i<n; i++) if (a[i] == x) return i; return n; // x not found index(44,a,7) = 1 index(40,a,7) = 7
266
تابع index() سه پارامتر دارد:
پارامتر a آرايهاي است که بايد در آن جستجو صورت گيرد و پارامتر n هم ايندکس عنصري است که مقدار مورد نظر در آن پيدا شده است. در اين تابع با استفاده از حلقۀ for عناصر آرايه a پيمايش شده و مقدار هر عنصر با x مقايسه ميشود. اگر اين مقدار با x برابر باشد، ايندکس آن عنصر بازگردانده شده و تابع خاتمه مييابد.
267
اگر مقدار x در هيچ يک از عناصر آرايه موجود نباشد، مقداري خارج از ايندکس آرايه بازگردانده ميشود که به اين معناست که مقدار x در آرايۀ a موجود نيست. در اولين اجراي آزمايشي، مشخص شده که مقدار 44 در a[1] واقع است و در اجراي آزمايشي دوم مشخص شده که مقدار 40 در آرايۀ a موجود نيست (يعني مقدار 44 در a[7] واقع است و از آنجا که آرايۀ a فقط تا a[6] عنصر دارد، مقدار 7 نشان ميدهد که 40 در آرايه موجود نيست).
268
7- مرتبسازي حبابي «مرتبسازي حبابي» يکي از سادهترين الگوريتمهاي مرتبسازي است. در اين روش، آرايه چندين مرتبه پويش ميشود و در هر مرتبه بزرگترين عنصر موجود به سمت بالا هدايت ميشود و سپس محدودۀ مرتبسازي براي مرتبۀ بعدي يکي کاسته ميشود. در پايان همۀ پويشها، آرايه مرتب شده است.
269
اولين عنصر آرايه با عنصر دوم مقايسه ميشود.
طريقۀ يافتن بزرگترين عنصر و انتقال آن به بالاي عناصر ديگر به اين شکل است اولين عنصر آرايه با عنصر دوم مقايسه ميشود. اگر عنصر اول بزرگتر بود، جاي اين دو با هم عوض ميشود. سپس عنصر دوم با عنصر سوم مقايسه ميشود. اگر عنصر دوم بزرگتر بود، جاي اين دو با هم عوض ميشود و به همين ترتيب مقايسه و جابجايي زوجهاي همسايه ادامه مييابد تا وقتي به انتهاي آرايه رسيديم، بزرگترين عضو آرايه در خانۀ انتهايي قرار خواهد گرفت. در اين حالت محدودۀ جستجو يکي کاسته ميشود و دوباره زوجهاي کناري يکي يکي مقايسه ميشوند تا عدد بزرگتر بعدي به مکان بالاي محدوده منتقل شود. اين پويش ادامه مييابد تا اين که وقتي محدوده جستجو به عنصر اول محدود شد، آرايه مرتب شده است.
270
مثال 13-6 مرتبسازي برنامۀ زير تابعي را آزمايش ميکند که اين تابع با استفاده از مرتبسازي حبابي يک آرايه را مرتب مينمايد: void print(float[],int); void sort(float[],int); int main() {float a[]={55.5,22.2,99.9,66.6,44.4,88.8,33.3, 77.7}; print(a,8); sort(a,8); } 55.5, 22.2, 99.9, 66.6, 44.4, 88.8, 33.3, 77.7 22.2, 33.3, 44.4, 55.5, 66.6, 77.7, 88.8, 99.9
271
void sort(float a[], int n)
{ // bubble sort: for (int i=1; i<n; i++) // bubble up max{a[0..n-i]}: for (int j=0; j<n-i; j++) if (a[j] > a[j+1]) swap (a[j],a[j+1]); //INVARIANT: a[n-1-i..n-1] is sorted }
272
تابع sort() از دو حلقۀ تودرتو استفاده ميكند.
1- حلقه for داخلي زوجهاي همسايه را با هم مقايسه ميكند و اگر آنها خارج از ترتيب باشند، جاي آن دو را با هم عوض ميکند. وقتي for داخلي به پايان رسيد، بزرگترين عنصر موجود در محدودۀ فعلي به انتهاي آن هدايت شده است. 2-سپس حلقۀ for بيروني محدودۀ جستجو را يکي کم ميکند و دوباره for داخلي را راه مياندازد تا بزرگترين عنصر بعدي به سمت بالاي آرايه هدايت شود.
273
8- الگوريتم جستجوي دودويي
در روش جستجوي دودويي به يک آرايۀ مرتب نياز است. هنگام جستجو آرايه از وسط به دو بخش بالايي و پاييني تقسيم ميشود. مقدار مورد جستجو با آخرين عنصر بخش پاييني مقايسه ميشود. اگر اين عنصر کوچکتر از مقدار جستجو بود، مورد جستجو در بخش پاييني وجود ندارد و بايد در بخش بالايي به دنبال آن گشت.
274
دوباره بخش بالايي به دو بخش تقسيم ميگردد و گامهاي بالا تکرار ميشود.
سرانجام محدودۀ جستجو به يک عنصر محدود ميشود که يا آن عنصر با مورد جستجو برابر است و عنصر مذکور يافت شده و يا اين که آن عنصر با مورد جستجو برابر نيست و لذا مورد جستجو در آرايه وجود ندارد. اين روش پيچيدهتر از روش جستجوي خطي است اما در عوض بسيار سريعتر به جواب ميرسيم.
275
int index(int, int[],int); int main()
مثال 14-6 جستجوي دودويي برنامۀ آزمون زير با برنامۀ آزمون مثال 12-6 يکي است اما تابعي که در زير آمده از روش جستجوي دودويي براي يافتن مقدار درون آرايه استفاده ميکند: int index(int, int[],int); int main() { int a[] = { 22, 33, 44, 55, 66, 77, 88 }; cout << "index(44,a,7) = " << index(44,a,7) << endl; cout << "index(60,a,7) = " << index(60,a,7) << endl; }
276
int index(int x, int a[], int n)
{ // PRECONDITION: a[0] <= a[1] <= ... <= a[n-1]; // binary search: int lo=0, hi=n-1, i; while (lo <= hi) { i = (lo + hi)/2; // the average of lo and hi if (a[i] == x) return i; if (a[i] < x) lo = i+1; // continue search in a[i+1..hi] else hi = i-1; // continue search in a[0..i-1] } return n; // x was not found in a[0..n-1] index(44,a,7) = 2 index(60,a,7) = 7
277
براي اين که بفهميم تابع چطور کار ميکند، فراخواني index(44,a,7) را دنبال ميکنيم.
وقتي حلقه شروع ميشود، x=44 و n=7 و lo=0 و hi=6 است. ابتدا i مقدار (0+6)/2 = 3 را ميگيرد.پس عنصر a[i] عنصر وسط آرايۀ a[0..6] است. مقدار a[3] برابر با 55 است که از مقدار x بزرگتر است. پس x در نيمۀ بالايي نيست و جستجو در نيمۀ پاييني ادامه مييابد. لذا hi با i-1 يعني 2 مقداردهي ميشود و حلقه تکرار ميگردد.
278
حالا hi=2 و lo=0 است و دوباره عنصر وسط آرايۀ a[0
حالا hi=2 و lo=0 است و دوباره عنصر وسط آرايۀ a[0..2] يعني a[1] با x مقايسه ميشود. a[1] برابر با 33 است که کوچکتر از x ميباشد. پس اين دفعه lo برابر با i+1 يعني 2 ميشود. در سومين دور حلقه، hi=2 و lo=2 است. پس عنصر وسط آرايۀ a[2..2] که همان a[2] است با x مقايسه ميشود. a[2] برابر با 44 است که با x برابر است. پس مقدار 2 بازگشت داده ميشود؛ يعني x مورد نظر در a[2] وجود دارد.
279
lo hi i a[i] ?? x 6 3 55 > 44 2 1 33 < ==
280
حال فراخواني index(60,a,7) را دنبال ميکنيم
حال فراخواني index(60,a,7) را دنبال ميکنيم. وقتي حلقه شروع ميشود، x=60 و n=7 و lo=0 و hi=6 است. عنصر وسط آرايۀ a[0..6] عنصر a[3]=55 است که از x کوچکتر است. پس lo برابر با i+1=4 ميشود و حلقه دوباره تکرار ميشود. اين دفعه hi=6 و lo=4 است . عنصر وسط آرايۀ a[4..6] عنصر a[5]=77 است که بزرگتر از x ميباشد. پس hi به i-1=4 تغيير مييابد و دوباره حلقه تکرار ميشود. اين بار hi=4 و lo=4 است و عنصر وسط آرايۀ a[4..4] عنصر a[4]=66 است که بزرگتر از x ميباشد. لذا hi به i-1=3 کاهش مييابد.
281
lo hi i a[i] ?? x 6 3 55 < 60 4 5 77 > 66 اکنون شرط حلقه غلط ميشود زيرا hi<lo است. بنابراين تابع مقدار 7 را برميگرداند يعني عنصر مورد نظر در آرايه موجود نيست.
282
در تابع فوق هر بار که حلقه تکرار ميشود، محدودۀ جستجو 50% کوچکتر ميشود. در آرايۀ n عنصري، روش جستجوي دودويي حداکثر به مقايسه نياز دارد تا به پاسخ برسد. حال آن که در روش جستجوي خطي به n مقايسه نياز است.
283
تفاوتهاي جستجوي دودويي و خطي
جستجوي دودويي سريعتر از جستجوي خطي است. دومين تفاوت در اين است که اگر چند عنصر داراي مقادير يکساني باشند، آنگاه جستجوي خطي هميشه کوچکترين ايندکس را برميگرداند ولي در مورد جستجوي دودويي نميتوان گفت که کدام ايندکس بازگردانده ميشود. سومين فرق در اين است که جستجوي دودويي فقط روي آرايههاي مرتب کارايي دارد و اگر آرايهاي مرتب نباشد، جستجوي دودويي پاسخ غلط ميدهد ولي جستجوي خطي هميشه پاسخ صحيح خواهد داد.
284
int main() { int a[] = { 22, 44, 66, 88, 44, 66, 55 }; }
* مثال 15-6 مشخص كردن اين كه آيا آرايه مرتب است يا خير برنامۀ زير يک تابع بولي را آزمايش ميکند. اين تابع مشخص مينمايد که آيا آرايۀ داده شده غير نزولي است يا خير: bool isNondecreasing(int a[], int n); int main() { int a[] = { 22, 44, 66, 88, 44, 66, 55 }; cout<<"isNondecreasing(a,4) = " << isNondecreasing(a,4)<< endl; cout<<"isNondecreasing(a,7) = " << isNondecreasing(a,7) << endl; }
285
bool isNondecreasing(int a[], int n)
{ // returns true iff a[0] <= a[1] <= ... <= a[n-1]: for (int i=1; i<n; i++) if (a[i]<a[i-1]) return false; return true; } isNondecreasing(a,4) = 1 isNondecreasing(a,7) = 0
286
اين تابع يک بار کل آرايه را پيمايش کرده و زوجهاي a[i-1] و a[i] را مقايسه ميکند.
اگر زوجي يافت شود که در آن a[i]<a[i-1] باشد، مقدار false را بر ميگرداند به اين معني که آرايه مرتب نيست. ببينيد که مقادير true و false به شکل اعداد 1 و 0 در خروجي چاپ ميشوند زيرا مقادير بولي در حقيقت به شکل اعداد صحيح در حافظه ذخيره ميشوند.
287
اگر پيششرط مثال 14-6 يعني مرتب بودن آرايه رعايت نشود، جستجوي دودويي پاسخ درستي نميدهد. به اين منظور ابتدا بايد اين پيششرط بررسي شود. با استفاده از تابع assert() ميتوان اجراي يک برنامه را به يک شرط وابسته کرد. اين تابع يک آرگومان بولي ميپذيرد. اگر مقدار آرگومان false باشد، برنامه را خاتمه داده و موضوع را به سيستم عامل گزارش ميکند. اگر مقدار آرگومان true باشد، برنامه بدون تغيير ادامه مييابد. تابع asset() در سرفايل <cassert> تعريف شده است.
288
مثال 16-6 استفاده از تابع assert() براي رعايت كردن يك پيششرط
برنامۀ زير نسخۀ بهبوديافتهاي از تابع search() مثال 14-6 را آزمايش ميکند. در اين نسخه، از تابع isNonDecreasing() مثال 15-6 استفاده شده تا مشخص شود آرايه مرتب است يا خير. نتيجه اين تابع به تابع assert() ارسال ميگردد تا اگر آرايه مرتب نباشد برنامه به بيراهه نرود.
289
int main() using namespace std; int index(int x, int a[], int n);
#include <cassert> // defines the assert() function #include <iostream> // defines the cout object using namespace std; int index(int x, int a[], int n); int main() { int a[] = { 22, 33, 44, 55, 66, 77, 88, 60 }; cout<<"index(44,a,7) = " << index(44,a,7) << endl; cout<<"index(44,a,8) = " << index(44,a,8) << endl; cout<<"index(60,a,8) = " << index(60,a,8) << endl; }
290
bool isNondecreasing(int a[], int n); int index(int x, int a[], int n)
{ assert(isNondecreasing(a,n)); int lo=0, hi=n-1, i; while (lo <= hi) { i = (lo + hi)/2; if (a[i] == x) return i; if (a[i] < x) lo = i+1; else hi = i-1; } return n; } index(44,a,7) = 2
291
آرايۀ a[] که در اين برنامه استفاده شده كاملا مرتب نيست اما هفت عنصر اول آن مرتب است. بنابراين در فراخوانيindex(44,a,7) تابع بولي مقدار true را به assert() ارسال ميکند و برنامه ادمه مييابد. اما در دومين فراخواني index(44,a,8) باعث ميشود که تابع isNondecreasing() مقدار false را به تابع assert() ارسال کند كه در اين صورت برنامه متوقف ميشود و ويندوز پنجرۀ هشدار مقابل را نمايش ميدهد.
292
با استفاده از انواع شمارشي نيز ميتوان آرايهها را پردازش نمود.
9- استفاده از انواع شمارشي در آرايه انواع شمارشي در جلسه دوم توضيح داده شدهاند. با استفاده از انواع شمارشي نيز ميتوان آرايهها را پردازش نمود. مثال 17-7 شمارش با استفاده از روزهاي هفته اين برنامه يك آرايه به نام high[] با هفت عنصرازنوعfloat تعريف ميكند كه هر عنصر حداکثر دما در يک روز هفته را نشان ميدهد: int main() { enum Day { SUN, MON, TUE, WED, THU, FRI, SAT }; float high[SAT+1] = {28.6, 29.1, 29.9, 31.3, 30.4, 32.0, 30.7}; for (int day = SUN; day <= SAT; day++) cout << "The high temperature for day " << day << " was "<< high[day] << endl; } The high temperature for day 0 was 28.6 The high temperature for day 1 was 29.1 The high temperature for day 2 was 29.9 The high temperature for day 3 was 31.3 The high temperature for day 4 was 30.4 The high temperature for day 5 was 32.0 The high temperature for day 6 was 30.7
293
for (int day = SUN; day <= SAT; day++)
به خاطر بياوريد که انواع شمارشي به شکل مقادير عددي ذخيره ميشوند. اندازۀ آرايه، SAT+1 است زيرا SAT مقدار صحيح 6 را دارد و آرايه به هفت عنصر نيازمند است. متغير day از نوع int است پس ميتوان مقادير Day را به آن تخصيص داد. استفاده از انواع شمارشي در برخي از برنامهها باعث ميشود که کد برنامه «خود استناد» شود. مثلا در مثال 17-6 کنترل حلقه به شکل for (int day = SUN; day <= SAT; day++) باعث ميشود که هر بينندهاي حلقۀ for بالا را به خوبي درک کند.
294
10- تعريف انواع انواع شمارشي يكي از راههايي است که کاربر ميتواند نوع ساخت خودش را تعريف کند. براي مثال دستور زير : enum Color{ RED,ORANGE,YELLOW, GREEN, BLUE, VIOLET }; يک نوع جديد به نام Color تعريف ميکند که متغيرهايي از اين نوع ميتوانند مقادير RED يا ORANGE يا YELLOW يا GREEN يا BLUE يا VIOLET را داشته باشند. پس با استفاده از اين نوع ميتوان متغيرهايي به شکل زير تعريف نمود: Color shirt = BLUE; Color car[] = { GREEN, RED, BLUE, RED }; Floatwavelength[VIOLET+1]={420,480,530,570,600,620}; در اينجا shirt متغيري از نوع Color است و با مقدار BLUE مقداردهي شده. car يک آرايۀ چهار عنصري است و مقدار عناصر آن به ترتيب GREEN و RED و BLUE و RED ميباشد. همچنين wavelength آرايهاي از نوع float است که داراي VIOLET+1 عنصر يعني 5+1=6 عنصر است.
295
در C++ ميتوان نام انواع استاندارد را تغيير داد.
کلمۀ کليدي typedef يک نام مستعار براي يک نوع استاندارد موجود تعريف ميکند. نحو استفاده از آن به شکل زير است: typedef type alias; كه type يک نوع استاندارد و alias نام مستعار براي آن است.
296
typedef element-type alias[];
براي مثال کساني که با پاسکال برنامه مينويسند به جاي نوع long از عبارت Integer استفاده ميکنند و به جاي نوع double از عبارت Real استفاده مينمايند. اين افراد ميتوانند به شکل زير از نام مستعار استفاده کنند: typedef long Integer; typedef double Real; و پس از آن کدهاي زير معتبر خواهند بود: Integer n = 22; const Real PI = ; Integer frequency[64]; اگر دستور typedef را به شکل زير بکار ببريم ميتوانيم آرايهها را بدون علامت براکت تعريف کنيم: typedef element-type alias[]; مثل تعريف زير : typedef float sequence[]; سپس ميتوانيم آرايۀ a را به شکل زير اعلان کنيم: sequence a = {55.5, 22.2, 99.9};
297
دستور typedef نوع جديدي را اعلان نميکند، بلکه فقط به يک نوع موجود نام مستعاري را نسبت ميدهد.
برنامۀ زير همان برنامۀ مثال 13-6 است با اين فرق که از typedef استفاده شده تا بتوان از نام مستعار sequrnce به عنوان يک نوع استفاده کرد. سپس اين نوع در فهرست پارامترها و اعلان a در تابع main() به کار رفته است:
298
typedef float Sequence[];
void sort(Sequence,int); void print(Sequence,int); int main() { Sequence a = {55.5, 22.2, 99.9, 66.6, 44.4, 88.8, 33.3, 77.7}; print(a,8); sort(a,8); }
299
void sort(Sequence a, int n) { for (int i=n-1; i>0; i--)
for (int j=0; j<i; j++) if (a[j] > a[j+1]) swap(a[j],a[j+1]); } دوباره به دستور typedef نگاه کنيد: typedef float Seguence[]; علامت براكتها [] نشان ميدهند که هر چيزي که از نوع Sequence تعريف شود، يک آرايه است و عبارت float نيز بيان ميکند که اين آرايه از نوع float است.
300
يک آرايۀ سه بعدي آرايهاي است که هر خانه از آن يک آرايۀ دو بعدي باشد.
11- آرايههاي چند بعدي همۀ آرايههايي كه تاکنون تعريف کرديم، يک بعدي هستند، خطي هستند، رشتهاي هستند. ميتوانيم آرايهاي تعريف کنيم که از نوع آرايه باشد، يعني هر خانه از آن آرايه، خود يک آرايه باشد. به اين قبيل آرايهها، آرايههاي چندبعدي ميگوييم. يک آرايۀ دو بعدي آرايهاي است که هر خانه از آن، خود يک آرايۀ يک بعدي باشد. يک آرايۀ سه بعدي آرايهاي است که هر خانه از آن يک آرايۀ دو بعدي باشد.
301
مقدار 99 را در عنصري قرار ميدهد که ايندکس آن عنصر(1,2,3) است.
شکل دستيابي به عناصر در آرايههاي چند بعدي مانند آرايههاي يک بعدي است. مثلا دستور a[1][2][3] = 99; مقدار 99 را در عنصري قرار ميدهد که ايندکس آن عنصر(1,2,3) است. دستور int a[5]; آرايهاي با پنج عنصر از نوع int تعريف ميکند. اين يک آرايۀ يک بعدي است. دستور int a[3][5]; آرايهاي با سه عنصر تعريف ميکند که هر عنصر، خود يک آرايۀ پنج عنصري از نوع int است. اين يک آرايۀ دو بعدي است که در مجموع پانزده عضو دارد. دستور int a[2][3][5]; آرايهاي با دو عنصر تعريف ميکند که هر عنصر، سه آرايه است که هر آرايه پنج عضو از نوع int دارد. اين يک آرايۀ سه بعدي است که در مجموع سي عضو دارد. آرايههاي چند بعدي مثل آرايههاي يک بعدي به توابع فرستاده ميشوند با اين تفاوت که هنگام اعلان و تعريف تابع مربوطه، بايد تعداد عناصر بُعد دوم تا بُعد آخر حتما ذکر شود.
302
void print(int a[][5]); int main() { int a[3][5]; read(a); print(a); }
مثال 19-6 نوشتن و خواندن يك آرايۀ دو بعدي برنامۀ زير نشان ميدهد که يک آرايۀ دوبعدي چگونه پردازش ميشود: void read(int a[][5]); void print(int a[][5]); int main() { int a[3][5]; read(a); print(a); }
303
void read(int a[][5]) { cout << "Enter 15 integers, 5 per row:\n"; for (int i=0; i<3; i++) { cout << "ROW " << i << ": "; for (int j=0; j<5; j++) cin >> a[i][j]; }
304
void print(const int a[][5])
{ for (int i=0; i<3; i++) { for (int j=0; j<5; j++) cout << " " << a[i][j]; cout << endl; }
305
Enter 15 integers, 5 per row:
دقت کنيد که در فهرست پارامترهاي توابع بالا، بعد اول نامشخص است اما بعد دوم مشخص شده. علت هم اين است که آرايۀ دو بعدي a[][] در حقيقت آرايهاي يکبعدي از سه آرايۀ پنج عنصري است. کامپايلر نياز ندارد بداند که چه تعداد از اين آرايههاي پنج عنصري موجود است، اما بايد بداند که آنها پنج عنصري هستند.
306
void printQuizAverages(Score); void printClassAverages(Score);
وقتي يک آرايۀ چند بعدي به تابع ارسال ميشود، بُعد اول مشخص نيست اما همۀ ابعاد ديگر بايد مشخص باشند. مثال 20-6 پردازش يك آرايۀ دوبعدي از نمرات امتحاني const NUM_STUDENTS = 3; const NUM_QUIZZES = 5; typedef int Score[NUM_STUDENTS][NUM_QUIZZES]; void read(Score); void printQuizAverages(Score); void printClassAverages(Score);
307
int main() { Score score; cout << "Enter " << NUM_QUIZZES << " quiz scores for each student:\n"; read(score); cout << "The quiz averages are:\n"; printQuizAverages(score); cout << "The class averages are:\n"; printClassAverages(score);}
308
void read(Score score)
{ for (int s=0; s<NUM_STUDENTS; s++) { cout << "Student " << s << ": "; for(int q=0; q<NUM_QUIZZES; q++) cin >> score[s][q]; }
309
void printQuizAverages(Score score)
{ for (int s=0; s<NUM_STUDENTS; s++) { float sum = 0.0; for (int q=0; q<NUM_QUIZZES; q++) sum += score[s][q]; cout << "\tStudent " << s << ": " << sum/NUM_QUIZZES << endl; }}
310
void printClassAverages(Score score)
{ for (int q=0; q<NUM_QUIZZES; q++) { float sum = 0.0; for (int s=0; s<NUM_STUDENTS; s++) sum += score[s][q]; cout << "\tQuiz " << q << ": " << sum/NUM_STUDENTS << endl; }
311
Enter 5 quiz scores for each student:
The quize averages are: student 0: 8.2 student 1: 8.8 student 2: 7 The class averages are: Quiz 0: Quiz 1: Quiz 2: Quiz 3: Quiz 4: در برنامۀ فوق با استفاده از دستور typedef براي آرايههاي دوبعدي 3*5 نام مستعار Score انتخاب شده. اين باعث ميشود که توابع خواناتر باشند. هر تابع از دو حلقۀ for تودرتو استفاده کرده که حلقۀ بيروني، بعد اول را پيمايش ميکند و حلقۀ دروني بعد دوم را پيمايش مي نمايد. تابع printQuizAverages() ميانگين هر سطر از نمرات را محاسبه و چاپ مينمايد و تابع printClassAverages() ميانگين هر ستون از نمرهها را چاپ ميكند.
312
cout << "This array has " << numZeros(a,2,4,3)
مثال 21-6 پردازش يك آرايۀ سه بعدي اين برنامه تعداد صفرها را در يك آرايۀ سه بعدي ميشمارد: int numZeros(int a[][4][3], int n1, int n2, int n3); int main() { int a[2][4][3]={{{5,0,2}, {0,0,9},{4,1,0},{7,7,7} }, { {3,0,0}, {8,5,0}, {0,0,0}, {2,0,9} } }; cout << "This array has " << numZeros(a,2,4,3) << " zeros:\n"; }
313
int numZeros(int a[][4][3], int n1, int n2, int n3) { int count = 0;
for (int i = 0; i < n1; i++) for (int j = 0; j < n2; j++) for (int k = 0; k < n3; k++) if (a[i][j][k] == 0) ++count; return count; } This array has 11 zeros:
314
توجه كنيد كه آرايه چگونه مقداردهي شده است
توجه كنيد كه آرايه چگونه مقداردهي شده است. اين قالب مقداردهي به خوبي نمايان ميکند که آرايۀ مذکور يک آرايه دو عنصري است که هر عنصر، خود يک آرايۀ چهار عضوي است که هر عضو شامل آرايهاي سه عنصري ميباشد. پس اين آرايه در مجموع 24 عنصر دارد. آرايۀ مذکور را به شکل زير نيز ميتوانيم مقداردهي کنيم: int a[2][4][3]={5,0,2,0,0,9,4,1,0,7,7,7,3,0,0,8,5,0,0,0,0,2,0,9}; و يا مانند اين: int a[2][4][3] = {{5,0,2,0,0,9,4,1,0,7,7,7},{3,0,0,8,5,0,0,0,0,2,0,9}}; هر سۀ اين قالبها براي کامپايلر يک مفهوم را دارند اما با نگاه کردن به دو قالب اخير به سختي ميتوان فهميد که کدام عنصر از آرايه، کدام مقدار را خواهد داشت.
315
پايان جلسه ششم
316
«اشارهگرها و ارجاعها »
جلسه هفتم «اشارهگرها و ارجاعها »
317
آنچه در اين جلسه مي خوانيد:
1- عملگر ارجاع 2- ارجاعها 3- اشارهگرها 4- مقداريابي 5- چپ مقدارها، راست مقداره 6- بازگشت از نوع ارجاع 7- آرايهها و اشارهگرها ›››
318
8- عملگر new 9- عملگر delete 10- آرايههاي پويا 11- اشارهگر ثابت 12- آرايهاي از اشارهگرها 13- اشارهگري به اشارهگر ديگر 14- اشارهگر به توابع 15- NUL و NULL
319
»»» هدف کلي: آشنايي با اشارهگرها و نحوۀ کار با آدرسهاي حافظه
هدفهاي رفتاري: انتظار ميرود پس از پايان اين جلسه بتوانيد: - «ارجاع» را تعريف کنيد و با استفاده از عملگر ارجاع به متغيرها دستيابي کنيد. - «اشارهگر» را بشناسيد و بتوانيد اشارهگرهايي به انواع مختلف ايجاد کرده و آنها را مقداريابي کنيد. »»»
320
- «چپمقدارها» و «راستمقدارها» را تعريف کرده و آنها را از يکديگر تميز دهيد.
- بازگشتهايي از نوع ارجاع ايجاد نماييد. - طريقۀ استفاده از عملگرهاي new و delete و وظيفۀ هر يک را بدانيد. - «آرايههاي پويا» را تعريف کرده و مزيت آنها را نسبت به آرايههاي ايستا ذکر کنيد. - آرايههاي پويا را در برنامههايتان ايجاد کرده و مديريت نماييد. - تفاوت بين NUL و NULL را توضيح دهيد.
321
1- مقدمه حافظۀ رايانه را ميتوان به صورت يک آرايۀ بزرگ در نظر گرفت. براي مثال رايانهاي با 256 مگابايت RAM در حقيقت حاوي آرايهاي به اندازۀ 268،435،456 (=228) خانه است که اندازۀ هر خانه يک بايت است. اين خانهها داراي ايندکس صفر تا 268،435،455 هستند. به ايندکس هر بايت، آدرس حافظۀ آن ميگويند.
322
آدرسهاي حافظه را با اعداد شانزدهدهي نشان ميدهند
آدرسهاي حافظه را با اعداد شانزدهدهي نشان ميدهند. پس رايانۀ مذکور داراي محدوده آدرس 0x تا 0x0fffffff ميباشد. هر وقت که متغيري را اعلان ميکنيم، سه ويژگي اساسي به آن متغير نسبت داده ميشود: «نوع متغير» و «نام متغير» و «آدرس حافظه» آن. مثلا اعلان int n; نوع int و نام n و آدرس چند خانه از حافظه که مقدار n در آن قرار ميگيرد را به يکديگر مرتبط ميسازد. فرض کنيد آدرس اين متغير 0x0050cdc0 است. بنابراين ميتوانيم n را مانند شکل مقابل مجسم کنيم:
323
0x0050cdc0 n int خود متغير به شکل جعبه نمايش داده شده. نام متغير، n، در بالاي جعبه است و آدرس متغير در سمت چپ جعبه و نوع متغير، int، در زير جعبه نشان داده شده. در بيشتر رايانهها نوع int چهار بايت از حافظه را اشغال مينمايد. بنابراين همان طور که در شکل مقابل نشان داده شده است، متغير n يک بلوک چهاربايتي از حافظه را اشغال ميکند که شامل بايتهاي 0x0050cdc0 تا 0x0050cdc3 است. توجه کنيد که آدرس شي، آدرس اولين بايت از بلوکي است که شي در آن جا ذخيره شده.
324
اگر متغير فوق به شکل int n=32; مقداردهي اوليه شود، آنگاه بلوک حافظه به شکل زير خواهد بود. مقدار 32 در چهار بايتي که براي آن متغير منظور شده ذخيره ميشود. 0x0050cdb8 0x0050cdb9 0x0050cdc0 0x0050cdc1 0x0050cdc2 0x0050cdc3 0x0050cdc4 0x0050cdc5 n 32 0x0050cdc0 32 int
325
2- عملگر ارجاع در C++ براي بدست آوردن آدرس يک متغير ميتوان از عملگر ارجاع1 & استفاده نمود. به اين عملگر «علمگر آدرس» نيز ميگويند. عبارت &n آدرس متغير n را به دست ميدهد. int main() { int n=44; cout << " n = " << n << endl; cout << "&n = " << &n << endl; } n = 44 &n = 0x00c9fdc3
326
خروجي نشان ميدهد كه آدرس n در اين اجرا برابر با 0x00c9fdc3 است
خروجي نشان ميدهد كه آدرس n در اين اجرا برابر با 0x00c9fdc3 است. ميتوان فهميد که اين مقدار بايد يک آدرس باشد زيرا به شکل شانزدهدهي نمايش داده شده. اعداد شانزدهدهي را از روي علامت 0x ميتوان تشخيص داد. معادل دهدهي عدد بالا مقدار 13,237,699 ميباشد.
327
3- ارجاعها نحو اعلان يک ارجاع به شکل زير است:
يك «ارجاع» يك اسم مستعار يا واژۀ مترادف براي متغير ديگر است. نحو اعلان يک ارجاع به شکل زير است: type& ref_name = var_name; type نوع متغير است، ref_name نام مستعار است و var_name نام متغيري است که ميخواهيم براي آن نام مستعار بسازيم. براي مثال در اعلان : int& rn=n; // r is a synonym for n rn يک ارجاع يا نام مستعار براي n است. البته n بايد قبلا اعلان شده باشد.
328
مثال 2-7 استفاده از ارجاعها
در برنامۀ زير rn به عنوان يک ارجاع به n اعلان ميشود: int main() { int n=44; int& rn=n; // rn is a synonym for n cout << "n = " << n << ", rn = " << rn << endl; --n; rn *= 2; } n = 44, rn = 44 n = 43, rn = 43 n = 86, rn = 86 n و rn نامهاي متفاوتي براي يک متغير است. اين دو هميشه مقدار يکساني دارند. اگر n کاسته شود، rn نيز کاسته شده و اگر rn افزايش يابد، n نيز افزايش يافته است.
329
همانند ثابتها، ارجاعها بايد هنگام اعلان مقداردهي اوليه شوند با اين تفاوت که مقدار اوليۀ يک ارجاع، يک متغير است نه يک ليترال. بنابراين کد زير اشتباه است: int& rn=44; // ERROR: 44 is not a variable; گرچه برخي از کامپايلرها ممکن است دستور بالا را مجاز بدانند ولي با نشان دادن يک هشدار اعلام ميکنند که يک متغير موقتي ايجاد شده تا rn به حافظۀ آن متغير، ارجاع داشته باشد.
330
درست است که ارجاع با يک متغير مقداردهي ميشود، اما ارجاع به خودي خود يک متغير نيست.
يک متغير، فضاي ذخيرهسازي و نشاني مستقل دارد، حال آن که ارجاع از فضاي ذخيرهسازي و نشاني متغير ديگري بهره ميبرد.
331
* مثال 3-7 ارجاعها متغيرهاي مستقل نيستند int main() { int n=44;
int& rn=n; // rn is a synonym for n cout << " &n = " << &n << ", &rn = " << &rn << endl; int& rn2=n; // rn2 is another synonym for n int& rn3=rn; // rn3 is another synonym for n cout << "&rn2 = " << &rn2 << ", &rn3 = " << &rn3 << endl; } &n = 0x0064fde4, &rn = 0x0064fde4 &rn2 = 0x0064fde4, &rn3 = 0x0064fde4
332
تنها فرق اين است که دامنۀ پارامتر ارجاع به همان تابع محدود شده است.
در برنامۀ فوق فقط يک شي وجود دارد و آن هم n است. rn و rn2 و rn3 ارجاعهايي به n هستند. خروجي نيز تاييد ميکند که آدرس rn و rn2 و rn3 با آدرس n يکي است. يک شي ميتواند چند ارجاع داشته باشد. ارجاعها بيشتر براي ساختن پارامترهاي ارجاع در توابع به کار ميروند. تابع ميتواند مقدار يک آرگومان را که به طريق ارجاع ارسال شده تغيير دهد زيرا آرگومان اصلي و پارامتر ارجاع هر دو يک شي هستند. تنها فرق اين است که دامنۀ پارامتر ارجاع به همان تابع محدود شده است.
333
اما آدرس حافظه را در چه نوع متغيري بايد قرار دهيم؟
4- اشارهگرها ميدانيم که اعداد صحيح را بايد در متغيري از نوع int نگهداري کنيم و اعداد اعشاري را در متغيرهايي از نوع float. به همين ترتيب کاراکترها را بايد در متغيرهايي از نوع char نگهداريم و مقدارهاي منطقي را در متغيرهايي از نوع bool. اما آدرس حافظه را در چه نوع متغيري بايد قرار دهيم؟
334
متغيري که يک آدرس در آن ذخيره ميشود اشارهگر ناميده ميشود.
عملگر ارجاع & آدرس حافظۀ يک متغير موجود را به دست ميدهد. ميتوان اين آدرس را در متغير ديگري ذخيره نمود. متغيري که يک آدرس در آن ذخيره ميشود اشارهگر ناميده ميشود. براي اين که يک اشارهگر اعلان کنيم، ابتدا بايد مشخص کنيم که آدرس چه نوع دادهاي قرار است در آن ذخيره شود. سپس از عملگر اشاره * استفاده ميکنيم تا اشارهگر را اعلان کنيم.
335
براي مثال دستور : float* px; اشارهگري به نام px اعلان ميکند که اين اشارهگر، آدرس متغيرهايي از نوع float را نگهداري مينمايد. به طور کلي براي اعلان يک اشارهگر از نحو زير استفاده ميکنيم: type* pointername; که type نوع متغيرهايي است که اين اشارهگر آدرس آنها را نگهداري ميکند و pointername نام اشارهگر است. آدرس يک شي از نوع int را فقط ميتوان در اشارهگري از نوع int* ذخيره کرد و آدرس يک شي از نوع float را فقط ميتوان در اشارهگري از نوع float* ذخيره نمود. دقت کنيد که يک اشارهگر، يک متغير مستقل است.
336
int* pn=&n; // pn holds the address of n
* مثال 4-7 به کارگيري اشارهگرها برنامۀ زير يک متغير از نوع int به نام n و يک اشارهگر از نوع int* به نام pn را اعلان ميکند: int main() { int n=44; cout << "n = " << n << ", &n = " << &n << endl; int* pn=&n; // pn holds the address of n cout << " pn = " << pn << endl; cout << "&pn = " << &pn << endl;} n = 44, &n = 0x0064fddc pn = 0x0064fddc &pn = 0x0064fde0
337
متغير n با مقدار 44 مقداردهي شده و آدرس آن 0x0064fddc ميباشد
متغير n با مقدار 44 مقداردهي شده و آدرس آن 0x0064fddc ميباشد. اشارهگر pn با مقدار &n يعني آدرس n مقداردهي شده. پس مقدار درون pn برابر با 0x0064fddc است (خط دوم خروجي اين موضوع را تاييد ميکند) . 44 n int 0x0064fddc pn int* 0x0064fde0
338
وقتي ميگوييم «pn به n اشاره ميکند» يعني درون pn آدرس n قرار دارد.
اما pn يک متغير مستقل است و آدرس مستقلي دارد. &pn آدرس pn را به دست ميدهد. خط سوم خروجي ثابت ميکند که متغير pn مستقل از متغير n است. تصوير زير به درک بهتر اين موضوع کمک ميکند. در اين تصوير ويژگيهاي مهم n و pn نشان داده شده. pn يک اشارهگر به n است و n مقدار 44 دارد. وقتي ميگوييم «pn به n اشاره ميکند» يعني درون pn آدرس n قرار دارد. وقتي ميگوييم «pn به n اشاره ميکند» يعني درون pn آدرس n قرار دارد. n 44 int int* pn
339
به اين کار مقداريابي اشارهگر ميگوييم.
5-مقداريابي فرض کنيد n داراي مقدار 22 باشد و pn اشارهگري به n باشد. با اين حساب بايد بتوان از طريق pn به مقدار 22 رسيد. با استفاده از * ميتوان مقداري که اشارهگر به آن اشاره دارد را به دست آورد. به اين کار مقداريابي اشارهگر ميگوييم.
340
ظاهرا *pn يک اسم مستعار براي n است زيرا هر دو يک مقدار دارند.
مثال 5-7 مقداريابي يك اشارهگر اين برنامه همان برنامۀ مثال 4-7 است. فقط يک خط کد بيشتر دارد: int main() { int n=44; cout << "n = " << n << ", &n = " << &n << endl; int* pn=&n; // pn holds the address of n cout << " pn = " << pn << endl; cout << "&pn = " << &pn << endl; cout << "*pn = " << *pn << endl; } n = 44, &n = 0x0064fdcc pn = 0x0064fdcc &pn = 0x0064fdd0 *pn = 44 ظاهرا *pn يک اسم مستعار براي n است زيرا هر دو يک مقدار دارند.
341
* مثال 6-7 اشارهگري به اشارهگرها
اين کد ادامۀ ساختار برنامۀ مثال 4-7 است: int main() { int n=44; cout << " n = " << n << endl; cout << " &n = " << &n << endl; int* pn=&n; // pn holds the address of n cout << " pn = " << pn << endl; cout << " &pn = " << &pn << endl; cout << " *pn = " << *pn << endl; int** ppn=&pn; // ppn holds the address of pn cout << " ppn = " << ppn << endl; cout << " &ppn = " << &ppn << endl; cout << " *ppn = " << *ppn << endl; cout << "**ppn = " << **ppn << endl; } يک اشارهگر به هر چيزي ميتواند اشاره کند، حتي به يک اشارهگر ديگر. به مثال زير دقت کنيد.
342
pn int* ppn int** n = 44 n &n = 0x0064fd78 int pn = 0x0064fd78
&pn = 0x0064fd7c *pn = 44 ppn = 0x0064fd7c &ppn = 0x0064fd80 *ppn = 0x0064fd78 **ppn = 44 n 44 int int* pn ppn int** در برنامۀ بالا متغير n از نوع int تعريف شده. pn اشارهگري است که به n اشاره دارد. پس نوع pn بايد int* باشد. ppn اشارهگري است که به pn اشاره ميکند. پس نوع ppn بايد int** باشد. همچنين چون ppn به pn اشاره دارد، پس *ppn مقدار pn را نشان ميدهد و چون pn به n اشاره دارد، پس *pn مقدار n را ميدهد.
343
عملگر مقداريابي. و عملگر ارجاع & معکوس يکديگر رفتار ميکنند
عملگر مقداريابي * و عملگر ارجاع & معکوس يکديگر رفتار ميکنند. اگر اين دو را با هم ترکيب کنيم، يکديگر را خنثي مينمايند. اگر n يک متغير باشد، &n آدرس آن متغير است. از طرفي با استفاده از عملگر * ميتوان مقداري که در آدرس &n قرار گرفته را به دست آورد. بنابراين *&n برابر با خود n خواهد بود. همچنين اگر p يک اشارهگر باشد، *p مقداري که p به آن اشاره دارد را ميدهد. از طرفي با استفاده از عملگر & ميتوانيم آدرس چيزي که در *p قرار گرفته را بدست آوريم. پس &*p برابر با خود p خواهد بود. ترتيب قرارگرفتن اين عملگرها مهم است. يعني *&n با &*n برابر نيست. علت اين امر را توضيح دهيد.
344
عملگر. دو کاربرد دارد. اگر پسوندِ يک نوع باشد (مثل int
عملگر * دو کاربرد دارد. اگر پسوندِ يک نوع باشد (مثل int*) يک اشارهگر به آن نوع را تعريف ميکند و اگر پيشوندِ يک اشارهگر باشد (مثل *p) آنگاه مقداري که p به آن اشاره ميکند را برميگرداند. عملگر & نيز دو کاربرد دارد. اگر پسوند يک نوع باشد (مثل int&) يک نام مستعار تعريف ميکند و اگر پيشوند يک متغير باشد (مثل &n) آدرس آن متغير را ميدهد.
345
6- چپ مقدارها، راست مقدارها
يک دستور جايگزيني دو بخش دارد: بخشي که در سمت چپ علامت جايگزيني قرار ميگيرد و بخشي که در سمت راست علامت جايگزيني قرار ميگيرد. مثلا دستور n = 55; متغير n در سمت چپ قرار گرفته و مقدار 55 در سمت راست. اين دستور را نميتوان به شکل 55 = n; نوشت زيرا مقدار 55 يک ثابت است و نميتواند مقدار بگيرد. پس هنگام استفاده از عملگر جايگزيني بايد دقت کنيم که چه چيزي را در سمت چپ قرار بدهيم و چه چيزي را در سمت راست.
346
چيزهايي که ميتوانند در سمت چپ جايگزيني قرار بگيرند «چپمقدار» خوانده ميشوند و چيزهايي که ميتوانند در سمت راست جايگزيني قرار بگيرند «راستمقدار» ناميده ميشوند. متغيرها (و به طور کلي اشيا) چپمقدار هستند و ليترالها (مثل 15 و "ABC") راست مقدار هستند.
347
يک ثابت در ابتدا به شکل يک چپمقدار نمايان ميشود:
const int MAX = 65535; // MAX is an lvalue اما از آن پس ديگر نميتوان به عنوان چپ مقدار از آنها استفاده کرد: MAX = 21024; // ERROR: MAX is constant به اين گونه چپمقدارها، چپمقدارهاي «تغيير ناپذير» گفته ميشود. مثل آرايهها: int a[] = {1,2,3}; // O.K a[] = {1,2,3}; // ERROR
348
مابقي چپمقدارها که ميتوان آنها را تغيير داد، چپمقدارهاي «تغيير پذير» ناميده ميشوند. هنگام اعلان يک ارجاع به يک چپمقدار نياز داريم: int& r = n; // O.K. n is an lvalue اما اعلانهاي زير غيرمعتبرند زيرا هيچ کدام چپمقدار نيستند: int& r = 44; // ERROR: 44 is not an lvalue int& r = n++; // ERROR: n++ is not an lvalue int& r = cube(n); // ERROR: cube(n) is not an lvalue1 – L_values 2- R_values يک تابع، چپمقدار نيست اما اگر نوع بازگشتي آن يک ارجاع باشد، ميتوان تابع را به يک چپمقدار تبديل کرد.
349
7- بازگشت از نوع ارجاع در بحث توابع، ارسال از طريق مقدار و ارسال از طريق ارجاع را ديديم. اين دو شيوۀ تبادل در مورد بازگشت از تابع نيز صدق ميکند: بازگشت از طريق مقدار و بازگشت از طريق ارجاع. توابعي که تاکنون ديديم بازگشت به طريق مقدار داشتند. يعني هميشه يک مقدار به فراخواننده برميگشت. ميتوانيم تابع را طوري تعريف کنيم که به جاي مقدار، يک ارجاع را بازگشت دهد. مثلا به جاي اين که مقدار m را بازگشت دهد، يک ارجاع به m را بازگشت دهد.
350
وقتي بازگشت به طريق مقدار باشد، تابع يک راستمقدار خواهد بود زيرا مقدارها ليترال هستند و ليترالها راستمقدارند. به اين ترتيب تابع را فقط در سمت راست يک جايگزيني ميتوان به کار برد مثل: m = f(); وقتي بازگشت به طريق ارجاع باشد، تابع يک چپمقدار خواهد بود زيرا ارجاعها چپمقدار هستند. در اين حالت تابع را ميتوان در سمت چپ يک جايگزيني قرار داد مثل : f() = m;
351
* مثال 7-8 بازگشت از نوع ارجاع int& max(int& m, int& n)
براي اين که نوع بازگشتي تابع را به ارجاع تبديل کنيم کافي است عملگر ارجاع را به عنوان پسوند نوع بازگشتي درج کنيم. * مثال 7-8 بازگشت از نوع ارجاع int& max(int& m, int& n) { return (m > n ? m : n);} int main() { int m = 44, n = 22; cout << m << ", " << n << ", " << max(m,n) << endl; max(m,n) = 55; } 44, 22, 44 55, 22, 55
352
تابع max() از بين m و n مقدار بزرگتر را پيدا کرده و سپس ارجاعي به آن را باز ميگرداند.
بنابراين اگر m از n بزرگتر باشد، تابع max(m,n) آدرس m را برميگرداند. پس وقتي مينويسيم max(m,n) = 55; مقدار 55 در حقيقت درون متغير m قرار ميگيرد (اگر m>n باشد). به بياني ساده، فراخواني max(m,n) خود m را بر ميگرداند نه مقدار آن را.
353
اخطار: وقتي يک تابع پايان مييابد، متغيرهاي محلي آن نابود ميشوند. پس هيچ وقت ارجاعي به يک متغير محلي بازگشت ندهيد زيرا وقتي کار تابع تمام شد، آدرس متغيرهاي محلياش غير معتبر ميشود و ارجاع بازگشت داده شده ممکن است به يک مقدار غير معتبر اشاره داشته باشد. تابع max() در مثال بالا يک ارجاع به m يا n را بر ميگرداند. چون m و n خودشان به طريق ارجاع ارسال شدهاند، پس محلي نيستند و بازگرداندن ارجاعي به آنها خللي در برنامه وارد نميکند.
354
به اعلان تابع max() دقت کنيد:
int& max(int& m, int& n) نوع بازگشتي آن با استفاده از عملگر ارجاع & به شکل يک ارجاع درآمده است. مثال 9-7 به کارگيري يك تابع به عنوان عملگر زيرنويس آرايه
355
float& component(float* v, int k) { return v[k-1];} int main()
for (int k = 1; k <= 4; k++) component(v,k) = 1.0/k; for (int i = 0; i < 4; i++) cout << "v[" << i << "] = " << v[i] << endl; } v[0] = 1 v[1] = 0.5 v[2] = v[3] = 0.25
356
تابع component() باعث ميشود که ايندکس آرايه v از «شمارهگذاري از صفر» به «شمارهگذاري از يک» تغيير کند. بنابراين component(v,3) معادل v[2] است. اين کار از طريق بازگشت از طريق ارجاع ممکن شده است.
357
8- آرايهها و اشارهگرها
گرچه اشارهگرها از انواع عددي صحيح نيستند اما بعضي از اعمال حسابي را ميتوان روي اشارهگرها انجام داد. حاصل اين ميشود که اشارهگر به خانۀ ديگري از حافظه اشاره ميکند. اشارهگرها را ميتوان مثل اعداد صحيح افزايش و يا کاهش داد و ميتوان يک عدد صحيح را به آنها اضافه نمود يا از آن کم کرد. البته ميزان افزايش يا کاهش اشارهگر بستگي به نوع دادهاي دارد که اشارهگر به آن اشاره دارد. مثال 10-7 پيمايش آرايه با استفاده از اشارهگر اين مثال نشان ميدهد كه چگونه ميتوان از اشارهگر براي پيمايش يک آرايه استفاده نمود:
358
cout << "a = " << a << endl;
int main() { const int SIZE = 3; short a[SIZE] = {22, 33, 44}; cout << "a = " << a << endl; cout << "sizeof(short) = " << sizeof(short) << endl; short* end = a + SIZE; // converts SIZE to offset 6 short sum = 0; for (short* p = a; p < end; p++) { sum += *p; cout << "\t p = " << p; cout << "\t *p = " << *p; cout << "\t sum = " << sum << endl; } cout << "end = " << end << endl; a = 0x3fffd1a sizeof(short) = 2 p = 0x3fffd1a *p = sum = 22 p = 0x3fffd1c *p = sum = 55 p = 0x3fffd1e *p = sum = 99 end = 0x3fffd20
359
اين مثال نشان ميدهد که هر گاه يک اشارهگر افزايش يابد، مقدار آن به اندازۀ تعداد بايتهاي شيئي که به آن اشاره ميکند، افزايش مييابد. مثلا اگر p اشارهگري به double باشد و sizeof(double) برابر با هشت بايت باشد، هر گاه که p يک واحد افزايش يابد، اشارهگر p هشت بايت به پيش ميرود.
360
مثلا کد زير : float a[8]; float* p = a; // p points to a[0] ++p; // increases the value of p by sizeof(float)
361
اگر floatها 4 بايت را اشغال كنند آنگاه ++p مقدار درون p را 4 بايت افزايش ميدهد و p += 5; مقدار درون p را 20 بايت افزايش ميدهد. با استفاده از خاصيت مذکور ميتوان آرايه را پيمايش نمود: يک اشارهگر را با آدرس اولين عنصر آرايه مقداردهي کنيد، سپس اشارهگر را پي در پي افزايش دهيد. هر افزايش سبب ميشود که اشارهگر به عنصر بعدي آرايه اشاره کند. يعني اشارهگري که به اين نحو به کار گرفته شود مثل ايندکس آرايه عمل ميکند.
362
float* p = a; // p points to a[0] p += 5; // now p points to a[5]
همچنين با استفاده از اشارهگر ميتوانيم مستقيما به عنصر مورد نظر در آرايه دستيابي کنيم: float* p = a; // p points to a[0] p += 5; // now p points to a[5] يک نکتۀ ظريف در ارتباط با آرايهها و اشارهگرها وجود دارد: اگر اشارهگر را بيش از ايندکس آرايه افزايش دهيم، ممکن است به بخشهايي از حافظه برويم که هنوز تخصيص داده نشدهاند يا براي کارهاي ديگر تخصيص يافتهاند. تغيير دادن مقدار اين بخشها باعث بروز خطا در برنامه و کل سيستم ميشود. هميشه بايد مراقب اين خطر باشيد.
363
float* p = a[7]; // points to last element in the array
کد زير نشان ميدهد که چطور اين اتفاق رخ ميدهد. float a[8]; float* p = a[7]; // points to last element in the array ++p; //now p points to memory past last element! *p = 22.2; // TROUBLE! مثال بعدي نشان ميدهد كه ارتباط تنگاتنگي بين آرايهها و اشارهگرها وجود دارد. نام آرايه در حقيقت يک اشارهگر ثابت (const) به اولين عنصر آرايه است. همچنين خواهيم ديد که اشارهگرها را مانند هر متغير ديگري ميتوان با هم مقايسه نمود.
364
* مثال 11-7 پيمايش عناصر آرايه از طريق آدرس int main()
{ short a[] = {22, 33, 44, 55, 66}; cout << "a = " << a << ", *a = " << *a << endl; for (short* p = a; p < a +5; p++) cout << "p = " << p << ", *p = " << *p << endl; } a = 0x3fffd08, *a = 22 p = 0x3fffd08, *p = 22 p = 0x3fffd0a, *p = 33 p = 0x3fffd0c, *p = 44 p = 0x3fffd0e, *p = 55 p = 0x3fffd10, *p = 66 p = 0x3fffd12, *p = 77
365
پس حلقه تا زماني که p < 0x3fffd12 باشد ادامه مييابد.
در نگاه اول، a و p مانند هم هستند: هر دو به نوع short اشاره ميکنند و هر دو داراي مقدار 0x3fffd08 هستند. اما a يک اشارهگر ثابت است و نميتواند افزايش يابد تا آرايه پيمايش شود. پس به جاي آن p را افزايش ميدهيم تا آرايه را پيمايش کنيم. شرط (p < a+5) حلقه را خاتمه ميدهد. a+5 به شکل زير ارزيابي ميشود: 0x3fffd08 + 5*sizeof(short) = 0x3fffd08 + 5*2 = 0x3fffd08 + 0xa = 0x3fffd12 پس حلقه تا زماني که p < 0x3fffd12 باشد ادامه مييابد.
366
عملگر زيرنويس [] مثل عملگر مقداريابي. رفتار ميکند
عملگر زيرنويس [] مثل عملگر مقداريابي * رفتار ميکند. هر دوي اينها ميتوانند به عناصر آرايه دسترسي مستقيم داشته باشند. a[0] == *a a[1] == *(a + 1) a[2] == *(a + 2) ... پس با استفاده از کد زير نيز ميتوان آرايه را پيمايش نمود: for (int i = 0; i < 8; i++) cout << *(a + i) << endl;
367
short* loc(short* a1, short* a2, int n1, int n2)
مثال 12-7 مقايسۀ الگو در اين مثال، تابع loc() در ميان n1 عنصر اول آرايۀ a1 به دنبال n2 عنصر اول آرايۀ a2 ميگردد. اگر پيدا شد، يک اشارهگر به درون a1 برميگرداند که a2 از آنجا شروع ميشود وگرنه اشارهگر NULL را برميگرداند. short* loc(short* a1, short* a2, int n1, int n2) { short* end1 = a1 + n1; for (short* p1 = a1; p1 <end1; p1++) if (*p1 == *a2) { for (int j = 0; j < n2; j++) if (p1[j] != a2[j]) break; if (j == n2) return p1; } return 0;
368
int main() { short a1[9] = {11, 11, 11, 11, 11, 22, 33, 44, 55}; short a2[5] = {11, 11, 11, 22, 33}; cout << "Array a1 begins at location\t" << a1 << endl; cout << "Array a2 begins at location\t" << a2 << endl; short* p = loc(a1, a2, 9, 5); if (p) { cout << "Array a2 found at location\t" << p << endl; for (int i = 0; i < 5; i++) cout << "\t" << &p[i] << ": " << p[i] << "\t" << &a2[i] << ": " << a2[i] << endl; } else cout << "Not found.\n";}
369
Array a1 begins at location 0x3fffd12
Array a2 found at location x3fffd16 0x3fffd16: x3fffd08: 11 0x3fffd18: x3fffd0a: 11 0x3fffd1a: x3fffd0c: 11 0x3fffd1c: x3fffd0e: 22 0x3fffd1e: x3fffd10: 33
370
9-7 عملگر new وقتي يك اشارهگر شبيه اين اعلان شود:
float* p; // p is a pointer to a float يک فضاي چهاربايتي به p تخصيص داده ميشود (معمولا sizeof(float) چهار بايت است). حالا p ايجاد شده است اما به هيچ جايي اشاره نميکند زيرا هنوز آدرسي درون آن قرار نگرفته. به چنين اشارهگري اشارهگر سرگردان ميگويند. اگر سعي کنيم يک اشارهگر سرگردان را مقداريابي يا ارجاع کنيم با خطا مواجه ميشويم.
371
مثلا دستور: *p = ; // ERROR: no storage has been allocated for *P خطاست. زيرا p به هيچ آدرسي اشاره نميکند و سيستم عامل نميداند که مقدار را کجا ذخيره کند. براي رفع اين مشکل ميتوان اشارهگرها را هنگام اعلان، مقداردهي کرد: float x = 0; // x cintains the value 0 float* p = &x // now p points to x *p = ; // O.K. assigns this value to address that p points to
372
در اين حالت ميتوان به *p دستيابي داشت زيرا حالا p به x اشاره ميکند و آدرس آن را دارد. راه حل ديگر اين است که يک آدرس اختصاصي ايجاد شود و درون p قرار بگيرد. بدين ترتيب p از سرگرداني خارج ميشود. اين کار با استفاده از عملگر new صورت ميپذيرد: float* p; p = new float; // allocates storage for 1 float *p = ; // O.K. assigns this value to that storage دقت کنيد که عملگر new فقط خود p را مقداردهي ميکند نه آدرسي که p به آن اشاره ميکند. ميتوانيم سه خط فوق را با هم ترکيب کرده و به شکل يک دستور بنويسيم: float* p = new float( );
373
با اين دستور، اشارهگر p از نوع float
با اين دستور، اشارهگر p از نوع float* تعريف ميشود و سپس يک بلوک خالي از نوع float منظور شده و آدرس آن به p تخصيص مييابد و همچنين مقدار در آن آدرس قرار ميگيرد. اگر عملگر new نتواند خانۀ خالي در حافظه پيدا کند، مقدار صفر را برميگرداند. اشارهگري که اين چنين باشد، «اشارهگر تهي» يا NULL مينامند.
374
با استفاده از کد هوشمند زير ميتوانيم مراقب باشيم که اشارهگر تهي ايجاد نشود:
double* p = new double; if (p == 0) abort(); // allocator failed: insufficent memory else *p = ; در اين قطعه کد، هرگاه اشارهگري تهي ايجاد شد، تابع abort() فراخواني شده و اين دستور لغو ميشود.
375
تاکنون دانستيم که به دو طريق ميتوان يک متغير را ايجاد و مقداردهي کرد
تاکنون دانستيم که به دو طريق ميتوان يک متغير را ايجاد و مقداردهي کرد. روش اول: float x = ; // allocates named memory و روش دوم: float* p = new float( ); // allocates unnamed memory در حالت اول، حافظۀ مورد نياز براي x هنگام کامپايل تخصيص مييابد. در حالت دوم حافظۀ مورد نياز در زمان اجرا و به يک شيء بينام تخصيص مييابد که با استفاده از *p قابل دستيابي است.
376
10- عملگر delete عملگر delete عملي برخلاف عملگر new دارد. کارش اين است که حافظۀ اشغال شده را آزاد کند. وقتي حافظهاي آزاد شود، سيستم عامل ميتواند از آن براي کارهاي ديگر يا حتي تخصيصهاي جديد استفاده کند. عملگر delete را تنها روي اشارهگرهايي ميتوان به کار برد که با دستور new ايجاد شدهاند. وقتي حافظۀ يک اشارهگر آزاد شد، ديگر نميتوان به آن دستيابي نمود مگر اين که دوباره اين حافظه تخصيص يابد: float* p = new float( ); delete p; // deallocates q *p = ; // ERROR: q has been deallocated
377
وقتي اشاره گر p در کد بالا آزاد شود، حافظهاي که توسط new به آن تخصيص يافته بود، آزاد شده و به ميزان sizeof(float) به حافظۀ آزاد اضافه ميشود. وقتي اشارهگري آزاد شد، به هيچ چيزي اشاره نميکند؛ مثل متغيري که مقداردهي نشده. به اين اشارهگر، اشارهگر سرگردان ميگويند. اشارهگر به يک شيء ثابت را نميتوان آزاد کرد: const int* p = new int; delete p; // ERROR: cannot delete pointer to const objects علت اين است که «ثابتها نميتوانند تغيير کنند».
378
اگر متغيري را صريحا اعلان کردهايد و سپس اشارهگري به آن نسبت دادهايد، از عملگر delete استفاده نکنيد. اين کار باعث اشتباه غير عمدي زير ميشود: float x = ; // x contains the value float* p = &x; // p contains the address of x delete p; // WARNING: this will make x free کد بالا باعث ميشود که حافظۀ تخصيصيافته براي x آزاد شود. اين اشتباه را به سختي ميتوان تشخيص داد و اشکالزدايي کرد.
379
11- آرايههاي پويا نام آرايه در حقيقت يك اشارهگر ثابت است كه در زمان كامپايل، ايجاد و تخصيص داده ميشود: float a[20]; //a is a const pointer to a block of 20 floats float* const p = new float[20]; // so is p هم a و هم p اشارهگرهاي ثابتي هستند که به بلوکي حاوي 20 متغير float اشاره دارند. به اعلان a بستهبندي ايستا1 ميگويند زيرا اين کد باعث ميشود که حافظۀ مورد نياز براي a در زمان کامپايل تخصيص داده شود. وقي برنامه اجرا شود، به هر حال حافظۀ مربوطه تخصيص خواهد يافت حتي اگر از آن هيچ استفادهاي نشود.
380
ميتوانيم با استفاده از اشارهگر، آرايۀ فوق را طوري تعريف کنيم که حافظه مورد نياز آن فقط در زمان اجرا تخصيص يابد: float* p = new float[20]; دستور بالا، 20 خانۀ خالي حافظه از نوع float را در اختيار گذاشته و اشارهگر p را به خانۀ اول آن نسبت ميدهد. به اين آرايه، «آرايۀ پويا2» ميگويند. به اين طرز ايجاد اشيا بستهبندي پويا3 يا «بستهبندي زمان جرا» ميگويند.
381
آرايۀ ايستاي a و آرايۀ پوياي p را با يکديگر مقايسه کنيد
آرايۀ ايستاي a و آرايۀ پوياي p را با يکديگر مقايسه کنيد. آرايۀ ايستاي a در زمان کامپايل ايجاد ميشود و تا پايان اجراي برنامه، حافظۀ تخصيصي به آن مشغول ميماند. ولي آرايۀ پوياي p در زمان اجرا و هر جا که لازم شد ايجاد ميشود و پس از اتمام کار نيز ميتوان با عملگر delete حافظۀ تخصيصي به آن را آزاد کرد: delete [] p; براي آزاد کردن آرايۀ پوياي p براکتها [] قبل از نام p بايد حتما قيد شوند زيرا p به يک آرايه اشاره دارد.
382
void get(double*& a, int& n)
مثال 15-7 استفاده از آرايههاي پويا تابع get() در برنامۀ زير يک آرايۀ پويا ايجاد ميكند: void get(double*& a, int& n) { cout << "Enter number of items: "; cin >> n; a = new double[n]; cout << "Enter " << n << " items, one per line:\n"; for (int i = 0; i < n; i++) { cout << "\t" << i+1 << ": "; cin >> a[i]; }} void print(double* a, int n) { for (int i = 0; i < n; i++) cout << a[i] << " " ; cout << endl; }
383
int main() { double* a;// a is simply an unallocated pointer int n; get(a,n); // now a is an array of n doubles print(a,n); delete [] a;// now a is simply an unallocated pointer again get(a,n); // now a is an array of n doubles print(a,n); }
384
Enter number of items: 4 Enter 4 items, one per line: 1: 44.4 2: 77.7 3: 22.2 4: 88.8 Enter number of items: 2 Enter 2 items, one per line: 1: 3.33 2: 9.99
385
12- اشارهگر ثابت «اشارهگر به يک ثابت» با «اشارهگر ثابت» تفاوت دارد. اين تفاوت در قالب مثال زير نشان داده شده است. مثال 16-7 اشارهگرهاي ثابت و اشارهگرهايي به ثابتها در اين کد چهار اشارهگر اعلان شده. اشارهگر p، اشارهگر ثابت cp، اشاره به يک ثابت pc، اشارهگر ثابت به يک ثابت cpc :
386
int n = 44; // an int int* p = &n; // a pointer to an int ++(*p); // OK: increments int *p ++p; // OK: increments pointer p int* const cp = &n; // a const pointer to an int ++(*cp); // OK: increments int *cp ++cp; // illegal: pointer cp is const const int k = 88; // a const int const int * pc = &k; // a pointer to a const int ++(*pc); // illegal: int *pc is const ++pc; // OK: increments pointer pc const int* const cpc = &k; // a const pointer to a const int ++(*cpc); // illegal: int *pc is const ++cpc; // illegal: pointer cpc is const
387
اشارهگر p اشارهگري به متغير n است
اشارهگر p اشارهگري به متغير n است. هم خود p قابل افزايش است (++p) و هم مقداري که p به آن اشاره ميکند قابل افزايش است (++(*P)). اشاره گر cp يک اشارهگر ثابت است. يعني آدرسي که در cp است قابل تغيير نيست ولي مقداري که در آن آدرس است را ميتوان دستکاري کرد. اشارهگر pc اشارهگري است که به آدرس يک ثابت اشاره دارد. خود pc را ميتوان تغيير داد ولي مقداري که pc به آن اشاره دارد قابل تغيير نيست. در آخر هم cpc يک اشارهگر ثابت به يک شيء ثابت است. نه مقدار cpc قابل تغيير است و نه مقداري که آدرس آن در cpc است.
388
13- آرايهاي از اشارهگرها
ميتوانيم آرايهاي تعريف کنيم که اعضاي آن از نوع اشارهگر باشند. مثلا دستور: float* p[4]; آرايۀ p را با چهار عنصر از نوع float* (يعني اشارهگري به float) اعلان ميکند. عناصر اين آرايه را مثل اشارهگرهاي معمولي ميتوان مقداردهي کرد: p[0] = new float( ); p[1] = new float(1.19);
389
اين آرايه را مي توانيم شبيه شکل مقابل مجسم کنيم:
مثال بعد نشان ميدهد که آرايهاي از اشارهگرها به چه دردي ميخورد. از اين آرايه ميتوان براي مرتبکردن يک فهرست نامرتب به روش حبابي استفاده کرد. به جاي اين که خود عناصر جابجا شوند، اشارهگرهاي آنها جابجا ميشوند. p 1 2 3 double 1.19 double
390
مثال 17-7 مرتبسازي حبابي غيرمستقيم
void sort(float* p[], int n) { float* temp; for (int i = 1; i < n; i++) for (int j = 0; j < n-i; j++) if (*p[j] > *p[j+1]) { temp = p[j]; p[j] = p[j+1]; p[j+1] = temp; }
391
تابع sort() آرايهاي از اشارهگرها را ميگيرد
تابع sort() آرايهاي از اشارهگرها را ميگيرد. سپس درون حلقههاي تودرتوي for بررسي ميکند که آيا مقاديري که اشارهگرهاي مجاور به آنها اشاره دارند، مرتب هستند يا نه. اگر مرتب نبودند، جاي اشارهگرهاي آنها را با هم عوض ميکند. در پايان به جاي اين که يک فهرست مرتب داشته باشيم، آرايهاي داريم که اشارهگرهاي درون آن به ترتيب قرار گرفته اند.
392
14-7 اشارهگري به اشارهگر ديگر
يك اشارهگر ميتواند به اشارهگر ديگري اشاره کند. مثلا: char c = 't'; char* pc = &c; char** ppc = &pc; char*** pppc = &ppc; ***pppc = 'w'; // changes value of c to 'w' حالا pc اشارهگري به متغير کاراکتري c است. ppc اشارهگري به اشارهگر pc است و اشارهگر pppc هم به اشارهگر ppc اشاره دارد. مثل شکل مقابل:
393
با اين وجود ميتوان با اشارهگر pppc مستقيما به متغير c رسيد.
'\t' c با اين وجود ميتوان با اشارهگر pppc مستقيما به متغير c رسيد.
394
15- اشارهگر به توابع اين بخش ممکن است کمي عجيب به نظر برسد. حقيقت اين است که نام يک تابع مثل نام يک آرايه، يک اشارهگر ثابت است. نام تابع، آدرسي از حافظه را نشان ميدهد که کدهاي درون تابع در آن قسمت جاي گرفتهاند. پس بنابر قسمت قبل اگر اشارهگري به تابع اعلان کنيم، در اصل اشارهگري به اشارهگر ديگر تعريف کردهايم. اما اين تعريف، نحو متفاوتي دارد: int f(int); // declares function f int (*pf)(int); // declares function pointer pf pf = &f; // assigns address of f to pf
395
اشارهگر pf همراه با * درون پرانتز قرار گرفته، يعني اين که pf اشارهگري به يک تابع است. بعد از آن يک int هم درون پرانتز آمده است، به اين معني که تابعي که pf به آن اشاره مينمايد، پارامتري از نوع int دارد. اشارهگر pf را ميتوانيم به شکل زير تصور کنيم:
396
فايدۀ اشارهگر به توابع اين است که به اين طريق ميتوانيم توابع مرکب بسازيم. يعني ميتوانيم يک تابع را به عنوان آرگومان به تابع ديگر ارسال کنيم! اين کار با استفاده از اشارهگر به تابع امکان پذير است. pf f int f(int n) { ... }
397
int sum(int (*)(int), int); int square(int); int cube(int); int main()
مثال 18-7 تابع مرکب جمع تابع sum() در اين مثال دو پارامتر دارد: اشارهگر تابع pf و عدد صحيح n : int sum(int (*)(int), int); int square(int); int cube(int); int main() { cout << sum(square,4) << endl; // cout << sum(cube,4) << endl; // }
398
تابع sum() يک پارامتر غير معمول دارد
تابع sum() يک پارامتر غير معمول دارد. نام تابع ديگري به عنوان آرگومان به آن ارسال شده. هنگامي که sum(square,4) فراخواني شود، مقدار square(1)+square(2)+square(3)+square(4) بازگشت داده ميشود. چونsquare(k) مقدار k*k را برميگرداند، فراخواني sum(square,4) مقدار =30 را محاسبه نموده و بازميگرداند. تعريف توابع و خروجي آزمايشي به شکل زير است:
399
int sum(int (*pf)(int k), int n)
{ // returns the sum f(0) + f(1) + f(2) f(n-1): int s = 0; for (int i = 1; i <= n; i++) s += (*pf)(i); return s; } int square(int k) { return k*k; int cube(int k) { return k*k*k; 30 100
400
pf در فهرست پارامترهاي تابع sum() يک اشارهگر به تابع است
pf در فهرست پارامترهاي تابع sum() يک اشارهگر به تابع است. اشارهگر به تابعي که آن تابع پارامتري از نوع int دارد و مقداري از نوع int را برميگرداند. k در تابع sum اصلا استفاده نشده اما حتما بايد قيد شود تا کامپايلر بفهمد که pf به تابعي اشاره دارد که پارامتري از نوع int دارد. عبارت (*pf)(i) معادل با square(i) يا cube(i) خواهد بود، بسته به اين که کدام يک از اين دو تابع به عنوان آرگومان به sum() ارسال شوند.
401
نام تابع، آدرس شروع تابع را دارد
نام تابع، آدرس شروع تابع را دارد. پس square آدرس شروع تابع square() را دارد. بنابراين وقتي تابع sum() به شکل sum(square,4) فراخواني شود، آدرسي که درون square است به اشارهگر pf فرستاده ميشود. با استفاده از عبارت (*pf)(i) مقدار i به آرگومان تابعي فرستاده ميشود که pf به آن اشاره دارد.
402
16- NUL و NULL ثابت صفر (0) از نوع int است اما اين مقدار را به هر نوع بنيادي ديگر ميتوان تخصيص داد: char c = 0; // initializes c to the char '\0' short d = 0; // initializes d to the short int 0 int n = 0; // initializes n to the int 0 unsigned u = 0; // initializes u to the unsigned int 0 float x = 0; // initializes x to the float 0.0 double z = 0; // initializes z to the double 0.0
403
مقدار صفر معناهاي گوناگوني دارد
مقدار صفر معناهاي گوناگوني دارد. وقتي براي اشياي عددي به کار رود، به معناي عدد صفر است. وقتي براي اشياي کاراکتري به کار رود، به معناي کاراکتر تهي يا NUL است. NUL معادل کاراکتر '\0' نيز هست. وقتي مقدار صفر براي اشارهگرها به کار رود، به معناي «هيچ چيز» يا NULL است. NULL يک کلمۀ کليدي است و کامپايلر آن را ميشناسد. هنگامي که مقدار NULL يا صفر در يک اشارهگر قرار ميگيرد، آن اشارهگر به خانه 0x0 در حافظه اشاره دارد. اين خانۀ حافظه، يک خانۀ استثنايي است که قابل پردازش نيست. نه ميتوان آن خانه را مقداريابي کرد و نه ميتوان مقداري را درون آن قرار داد. به همين دليل به NULL «هيچ چيز» ميگويند.
404
وقتي اشارهگري را بدون استفاده از new اعلان ميکنيم، خوب است که ابتدا آن را NULL کنيم تا مقدار زبالۀ آن پاک شود. اما هميشه بايد به خاطر داشته باشيم که اشارهگر NULL را نبايد مقداريابي نماييم: int* p = 0; // p points to NULL *p = 22; // ERROR: cannot dereference the NULL pointer پس خوب است هنگام مقداريابي اشارهگرها، احتياط کرده و بررسي کنيم که آن اشارهگر NULL نباشد: if (p) *p = 22; // O.K. حالا دستور *p=22; وقتي اجرا ميشود که p صفر نباشد. ميدانيد که شرط بالا معادل شرط زير است: if (p != NULL) *p = 22;
405
اشارهگرها را نميتوان ناديده گرفت.
آنها سرعت پردازش را زياد ميکنند و کدنويسي را کم. با استفاده از اشارهگرها ميتوان به بهترين شکل از حافظه استفاده کرد. با به کارگيري اشارهگرها ميتوان اشيايي پيچيدهتر و کارآمدتر ساخت.
406
پايان جلسه هفتم
407
« رشتههاي كاراكتري و فايلها در ++C استاندارد»
جلسه هشتم « رشتههاي كاراكتري و فايلها در ++C استاندارد»
408
آنچه در اين جلسه مي خوانيد
مروري بر اشارهگرها رشتههاي كاراكتري در C ورودي/خروجي رشتههاي کاراکتري چند تابع عضو cin و cout توابع كاراكتري C استاندارد آرايهاي از رشتهها توابع استاندارد رشتههاي کاراکتري ›››
409
رشتههاي کاراکتري در C++ استاندارد نگاهي دقيقتر به تبادل دادهها
ورودي قالببندي نشده نوع string در ++C استاندارد فايلها هدف کلي: آشنايي با کلاسها و اصول اوليۀ بهکارگيري آنها.
410
هدف کلي: معرفي رشتههاي کاراکتري به سبک c و c++ و نحوۀ ايجاد و دستکاري آنها و همچنين نحوۀ استفاده از فايلهاي متني براي ذخيرهسازي و بازيابي اطلاعات.
411
هدفهاي رفتاري: انتظار ميرود پس از پايان اين جلسه بتوانيد:
- رشتههاي کاراکتري به سبک C استاندارد را ايجاد نماييد. - توابع معرفي شده عضو cin و cout را شناخته و وظيفۀ هر يک را شرح دهيد. - رشتههاي کاراکتري به سبک C++ استاندارد را ايجاد نماييد. - مفهوم «ورودي قالببندي شده» و «ورودي قالببندي نشده» را دانسته و هر کدام را در مکانهاي مناسب به کار ببريد. - نوع string را شناخته و رشتههايي از اين نوع ايجاد کنيد و با استفاده از توابع خاص، اين رشتهها را دستکاري نماييد. - اطلاعات کاراکتري و رشتهاي را در يک فايل متني نوشته يا از آن بخوانيد.
412
مقدمه: دادههايي که در رايانهها پردازش ميشوند هميشه عدد نيستند. معمولا لازم است که اطلاعات کاراکتري مثل نام افراد – نشانيها – متون – توضيحات – کلمات و ... نيز پردازش گردند، جستجو شوند، مقايسه شوند، به يکديگر الصاق شوند يا از هم تفکيک گردند. در اين جلسه بررسي ميکنيم که چطور اطلاعات کاراکتري را از ورودي دريافت کنيم و يا آنها را به شکل دلخواه به خروجي بفرستيم. در همين راستا توابعي معرفي ميکنيم که انجام اين کارها را آسان ميکنند.
413
مروري بر اشارهگرها: يك اشارهگر متغيري است که حاوي يک آدرس از حافظه ميباشد. نوع اين متغير از نوع مقداري است که در آن آدرس ذخيره شده. با استفاده از عملگر ارجاع & ميتوان آدرس يک شي را پيدا کرد. همچنين با استفاده از عملگر مقداريابي * ميتوانيم مقداري که در يک آدرس قرار دارد را مشخص کنيم. به تعاريف زير نگاه کنيد: int n = 44; int* p = &n;
414
رشتههاي كاراكتري در C در زبان C++ يك «رشتۀ کاراکتري» آرايهاي از کاراکترهاست که اين آرايه داراي ويژگي مهم زير است: 1- يك بخش اضافي در انتهاي آرايه وجود دارد که مقدار آن، کاراکتر NUL يعني '\0‘ است. پس تعداد کل کاراکترها در آرايه هميشه يکي بيشتر از طول رشته است. 2 – رشتۀ کاراکتري را ميتوان با ليترال رشتهاي به طور مستقيم مقدارگذاري کرد مثل: char str[] = "string"; توجه كنيد كه اين آرايه هفت عنصر دارد: 's' و 't' و 'r' و 'i' و 'n' و 'g' و '\0'
415
3– کل يک رشتۀ کاراکتري را ميتوان مثل يک متغير معمولي چاپ کرد. مثل:
cout << str; در اين صورت، همۀ کاراکترهاي درون رشتۀ کاراکتري str يکي يکي به خروجي ميروند تا وقتي که به کاراکتر انتهايي NUL برخورد شود. 4 – يک رشتۀ کاراکتري را ميتوان مثل يک متغير معمولي از ورودي دريافت کرد مثل: cin >> str; در اين صورت، همۀ کاراکترهاي وارد شده يکي يکي درون str جاي ميگيرند تا وقتي که به يک فضاي خالي در کاراکترهاي ورودي برخورد شود. برنامهنويس بايد مطمئن باشد که آرايۀ str براي دريافت همۀ کاراکترهاي وارد شده جا دارد.
416
5 – توابع تعريف شده در سرفايل <cstring> را ميتوانيم براي دستکاري رشتههاي کاراکتري به کار بگيريم. اين توابع عبارتند از: تابع طول رشته strlen() توابع کپي رشته strcpy() و strncpy() توابع الصاق رشتهها strcat() و strncat() توابع مقايسۀ رشتهها strcmp() و strncmp() و تابع استخراج نشانه strtok() .
417
for (int i = 0; i < 5; i++)
رشتههاي کاراکتري با كاراكتر NUL خاتمه مييابند برنامۀ کوچک زير نشان ميدهد که کاراکتر '\0' به رشتههاي کاراکتري الصاق ميشود: int main() { char s[] = "ABCD"; for (int i = 0; i < 5; i++) cout << "s[" << i << "] = '" << s[i] << "'\n"; }
418
رشتۀ کاراکتري s داراي پنج عضو است که عضو پنجم، کاراکتر '\0' ميباشد
وقتي کاراکتر '\0' به cout فرستاده ميشود، هيچ چيز چاپ نميشود. حتي جاي خالي هم چاپ نميشود. خط آخر خروجي، عضو پنجم را نشان مي دهد که ميان دو علامت آپستروف هيچ چيزي چاپ نشده. S A 1 B 2 C 3 D 4 Ø
419
ورودي/خروجي رشتههاي کاراکتري:
در C++ به چند روش ميتوان رشتههاي کاراکتري را دريافت کرده يا نمايش داد. يک راه استفاده از عملگرهاي کلاس string است که در بخشهاي بعدي به آن خواهيم پرداخت. روش ديگر، استفاده از توابع کمکي است که آن را در ادامه شرح ميدهيم.
420
if (*word) cout << "\t\"" << word << "\"\n";
مثال روش سادۀ دريافت و نمايش رشتههاي کاراکتري: در برنامۀ زير يک رشتۀ کاراکتري به طول 79 کاراکتر اعلان شده و کلماتي که از ورودي خوانده ميشود در آن رشته قرار ميگيرد: int main() { char word[80]; do { cin >> word; if (*word) cout << "\t\"" << word << "\"\n"; } while (*word); }
421
چند تابع عضو cin و cout cin.getline() cin.get() cin.ignore()
cin.putback() cin.peek() همۀ اين توابع شامل پيشوند cin هستند زيرا آنها عضوي از cin ميباشند. به cout شيء فرآيند خروجي ميگويند. اين شي نيز شامل تابع cout.put() است. نحوۀ کاربرد هر يک از اين توابع عضو را در ادامه خواهيم ديد. فراخواني cin.getline(str,n); باعث ميشود که n کاراکتر به درون str خوانده شود و مابقي کاراکترهاي وارد شده ناديده گرفته ميشوند.
422
با دو پارامتر cin.getline() تابع
اين برنامه ورودي را خط به خط به خروجي ميفرستد: int main() { char line[80]; do { cin.getline(line,80); if (*line) cout << "\t[" << line << "]\n"; } while (*line); }
423
با سه پارامتر cin.getlineتابع()
برنامه زير، متن ورودي را جمله به جمله تفکيک مينمايد: int main() { char clause[20]; do { cin.getline(clause, 20, ','); if (*clause) cout << "\t[" << clause << "]\n"; } while (*clause); }
424
cout << count << " e's were counted.\n"; }
تابع cin.get() اين برنامه تعداد حرف 'e' در جريان ورودي را شمارش ميكند. تا وقتي cin.get(ch) کاراکترها را با موفقيت به درون ch ميخواند، حلقه ادامه مييابد: int main() { char ch; int count = 0; while (cin.get(ch)) if (ch = = 'e') ++count; cout << count << " e's were counted.\n"; }
425
تابع cout.put() برنامۀ زير، اولين حرف از هر کلمۀ ورودي را به حرف بزرگ تبديل کرده و آن را مجددا در خروجي چاپ ميکند: int main() { char ch, pre = '\0'; while (cin.get(ch)) { if (pre = = ' ' || pre = = '\n') cout.put(char(toupper(ch))); else cout.put(ch); pre = ch; }
426
cin.ignore() و cin.putback() توابع
با استفاده از برنامۀ زير، تابعي آزمايش ميشود که اين تابع اعداد صحيح را از ورودي استخراج ميکند: int nextInt(); int main() { int m = nextInt(), n = nextInt(); cin.ignore(80,'\n'); // ignore rest of input line cout << m << " + " << n << " = " << m+n << endl; } int nextInt() { char ch; int n; while (cin.get(ch)) if (ch >= '0' && ch <= '9') // next character is a digit { cin.putback(ch); // put it back so it can be cin >> n; // read as a complite int break; return n;
427
تابع cin.peek() اين نسخه از تابع nextInt() معادل آن است كه در مثال قبلي بود: int nextInt() { char ch; int n; while (ch = cin.peek()) if (ch >= '0' && ch <= '9') { cin >> n; break; } else cin.get(ch); return n;
428
توابع كاراكتري C استاندارد
در مثال 6-8 به تابعtoupper() اشاره شد. اين فقط يکي از توابعي است که براي دستکاري کاراکترها استفاده ميشود. ساير توابعي که در سرفايل <ctype.h> يا <cctype> تعريف شده به شرح زير است: شرح نام تابع int isalnum(int c); اگر c کاراکتر الفبايي يا عددي باشد مقدار غيرصفر وگرنه صفر را برميگرداند isalnum() int isalpha(int c); اگر c کاراکتر الفبايي باشد مقدار غيرصفر و در غير آن، صفر را برميگرداند isalpha()
429
شرح نام تابع int iscntrl(int c); اگر c کاراکتر کنترلي باشد مقدار غيرصفر و در غير آن، صفر را برميگرداند iscntrl() int isdigit(int c); اگر c کاراکتر عددي باشد، مقدار غيرصفر و در غير آن، صفر را برميگرداند isdigit() int isgraph(int c); اگر c کاراکتر چاپي و غيرخالي باشد مقدار غيرصفر وگرنه صفر را برميگرداند isgraph() int islower(int c); اگر c حرف کوچک باشد مقدار غيرصفر و در غير آن، صفر را برميگرداند islower() int isprint(int c); اگر c کاراکتر قابل چاپ باشد مقدار غيرصفر و در غير آن، صفر را برميگرداند isprint()
430
ispunct() isspace() isupper() isxdigit() tolower() toupper()
شرح نام تابع int ispunct(int c); اگر c کاراکتر چاپي به غير از حروف و اعداد و فضاي خالي باشد، مقدار غيرصفر برميگرداند وگرنه مقدار صفر را برميگرداند ispunct() int isspace(int c); اگر c کاراکتر فضاي سفيد شامل فضاي خالي ' ' و عبور فرم '\f' و خط جديد '\n' و بازگشت نورد '\r' و پرش افقي '\t' و پرش عمودي '\v' باشد، مقدار غيرصفر را برميگرداند وگرنه صفر را برميگرداند isspace() int isupper(int c); اگر c حرف بزرگ باشد، مقدار غيرصفر برميگرداند وگرنه صفر را برميگرداند isupper() int isxdigit(int c); اگر c يکي از ده کاراکتر عددي يا يکي از دوازده حرف عدد شانزدهدهي شامل 'a' و 'b' و 'c' و 'd' و 'e' و 'f' و 'A' و 'B' و 'C' و 'D' و 'E' و 'F' باشد، مقدار غيرصفر برميگرداند وگرنه مقدار صفر را برميگرداند isxdigit() int tolower(int c); اگر c حرف بزرگ باشد، کاراکتر کوچک معادل آن را برميگرداند وگرنه خود c را برميگرداند tolower() int toupper(int c); اگر c حرف کوچک باشد، کاراکتر بزرگ معادل آن را برميگرداند وگرنه خود c را برميگرداند toupper()
431
توجه کنيد که همۀ توابع فوق يک پارامتر از نوع int دريافت ميکنند و يک مقدار int را برميگردانند. علت اين است که نوع char در اصل يک نوع صحيح است. در عمل وقتي توابع فوق را به کار ميبرند، يک مقدار char به تابع ميفرستند و مقدار بازگشتي را نيز در يک char ذخيره ميکنند. به همين خاطر اين توابع را به عنوان «توابع کاراکتري» در نظر ميگيريم.
432
آرايهاي از رشتهها به خاطر داريد که گفتيم يک آرايۀ دوبعدي در حقيقت آرايهاي يک بعدي است که هر کدام از اعضاي آن يک آرايۀ يک بعدي ديگر است. مثلا در آرايۀ دو بعدي که به شکل مقابل اعلان شده باشد: char name[5][20]; اين آرايه در اصل پنج عضو دارد که هر عضو ميتواند بيست کاراکتر داشته باشد. اگر آرايۀ فوق را با تعريف رشتههاي کاراکتري مقايسه کنيم، نتيجه اين ميشود که آرايۀ بالا يک آرايۀ پنج عنصري است که هر عنصر آن يک رشتۀ کاراکتري بيست حرفي است. اين آرايه را ميتوانيم به شکل مقابل تصور کنيم.
433
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 از طريق name[0] و name[1] و name[2] و name[3] و name[4] ميتوانيم به هر يک از رشتههاي کاراکتري در آرايۀ بالا دسترسي داشته باشيم. يعني آرايۀ name گرچه به صورت يک آرايۀ دوبعدي اعلان شده ليکن به صورت يک آرايۀ يک بعدي با آن رفتار ميشود.
434
آرايهاي از رشتههاي کاراکتري
برنامۀ زير چند رشتۀ کاراکتري را از ورودي ميخواند و آنها را در يک آرايه ذخيره کرده و سپس مقادير آن آرايه را چاپ ميکند: int main() { char name[5][20]; int count=0; cout << "Enter at most 4 names with at most 19 characters:\n"; while (cin.getline(name[count++], 20)) ; --count; cout << "The names are:\n"; for (int i=0; i<count; i++) cout << "\t" << i << ". [" << name[i] << "]" << endl; }
435
يك آرايۀ رشتهاي پويا
اين برنامه نشان ميدهد که چگونه ميتوان از کاراکتر '$' به عنوان کاراکتر نگهبان در تابع getline() استفاده کرد. مثال زير تقريبا معادل مثال 9-9 است. برنامۀ زير مجموعهاي از اسامي را ميخواند، طوري که هر اسم روي يک خط نوشته ميشود و هر اسم با کاراکتر '\n' پايان مييابد. اين اسامي در آرايۀ name ذخيره ميشوند. سپس نامهاي ذخيره شده در آرايۀ name چاپ ميشوند:
436
int main() { char buffer[80]; cin.getline(buffer,80,'$'); char* name[4]; name[0] = buffer; int count = 0; for (char* p=buffer; *p ! '\0'; p++) if (*p == '\n') { *p = '\0'; // end name[count] name[++count] = p+1; // begin next name } cout << "The names are:\n"; for (int i=0; i<count; i++) cout << "\t" << i << . [" << name[i] << "]" << endl;
437
مقداردهي يك آرايۀ رشتهاي
اين برنامه هم آرايۀ رشتهاي name را مقداردهي کرده و سپس مقادير آن را چاپ مينمايد: int main() {char* name[]= { "Mostafa Chamran", "Mehdi Zeinoddin", "Ebrahim Hemmat" }; cout << "The names are:\n"; for (int i = 0; i < 3; i++) cout << "\t" << i << ". [" << name[i] << "]" << endl; }
438
توابع استاندارد رشتههاي کاراکتري:
سرفايل <cstring> که به آن «کتابخانۀ رشتههاي کاراکتري» هم ميگويند، شامل خانوادۀ توابعي است که براي دستکاري رشتههاي کاراکتري خيلي مفيدند. مثال بعدي سادهترين آنها يعني تابع طول رشته را نشان ميدهد. اين تابع، طول يک رشتۀ کاراکتري ارسال شده به آن (يعني تعداد کاراکترهاي آن رشته) را برميگرداند.
439
تابع strlen(): برنامۀ زير يک برنامۀ آزمون ساده براي تابع strlen() است. وقتي strlen(s) فراخواني ميشود، تعداد کاراکترهاي درون رشتۀ s که قبل از کاراکتر NUL قرار گرفتهاند، بازگشت داده ميشود: #include <cstring> int main() { char s[] = "ABCDEFG"; cout << "strlen(" << s << ") = " << strlen(s) << endl; cout << "strlen(\"\") = " << strlen("") << endl; char buffer[80]; cout << "Enter string: "; cin >> buffer; cout << "strlen(" << buffer << ") = " << strlen(buffer) << endl; }
440
توابع strrchr(), strchr(), strstr():
#include <cstring> int main() { char s[] = "The Mississippi is a long river."; cout << "s = \"" << s << "\"\n"; char* p = strchr(s, ' '); cout << "strchr(s, ' ') points to s[" << p - s << "].\n"; p = strchr(s, 'e'); cout << "strchr(s, 'e') points to s[" << p - s << "].\n"; p = strrchr(s, 'e'); cout << "strrchr(s, 'e') points to s[" << p - s << "].\n"; p = strstr(s, "is"); cout << "strstr(s, \"is\") points to s[" << p – s << "].\n"; p = strstr(s, "isi"); if (p == NULL) cout << "strstr(s, \"isi\") returns NULL\n"; }
441
تابع strcpy(): برنامۀ زير نشان ميدهد که فراخواني strcpy(s1, s2) چه تاثيري دارد: #include <iostream> #include <cstring> int main() { char s1[] = "ABCDEFG"; char s2[] = "XYZ"; cout << "Before strcpy(s1,s2):\n"; cout << "\ts1 = [" << s1 << "], length = " << strlen(s1) << endl; cout << "\ts2 = [" << s2 << "], length = " << strlen(s2) strcpy(s1,s2); cout << "After strcpy(s1,s2):\n"; }
442
تابع :strncpy() برنامۀ زير بررسي ميکند که فراخوانيstrncpy(s1, s2, n) چه اثري دارد: int main() { char s1[] = "ABCDEFG"; char s2[] = "XYZ"; cout << "Before strncpy(s1,s2,2):\n"; cout << "\ts1 = [" << s1 << "], length = " << strlen(s1) << endl; cout << "\ts2 = [" << s2 << "], length = " << strlen(s2) strncpy(s1,s2,2); cout << "After strncpy(s1,s2,2):\n"; }
443
تابع الصاق رشته :strcat()
برنامۀ زير بررسي ميکند که فراخواني strcat(s1, s2) چه تاثيري دارد: int main() { char s1[] = "ABCDEFG"; char s2[] = "XYZ"; cout << "Before strcat(s1,s2):\n"; cout << "\ts1 = [" << s1 << "], length = " << strlen(s1) << endl; cout << "\ts2 = [" << s2 << "], length = " << strlen(s2) strcat(s1,s2); cout << "After strcat(s1,s2):\n"; }
444
رشتههاي کاراکتري در C++ استاندارد :
هستند. اما اين سرعت پردازش، هزينهاي هم دارد: خطر خطاهاي زمان اجرا. ين خطاها معمولا از اين ناشي ميشوند که فقط بر کاراکتر NUL به عنوان پايان رشته تکيه ميشود. C++ رشتههاي کاراکتري خاصي نيز دارد که امنتر و مطمئنتر هستند. در اين رشتهها، طول رشته نيز درون رشته ذخيره ميشود و لذا فقط به کاراکتر NUL براي مشخص نمودن انتهاي رشته اکتفا نميشود.
445
نگاهي دقيقتر به تبادل دادهها
وقتي ميخواهيم دادههايي را وارد کنيم، اين دادهها را در قالب مجموعهاي از کاراکترها تايپ ميکنيم. همچنين وقتي ميخواهيم نتايجي را به خارج از برنامه بفرستيم، اين نتايج در قالب مجموعهاي از کاراکترها نمايش داده ميشوند. لازم است که اين کاراکترها به نحوي براي برنامه تفسير شوند. مثلا وقتي قصد داريم يک عدد صحيح را وارد کنيم، چند کاراکتر عددي تايپ ميکنيم
446
حالا ساز و کاري لازم است که از اين کاراکترها يک مقدار صحيح بسازد و به برنامه تحويل دهد. همچنين وقتي قصد داريم يک عدد اعشاري را به خروجي بفرستيم، بايد با استفاده از راهکاري، آن عدد اعشاري به کاراکترهايي تبديل شود تا در خروجي نمايش يابد. C++ بر عهده دارند.
447
جريانها اين وظايف را در C++ بر عهده دارند
جريانها اين وظايف را در C++ بر عهده دارند. جريانها شبيه پالايهاي هستند که دادهها را به کاراکتر تبديل ميکنند و کاراکترها را به دادههايي از يک نوع بنيادي تبديل مينمايند. به طور کلي، وروديها و خروجيها را يک کلاس جريان به نام stream کنترل ميکند. اين کلاس خود به زيرکلاسهايي تقسيم ميشود:
448
شيء istream جرياني است که دادههاي مورد نياز را از کاراکترهاي وارد شده از صفحه کليد، فراهم ميکند. شيء ostream جرياني است که دادههاي حاصل را به کاراکترهاي خروجي قابل نمايش روي صفحۀ نمايشگر تبديل مينمايد. شيء ifstream جرياني است که دادههاي مورد نياز را از دادههاي داخل يک فايل، فراهم ميکند. شيء ofstream جرياني است که دادههاي حاصل را درون يک فايل ذخيره مينمايد. اين جريانها و طريقۀ استفاده از آنها را در ادامه خواهيم ديد.
449
cout << "n = " << n << endl; }
استفاده از عملگر بيرونكشي براي كنترل کردن يك حلقه : int main() { int n; while (cin >> n) cout << "n = " << n << endl; }
450
ورودي قالببندي نشده:
سرفايل <iostream> توابع مختلفي براي ورودي دارد. اين توابع براي وارد کردن کاراکترها و رشتههاي کاراکتري به کار ميروند که کاراکترهاي فضاي سفيد را ناديده نميگيرند. رايجترين آنها، تابع cin.get() براي دريافت يک کاراکتر تکي و تابع cin.getline() براي دريافت يک رشتۀ کاراکتري است.
451
{ if (c >= 'a' && c <= 'z') c += 'A' - 'a'; // capitalize c
دريافت كاراكترها با استفاده از تابع :cin.get() while (cin.get(c)) { if (c >= 'a' && c <= 'z') c += 'A' - 'a'; // capitalize c cout.put(c); }
452
وارد كردن يک رشتۀ کاراکتري به وسيلۀ تابع :cin.getline()
برنامۀ زير نشان ميدهد که چطور ميتوان دادههاي متني را خط به خط از ورودي خوانده و درون يک آرايۀ رشتهاي قرار داد::: const int LEN=32; // maximum word length const int SIZE=10; // array size typedef char Name[LEN]; // defines Name to be a C_string type int main() { Name martyr[SIZE]; // defines martyr to be an array of 10 names int n=0; while(cin.getline(martyr[n++], LEN) && n<SIZE) ; --n; for (int i=0; i<n; i++) cout << '\t' << i+1 << ". " << martyr[i] << endl; }
453
نوع string در ++C استاندارد:
در C++ استاندارد نوع دادهاي خاصي به نام string وجود دارد که مشخصات اين نوع در سرفايل <string> تعريف شده است. براي آشنايي با اين نوع جديد، از طريقۀ اعلان و مقداردهي آن شروع ميکنيم. اشيايي که از نوع string هستند به چند طريق ميتوانند اعلان و مقداردهي شوند: string s1; // s1 contains 0 characters string s2 = "PNU University"; // s2 contains 14 characters string s3(60, '*'); // s3 contains 60 asterisks string s4 = s3; // s4 contains 60 asterisks string s5(s2, 4, 2); // s5 is the 2-character string "Un"
454
استفاده از نوع string
کد زير يک مجموعه کاراکتر را از ورودي ميگيرد و سپس بعد از هر کاراکتر "E" يک علامت ويرگول ',' اضافه مينمايد. مثلا اگر عبارت "The SOFTWARE MOVEMENT is began" وارد شود، برنامۀ زير، آن را به جملۀ زير تبديل ميکند: The SOFTWARE, MOVE,ME,NT is began متن برنامه اين چنين است: string word; int k; while (cin >> word) { k = word.find("E") + 1; if (k < word.length()) word.relace(k, 0, ","); cout << word << ' '; }
455
فايلها يکي از مزيتهاي رايانه، قدرت نگهداري اطلاعات حجيم است. فايلها اين قدرت را به رايانه ميدهند. اگر چيزي به نام فايل وجود نميداشت، شايد رايانهها به شکل امروزي توسعه و کاربرد پيدا نميکردند. چون اغلب برنامههاي امروزي با فايلها سر و کار دارند، يک برنامهنويس لازم است که با فايل آشنا باشد و بتواند با استفاده از اين امکان ذخيره و بازيابي، کارايي برنامههايش را ارتقا دهد.
456
پردازش فايل در C++ بسيار شبيه تراکنشهاي معمولي ورودي و خروجي است زيرا اينها همه از اشياي جريان مشابهي بهره ميبرند. جريان fstream براي تراکنش برنامه با فايلها به کار ميرود. fstream نيز به دو زيرشاخۀ ifstream و ofstream تقسيم ميشود. جريان ifstream براي خواندن اطلاعات از يک فايل به کار ميرود و جريان ofstream براي نوشتن اطلاعات درون يک فايل استفاده ميشود.
457
ifstream readfile("INPUT.TXT"); ofstream writefile("OUTPUT.TXT");
پس بايد دستور پيشپردازندۀ #include <fstream> را به ابتداي برنامه بيافزاييد. سپس ميتوانيد عناصري از نوع جريان فايل به شکل زير تعريف کنيد: ifstream readfile("INPUT.TXT"); ofstream writefile("OUTPUT.TXT"); طبق کدهاي فوق، readfile عنصري است که دادهها را از فايلي به نام INPUT.TXT ميخواند و writefile نيز عنصري است که اطلاعاتي را در فايلي به نام OUTPUT.TXT مينويسد. اکنون ميتوان با استفاده از عملگر >> دادهها را به درون readfile خواند و با عملگر << اطلاعات را درون writefile نوشت. به مثال زير توجه کنيد.
458
يک دفتر تلفن برنامۀ زير، چند نام و تلفن مربوط به هر يک را به ترتيب از کاربر دريافت کرده و در فايلي به نام PHONE.TXT ذخيره ميکند. کاربر براي پايان دادن به ورودي بايد عدد 0 را تايپ کند. #include <fstream> #include <iostream> using namespace std; int main() { ofstream phonefile("PHONE.TXT"); long number; string name; cout << "Enter a number for each name. (0 for quit): "; for ( ; ; ) { cout << "Number: "; cin >> number; if (number == 0) break; phonefile << number << ' '; cout << "Name: "; cin >> name; phonefile << name << ' '; cout << endl; }
459
جستجوي يک شماره در دفتر تلفن
اين برنامه، فايل توليد شده توسط برنامۀ قبل را به کار ميگيرد و درون آن به دنبال يک شماره تلفن ميگردد: #include <fstream> #include <iostream> using namespace std; int main() { ifstream phonefile("PHONE.TXT"); long number; string name, searchname; bool found=false; cout << "Enter a name for findind it's phone number: "; cin >> searchname; cout << endl; while (phonefile >> number) { phonefile >> name; if (searchname == name) { cout << name << ' ' << number << endl; found = true; } if (!found) cout << searchname << " is not in this phonebook." << endl; }
460
پايان جلسه هشتم
461
جلسه نهم «شيگرايي»
462
آنچه در اين جلسه مي خوانيد:
1- اعلان كلاسها 2- سازندهها 3- فهرست مقداردهي در سازندهها 4- توابع دستيابي 5- توابع عضو خصوصي 6- سازندۀ كپي ›››
463
7- نابود کننده 8 - اشياي ثابت 9- اشارهگر به اشيا 10- اعضاي دادهاي ايستا 11- توابع عضو ايستا
464
آشنايي با کلاسها و اصول اوليۀ بهکارگيري آنها.
هدف کلي : آشنايي با کلاسها و اصول اوليۀ بهکارگيري آنها.
465
هدفهاي رفتاري: انتظار ميرود پس از پايان اين جلسه بتوانيد:
- نحوۀ اعلان «کلاسها» را بدانيد و اعضاي يک کلاس را برشماريد. - تفاوت بين اعضاي «عمومي» و «خصوصي» را شرح داده و نحوۀ اعلان هر کدام را بيان کنيد. - «تابع سازنده» را شناخته و وظيفۀ آن را شرح دهيد. - روشهاي گوناگون مقداردهي با استفاده از فهرست مقداردهي را بدانيد. - «تابع سازندۀ کپي» را معرفي کرده و وظيفۀ آن را شرح دهيد. - اعضاي «ايستا» را تعريف کرده و نحوۀ اعلان و استفاده از آنها را بدانيد.
466
اشيا را ميتوان با توجه به مشخصات ورفتار آنها دسته بندي کرد.
مقدمه «شيگرايي» رهيافت جديدي بود که براي پاره اي از مشکلات برنامه نويسي راه حل داشت. اين مضمون از دنياي فلسفه به جهان برنامهنويسي آمد و کمک کرد تا معضلات توليد و پشتيباني نرمافزار کمتر شود. اشيا را ميتوان با توجه به مشخصات ورفتار آنها دسته بندي کرد.
467
در بحث شيگرايي به دستهها «کلاس» ميگويند و به نمونههاي هر کلاس «شي» گفته ميشود.
مشخصات هر شي را «صفت» مينامند و به رفتارهاي هر شي «متد» ميگويند.
468
برنامهنويسي شيگرا بر سه ستون استوار است:
الف. بستهبندي: يعني اين که دادههاي مرتبط، با هم ترکيب شوند و جزييات پيادهسازي مخفي شود. ب. وراثت: در دنياي واقعي، وراثت به اين معناست که يک شي وقتي متولد ميشود، خصوصيات و ويژگيهايي را از والد خود به همراه دارد .
469
امتياز وراثت در اين است که از کدهاي مشترک استفاده ميشود و علاوه بر اين که ميتوان از کدهاي قبلي استفاده مجدد کرد، در زمان نيز صرفهجويي شده و استحکام منطقي برنامه هم افزايش مييابد.
470
ج. چند ريختي: که به آن چندشکلي هم ميگويند به معناي يک چيز بودن و چند شکل داشتن است. چندريختي بيشتر در وراثت معنا پيدا ميکند.
471
اعلان کلاس با کلمۀ کليدي class شروع ميشودسپس نام کلاس ميآيد.
اعلان كلاسها کد زير اعلان يک کلاس را نشان ميدهد. class Ratio { public: void assign(int, int); viod print(); private: int num, den; }; اعلان کلاس با کلمۀ کليدي class شروع ميشودسپس نام کلاس ميآيد.
472
اعلان اعضاي کلاس درون يک بلوک انجام ميشود و سرانجام يک سميکولن بعد از بلوک نشان ميدهد که اعلان کلاس پايان يافته است. عبارت public و عبارت private . هر عضوي که ذيل عبارت public اعلان شود، يک «عضو عمومي» محسوب ميشود و هر عضوي که ذيل عبارت private اعلان شود، يک «عضو خصوصي» محسوب ميشود.
473
سازندهها وظيفۀ تابع سازنده اين است که حافظۀ لازم را براي شيء جديد تخصيص داده و آن را مقداردهي نمايد و با اجراي وظايفي که در تابع سازنده منظور شده، شيء جديد را براي استفاده آماده کند. هر کلاس ميتواند چندين سازنده داشته باشد. در حقيقت تابع سازنده ميتواند چندشکلي داشته باشد.
474
سازندهها، از طريق فهرست پارامترهاي متفاوت از يکديگر تفکيک ميشوند
سازندهها، از طريق فهرست پارامترهاي متفاوت از يکديگر تفکيک ميشوند. به مثال بعدي نگاه کنيد. مثال 5-9 افزودن چند تابع سازندۀ ديگر به كلاس Ratio class Ratio { public: Ratio() { num = 0; den = 1; } Ratio(int n) { num = n; den = 1; } Ratio(int n, int d) { num = n; den = d; } void print() { cout << num << '/' << den; } private: int num, den; };
475
سومين سازنده نيز همان سازندۀ مثال 4-2 است.
اين نسخه از كلاس Ratio سه سازنده دارد: اولي هيچ پارامتري ندارد و شيء اعلان شده را با مقدار پيشفرض 0 و 1 مقداردهي ميکند. دومين سازنده يک پارامتر از نوع int دارد و شيء اعلان شده را طوري مقداردهي ميکند که حاصل کسر با مقدار آن پارامتر برابر باشد. سومين سازنده نيز همان سازندۀ مثال 4-2 است.
476
يک کلاس ميتواند سازندههاي مختلفي داشته باشد
يک کلاس ميتواند سازندههاي مختلفي داشته باشد. سادهترين آنها، سازندهاي است که هيچ پارامتري ندارد. به اين سازنده سازندۀ پيشفرض ميگويند. اگر در يک کلاس، سازندۀ پيشفرض ذکر نشود، کامپايلر به طور خودکار آن را براي کلاس مذکور ايجاد ميکند.
477
فهرست مقداردهي در سازندهها
سازندهها اغلب به غير از مقداردهي دادههاي عضو يک شي، کار ديگري انجام نميدهند. به همين دليل در C++ يک واحد دستوري مخصوص پيشبيني شده که توليد سازنده را تسهيل مينمايد. اين واحد دستوري فهرست مقداردهي نام دارد.
478
به سومين سازنده در مثال 5-9 دقت کنيد
به سومين سازنده در مثال 5-9 دقت کنيد. اين سازنده را ميتوانيم با استفاده از فهرست مقداردهي به شکل زير خلاصه کنيم: Ratio(int n, int d) : num(n), den(d) { } مثال 6-9 استفاده از فهرست مقداردهي در كلاس Ratio class Ratio { public: Ratio() : num(0) , den(1) { } Ratio(int n) : num(n) , den(1) { } Ratio(int n, int d) : num(n), den(d) { } private: int num, den; };
479
توابع دستيابي دادههاي عضو يک کلاس معمولا به صورت خصوصي (private) اعلان ميشوند تا دستيابي به آنها محدود باشد اما همين امر باعث ميشود که نتوانيم در مواقع لزوم به اين دادهها دسترسي داشته باشيم. براي حل اين مشکل از توابعي با عنوان توابع دستيابي استفاده ميکنيم.
480
تابع دستيابي يک تابع عمومي عضو کلاس است و به همين دليل اجازۀ دسترسي به اعضاي دادهاي خصوصي را دارد.
با استفاده از توابع دستيابي فقط ميتوان اعضاي دادهاي خصوصي را خواند ولي نميتوان آنها را دستکاري کرد.
481
مثال 8-9 افزودن توابع دستيابي به كلاس Ratio
class Ratio { public: Ratio(int n=0, int d=1) : num(n) , den(d) { } int numerator() { return num; } int denomerator() { return den; } private: int num, den; }; در اينجا توابع numerator() و denumerator() مقادير موجود در دادههاي عضو خصوصي را نشان ميدهند.
482
توابع عضو خصوصي توابع عضو را گاهي ميتوانيم به شکل يک عضو خصوصي کلاس معرفي کنيم. واضح است که چنين تابعي از داخل برنامۀ اصلي به هيچ عنوان قابل دستيابي نيست. اين تابع فقط ميتواند توسط ساير توابع عضو کلاس دستيابي شود. به چنين تابعي يک تابع سودمند محلي ميگوييم.
483
در برنامه زير، کلاس Ratio داراي يک تابع عضو خصوصي به نام toFloat() است
مثال 9-9 استفاده از توابع عضو خصوصي class Ratio { public: Ratio(int n=0, int d=1) : num(n), den(d) { } void print() { cout << num << '/' << den << endl; } void printconv() { cout << toFloat() << endl; } private: int num, den; double toFloat(); }; اين تابع فقط درون بدنۀ تابع عضو printconv() استفاده شده و به انجام وظيفۀ آن کمک مينمايد و هيچ نقشي در برنامۀ اصلي ندارد.
484
سازندۀ كپي int x; int x=k;
ميدانيم که به دو شيوه ميتوانيم متغير جديدي تعريف نماييم: int x; int x=k; در روش اول متغيري به نام x از نوع int ايجاد ميشود. در روش دوم هم همين کار انجام ميگيرد با اين تفاوت که پس از ايجاد x مقدار موجود در متغير k که از قبل وجود داشته درون x کپي ميشود. اصطلاحا x يک کپي از k است.
485
Ratio y(x); کد بالا يک شي به نام y از نوع Ratio ايجاد ميکند و تمام مشخصات شيء x را درون آن قرار ميدهد. اگر در تعريف کلاس، سازندۀ کپي ذکر نشود (مثل همۀ کلاسهاي قبلي) به طور خودکار يک سازندۀ کپي پيشفرض به کلاس افزوده خواهد شد.
486
مثال 10-9 افزودن يك سازندۀ كپي به كلاس Ratio
{ public: Ratio(int n=0, int d=1) : num(n), den(d) { } Ratio(const Ratio& r) : num(r.num), den(r.den) { } void print() { cout << num << '/' << den; } private: int num, den; }; در مثال بالا، تابع سازندۀ کپي طوري تعريف شده که عنصرهاي num و den از پارامتر r به درون عنصرهاي متناظر در شيء جديد کپي شوند.
487
1 – وقتي که يک شي هنگام اعلان از روي شيء ديگر کپي شود.
سازندۀ کپي در سه وضعيت فرا خوانده ميشود: 1 – وقتي که يک شي هنگام اعلان از روي شيء ديگر کپي شود. 2 – وقتي که يک شي به وسيلۀ مقدار به يک تابع ارسال شود. 3 – وقتي که يک شي به وسيلۀ مقدار از يک تابع بازگشت داده شود .
488
هر کلاس فقط يک نابودکننده دارد.
نابود کننده هر کلاس فقط يک نابودکننده دارد. وقتي که يک شي ايجاد ميشود، تابع سازنده به طور خودکار براي ساختن آن فراخواني ميشود. وقتي که شي به پايان زندگياش برسد، تابع عضو ديگري به طور خودکار فراخواني ميشود تا نابودکردن آن شي را مديريت کند. اين تابع عضو، نابودکننده ناميده ميشود . سازنده وظيفه دارد تا منابع لازم را براي شي تخصيص دهد و نابودکننده وظيفه دارد آن منابع را آزاد کند.
489
مثال 12-9 افزودن يك نابودكننده به كلاس Ratio
class Ratio { public: Ratio() { cout << "OBJECT IS BORN.\n"; } ~Ratio() { cout << "OBJECT DIES.\n"; } private: int num, den; };
490
اشياي ثابت اگر قرار است شيئي بسازيد که در طول اجراي برنامه هيچگاه تغيير نميکند، بهتر است منطقي رفتار کنيد و آن شي را به شکل ثابت اعلان نماييد. اعلانهاي زير چند ثابت آشنا را نشان ميدهند: اشيا را نيز ميتوان با استفاده از عبارت const به صورت يک شيء ثابت اعلان کرد: const Ratio PI(22,7); const char BLANK = ' '; const int MAX_INT = ; const double PI = ; void int(float a[], const int SIZE);
491
اشارهگر به اشيا ميتوانيم اشارهگر به اشياي کلاس نيز داشته باشيم. از آنجا که يک کلاس ميتواند اشياي دادهاي متنوع و متفاوتي داشته باشد، اشارهگر به اشيا بسيار سودمند و مفيد است. اشارهگر به اشيا براي ساختن فهرستهاي پيوندي و درختهاي دادهاي به کار ميرود.
492
مثال 13-9 استفاده از اشارهگر به اشيا
class X { public: int data; }; main() { X* p = new X; (*p).data = 22; // equivalent to: p->data = 22; cout << "(*p).data = " << (*p).data << " = " << p->data << endl; p->data = 44; cout << " p->data = " << (*p).data << " = " << p->data << endl; }
493
در اين مثال، p اشارهگري به شيء x است. پس. p يک شيء x است و (. p)
در اين مثال، p اشارهگري به شيء x است. پس *p يک شيء x است و (*p).data دادۀ عضو آن شي را دستيابي ميکند. حتما بايد هنگام استفاده از *p آن را درون پرانتز قرار دهيد زيرا عملگر انتخاب عضو (.) تقدم بالاتري نسبت به عملگر مقداريابي (*) دارد. اگر پرانتزها قيد نشوند و فقط *p.data نوشته شود، کامپايلر اين خط را به صورت *(p.data) تفسير خواهد کرد که اين باعث خطا ميشود.
494
Node(int d, Node* p=0) : data(d), next(p) { } int data; Node* next; };
مثال بعدي اهميت بيشتري دارد و کاربرد اشارهگر به اشيا را بهتر نشان ميدهد. مثال 14-9 فهرستهاي پيوندي با استفاده از كلاس Node به کلاسي که در زير اعلان شده دقت کنيد: class Node { public: Node(int d, Node* p=0) : data(d), next(p) { } int data; Node* next; };
495
عبارت بالا کلاسي به نام Node تعريف ميکند که اشياي اين کلاس داراي دو عضو دادهاي هستند که يکي متغيري از نوع int است و ديگري يک اشارهگر از نوع همين کلاس. اين کار واقعا ممکن است و باعث ميشود بتوانيم يک شي را با استفاده از همين اشارهگر به شيء ديگر پيوند دهيم و يک زنجيره بسازيم.
496
اگر اشياي q و r و s از نوع Node باشند، ميتوانيم پيوند اين سه شي را به صورت زير مجسم کنيم:
int data q Node* next r s
497
for ( ; p->next; p = p->next)
به تابع سازنده نيز دقت کنيد که چطور هر دو عضو دادهاي شيء جديد را مقداردهي ميکند. int main() { int n; Node* p; Node* q=0; while (cin >> n) { p = new Node(n, q); q = p; } for ( ; p->next; p = p->next) cout << p->data << " -> "; cout << "*\n";
498
ج - پس از دومين تکرار حلقه
int data q شکل زير روند اجراي برنامه را نشان ميدهد. Node* next الف - قبل از شروع حلقه int data q ب - پس از اولين تکرار حلقه int data p Node* next Node* next int data q int data int data p Node* next Node* next Node* next ج - پس از دومين تکرار حلقه
499
اعضاي دادهاي ايستا هر وقت که شيئي از روي يک کلاس ساخته ميشود، آن شي مستقل از اشياي ديگر، دادههاي عضو خاص خودش را دارد. گاهي لازم است که مقدار يک عضو دادهاي در همۀ اشيا يکسان باشد. اگر اين عضو مفروض در همۀ اشيا تکرار شود، هم از کارايي برنامه ميکاهد و هم حافظه را تلف ميکند. در چنين مواقعي بهتر است آن عضو را به عنوان يک عضو ايستا اعلان کنيم.
500
static int n; // declaration of n as a static data member
class X { public: static int n; // declaration of n as a static data member }; int X::n = 0; // definition of n خط آخر نشان ميدهد که متغيرهاي ايستا را بايد به طور مستقيم و مستقل از اشيا مقداردهي کرد.
501
متغيرهاي ايستا به طور پيشفرض با صفر مقداردهي اوليه ميشوند
متغيرهاي ايستا به طور پيشفرض با صفر مقداردهي اوليه ميشوند. بنابراين مقداردهي صريح به اين گونه متغيرها ضروري نيست مگر اين که بخواهيد يک مقدار اوليۀ غير صفر داشته باشيد. مثال 15-9 يك عضو دادهاي ايستا کد زير، کلاسي به نام widget اعلان ميکند که اين کلاس يک عضو دادهاي ايستا به نام count دارد. اين عضو، تعداد اشياي widget که موجود هستند را نگه ميدارد. هر وقت که يک شيء widget ساخته ميشود، از طريق سازنده مقدار count يک واحد افزايش مييابد و هر زمان که يک شيء widget نابود ميشود، از طريق نابودکننده مقدار count يک واحد کاهش مييابد:
502
class Widget { public: Widget() { ++count; } ~Widget() { --count; } static int count; }; int Widget::count = 0; main() { Widget w, x; cout << "Now there are " << w.count << " widgets.\n"; { Widget w, x, y, z; } Widget y; Now there are 2 widgets. Now there are 6 widgets. Now there are 3 widgets.
503
توجه کنيد که چگونه چهار شيء widget درون بلوک داخلي ايجاد شده است
توجه کنيد که چگونه چهار شيء widget درون بلوک داخلي ايجاد شده است. هنگامي که اجراي برنامه از آن بلوک خارج ميشود، اين اشيا نابود ميشوند و لذا تعداد کل widgetها از 6 به 2 تقليل مييابد. يک عضو دادهاي ايستا مثل يک متغير معمولي است: فقط يک نمونه از آن موجود است بدون توجه به اين که چه تعداد شي از آن کلاس موجود باشد. از آنجا که عضو دادهاي ايستا عضوي از کلاس است، ميتوانيم آن را به شکل يک عضو خصوصي نيز اعلان کنيم.
504
* مثال 16-9 يك عضو دادهاي ايستا و خصوصي
class Widget { public: Widget() { ++count; } ~Widget() { --count; } int numWidgets() { return count; } private: static int count; };
505
int Widget::count = 0; main() { Widget w, x; cout << "Now there are " << w.numWidgets() << " widgets.\n"; { Widget w, x, y, z; } Widget y;
506
اين برنامه مانند مثال 15-9 کار ميکند با اين تفاوت که متغير ايستاي count به شکل يک عضو خصوصي اعلان شده و به همين دليل به تابع دستيابي numWidgets() نياز داريم تا بتوانيم درون برنامۀ اصلي به متغير count دسترسي داشته باشيم. ميتوانيم کلاس Widget و اشياي x و y و w را مانند مقابل تصور کنيم: Widget() ~Widget() numWidgets() count 3 Widget x y w
507
12- توابع عضو ايستا با دقت در مثال قبلي به دو ايراد بر ميخوريم: اول اين که گرچه متغير count يک عضو ايستا است ولي براي خواندن آن حتما بايد از يک شيء موجود استفاده کنيم. در مثال قبلي از شيء w براي خواندن آن استفاده کردهايم. اين باعث ميشود که مجبور شويم هميشه مواظب باشيم عضو ايستاي مفروض از طريق يک شي که الان موجود است فراخواني شود.
508
static int num() { return count; } private: static int count; };
* مثال 17-9 يك تابع عضو ايستا کد زير همان کد مثال قبلي است با اين فرق که در اين کد، تابع دستيابي کننده نيز به شکل ايستا اعلان شده است: class Widget { public: Widget() { ++count; } ~Widget() { --count; } static int num() { return count; } private: static int count; };
509
int Widget::count = 0; int main() { cout << "Now there are " << Widget::num() << " widgets.\n"; Widget w, x; cout << "Now there are " << Widget::num() << " widgets.\n"; { Widget w, x, y, z; } Widget y;
510
وقتي تابع num() به صورت ايستا تعريف شود، از اشياي کلاس مستقل ميشود و براي فراخواني آن نيازي به يک شيء موجود نيست و ميتوان با کد Widget::num() به شکل مستقيم آن را فراخواني کرد.
511
پايان جلسه نهم
512
جلسه دهم «سربارگذاري عملگرها »
513
آنچه در اين جلسه مي خوانيد:
1- توابع دوست 2- سربارگذاري عملگر جايگزيني (=) 3- اشارهگر this 4- سربارگذاري عملگرهاي حسابي 5- سربارگذاري عملگرهاي جايگزيني حسابي 6- سربارگذاري عملگرهاي رابطهاي 7- سربارگذاري عملگرهاي افزايشي و كاهشي
514
هدف کلي: بيان اهميت سربارگذاري عملگرها براي يک کلاس و نحوۀ انجام اين کار.
515
هدفهاي رفتاري: انتظار ميرود پس از پايان اين جلسه بتوانيد:
- «سربارگذاري» را تعريف کرده و اهميت آن را شرح دهيد. - «تابع دوست» را تعريف کنيد و علت و اهميت استفاده از چنين توابعي را بيان نماييد. - اشارهگر this را بشناسيد و علت استفاده از چنين اشارهگري را بيان نماييد. - نحوۀ سربارگذاري عملگر جايگزيني را بيان کنيد. - نحوۀ سربارگذاري عملگرهاي حسابي را بيان کنيد. - نحوۀ سربارگذاري عملگرهاي جايگزيني حسابي را بيان کنيد. - نحوۀ سربارگذاري عملگرهاي رابطهاي را بيان کنيد. - نحوۀ سربارگذاري عملگرهاي افزايشي و کاهشي را بيان کنيد.
516
مقدمه: در C++ مجموعهاي از 45 عملگر مختلف وجود دارد که براي کارهاي متنوعي استفاده ميشوند. همۀ اين عملگرها براي کار کردن با انواع بنيادي (مثل int و float و char) سازگاري دارند. هنگامي که کلاسي را تعريف ميکنيم، در حقيقت يک نوع جديد را به انواع موجود اضافه کردهايم. ممکن است بخواهيم اشياي اين کلاس را در محاسبات رياضي به کار ببريم. اما چون عملگرهاي رياضي (مثل + يا = يا *= ) چيزي راجع به اشياي کلاس جديد نميدانند، نميتوانند به درستي کار کنند. C++ براي رفع اين مشکل چاره انديشيده و امکان سربارگذاري عملگرها را تدارک ديده است. سربارگذاري عملگرها به اين معناست که به عملگرها تعاريف جديدي اضافه کنيم تا بتوانند با اشياي کلاس مورد نظر به درستي کار کنند.
517
1- توابع دوست: class Ratio { friend int numReturn(Ratio); public: Ratio(); ~Ratio(); private: int num, den; } int numReturn(Ratio r) { return r.num; int main() { Ratio x(22, 7);1 – Friend function cout << numReturn(x) << endl; به کد زير نگاه کنيد: اعضايي از کلاس که به شکل خصوصي (private) اعلان ميشوند فقط از داخل همان کلاس قابل دستيابياند و از بيرون کلاس (درون بدنۀ اصلي) امکان دسترسي به آنها نيست. اما يک استثنا وجود دارد. تابع دوست تابعي است که عضو يک کلاس نيست اما اجازه دارد به اعضاي خصوصي آن دسترسي داشته باشد.
518
2-سربارگذاري عملگر جايگزيني(=):
در بين عملگرهاي گوناگون، عملگر جايگزيني شايد بيشترين کاربرد را داشته باشد. هدف اين عملگر، کپي کردن يک شي در شيء ديگر است. مانند سازندۀ پيشفرض، سازندۀ کپي و نابودکننده، عملگر جايگزيني نيز به طور خودکار براي يک کلاس ايجاد ميشود اما اين تابع را ميتوانيم به شکل صريح درون کلاس اعلان نماييم.
519
مثال: افزودن عملگر جايگزيني به كلاس:
کد زير يک رابط کلاس براي Ratio است که شامل سازندۀ پيشفرض، سازندۀ کپي و عملگر جايگزيني ميباشد: class Ratio { public: Ratio(int = 0, int = 1); Ratio(const Ratio&); void operator=(const Ratio&); private: int num, den; };
520
به نحو اعلان عملگر جايگزيني دقت نماييد.
نام اين تابع عضو، operator= است و فهرست آرگومان آن مانند سازندۀ کپي ميباشد يعني يک آرگومان منفرد دارد که از نوع همان کلاس است که به طريقۀ ارجاع ثابت ارسال ميشود. عملگر جايگزيني را ميتوانيم به شکل زير تعريف کنيم: void Ratio::operator=(const Ratio& r) { num = r.num; den = r.den; } کد فوق اعضاي دادهاي شيء r را به درون اعضاي دادهاي شيئي که مالک فراخواني اين عملگر است، کپي ميکند.
521
3-اشارهگر :this در C++ ميتوانيم عملگر جايگزيني را به شکل زنجيرهاي مثل زير به کار ببريم: x = y = z = 3.14; اجراي کد بالا از راست به چپ صورت ميگيرد. يعني ابتدا مقدار 3.14 درون z قرار ميگيرد و سپس مقدار z درون y کپي ميشود و سرانجام مقدار y درون x قرار داده ميشود. عملگر جايگزيني که در مثال قبل ذکر شد، نميتواند به شکل زنجيرهاي به کار رود.
522
مثال 2-10 سربارگذاري عملگر جايگزيني به شکل صحيح:
class Ratio { public: Ratio(int =0, int =1); // default constructor Ratio(const Ratio&); // copy constructor Ratio& operator=(const Ratio&); // assignment operator // other declarations go here private: int num, den; }; Ratio& Ratio::operator=(const Ratio& r) { num = r.num; den = r.den; return *this; }
523
توجه داشته باشيد که عمل جايگزيني با عمل مقداردهي تفاوت دارد، هر چند هر دو از عملگر يکساني استفاده ميکنند. مثلا در کد زير: Ratio x(22,7); // this is an initialization Ratio y(x); // this is an initialization Ratio z = x; // this is an initialization Ratio w; w = x; // this is an assignment سه دستور اول، دستورات مقداردهي هستند ولي دستور آخر يک دستور جايگزيني است. دستور مقداردهي، سازندۀ کپي را فرا ميخواند ولي دستور جايگزيني عملگر جايگزيني را فراخواني ميکند.
524
4-سربارگذاري عملگرهاي حسابي:
چهار عملگر حسابي + و – و * و / در همۀ زبانهاي برنامهنويسي وجود دارند و با همۀ انواع بنيادي به کار گرفته ميشوند. قصد داريم سرباري را به اين عملگرها اضافه کنيم تا بتوانيم با استفاده از آنها، اشياي ساخت خودمان را در محاسبات رياضي به کار ببريم. عملگرهاي حسابي به دو عملوند نياز دارند. مثلا عملگر ضرب (*) در رابطۀ زير: z = x*y; با توجه به رابطۀ فوق و آنچه در بخش قبلي گفتيم، عملگر ضرب سربارگذاري شده بايد دو پارامتر از نوع يک کلاس و به طريق ارجاع ثابت بگيرد و يک مقدار بازگشتي از نوع همان کلاس داشته باشد. پس انتظار داريم قالب سربارگذاري عملگر ضرب براي کلاس Ratio به شکل زير باشد: Ratio operator*(Ratio x, Ratio y) { Ratio z(x.num*y.num, x.den*y.den); return z; }
525
اگر تابعي عضو کلاس نباشد، نميتواند به اعضاي خصوصي آن کلاس دستيابد
اگر تابعي عضو کلاس نباشد، نميتواند به اعضاي خصوصي آن کلاس دستيابد. براي رفع اين محدوديتها، تابع سربارگذاري عملگر ضرب را بايد به عنوان تابع دوست کلاس معرفي کنيم. لذا قالب کلي براي سربارگذاري عملگر ضرب درون کلاس مفروض T به شکل زير است: Class T { friend T operator*(const T&, const T&); public: // public members private: // private members }
526
T operator*(const T& x, const T& y) { T z;
در سربارگذاري عملگرهاي حسابي + و – و / نيز از قالبهاي کلي فوق استفاده ميکنيم با اين تفاوت که در نام تابع سربارگذاري، به جاي علامت ضرب * بايد علامت عملگر مربوطه را قرار دهيم و دستورات بدنۀ تابع را نيز طبق نياز تغيير دهيم. و از آنجا که تابع دوست عضوي از کلاس نيست، تعريف بدنۀ آن بايد خارج از کلاس صورت پذيرد. در تعريف بدنۀ تابع دوست به کلمۀ کليدي friend نيازي نيست و عملگر جداسازي حوزه :: نيز استفاده نميشود: T operator*(const T& x, const T& y) { T z; // required operations for z = x*y return z; }
527
مثال 3-10 سربارگذاري عملگر ضرب براي کلاس :Ratio
class Ratio { friend Ratio operator*(const Ratio&, const Ratio&); public: Ratio(int = 0, int = 1); Ratio(const Ratio&); Ratio& operator=(const Ratio&); // other declarations go here private: int num, den; }; Ratio operator*(const Ratio& x, const Ratio& y) { Ratio z(x.num * y.num , x.den * y.den); return z; } int main() { Ratio x(22,7) ,y(-3,8) ,z; z = x; // assignment operator is called z.print(); cout << endl; x = y*z; // multiplication operator is called x.print(); cout << endl;
528
5-سربارگذاري عملگرهاي جايگزيني حسابي:
به خاطر بياوريد که عملگرهاي جايگزيني حسابي، ترکيبي از عملگر جايگزيني و يک عملگر حسابي ديگر است. مثلا عملگر *= ترکيبي از دو عمل ضرب * و سپس جايگزيني = است. نکتۀ قابل توجه در عملگرهاي جايگزيني حسابي اين است که اين عملگرها بر خلاف عملگرهاي حسابي ساده، فقط يک عملوند دارند. پس تابع سربارگذاري عملگرهاي جايگزيني حسابي بر خلاف عملگرهاي حسابي، ميتواند عضو کلاس باشد. سربارگذاري عملگرهاي جايگزيني حسابي بسيار شبيه سربارگذاري عملگر جايگزيني است. قالب کلي براي سربارگذاري عملگر *= براي کلاس مفروض T به صورت زير است: class T { public: T& operator*=(const T&); // other public members private: // private members };
529
T& T::operator*=(const T& x) { // required operations return *this; }
بدنۀ تابع سربارگذاري به قالب زير است: T& T::operator*=(const T& x) { // required operations return *this; } استفاده از اشارهگر *this باعث ميشود که بتوانيم عملگر *= را در يک رابطۀ زنجيرهاي به کار ببريم. در C++ چهار عملگر جايگزيني حسابي += و -= و *= و /= وجود دارد. قالب کلي براي سربارگذاري همۀ اين عملگرها به شکل قالب بالا است فقط در نام تابع به جاي *= بايد علامت عملگر مربوطه را ذکر کرد و دستورات بدنۀ تابع را نيز به تناسب، تغيير داد. مثال بعدي نشان ميدهد که عملگر *= چگونه براي کلاس Ratio سربارگذاري شده است.
530
مثال 4-10 كلاس Ratio با عملگر *= سربارگذاري شده:
class Ratio { public: Ratio(int = 0, int = 1); Ratio& operator=(const Ratio&); Ratio& operator*=(const Ratio&); // other declarations go here private: int num, den; }; Ratio& Ratio::operator*=(const Ratio& r) { num = num*r.num; den = den*r.den; return *this; } بديهي است که عملگر سربارگذاري شدۀ جايگزيني حسابي، بايد با عملگر سربارگذاري شدۀ حسابي معادلش، نتيجۀ يکساني داشته باشد. مثلا اگر x و y هر دو از کلاس Ratio باشند، آنگاه دو خط کد زير بايد نتيجۀ مشابهي داشته باشند: x *= y; x = x*y;
531
6-سربارگذاري عملگرهاي رابطهاي:
شش عملگر رابطهاي در C++ وجود دارد که عبارتند از: > و < و => و <= و == و != . اين عملگرها به همان روش عملگرهاي حسابي،يعني به شکل توابع دوست سربارگذاري ميشوند. اما نوع بازگشتيشان فرق ميکند.
532
حاصل عبارتي که شامل عملگر رابطهاي باشد، همواره يک مقدار بولين است
حاصل عبارتي که شامل عملگر رابطهاي باشد، همواره يک مقدار بولين است. يعني اگر آن عبارت درست باشد، حاصل true است و اگر آن عبارت نادرست باشد، حاصل false است. چون نوع بولين در حقيقت يک نوع عددي صحيح است، ميتوان به جاي true مقدار 1 و به جاي false مقدار 0 را قرار داد. به همين جهت نوع بازگشتي را براي توابع سربارگذاري عملگرهاي رابطهاي، از نوع int قرار دادهاند.
533
{ friend int operator==(const T&, const T&); public: // public members
قالب کلي براي سربارگذاري عملگر رابطهاي == به شکل زير است: class T { friend int operator==(const T&, const T&); public: // public members private: // private members }
534
int operator==(const T& x,const T& y)
همچنين قالب کلي تعريف بدنۀ اين تابع به صورت زير ميباشد: int operator==(const T& x,const T& y) { // required operations to finding result return result; } که به جاي result يک مقدار بولين يا يک عدد صحيح قرار ميگيرد. ساير عملگرهاي رابطهاي نيز از قالب بالا پيروي ميکنند.
535
مثال 5-10 سربارگذاري عملگر تساوي (==) براي كلاس :Ratio
class Ratio { friend int operator==(const Ratio&, const Ratio&); frined Ratio operator*(const Ratio&, const Ratio&); // other declarations go here public: Ratio(int = 0, int = 1); Ratio(const Ratio&); Ratio& operator=(const Ratio&); private: int num, den; }; int operator==(const Ratio& x, const Ratio& y) { return (x.num * y.den == y.num * x.den); } چون اشياي کلاس Ratio به صورت کسر هستند، بررسي تساوي x==y معادل بررسي است که براي بررسي اين تساوي ميتوانيم مقدار (a*d==b*c) را بررسي کنيم. بدنۀ تابع سربارگذاري در مثال همين رابطه را بررسي ميکند.
536
7-سربارگذاري عملگرهاي افزايشي و كاهشي:
عملگر افزايشي ++ و کاهشي -- هر کدام دو شکل دارند: 1- شکل پيشوندي. 2-شکل پسوندي. هر کدام از اين حالتها را ميتوان سربارگذاري کرد.
537
قالب کلي براي سربارگذاري عملگر پيشافزايشي به شکل زير است:
T T::operator++() { // required operations return *this; } اينجا هم از اشارهگر *this استفاده شده. علت هم اين است که مشخص نيست چه چيزي بايد بازگشت داده شود. به همين دليل اشارهگر *this به کار رفته تا شيئي که عمل پيشافزايش روي آن صورت گرفته، بازگشت داده شود.
538
افزودن عملگر پيشافزايشي به كلاس :Ratio
اگر y يک شي از کلاس Ratio باشد و عبارت ++y ارزيابي گردد، مقدار 1 به y افزوده ميشود اما چون y يک عدد کسري است، افزودن مقدار 1 به اين کسر اثر متفاوتي دارد. فرض کنيد y=22/7 باشد. حالا داريم:
539
افزودن عملگر پسافزايشي به كلاس Ratio:
در عبارت y = x++; از عملگر پسافزايشي استفاده کردهايم. تابع اين عملگر بايد طوري تعريف شود که مقدار x را قبل از اين که درون y قرار بگيرد، تغيير ندهد. ميدانيم که اشارهگر *this به شيء جاري (مالک فراخواني) اشاره دارد. کافي است مقدار اين اشارهگر را در يک محل موقتي ذخيره کنيم و عمل افزايش را روي آن مقدار موقتي انجام داده و حاصل آن را بازگشت دهيم. به اين ترتيب مقدار *this تغييري نميکند و پس از شرکت در عمل جايگزيني، درون y قرار ميگيرد: int main() { Ratio x(22,7) , y = x++; cout << "y = "; y.print(); cout << ", x = "; x.print();} Ratio Ratio::operator++(int) { Ratio temp = *this; num += den; return temp; } class Ratio { public: Ratio(int n=0, int d=1) : num(n) , den(d) { } Ratio operator++(); //pre-increment Ratio operator++(int); //post-increment void print() { cout << num << '/' << den << endl; } private: int num, den; };
540
عملگرهاي پيشکاهشي و پسکاهشي نيز به همين شيوۀ عملگرهاي پيشافزايشي و پسافزايشي سربارگذاري ميشوند. غير از اينها، عملگرهاي ديگري نيز مثل عملگر خروجي (<<) ، عملگر ورودي (>>) ، عملگر انديس ([]) و عملگر تبديل نيز وجود دارند که ميتوان آنها را براي سازگاري براي کلاسهاي جديد سربارگذاري کرد.
541
پايان جلسه دهم
542
جلسه يازدهم «تركيب و وراثت»
543
«تركيب و وراثت» مقدمه تركيب وراثت اعضاي حفاظت شد غلبه کردن بر وراثت
اشارهگرها در وراثت توابع مجازي و چندريختي نابودكنندۀ مجازي <<<
544
كلاسهاي پايۀ انتزاعي پرسشهاي گزينهاي پرسشهاي تشريحي تمرينهاي برنامهنويسي ضميمه الف : پاسخنامۀ پرسشهاي گزينهاي ضميمه ب:جدول اسکي ضميمه ج : کلمات کليدي C++ استاندارد ضميمه د : عملگرهاي C++ استاندارد ضميمه هـ : فهرست منابع و مأخذ
545
هدف کلي: بيان اهميت ترکيب و وراثت در شيگرايي و چگونگي انجام اين کارها. هدفهاي رفتاري: انتظار ميرود پس از پايان اين جلسه بتوانيد: - علت استفاده از «ترکيب» و «وراثت» را در برنامههاي شيگرا توضيح دهيد. - نحوۀ ترکيب دو يا چند کلاس را براي ايجاد کلاس جديد، بدانيد. - وراثت را تعريف کنيد.
546
- «اعضاي حفاظت شدۀ کلاس» را تعريف کنيد و تفاوت اين اعضا با اعضاي عمومي و خصوصي کلاس را شرح دهيد.
- نحوۀ غلبه کردن بر وراثت را شرح دهيد. - «تابع مجازي» را تعريف کنيد و علت استفاده از توابع مجازي را بدانيد. - «چندريختي» را تعريف کنيد و شيوۀ پيادهسازي چندريختي در کلاسها را بدانيد. - «کلاس پايۀ انتزاعي» را تعريف کنيد و علت تعريف اين کلاسها را ذکر کنيد.
547
مقدمه اغلب اوقات براي ايجاد يک کلاس جديد، نيازي نيست که همه چيز از اول طراحي شود. ميتوانيم براي ايجاد کلاس مورد نظر، از تعاريف کلاسهايي که قبلا ساختهايم، استفاده نماييم. اين باعث صرفهجويي در وقت و استحکام منطق برنامه ميشود. در شيگرايي به دو شيوه ميتوان اين کار را انجام داد: ترکيب1 و وراثت2. در اين جلسه خواهيم ديد که چگونه و چه مواقعي ميتوانيم از اين دو شيوه بهره ببريم.
548
تركيب: ترکيب کلاسها (يا تجميع کلاسها) يعني استفاده از يک يا چند کلاس ديگر در داخل تعريف يک کلاس جديد. هنگامي که عضو دادهاي کلاس جديد، شيئي از کلاس ديگر باشد، ميگوييم که اين کلاس جديد ترکيبي از ساير کلاسهاست. به تعريف دو کلاس زير نگاه کنيد.
549
كلاس Date کد زير، کلاس Date را نشان ميدهد که اشياي اين کلاس براي نگهداري تاريخ استفاده ميشوند. class Date { public: Date(int y=0, int m=0, int d=0) : year(y), month(m), day(d) {}; void setDate(int y, int m, int d) { year = y; month = m; day = d; } void getDate() { cin >> year >> month >> day ; } void showDate() { cout << year << '/' << month << '/' << day ; } private: int year, month, day; }
550
كلاس Book کد زير، کلاس Book را نشان ميدهد که اشياي اين کلاس برخي از مشخصات يک کتاب را نگهداري ميکنند: class Book { public: Book(char* n = " ", int i = 0, int p = 0) : name(n), id(i), page(p) { } void printName() { cout << name; } void printId() { cout << id; } void printPage() { cout << page; } private: string name, author; int id, page; }
551
بهبود دادن کلاس Book #include "Date.h" class Book { public:
Book(char* n = " ", int i = 0, int p = 0) : name(n), id(i), page(p) { } void printName() { cout << name; } void printId() { cout << id; } void printPage() { cout << page; } void setDOP(int y, int m, int d) { publish.setDate(y, m, d) ; } void showDOP() { publish.showDate(); } private: string name, author; int id, page; Date publish; }
552
وراثت : وراثت روش ديگري براي ايجاد کلاس جديد از روي کلاس قبلي است. گاهي به وراثت «اشتقاق» نيز ميگويند. اگر از قبل با برنامهنويسي مبتني بر پنجرهها آشنايي مختصر داشته باشيد، احتمالا عبارت «کلاس مشتقشده» را فراوان ديدهايد. اين موضوع به خوبي اهميت وراثت را آشکار مينمايد.
553
اعضاي حفاظت شده: گرچه کلاس Ebook در مثال قبل نميتواند مستقيما به اعضاي خصوصي کلاس والدش دسترسي داشته باشد، اما با استفاده از توابع عضو عمومي که از کلاس والد به ارث برده، ميتواند به اعضاي خصوصي آن کلاس دستيابي کند. اين محدوديت بزرگي محسوب ميشود. اگر توابع عضو عمومي کلاس والد انتظارات کلاس فرزند را برآورده نسازند، کلاس فرزند ناکارآمد ميشود. اوضاع زماني وخيمتر ميشود که هيچ تابع عمومي براي دسترسي به يک دادۀ خصوصي در کلاس والد وجود نداشته باشد.
554
غلبه کردن بر وراثت : اگر Y زير کلاسي از X باشد، آنگاه اشياي Y همۀ اعضاي عمومي و حفاظت شدۀ کلاس X را ارث ميبرند. مثلا تمامي اشياي Ebook تابع دستيابي printName() از کلاس Book را به ارث ميبرند. به تابع printName() يک «عضو موروثي» ميگوييم. گاهي لازم است يک نسخۀ محلي از عضو موروثي داشته باشيم. يعني کلاس فرزند، عضوي هم نام با عضو موروثي داشته باشد که مخصوص به خودش باشد و ارثي نباشد. براي مثال فرض کنيد کلاس X يک عضو عمومي به نام p داشته باشد و کلاس Y زير کلاس X باشد.
555
در اين حالت اشياي کلاس Y عضو موروثي p را خواهند داشت
در اين حالت اشياي کلاس Y عضو موروثي p را خواهند داشت. حال اگر يک عضو به همان نام p در زيرکلاس Y به شکل صريح اعلان کنيم، اين عضو جديد، عضو موروثي همنامش را مغلوب ميکند. به اين عضو جديد، «عضو غالب» ميگوييم. بنابراين اگر y1 يک شي از کلاس Y باشد، y1.p به عضو p غالب اشاره دارد نه به p موروثي. البته هنوز هم ميتوان به p موروثي دسترسي داشت. عبارت y1.X::p به p موروثي دستيابي دارد.
556
هم ميتوان اعضاي دادهاي موروثي را مغلوب کرد و هم اعضاي تابعي موروثي را.
يعني اگر کلاس X داراي يک عضو تابعي عمومي به نام f() باشد و در زيرکلاس Y نيز تابع f() را به شکل صريح اعلان کنيم، آنگاه y1.f() به تابع غالب اشاره دارد و y1.X::f() به تابع موروثي اشاره دارد. در برخي از مراجع به توابع غالب override ميگويند و دادههاي غالب را dominate مينامند. ما در اين کتاب هر دو مفهوم را به عنوان اعضاي غالب به کار ميبريم. به مثال زير نگاه کنيد.
557
اعضاي دادهاي و تابعي غالب :
class X { public: void f() { cout << "Now X::f() is running\n"; } int a; }; class Y : public X void f() { cout << "Now Y::f() is running\n"; } // this f() overrides X::f() // this a dominates X::a
558
سازندهها و نابودکنندههاي والد:
class X { public: X() { cout << "X::X() constructor executing\n"; } ~X() { cout << "X::X() destructor executing\n"; } }; clas Y : public X Y() { cout << "Y::Y() constructor executing\n"; } ~Y() { cout << "Y::Y() destructor executing\n"; } clas Z : public Y Z(int n) {cout << "Z::Z(int) constructor executing\n";} ~Z() { cout << "Z::Z() destructor executing\n"; } int main() { Z z(44); }
559
اشارهگرها در وراثت : در شيگرايي خاصيت جالبي وجود دارد و آن اين است که اگر p اشارهگري از نوع کلاس والد باشد، آنگاه p را ميتوان به هر فرزندي از آن کلاس نيز اشاره داد. به کد زير نگاه کنيد: class X { public: void f(); } class Y : public X // Y is a subclass of X } int main() { X* p; // p is a pointer to objects of base class X Y y; p = &y; // p can also point to objects of subclass Y
560
اشارهگري از کلاس والد به شيئي از کلاس فرزند:
در برنامۀ زير، کلاس Y زيرکلاسي از X است. هر دوي اين کلاسها داراي يک عضو تابعي به نام f() هستند و p اشارهگري از نوع X* تعريف شده: class X { public: void f() { cout << "X::f() executing\n"; } }; class Y : public X void f() { cout << "Y::f() executing\n"; } } int main() { X x; Y y; X* p = &x; p->f(); // invokes X::f() because p has type X* p = &y;
561
توابع مجازي و چندريختي :
تابع مجازي تابعي است که با کلمۀ کليدي virtual مشخص ميشود. وقتي يک تابع به شکل مجازي اعلان ميشود، يعني در حداقل يکي از کلاسهاي فرزند نيز تابعي با همين نام وجود دارد. توابع مجازي امکان ميدهند که هنگام استفاده از اشارهگرها، بتوانيم بدون در نظر گرفتن نوع اشارهگر، به توابع شيء جاري دستيابي کنيم. به مثال زير دقت کنيد.
562
استفاده از توابع مجازي:
class X { public:1 – Virtual function virtual void f() { cout << "X::f() executing\n"; } }; class Y : public X { public: void f() { cout << "Y::f() executing\n"; } } int main() { X x; Y y; X* p = &x; p->f(); // invokes X::f() p = &y; p->f(); // invokes Y::f()
563
چندريختي از طريق توابع مجازي :
سه کلاس زير را در نظر بگيريد. بدون استفاده از توابعمجازي، برنامه آن طور که مورد انتظار است کار نميکند: class Ebook : public Book { public: Ebook(char* s, float g) : Book(s), size(g) {} void print() { cout << "Here is an Ebook with name " << name << " and size " << size << " MB.\n"; } private: float size; } class Book { public: Book(char* s) { name = new char[strlen(s+1)]; strcpy(name, s); } void print() { cout << "Here is a book with name " << name << ".\n"; protected: char* name; }; class Notebook : public Book { public: Notebook(char* s, int n) : Book(s) , pages(n) {} void print() { cout << "Here is a Notebook with name " << name << " and " << pages << " pages.\n"; } private: int pages; };
564
نابودكنندۀ مجازي: با توجه به تعريف توابع مجازي، به نظر ميرسد که نميتوان توابع سازنده و نابودکننده را به شکل مجازي تعريف نمود زيرا سازندهها و نابودگرها در کلاسهاي والد و فرزند، همنام نيستند. در اصل، سازندهها را نميتوان به شکل مجازي تعريف کرد اما نابودگرها قصۀ ديگري دارند. مثال بعدي ايراد مهلکي را نشان ميدهد که با مجازي کردن نابودگر، برطرف ميشود.
565
حافظۀ گم شده : به برنامۀ زير دقت کنيد: class X { public:
x() { p = new int[2]; cout << "X(). "; } ~X() { delete [] p; cout << "~X().\n" } private: int* p; }; class Y : public X Y() { q = new int[1023]; cout << "Y() : Y::q = " << q << ". "; } ~Y() { delete [] q; cout << "~Y(). "; } int* q; int main() { for (int i=0; i<8; i++) { X* r = new Y; delete r; }
566
كلاسهاي پايۀ انتزاعي :
در شيگرايي رسم بر اين است که ساختار برنامه و کلاسها را طوري طراحي کنند که بتوان آنها را به شکل يک نمودار درختي شبيه زير نشان داد: BOOK Paper BOOK EBOOK REFERENCE MAGAZINE COOKBOOK PDF CHM HLP HTML
Similar presentations
© 2025 SlidePlayer.com. Inc.
All rights reserved.