زبان برنامه نویسی سی/آرایه

آرایه ویرایش

آرایه یا Array در برنامه‌نویسی به نحوه ای از ایجاد داده‌ها ( عموماً متغیرها ) می‌گویند که به تعداد معین از نوع داده تعیین شده ، توسط برنامه‌نویس ، خانه‌هایی را بر روی حافظه به صورت متوالی ایجاد می‌نماید ( که به هر یک از آنها عنصر آن آرایه می‌گوئیم ) . تفاوت آرایه با ساختمان یا اجتماع ، بر روی حافظه اینست که فضای خالی در بین خانه‌ها ایجاد نمی‌شود و از طرفی ایجاد آرایه بسیار راحت‌تر از تعریف ساختمان یا اجتماع می‌باشد . شکل کلی تعریف آرایه به این صورت می‌باشد :

data-type name[number] = {value, value, ...};

ابتدا نوع داده خود را تعیین می‌نمائید که می‌تواند یک متغیر از نوع پایه یا یک نمونه از ساختمان یا اجتماع یا هر نوع داده دیگری باشد ، سپس نام داده خود را می نویسید و بدون فاصله در مقابل نام داده خود ، یک جفت کروشه به صورت باز و بسته می‌نویسیم که در داخل آن یک عدد ، قرار می‌گیرد که نشان دهنده تعداد آرایه ساخته شده و قابل استفاده برای برنامه‌نویس می‌باشد ؛ به عدد نوشته شده در داخل کروشه‌ها که تعداد عناصر آرایه را تعیین می‌نماید اندیس ( Index یا Subscript ) گفته می شود

توجه : اندیس تعیین شده توسط شما هرگز نباید 0 باشد

خانه های متوالی و پشت سرهم ذخیره شده برای شما از عنصر اول که همان اندیس 0 آرایه می باشد ، آغاز می‌گردد و تا عددی که شما تعیین کرده‌اید ، منهای یکی ادامه می‌یابد . به عنصر آخر آرایه همواره مقدار تهی یا همان NULL تخصیص می‌یابد تا مشخص شود که آرایه به پایان رسیده . بنابراین مثلاً شما می نویسید :

int a[5];

با تعریف بالا ۶ متغیر از نوع صحیح ایجاد می‌شود ( ۶ عنصر ) که شما از [0]a را مقدار می دهید تا [4]a که می شود ۵ عنصر از متغیر صحیح ( ۱۰ بایت برای شما که بر روی ۱۰ آفست از سگمنت‌های حافظه ذخیره می‌شود - در سیستم های ۳۲ بیتی قدیمی - و البته ۲۰ بایت در ۲۰ آفست از سگمنت‌های حافظه در سیستم‌های جدیدتر ) اما [5]a توسط کامپایلر مقداری تهی می‌گیرد که شما قادر به استفاده از آن نیستید ، پس به خوبی به یاد داشته باشید که زمانی که شما عدد 5 را به عنوان اندیس آرایه می نویسید ، ۶ داده از داده مشخص شده ایجاد می‌شود که از شماره 0 تا یکی مانده به اندیسِ نوشته شده در اختیار شما قرار می‌گیرد تا مقدار دهی گردد و آخرین متغیر یا داده که همان شماره اندیس نوشته شده می‌باشد ، با استاندارد زبان سی توسط کامپایلر مقدار تهی می گیرد . بنابراین آرایه بالا ۱۲ بایت به صورت استاندارد برای سیستم های ۳۲ بیتی قدیمی و ۲۴ بایت در سیستم‌های جدیدتر بر روی حافظه موقت ، فضا اشغال می‌نماید

به جای نوشتن عدد به عنوان اندیس ، می توان از نام یک متغیر عددی که تعریف شده است ( مقدار آن تعیین شده ) استفاده نمود ، برای این منظور به جای عدد داخل کروشه‌ها ، نام متغیر عددی خود را می‌نویسید . اما آنچه که بعد از علامت تساوی در بین آکولادها قرار گرفته و value نوشته گردیده ، مقدار و موجودی‌های اختصاص داده شده توسط شما برای هر یک از عنصر های آرایه می‌باشند که توسط علامت کالن ( , ) از یکدیگر جدا می‌شوند . تعداد مقادیری که می‌نویسید تا به آرایه اختصاص یابد نباید بیشتر از اندیس نوشته شده توسط شما باشد ، اما اگر کمتر باشد ، کامپایلر طبق استاندارد ، مقدار 0 را برای عنصر یا عنصرهای متناظر در نظر می‌گیرد . مقادیر به صورت متوالی از چپ به راست به عناصر آرایه ( به ترتیب از 0 تا عنصر آخر ) اختصاص می‌یابند( بنابراین شما با روش ابتدایی که ذکر نوشته شد نمی توانید یک یا چند عنصر را در بین عنصر‌های دیگر از قلم بیاندازید و فقط عنصرهای مورد نظر خود را مقدار دهی کنید )

مثال :

int a[5] = {1, 6, 8, 12};

مثال ۲ :

int i = 7;
;int b[i]

b[i] = {2, 4, 6, 8, 10, 12, 14}

در مثال اول [0]a برابر با مقدار 1 می باشد و [3]a برابر با 12 و [4]a که مقداری ندارد ، توسط کامپایلر مقدار 0 میگیرد و همان طور که گفته شد به [5]a مقدار NULL اختصاص خواهد یافت . در مثال دوم یک متغیر از نوع صحیح با مقدار 7 تعریف نمودیم ، سپس یک آرایه با نام b از نوع صحیح با اندیس i اعلان کردیم که 7 عنصر دارد ( چون مقدار i مقدار ۷ است ) و سپس b را با مقدار دهی تعریف کردیم . همچنین جهت دسترسی به هر یک از عنصرهای آرایه ، شماره آن را می‌نویسیم و به آن مقدار می‌دهیم یا اگر مقداری داده‌ایم مقدارش را عوض می‌کنیم یا با نوشتن شماره آن و قرار دادن مقدار آن بر روی یک داده دیگر و یا به عنوان آرگومان یک تابع مقدار آن را فرا می‌خوانیم . فراموش نکنید که تمام عنصرهایی که هنوز مقداری نگرفته‌اند به صورت پیش فرض دارای مقدار 0 خواهند بود ( تا زمانی که به آنها مقداری تخصیص بدهیم )

مثال :

int a[6];
a[4] = 18;

اینکه یک آرایه بدون اندیس نوشته شده و مقدار دهی شود ، توسط برخی کامپایلر ها پشتیبانی می‌شود و برخی دیگر آن را پشتیبانی نمی‌کنند و از شما خطا خواهند گرفت . اما در مورد کاراکتر ها مسئله متفاوت است ، طبق استاندارد هر متغیر از نوع کاراکتر می‌تواند بدون اندیس به صورت آرایه تعریف شود . اما طبق استاندارد ، یک آرایه ، تنها زمانی می تواند دارای اندیس نباشد ( به غیر از نبودن از نوع کاراکتر ) که یا :

  • پارامتر یک تابع باشد ( که در این صورت ، کامپایلر به صورت خودکار ، آن را به اشاره‌گری از نوع داده آرایه ، تبدیل می‌نماید )
  • به صورت خارجی ( با کلیدواژه extern ) اعلان گردد که در این صورت باید در جای دیگری از برنامه تعریف شده باشد ( عموماً در یک متن منبع دیگری از پروژه )
  • آرایه در جای دیگری در همان متن منبع برنامه مورد نظر ، تعریف گردد

در غیر این صورت ، کامپایلر از ما خطا خواهد گرفت !

گفتیم که آرایه‌های کاراکتری می‌توانند بدون اندیس نوشته شده و مقدار بگیرند . برای مقدار دهی آرایه‌های کاراکتری می توانیم از روش معمول استفاده نموده و تک تک کاراکترهای مورد نظر را با شیوه استاندارد مقداردهی ، کاراکتر بنویسیم ( یعنی با استفاده از علامت کوت ' ) یا از روش دیگر آن که رایج بوده و قابل استفاده است یعنی قرار دادن یک رشته به عنوان مقدار آرایه کاراکتری استفاده کنیم ، به مثال ذیل توجه نمائید :

char deb[] = "this is a string";

معمولاً بهتر است برای آرایه‌های کاراکتری از همین شیوه استفاده نمائید ، چرا که در صورت نوشتن اندیس باید حتماً دقت کنید که تعداد کاراکترهای نوشته شده توسط شما از اندیس تعیین شده بزرگ‌تر نباشد . حال به مثالی از تعریف یک نمونه از ساختمان به صورت آرایه بپردازیم :

struct point
{
int x, y;
} point_array[2];
point_array[0].x = 3;

نمونه‌ای که از ساختمان point ایجاد شد ، خود یک آرایه دو عنصری است . بنابراین برای استفاده از آن باید ابتدا نمونه به همراه اندیسی که عنصر آن را میخواهیم مقدار دهی نمائیم بنویسیم و سپس به وسیله عملگر نقطه « . » به عضو ساختمان دسترسی یافته و از آن بهره جوئیم

آرایه های چند بعدی

آرایه‌ها را می‌توان به صورت چند بعدی نیز تعریف نمود . استفاده از آن در کار با ماتریس‌ها کاملاً ملموس می‌باشد . برای این کار به جای تعیین یک اندیس ، چند اندیس ( به هر تعداد ابعادی که مورد نظر ماست ) تعیین می‌نمائیم . مثال :

float cib[8][7];

در مثال بالا یک متغیر اعشاری با 8 عنصر 7 عنصری اعلان نمودیم ( ۸ دسته ۷ تایی ) . آرایه ها را از آخرین اندیس ( سمت راست‌ترین ) به ترتیب یک به یک پر می‌کنید و سپس به اندیس قبل‌تر می‌رسید ( سمت چپ‌تر ) که با اندیس 0 برای اندیس‌های رو به رو ( بعدتر ، راست‌تر ) شروع می‌شود تا اندیس قبل‌تر ( چپ‌تر ) پر شود و پس از رسیدن به آخرین عنصر مجاز اندیس قبل‌تر ( چپ‌تر ) ، اندیس‌های رو به رویی ( مقابل قبلی ) یک به یک اضافه شوند تا به آخرین عنصر مجاز اندیس سمت راستی‌تر برسید و سپس به سمت ابتدایی‌تر ( چپ‌تر ) بروید و الی آخر ، تا در نهایت آخرین شماره اندیس‌های مجاز را بنویسید و عنصرهای خود را مقدار بدهید . اگر مثلاً یک متغیر long را بدین صورت اعلان کنیم :

long kid[5][8][7];

عناصر متغیر kid بدین ترتیب خوانده شده و باید مقدار دهی گردند :

kid[0][0][0] kid[0][0][1] kid[0][0][2] . . . kid[0][0][6] . . . kid[0][1][0] kid[0][2][0] kid[0][7][0] . . . kide[0][7][1] . . kid[0][7][6] kid[1][0][0] kid[2][0][0] . . kid[4][0][0] . . kid[4][1][0] . . kid[4][7][0] . . kid[4][7][1] . . kid[4][7][6]

در مقدار دهی آرایه‌های چند بعدی نیز می توانیم داخل یک آکولاد مقادیر خود را بنویسیم که توسط کامپایلر از اولین عنصر آرایه چند بعدی ما تا آخرین عنصر به ترتیب مقدار خواهند گرفت ، اما این روش توسط برخی کامپایلرها مورد حمایت قرار نمی‌گیرد . بنابراین باید از روش استاندارد آن که استفاده از آکولادهای تو در تو می باشد استفاده نمود

مثال :

int two_dimensions[2][5] = {{1, 2, 5, 8, 3}, {7, 6, 5, 12, 10}};

همان طور که می‌بینید دو دسته پنج تایی از اعداد را به عنوان مقدار به two_dimensions اختصاص داده‌ایم . حال اگر برای متغیر آرایه‌ای kid ( که در مثال قبل نوشتیم ) می‌خواستیم مقدار بدهیم باید ۵ تا دسته از ۸ دسته ۷ تایی را در آکولادهای تو در تو مقدار می‌دادیم

آرایه ها و اشاره گر ها

مبحث بعدی ما در این کتاب ، مبحث اشاره‌گر می‌باشد . اشاره‌گرها داده‌هایی هستند که به داده دیگری اشاره داده می‌شوند تا بتوانند به صورت مستقیم آنچه که بر روی حافظه آن داده به عنوان مقدار و موجودی آن قرار گرفته را در دسترس ما قرار دهند و امکانات زیادی برای ما در برنامه‌نویسی فراهم می‌آورد که در مبحث بعدی به آن می‌پردازیم . اما آرایه‌ها ارتباط تنگاتنگی با اشاره‌گرها دارند . آرایه‌ها در زبان C به صورت اشاره‌گر تعریف می‌شوند و از این روی استفاده از آرایه‌ها به وسیله اشاره‌گرها بسیار راحت‌تر است و در نوشتن برنامه های پیچیده‌تر و تو در توی ما لازمه استفاده بهینه از آرایه‌ها می‌باشد که اگر کتاب حاضر را همراهی کنید ، به این مبحث مهم می‌پردازیم تا به نقطه تداخل اشاره‌گرها با آرایه‌ها و روش دیگر دسترسی به آرایه‌ها برسیم