ऑब्जेक्टिव-सी क्विक गाइड

ऑब्जेक्टिव-सी सामान्य-प्रयोजन की भाषा है जिसे C टॉकिंग लैंग्वेज के शीर्ष पर विकसित किया जाता है, जिसमें स्माल टॉक प्रोग्रामिंग लैंग्वेज की विशेषताओं को जोड़कर इसे ऑब्जेक्ट-ओरिएंटेड भाषा बनाया जाता है। यह मुख्य रूप से आईओएस और मैक ओएस एक्स ऑपरेटिंग सिस्टम के साथ-साथ इसके अनुप्रयोगों को विकसित करने में उपयोग किया जाता है।

प्रारंभ में, Objective-C को NeXT द्वारा अपने NeXTSTEP OS के लिए विकसित किया गया था, जहाँ से Apple द्वारा इसके iOS और Mac OS X के लिए इसे ले लिया गया था।

ऑब्जेक्ट ओरिएंटेड प्रोग्रामिंग

ऑब्जेक्ट-ओरिएंटेड प्रोग्रामिंग के चार स्तंभों सहित ऑब्जेक्ट-ओरिएंटेड प्रोग्रामिंग का पूरी तरह से समर्थन करता है -

  • Encapsulation
  • डेटा छिपाना
  • Inheritance
  • Polymorphism

उदाहरण कोड

#import <Foundation/Foundation.h>

int main (int argc, const char * argv[]) {
   NSAutoreleasePool * pool = [[NSAutoreleasePool alloc] init];

   NSLog (@"hello world");
   [pool drain];
   return 0;
}

फाउंडेशन की रूपरेखा

फाउंडेशन फ्रेमवर्क सुविधाओं के बड़े सेट प्रदान करता है और वे नीचे सूचीबद्ध हैं।

  • इसमें NSArray, NSDictionary, NSSet और जैसे विस्तारित डेटाटिप्स की एक सूची शामिल है।

  • इसमें फ़ाइलों, स्ट्रिंग्स आदि में हेरफेर करने वाले फ़ंक्शंस का एक समृद्ध सेट होता है।

  • यह URL हैंडलिंग, उपयोगिताओं जैसे दिनांक स्वरूपण, डेटा हैंडलिंग, त्रुटि हैंडलिंग आदि के लिए सुविधाएँ प्रदान करता है।

लर्निंग ऑब्जेक्टिव-सी

ऑब्जेक्टिव-सी सीखते समय सबसे महत्वपूर्ण बात यह है कि अवधारणाओं पर ध्यान केंद्रित करें और भाषा तकनीकी विवरण में खो न जाएं।

एक प्रोग्रामिंग भाषा सीखने का उद्देश्य एक बेहतर प्रोग्रामर बनना है; यानी नई प्रणालियों को डिजाइन करने और लागू करने और पुराने बनाए रखने में अधिक प्रभावी बनने के लिए।

उद्देश्य-सी का उपयोग

उद्देश्य-सी, जैसा कि पहले उल्लेख किया गया है, आईओएस और मैक ओएस एक्स में उपयोग किया जाता है। इसमें आईओएस उपयोगकर्ताओं का बड़ा आधार है और मोटे तौर पर मैक ओएस एक्स उपयोगकर्ताओं में वृद्धि हुई है। और चूंकि Apple पहले गुणवत्ता पर ध्यान केंद्रित करता है और उन लोगों के लिए यह अद्भुत है जिन्होंने ऑब्जेक्टिव-सी सीखना शुरू कर दिया है।

स्थानीय पर्यावरण सेटअप

यदि आप अभी भी ऑब्जेक्ट-सी प्रोग्रामिंग भाषा के लिए अपना वातावरण सेट करने के लिए तैयार हैं, तो आपको अपने कंप्यूटर पर उपलब्ध दो सॉफ्टवेयर्स की आवश्यकता है, (ए) टेक्स्ट एडिटर और (बी) जीसीसी कंपाइलर।

पाठ संपादक

इसका उपयोग आपके प्रोग्राम को टाइप करने के लिए किया जाएगा। कुछ संपादकों के उदाहरणों में विंडोज नोटपैड, ओएस एडिट कमांड, ब्रीफ, एप्सिलॉन, ईएमएसीएस और विम या vi शामिल हैं।

टेक्स्ट एडिटर का नाम और संस्करण अलग-अलग ऑपरेटिंग सिस्टम पर भिन्न हो सकते हैं। उदाहरण के लिए, नोटपैड का उपयोग विंडोज पर किया जाएगा, और vim या vi का उपयोग विंडोज़ के साथ-साथ लिनक्स या यूनिक्स पर भी किया जा सकता है।

आपके द्वारा अपने संपादक के साथ बनाई गई फाइलों को सोर्स फाइल कहा जाता है और इसमें प्रोग्राम सोर्स कोड होता है। उद्देश्य-सी कार्यक्रमों के लिए स्रोत फाइलें आमतौर पर एक्सटेंशन के साथ नामित की जाती हैं ".m"।

अपनी प्रोग्रामिंग शुरू करने से पहले, सुनिश्चित करें कि आपके पास एक पाठ संपादक है और आपके पास एक कंप्यूटर प्रोग्राम लिखने के लिए पर्याप्त अनुभव है, इसे एक फ़ाइल में सहेजें, इसे संकलित करें और अंत में इसे निष्पादित करें।

जीसीसी कंपाइलर

स्रोत फ़ाइल में लिखा गया स्रोत कोड आपके प्रोग्राम के लिए मानव पठनीय स्रोत है। इसे मशीन भाषा में बदलने के लिए "संकलित" होने की आवश्यकता है, ताकि आपका सीपीयू वास्तव में दिए गए निर्देशों के अनुसार कार्यक्रम को निष्पादित कर सके।

इस जीसीसी कंपाइलर का उपयोग आपके स्रोत कोड को अंतिम निष्पादन कार्यक्रम में संकलित करने के लिए किया जाएगा। मुझे लगता है कि आपको प्रोग्रामिंग भाषा संकलक के बारे में बुनियादी जानकारी है।

जीसीसी संकलक विभिन्न प्लेटफार्मों पर मुफ्त में उपलब्ध है और विभिन्न प्लेटफार्मों पर स्थापित करने की प्रक्रिया नीचे बताई गई है।

UNIX / Linux पर स्थापना

प्रारंभिक कदम gcc ऑब्जेक्टिव-सी पैकेज के साथ gcc स्थापित है। इसके द्वारा किया जाता है -

$ su - $ yum install gcc
$ yum install gcc-objc

अगला कदम निम्नलिखित कमांड का उपयोग करके पैकेज निर्भरता स्थापित करना है -

$ yum install make libpng libpng-devel libtiff libtiff-devel libobjc 
   libxml2 libxml2-devel libX11-devel libXt-devel libjpeg libjpeg-devel

उद्देश्य-सी की पूरी सुविधाएँ प्राप्त करने के लिए, GNUStep को डाउनलोड और इंस्टॉल करें। इससे पैकेज डाउनलोड करके किया जा सकता हैhttp://main.gnustep.org/resources/downloads.php।

अब, हमें डाउनलोड किए गए फ़ोल्डर पर स्विच करने और फ़ाइल को अनपैक करने की आवश्यकता है -

$ tar xvfz gnustep-startup-
      
       .tar.gz 
      

अब, हमें फ़ोल्डर gnustep-स्टार्टअप पर स्विच करने की आवश्यकता है जो कि उपयोग करके बनाया जाता है -

$ cd gnustep-startup-<version>

अगला, हमें निर्माण प्रक्रिया को कॉन्फ़िगर करने की आवश्यकता है -

$ ./configure

फिर, हम द्वारा निर्माण कर सकते हैं -

$ make

हमें अंत में पर्यावरण की स्थापना की आवश्यकता है -

$ . /usr/GNUstep/System/Library/Makefiles/GNUstep.sh

हमारे पास एक हेल्लोवर्ल्ड है। ऑब्जेक्टिव-सी निम्नानुसार है -

#import <Foundation/Foundation.h>

int main (int argc, const char * argv[]) {
   NSAutoreleasePool * pool = [[NSAutoreleasePool alloc] init];
   
   NSLog (@"hello world");
   [pool drain];
   return 0;
}

अब, हम एक ऑब्जेक्टिव-सी फाइल को कह सकते हैं और कह सकते हैं कि helloWorld.m को cd का उपयोग करके फाइल वाले फोल्डर में स्विच करके और फिर निम्न चरणों का उपयोग करके -

$ gcc `gnustep-config --objc-flags` 
-L/usr/GNUstep/Local/Library/Libraries 
-lgnustep-base helloWorld.m -o helloWorld
$ ./helloWorld

हम निम्न आउटपुट देख सकते हैं -

2013-09-07 10:48:39.772 tutorialsPoint[12906] hello world

मैक ओएस पर स्थापना

यदि आप मैक ओएस एक्स का उपयोग करते हैं, तो जीसीसी प्राप्त करने का सबसे आसान तरीका ऐप्पल की वेब साइट से एक्सकोड विकास पर्यावरण डाउनलोड करना और सरल इंस्टॉलेशन निर्देशों का पालन करना है। एक बार जब आप Xcode सेट कर लेते हैं, तो आप C / C ++ के लिए GNU कंपाइलर का उपयोग कर पाएंगे।

Xcode वर्तमान में developer.apple.com/technologies/tools/ पर उपलब्ध है ।

विंडोज पर स्थापना

खिड़कियों पर ऑब्जेक्टिव-सी प्रोग्राम चलाने के लिए, हमें मिनगव और जीएनयूएसटीपी कोर को स्थापित करने की आवश्यकता है। दोनों पर उपलब्ध हैंhttps://www.gnu.org/software/gnustep/windows/installer.html।

सबसे पहले, हमें MSYS / MinGW सिस्टम पैकेज स्थापित करने की आवश्यकता है। फिर, हमें जीएनयूस्टेप कोर पैकेज स्थापित करने की आवश्यकता है। दोनों एक विंडोज इंस्टॉलर प्रदान करते हैं, जो कि स्व-व्याख्यात्मक है।

फिर Start -> All Programs -> GNUstep -> Shell का चयन करके Objective-C और GNUstep का उपयोग करें

HelloWorld.m वाले फ़ोल्डर पर जाएँ

हम प्रोग्राम का उपयोग करके संकलित कर सकते हैं -

$ gcc `gnustep-config --objc-flags` 
-L /GNUstep/System/Library/Libraries hello.m -o hello -lgnustep-base -lobjc

हम प्रोग्राम का उपयोग करके चला सकते हैं -

./hello.exe

हम निम्नलिखित उत्पादन प्राप्त करते हैं -

2013-09-07 10:48:39.772 tutorialsPoint[1200] hello world

इससे पहले कि हम ऑब्जेक्टिव-सी प्रोग्रामिंग लैंग्वेज के बेसिक बिल्डिंग ब्लॉक्स का अध्ययन करें, आइए हम एक नंगे न्यूनतम ऑब्जेक्टिव-सी प्रोग्राम संरचना को देखें ताकि हम इसे आगामी अध्यायों में एक संदर्भ के रूप में ले सकें।

उद्देश्य-सी हैलो विश्व उदाहरण

एक उद्देश्य-सी कार्यक्रम मूल रूप से निम्नलिखित भागों के होते हैं -

  • प्रीप्रोसेसर कमैंट्स
  • Interface
  • Implementation
  • Method
  • Variables
  • कथन और भाव
  • Comments

आइए एक सरल कोड देखें जो "हैलो वर्ल्ड" शब्दों को प्रिंट करेगा -

#import <Foundation/Foundation.h>

@interface SampleClass:NSObject
- (void)sampleMethod;
@end

@implementation SampleClass

- (void)sampleMethod {
   NSLog(@"Hello, World! \n");
}

@end

int main() {
   /* my first program in Objective-C */
   SampleClass *sampleClass = [[SampleClass alloc]init];
   [sampleClass sampleMethod];
   return 0;
}

आइए उपरोक्त कार्यक्रम के विभिन्न भागों को देखें -

  • कार्यक्रम की पहली पंक्ति #import <Foundation / Foundation.h> एक प्रीप्रोसेसर कमांड है, जो वास्तविक संकलन पर जाने से पहले Foundation.h फाइल को शामिल करने के लिए एक Objective-C कंपाइलर को बताता है।

  • अगली पंक्ति @interface SampleClass: NSObject से पता चलता है कि इंटरफ़ेस कैसे बनाया जाए। यह NSObject को विरासत में मिला है, जो सभी वस्तुओं का आधार वर्ग है।

  • अगली पंक्ति - (शून्य) नमूनामेथोड; कैसे एक विधि घोषित करने के लिए दिखाता है।

  • अगली पंक्ति @end एक इंटरफ़ेस के अंत को चिह्नित करती है।

  • अगली पंक्ति @implementation SampleClass दिखाता है कि इंटरफ़ेस SampleClass को कैसे लागू किया जाए।

  • अगली पंक्ति - (शून्य) नमूनामैथोड {} नमूनामेथोड के कार्यान्वयन को दर्शाता है।

  • अगली पंक्ति @end कार्यान्वयन के अंत को चिह्नित करती है।

  • अगली पंक्ति int main () मुख्य कार्य है जहां कार्यक्रम निष्पादन शुरू होता है।

  • अगली पंक्ति / _...*/ को संकलक द्वारा अनदेखा किया जाएगा और इसे कार्यक्रम में अतिरिक्त टिप्पणियां जोड़ने के लिए रखा गया है। तो इस तरह की लाइनों को कार्यक्रम में टिप्पणी कहा जाता है।

  • अगली पंक्ति NSLog (...) उद्देश्य-सी में उपलब्ध एक और फ़ंक्शन है जो संदेश का कारण बनता है "हैलो, वर्ल्ड!" स्क्रीन पर प्रदर्शित किया जाना है।

  • अगली पंक्ति return 0; मुख्य () फ़ंक्शन को समाप्त करता है और मान 0 लौटाता है।

संकलन और निष्पादन उद्देश्य-सी कार्यक्रम

अब जब हम प्रोग्राम को संकलित करते हैं और चलाते हैं, तो हमें निम्नलिखित परिणाम मिलेंगे।

2017-10-06 07:48:32.020 demo[65832] Hello, World!

आपने ऑब्जेक्टिव-सी प्रोग्राम की एक बुनियादी संरचना देखी है, इसलिए ऑब्जेक्ट-सी प्रोग्रामिंग भाषा के अन्य बुनियादी बिल्डिंग ब्लॉकों को समझना आसान होगा।

उद्देश्य-सी में टोकन

एक उद्देश्य-सी कार्यक्रम में विभिन्न टोकन शामिल हैं और एक टोकन या तो एक कीवर्ड, एक पहचानकर्ता, एक निरंतर, एक स्ट्रिंग शाब्दिक, या एक प्रतीक है। उदाहरण के लिए, निम्नलिखित उद्देश्य-सी स्टेटमेंट में छह टोकन शामिल हैं -

NSLog(@"Hello, World! \n");

व्यक्तिगत टोकन हैं -

NSLog
@
(
   "Hello, World! \n"
)
;

अर्धविराम;

ऑब्जेक्टिव-सी प्रोग्राम में, अर्धविराम एक स्टेटमेंट टर्मिनेटर है। यही है, प्रत्येक व्यक्तिगत बयान को अर्धविराम के साथ समाप्त किया जाना चाहिए। यह एक तार्किक इकाई के अंत का संकेत देता है।

उदाहरण के लिए, निम्नलिखित दो अलग-अलग कथन हैं -

NSLog(@"Hello, World! \n");
return 0;

टिप्पणियाँ

टिप्पणियाँ आपके उद्देश्य-सी कार्यक्रम में पाठ की मदद करने की तरह हैं और उन्हें संकलक द्वारा अनदेखा किया जाता है। वे / * से शुरू करते हैं और पात्रों के साथ समाप्त करते हैं * / जैसा कि नीचे दिखाया गया है -

/* my first program in Objective-C */

आप टिप्पणियों में टिप्पणियों के साथ नहीं हो सकते हैं और वे एक स्ट्रिंग या चरित्र शाब्दिक के भीतर नहीं होते हैं।

पहचानकर्ता

ऑब्जेक्टिव-सी आइडेंटिफ़ायर एक ऐसा नाम है जिसका इस्तेमाल वैरिएबल, फंक्शन या किसी अन्य यूज़र-डिफ़ाइंड आइटम की पहचान के लिए किया जाता है। एक पहचानकर्ता अक्षर A से Z या z या एक अंडरस्कोर _ से शुरू होता है, उसके बाद शून्य या अधिक अक्षर, अंडरस्कोर और अंक (0 से 9) होता है।

उद्देश्य-सी पहचानकर्ता के भीतर @, $, और% जैसे विराम चिह्न वर्णों को अनुमति नहीं देता है। उद्देश्य-सी एक हैcase-sensitiveप्रोग्रामिंग भाषा। इस प्रकार, उद्देश्य-सी में मैनपावर और मैनपावर दो अलग-अलग पहचानकर्ता हैं। यहां स्वीकार्य पहचानकर्ताओं के कुछ उदाहरण दिए गए हैं -

mohd       zara    abc   move_name  a_123
myname50   _temp   j     a23b9      retVal

कीवर्ड

निम्न सूची उद्देश्य-सी में कुछ आरक्षित शब्दों को दिखाती है। ये आरक्षित शब्द निरंतर या परिवर्तनशील या किसी अन्य पहचानकर्ता नाम के रूप में उपयोग नहीं किए जा सकते हैं।

ऑटो अन्य लंबा स्विच
टूटना enum रजिस्टर करें typedef
मामला निर्वासन वापसी संघ
चार नाव कम अहस्ताक्षरित
स्थिरांक के लिये पर हस्ताक्षर किए शून्य
जारी रखें के लिए जाओ इस आकार का परिवर्तनशील
चूक अगर स्थिर जबकि
कर पूर्णांक struct _Packed
दोहरा मसविदा बनाना इंटरफेस कार्यान्वयन
NSObject NSInteger NSNumber CGFloat
संपत्ति nonatomic; बनाए रखने के मजबूत
कमज़ोर unsafe_unretained; पढ़ना लिखना सिफ़ पढ़िये

ऑब्जेक्टिव-सी में व्हॉट्सएप

केवल व्हॉट्सएप युक्त एक लाइन, संभवतः एक टिप्पणी के साथ, एक रिक्त रेखा के रूप में जानी जाती है, और एक उद्देश्य-सी संकलक इसे पूरी तरह से अनदेखा करता है।

व्हॉट्सएप का अर्थ ऑब्जेक्टिव-सी में प्रयुक्त शब्द रिक्त, टैब, न्यूलाइन वर्ण और टिप्पणियों का वर्णन करने के लिए है। व्हॉट्सएप एक स्टेटमेंट के एक हिस्से को दूसरे से अलग करता है और कंपाइलर को यह पहचानने में सक्षम बनाता है कि एक स्टेटमेंट में कोई तत्व, जैसे इंट, एंड और अगला एलिमेंट कहां से शुरू होता है। इसलिए, निम्नलिखित कथन में -

int age;

कंपाइलर को भेद करने में सक्षम होने के लिए इंट और उम्र के बीच कम से कम एक व्हाट्सएप चरित्र (आमतौर पर एक स्थान) होना चाहिए। दूसरी ओर, निम्नलिखित कथन में,

fruit = apples + oranges;   // get the total fruit

फल और =, या सेब और सेब के बीच कोई व्हाट्सएप वर्ण आवश्यक नहीं है, हालांकि यदि आप पठनीय उद्देश्य की इच्छा रखते हैं तो आप कुछ को शामिल करने के लिए स्वतंत्र हैं।

ऑब्जेक्टिव-सी प्रोग्रामिंग भाषा में, डेटा प्रकार एक व्यापक प्रणाली को संदर्भित करते हैं जिसका उपयोग विभिन्न प्रकारों के चर या कार्यों को घोषित करने के लिए किया जाता है। एक चर का प्रकार यह निर्धारित करता है कि यह भंडारण में कितना स्थान रखता है और संग्रहीत बिट पैटर्न की व्याख्या कैसे की जाती है।

ऑब्जेक्टिव-सी के प्रकारों को निम्नानुसार वर्गीकृत किया जा सकता है -

अनु क्रमांक। प्रकार और विवरण
1

Basic Types −

वे अंकगणित प्रकार हैं और दो प्रकार के होते हैं: (ए) पूर्णांक प्रकार और (बी) फ्लोटिंग-पॉइंट प्रकार।

2

Enumerated types −

वे फिर से अंकगणित प्रकार हैं और वे चर को परिभाषित करने के लिए उपयोग किए जाते हैं जिन्हें केवल पूरे कार्यक्रम में कुछ असतत पूर्णांक मान निर्दिष्ट किए जा सकते हैं।

3

The type void −

प्रकार निर्दिष्ट शून्य इंगित करता है कि कोई मान उपलब्ध नहीं है।

4

Derived types −

वे (ए) सूचक प्रकार, (बी) सरणी प्रकार, (सी) संरचना प्रकार, (डी) संघ प्रकार और (ई) फ़ंक्शन प्रकार शामिल हैं।

सरणी प्रकार और संरचना प्रकार को सामूहिक रूप से समग्र प्रकार के रूप में संदर्भित किया जाता है। फ़ंक्शन का प्रकार फ़ंक्शन के रिटर्न मान के प्रकार को निर्दिष्ट करता है। हम निम्नलिखित अनुभाग में बुनियादी प्रकार देखेंगे, जबकि अन्य प्रकार आगामी अध्यायों में शामिल किए जाएंगे।

पूर्णांक प्रकार

निम्नलिखित तालिका आपको इसके आकार और मूल्य श्रेणियों के साथ मानक पूर्णांक प्रकारों के बारे में विवरण देती है -

प्रकार भंडारण का आकार मूल्य सीमा
चार 1 बाइट -128 से 127 या 0 से 255
अचिन्हित वर्ण 1 बाइट 0 से 255 रु
हस्ताक्षरित चार 1 बाइट -128 से 127
पूर्णांक 2 या 4 बाइट्स -32,768 से 32,767 या -2,147,483,648 से 2,147,483,647
अहस्ताक्षरित int 2 या 4 बाइट्स 0 से 65,535 या 0 से 4,294,967,295 रु
कम 2 बाइट्स -32,768 से 32,767
अहस्ताक्षरित छोटा 2 बाइट्स 0 से 65,535 रु
लंबा 4 निवाले -2,147,483,648 से 2,147,483,647
अहस्ताक्षरित लंबे 4 निवाले 0 से 4,294,967,295 रु

किसी विशेष प्लेटफ़ॉर्म पर किसी प्रकार या चर का सटीक आकार प्राप्त करने के लिए, आप निम्न का उपयोग कर सकते हैं sizeofऑपरेटर। एक्सप्रेशन साइज़ोफ़ (प्रकार) ऑब्जेक्ट के स्टोरेज साइज़ या बाइट्स में टाइप करता है। किसी भी मशीन पर int प्रकार का आकार प्राप्त करने के लिए एक उदाहरण निम्नलिखित है -

#import <Foundation/Foundation.h>

int main() {
   NSLog(@"Storage size for int : %d \n", sizeof(int));
   return 0;
}

जब आप उपरोक्त कार्यक्रम को संकलित और निष्पादित करते हैं, तो यह लिनक्स पर निम्नलिखित परिणाम उत्पन्न करता है -

2013-09-07 22:21:39.155 demo[1340] Storage size for int : 4

फ्लोटिंग-पॉइंट प्रकार

निम्नलिखित तालिका आपको भंडारण आकार और मूल्य श्रेणियों और उनकी परिशुद्धता के साथ मानक फ्लोट-पॉइंट प्रकारों के बारे में विवरण देती है -

प्रकार भंडारण का आकार मूल्य सीमा शुद्धता
नाव 4 बाइट 1.2E-38 से 3.4E + 38 6 दशमलव स्थान
दोहरा 8 बाइट 2.3E-308 से 1.7E + 308 15 दशमलव स्थान
लंबा दोहरा 10 बाइट 3.4E-4932 से 1.1E + 4932 19 दशमलव स्थान

शीर्ष लेख फ़ाइल float.h मैक्रोज़ को परिभाषित करती है जो आपको अपने प्रोग्राम में वास्तविक संख्याओं के द्विआधारी प्रतिनिधित्व के बारे में इन मूल्यों और अन्य विवरणों का उपयोग करने की अनुमति देती है। निम्न उदाहरण फ्लोट प्रकार और इसके श्रेणी मानों द्वारा लिया गया संग्रहण स्थान प्रिंट करेगा -

#import <Foundation/Foundation.h>

int main() {
   NSLog(@"Storage size for float : %d \n", sizeof(float));
   return 0;
}

जब आप उपरोक्त कार्यक्रम को संकलित और निष्पादित करते हैं, तो यह लिनक्स पर निम्नलिखित परिणाम उत्पन्न करता है -

2013-09-07 22:22:21.729 demo[3927] Storage size for float : 4

शून्य प्रकार

शून्य प्रकार निर्दिष्ट करता है कि कोई मूल्य उपलब्ध नहीं है। इसका उपयोग तीन प्रकार की स्थितियों में किया जाता है -

अनु क्रमांक। प्रकार और विवरण
1 Function returns as void

उद्देश्य-सी में विभिन्न कार्य हैं जो मूल्य नहीं लौटाते हैं या आप कह सकते हैं कि वे शून्य हैं। बिना रिटर्न वैल्यू वाले फ़ंक्शन में रिटर्न प्रकार शून्य होता है। उदाहरण के लिए,void exit (int status);

2 Function arguments as void

उद्देश्य-सी में विभिन्न कार्य हैं जो किसी भी पैरामीटर को स्वीकार नहीं करते हैं। बिना किसी पैरामीटर के एक फ़ंक्शन शून्य के रूप में स्वीकार कर सकता है। उदाहरण के लिए,int rand(void);

शून्य प्रकार इस बिंदु पर आपको समझा नहीं जा सकता है, इसलिए हमें आगे बढ़ने दें और हम इन अवधारणाओं को आगामी अध्यायों में शामिल करेंगे।

एक चर कुछ भी नहीं है लेकिन एक भंडारण क्षेत्र को दिया गया नाम है जो हमारे कार्यक्रमों में हेरफेर कर सकता है। ऑब्जेक्टिव-सी में प्रत्येक चर का एक विशिष्ट प्रकार है, जो चर की स्मृति के आकार और लेआउट को निर्धारित करता है; मूल्यों की सीमा जो उस मेमोरी में संग्रहीत की जा सकती है; और परिचालनों का सेट जो चर पर लागू किया जा सकता है।

एक चर का नाम अक्षरों, अंकों और अंडरस्कोर वर्ण से बना हो सकता है। यह या तो एक पत्र या एक अंडरस्कोर से शुरू होना चाहिए। ऊपरी और निचले अक्षर अलग-अलग हैं क्योंकि ऑब्जेक्टिव-सी केस-संवेदी है। पिछले अध्याय में बताए गए मूल प्रकारों के आधार पर, निम्नलिखित बुनियादी चर प्रकार होंगे -

अनु क्रमांक। टाइप और विवरण
1

char

आमतौर पर एक एकल ओकटेट (एक बाइट)। यह एक पूर्णांक प्रकार है।

2

int

मशीन के लिए पूर्णांक का सबसे प्राकृतिक आकार।

3

float

एकल-सटीक फ़्लोटिंग पॉइंट मान।

4

double

एक डबल-सटीक फ़्लोटिंग पॉइंट मान।

5

void

प्रकार की अनुपस्थिति का प्रतिनिधित्व करता है।

ऑब्जेक्टिव-सी प्रोग्रामिंग भाषा कई अन्य प्रकार के चर को भी परिभाषित करने की अनुमति देती है, जिसे हम बाद के अध्यायों जैसे Enumeration, Pointer, Array, Structure, Union आदि में कवर करेंगे। इस अध्याय के लिए, हम केवल मूल चर प्रकारों का अध्ययन करते हैं।

उद्देश्य-सी में परिवर्तनीय परिभाषा

एक चर परिभाषा का मतलब संकलक को यह बताना है कि चर के लिए भंडारण कहाँ और कितना बनाना है। एक चर परिभाषा एक डेटा प्रकार को निर्दिष्ट करती है और इसमें उस प्रकार के एक या अधिक चर की सूची शामिल होती है -

type variable_list;

यहाँ, type एक मान्य ऑब्जेक्टिव-सी डेटा प्रकार होना चाहिए जिसमें चार, w_char, int, float, double, bool या कोई उपयोगकर्ता-निर्धारित ऑब्जेक्ट, आदि शामिल हैं। variable_listअल्पविराम द्वारा अलग किए गए एक या अधिक पहचानकर्ता नामों से मिलकर बना हो सकता है। कुछ मान्य घोषणाएँ यहाँ दिखाई गई हैं -

int    i, j, k;
char   c, ch;
float  f, salary;
double d;

रेखा int i, j, k;दोनों घोषित करता है और चर को परिभाषित करता है i, j और k; जो संकलक को निर्देश देता है कि मैं i, j और k का नाम लिखें

वेरिएबल्स को उनकी घोषणा में आरंभीकृत किया जा सकता है (एक प्रारंभिक मूल्य सौंपा गया है)। इनिशियलाइज़र में एक समरूप चिन्ह होता है, जिसके बाद एक स्थिर अभिव्यक्ति होती है -

type variable_name = value;

कुछ उदाहरण हैं -

extern int d = 3, f = 5;    // declaration of d and f. 
int d = 3, f = 5;           // definition and initializing d and f. 
byte z = 22;                // definition and initializes z. 
char x = 'x';               // the variable x has the value 'x'.

एक इनिशलाइज़र के बिना परिभाषा के लिए: स्थिर भंडारण अवधि वाले वेरिएबल को संक्षेप में NULL के साथ आरंभीकृत किया जाता है (सभी बाइट्स का मान 0 होता है); अन्य सभी चर का प्रारंभिक मूल्य अपरिभाषित है।

उद्देश्य-सी में परिवर्तनीय घोषणा

एक चर घोषणा संकलक को आश्वासन देती है कि दिए गए प्रकार और नाम के साथ एक चर मौजूद है ताकि संकलक चर के बारे में पूर्ण विवरण की आवश्यकता के बिना आगे संकलन के लिए आगे बढ़े। एक चर घोषणा का संकलन के समय ही अर्थ होता है, संकलनकर्ता को कार्यक्रम के लिंकिंग के समय वास्तविक चर घोषणा की आवश्यकता होती है।

एक चर घोषणा उपयोगी है जब आप कई फ़ाइलों का उपयोग कर रहे हैं और आप अपने चर को फ़ाइलों में से एक में परिभाषित करते हैं, जो कार्यक्रम के लिंकिंग के समय उपलब्ध होगा। आप उपयोग करेंगेexternकिसी भी स्थान पर एक चर घोषित करने के लिए कीवर्ड। यद्यपि आप अपने उद्देश्य-सी कार्यक्रम में कई बार एक चर घोषित कर सकते हैं, लेकिन इसे केवल एक बार फ़ाइल, फ़ंक्शन या कोड के ब्लॉक में परिभाषित किया जा सकता है।

उदाहरण

निम्नलिखित उदाहरण का प्रयास करें, जहां चर शीर्ष पर घोषित किए गए हैं, लेकिन उन्हें मुख्य कार्य के अंदर परिभाषित और आरंभ किया गया है -

#import <Foundation/Foundation.h>

// Variable declaration:
extern int a, b;
extern int c;
extern float f;

int main () {
  /* variable definition: */
  int a, b;
  int c;
  float f;
 
  /* actual initialization */
  a = 10;
  b = 20;
  
  c = a + b;
  NSLog(@"value of c : %d \n", c);

  f = 70.0/3.0;
  NSLog(@"value of f : %f \n", f);
 
  return 0;
}

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

2013-09-07 22:43:31.695 demo[14019] value of c : 30 
2013-09-07 22:43:31.695 demo[14019] value of f : 23.333334

समान अवधारणा फ़ंक्शन घोषणा पर लागू होती है जहां आप इसकी घोषणा के समय एक फ़ंक्शन नाम प्रदान करते हैं और इसकी वास्तविक परिभाषा कहीं और दी जा सकती है। निम्नलिखित उदाहरण में, इसे C फ़ंक्शन का उपयोग करके समझाया गया है और जैसा कि आप जानते हैं कि ऑब्जेक्टिव-सी C स्टाइल फ़ंक्शन का भी समर्थन करता है -

// function declaration
int func();

int main() {
   // function call
   int i = func();
}

// function definition
int func() {
   return 0;
}

वस्तुनिष्ठ-सी में लवलीन और नियम

Objective-C में दो तरह के भाव होते हैं -

  • lvalue- अभिव्यक्ति जो एक स्मृति स्थान को संदर्भित करती है उसे "लवल्यू" अभिव्यक्ति कहा जाता है। एक लैवल्यू, बायें हाथ या दाईं ओर एक असाइनमेंट के रूप में दिखाई दे सकता है।

  • rvalue- रिवाल्यू शब्द एक डेटा मान को संदर्भित करता है जो मेमोरी में कुछ पते पर संग्रहीत होता है। एक अवतरण एक ऐसी अभिव्यक्ति है जिसमें कोई मान निर्दिष्ट नहीं किया जा सकता है जिसका अर्थ है कि एक दायीं ओर एक लकीर दिखाई दे सकती है, लेकिन असाइनमेंट के बाईं ओर नहीं।

चर अंतराल हैं और इसलिए एक असाइनमेंट के बाईं ओर दिखाई दे सकते हैं। न्यूमेरिकल शाब्दिक नियम हैं और इसलिए इन्हें असाइन नहीं किया जा सकता है और यह बाईं ओर नहीं दिखाई दे सकता है। निम्नलिखित एक मान्य कथन है -

int g = 20;

लेकिन निम्नलिखित एक मान्य कथन नहीं है और यह संकलन-समय त्रुटि उत्पन्न करेगा -

10 = 20;

स्थिरांक निश्चित मानों को संदर्भित करता है कि कार्यक्रम इसके निष्पादन के दौरान बदल नहीं सकता है। इन निश्चित मूल्यों को भी कहा जाता हैliterals

स्थिरांक किसी भी मूल डेटा प्रकार के हो सकते हैं जैसे पूर्णांक स्थिरांक, एक अस्थायी स्थिरांक, एक वर्ण स्थिरांक या एक स्ट्रिंग शाब्दिक । साथ ही एन्यूमरेशन कॉन्स्टेंट भी हैं।

constants केवल नियमित चर की तरह व्यवहार किया जाता है सिवाय इसके कि उनके मूल्यों को उनकी परिभाषा के बाद संशोधित नहीं किया जा सकता है।

पूर्णांक शाब्दिक

पूर्णांक शाब्दिक एक दशमलव, अष्टक या षोडश आधारी स्थिरांक हो सकता है। एक उपसर्ग आधार या मूलांक को निर्दिष्ट करता है: हेक्साडेसिमल के लिए 0x या 0X, ऑक्टल के लिए 0, और दशमलव के लिए कुछ भी नहीं।

एक पूर्णांक शाब्दिक में एक प्रत्यय भी हो सकता है जो क्रमशः अहस्ताक्षरित और लंबे समय के लिए यू और एल का संयोजन है। प्रत्यय अपरकेस या लोअरकेस हो सकता है और किसी भी क्रम में हो सकता है।

यहाँ पूर्णांक शाब्दिक के कुछ उदाहरण हैं -

212         /* Legal */
215u        /* Legal */
0xFeeL      /* Legal */
078         /* Illegal: 8 is not an octal digit */
032UU       /* Illegal: cannot repeat a suffix */

विभिन्न प्रकार के पूर्णांक शाब्दिक अन्य उदाहरण हैं -

85         /* decimal */
0213       /* octal */
0x4b       /* hexadecimal */
30         /* int */
30u        /* unsigned int */
30l        /* long */
30ul       /* unsigned long */

फ्लोटिंग-पॉइंट शाब्दिक

फ्लोटिंग-पॉइंट शाब्दिक में एक पूर्णांक भाग, एक दशमलव बिंदु, एक आंशिक भाग और एक घातांक भाग होता है। आप अस्थायी बिंदु शाब्दिक या तो दशमलव रूप या घातांक रूप में दर्शा सकते हैं।

दशमलव रूप का उपयोग करते समय, आपको दशमलव बिंदु, प्रतिपादक या दोनों को शामिल करना चाहिए और घातीय रूप का उपयोग करते समय, आपको पूर्णांक भाग, भिन्नात्मक भाग, या दोनों को शामिल करना होगा। हस्ताक्षरित प्रतिपादक को ई या ई द्वारा प्रस्तुत किया जाता है।

यहाँ फ़्लोटिंग पॉइंट शाब्दिक के कुछ उदाहरण दिए गए हैं -

3.14159       /* Legal */
314159E-5L    /* Legal */
510E          /* Illegal: incomplete exponent */
210f          /* Illegal: no decimal or exponent */
.e55          /* Illegal: missing integer or fraction */

चरित्र कांस्टेंट

चरित्र के शाब्दिक शब्दों को एकल उद्धरणों में संलग्न किया जाता है, जैसे 'x' और के एक साधारण चर में संग्रहीत किया जा सकता है char प्रकार।

एक चरित्र शाब्दिक एक सादा चरित्र हो सकता है (जैसे, 'x'), एक एस्केप अनुक्रम (जैसे, '\ t'), या एक सार्वभौमिक चरित्र (जैसे, '\ u02C0')।

C में कुछ वर्ण हैं जब वे एक बैकस्लैश द्वारा आगे बढ़े जाते हैं तो उनका विशेष अर्थ होगा और उनका उपयोग newline (\ n) या टैब (\ t) की तरह प्रतिनिधित्व करने के लिए किया जाता है। यहाँ, आपके पास कुछ ऐसे पलायन अनुक्रम कोड की सूची है -

निकास का क्रम जिसका अर्थ है
\\ \ चरित्र
\ ' 'चरित्र
\ " “चरित्र
\? ? चरित्र
\ए चेतावनी या घंटी
\ b बैकस्पेस
\ च फ़ीड बनाएं
\ n नई पंक्ति
\ r कैरिज रिटर्न
\ t क्षैतिज टैब
\ v लंबवत टैब
\ OOO एक से तीन अंकों की अष्टक संख्या
\ xhh। । । एक या अधिक अंकों की हेक्साडेसिमल संख्या

निम्नलिखित कुछ अनुक्रम अनुक्रम दिखाने के लिए उदाहरण है -

#import <Foundation/Foundation.h>

int main() {
   NSLog(@"Hello\tWorld\n\n");
   return 0;
}

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

2013-09-07 22:17:17.923 demo[17871] Hello	World

स्ट्रिंग शाब्दिक

स्ट्रिंग शाब्दिक या स्थिरांक दोहरे उद्धरण चिह्नों में संलग्न हैं ""। एक तार में वर्ण होते हैं जो वर्ण शाब्दिक के समान होते हैं: सादे वर्ण, बच के क्रम और सार्वभौमिक वर्ण।

आप स्ट्रिंग शाब्दिक का उपयोग करके कई लाइनों में एक लंबी लाइन को तोड़ सकते हैं और उन्हें व्हाट्सएप का उपयोग करके अलग कर सकते हैं।

यहाँ स्ट्रिंग शाब्दिकों के कुछ उदाहरण दिए गए हैं। तीनों रूप समान तार हैं।

"hello, dear"

"hello, \

dear"

"hello, " "d" "ear"

लगातार परिभाषित करना

स्थिरांक को परिभाषित करने के लिए C में दो सरल तरीके हैं -

  • का उपयोग करते हुए #define पूर्वप्रक्रमक।

  • का उपयोग करते हुए const कीवर्ड।

#Define प्रीप्रोसेसर

एक स्थिरांक को परिभाषित करने के लिए #define प्रीप्रोसेसर का उपयोग करने का तरीका निम्नलिखित है -

#define identifier value

निम्नलिखित उदाहरण इसे विस्तार से बताते हैं -

#import <Foundation/Foundation.h>

#define LENGTH 10   
#define WIDTH  5
#define NEWLINE '\n'

int main() {
   int area;
   area = LENGTH * WIDTH;
   NSLog(@"value of area : %d", area);
   NSLog(@"%c", NEWLINE);

   return 0;
}

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

2013-09-07 22:18:16.637 demo[21460] value of area : 50
2013-09-07 22:18:16.638 demo[21460]

कास्ट कीवर्ड

आप उपयोग कर सकते हैं const एक विशिष्ट प्रकार के साथ स्थिरांक घोषित करने के लिए उपसर्ग इस प्रकार है -

const type variable = value;

निम्नलिखित उदाहरण इसे विस्तार से बताते हैं -

#import <Foundation/Foundation.h>

int main() {
   const int  LENGTH = 10;
   const int  WIDTH  = 5;
   const char NEWLINE = '\n';
   int area;  
   
   area = LENGTH * WIDTH;
   NSLog(@"value of area : %d", area);
   NSLog(@"%c", NEWLINE);

   return 0;
}

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

2013-09-07 22:19:24.780 demo[25621] value of area : 50
2013-09-07 22:19:24.781 demo[25621]

ध्यान दें कि CAPITALS में स्थिरांक को परिभाषित करने के लिए यह एक अच्छा प्रोग्रामिंग अभ्यास है।

एक ऑपरेटर एक प्रतीक है जो संकलक को विशिष्ट गणितीय या तार्किक जोड़तोड़ करने के लिए कहता है। उद्देश्य-सी भाषा अंतर्निहित ऑपरेटरों में समृद्ध है और निम्नलिखित प्रकार के ऑपरेटर प्रदान करती है -

  • अंकगणितीय आपरेटर
  • संबंधपरक संकारक
  • लॉजिकल ऑपरेटर्स
  • बिटवाइज ऑपरेटर्स
  • असाइनमेंट ऑपरेटर्स
  • विविध संचालक

यह ट्यूटोरियल एक-एक करके अंकगणित, संबंधपरक, तार्किक, बिटवाइज़, असाइनमेंट और अन्य ऑपरेटरों को समझाएगा।

अंकगणितीय आपरेटर

निम्न तालिका में सभी सी-ऑब्जेक्टिव ऑपरेटर्स को ऑब्जेक्टिव-सी भाषा द्वारा समर्थित दिखाया गया है। चर मान लेंA 10 और चर रखता है B 20 रखती है, तो -

उदाहरण दिखाएं

ऑपरेटर विवरण उदाहरण
+ दो ऑपरेंड जोड़ता है A + B 30 देगा
- पहले से दूसरे ऑपरेंड को घटाता है A - B देगा -10
* दोनों ऑपरेंड को गुणा करता है A * B 200 देगा
/ विभाजक द्वारा अंशों को विभाजित करता है B / A 2 देगा
% मापांक ऑपरेटर और पूर्णांक विभाजन के बाद शेष ब% ए द 0 दे खेंगे
++ वृद्धि ऑपरेटर एक के बाद पूर्णांक मान बढ़ाता है A ++ 11 देगा
- कमी ऑपरेटर एक से पूर्णांक मान घटाता है A-- 9 देंगे

संबंधपरक संकारक

निम्न तालिका में उद्देश्य-सी भाषा द्वारा समर्थित सभी संबंधित परिचालकों को दिखाया गया है। चर मान लेंA 10 और चर रखता है B 20 रखती है, तो -

उदाहरण दिखाएं

ऑपरेटर विवरण उदाहरण
== जाँच करता है कि दो ऑपरेंड का मान बराबर है या नहीं; यदि हाँ, तो स्थिति सच हो जाती है। (ए == बी) सच नहीं है।
! = जाँच करता है कि दो ऑपरेंड का मान बराबर है या नहीं; यदि मूल्य समान नहीं हैं, तो स्थिति सच हो जाती है। (ए! = बी) सच है।
> जाँच करता है कि क्या बाएं ऑपरेंड का मूल्य सही ऑपरेंड के मूल्य से अधिक है; यदि हाँ, तो स्थिति सच हो जाती है। (ए> बी) सच नहीं है।
< चेक करता है कि क्या बाएं ऑपरेंड का मूल्य सही ऑपरेंड के मूल्य से कम है; यदि हाँ, तो स्थिति सच हो जाती है। (ए <बी) सच है।
> = जाँच करता है कि क्या बाएं ऑपरेंड का मूल्य सही ऑपरेंड के मूल्य से अधिक या बराबर है; यदि हाँ, तो स्थिति सच हो जाती है। (ए> = बी) सच नहीं है।
<= यह जाँचता है कि क्या बाएं ऑपरेंड का मूल्य सही ऑपरेंड के मूल्य से कम या उसके बराबर है; यदि हाँ, तो स्थिति सच हो जाती है। (ए <= बी) सच है।

लॉजिकल ऑपरेटर्स

निम्न तालिका में उद्देश्य-सी भाषा द्वारा समर्थित सभी तार्किक ऑपरेटरों को दिखाया गया है। चर मान लेंA 1 और चर रखता है B 0 रखती है, तो -

उदाहरण दिखाएं

ऑपरेटर विवरण उदाहरण
&& जिसे लॉजिकल एंड ऑपरेटर कहा जाता है। यदि दोनों ऑपरेंड नॉन जीरो हैं तो कंडीशन सही हो जाती है। (A && B) गलत है।
|| जिसे लॉजिकल या ऑपरेटर कहा जाता है। यदि दोनों संचालकों में से कोई भी गैर शून्य है तो स्थिति सत्य हो जाती है। (ए || बी) सच है
! जिसे लॉजिकल नॉट ऑपरेटर कहा जाता है। उपयोग अपने ऑपरेंड की तार्किक स्थिति को उलट देता है। यदि कोई शर्त सही है, तो तार्किक संचालक गलत नहीं करेगा। ! (A && B) सत्य है।

बिटवाइज ऑपरेटर्स

बिटवाइज ऑपरेटर बिट्स पर काम करता है और बिट ऑपरेशन द्वारा बिट करता है। सत्य सारणी &,; और ^ के लिए निम्नानुसार हैं -

पी क्यू पी क्यू पी | क्यू p ^ q
0 0 0 0 0
0 1 0 1 1
1 1 1 1 0
1 0 0 1 1

मान लें कि ए = 60; और बी = 13; अब द्विआधारी प्रारूप में वे निम्नानुसार होंगे -

ए = 0011 1100

B = 0000 1101

-----------------

A & B = 0000 1100

ए | बी = ००११ ११०१

ए ^ बी = 0011 0001

~ ए = ११०० ००११

उद्देश्य-सी भाषा द्वारा समर्थित बिटवाइज़ ऑपरेटरों को निम्न तालिका में सूचीबद्ध किया गया है। चर A मानें 60 और चर B 13 धारण करता है -

उदाहरण दिखाएं

ऑपरेटर विवरण उदाहरण
और बाइनरी और ऑपरेटर परिणाम के लिए थोड़ा सा कॉपी करता है यदि यह दोनों ऑपरेंड में मौजूद है। (A & B) 12 देगा, जो 0000 1100 है
| बाइनरी या ऑपरेटर थोड़ा सा कॉपी करता है अगर यह किसी भी ऑपरेंड में मौजूद है। (ए | बी) 61 देगा, जो 0011 1101 है
^ बाइनरी एक्सओआर ऑपरेटर बिट को कॉपी करता है यदि यह एक ऑपरेंड में सेट होता है लेकिन दोनों नहीं। (ए ^ बी) 49 देगा, जो 0011 0001 है
~ बाइनरी ओन्स कंप्लीट ऑपरेटर, एकात्मक है और इसमें 'फ्लिपिंग' बिट्स का प्रभाव होता है। (~ ए) -61 देगा, जो 2 के पूरक रूप में ११०० ००११ है।
<< बाइनरी लेफ्ट शिफ्ट ऑपरेटर। बाएं ऑपरेंड वैल्यू को दाएं ऑपरेंड द्वारा निर्दिष्ट बिट्स की संख्या से छोड़ दिया जाता है। A << 2 240 देगा, जो कि 1111 0000 है
>> बाइनरी राइट शिफ्ट ऑपरेटर। बाएं ऑपरेंड वैल्यू को दाएं ऑपरेंड द्वारा निर्दिष्ट बिट्स की संख्या से दाएं स्थानांतरित किया जाता है। A >> 2 15 देगा, जो 0000 1111 है

असाइनमेंट ऑपरेटर्स

उद्देश्य-सी भाषा द्वारा समर्थित असाइनमेंट ऑपरेटर निम्नलिखित हैं -

उदाहरण दिखाएं

ऑपरेटर विवरण उदाहरण
= सरल असाइनमेंट ऑपरेटर, राइट साइड ऑपरेंड्स से लेफ्ट साइड ऑपरेंड तक वैल्यू असाइन करता है C = A + B A + B का C में मान प्रदान करेगा
+ = जोड़ें और असाइनमेंट ऑपरेटर, यह बाएं ऑपरेंड में दाईं ओर ऑपरेंड जोड़ता है और परिणाम को बाएं ऑपरेंड में असाइन करता है C + = A C = C + A के बराबर है
- = घटाव और असाइनमेंट ऑपरेटर, यह बाएं ऑपरेंड से दाएं ऑपरेंड को घटाता है और बाएं ऑपरेंड को परिणाम प्रदान करता है C - = A C = C - A के बराबर है
* = गुणा और असाइनमेंट ऑपरेटर, यह बाएं ऑपरेंड के साथ दाएं ऑपरेंड को गुणा करता है और बाएं ऑपरेंड को परिणाम प्रदान करता है C * = A C = C * A के बराबर है
/ = डिवाइड और असाइनमेंट ऑपरेटर, यह बाएं ऑपरेंड को दाएं ऑपरेंड के साथ विभाजित करता है और परिणाम को बाएं ऑपरेंड को असाइन करता है C / = A C = C / A के बराबर है
% = मापांक और असाइनमेंट ऑपरेटर, यह दो ऑपरेंड का उपयोग करके मापांक लेता है और बाएं ऑपरेंड को परिणाम प्रदान करता है C% = A C = C% A के बराबर है
<< = लेफ्ट शिफ्ट और असाइनमेंट ऑपरेटर C << = 2 C = C << 2 के समान है
>> = राइट शिफ्ट और असाइनमेंट ऑपरेटर C >> = 2 C = C >> 2 के समान है
और = बिटवाइज और असाइनमेंट ऑपरेटर C & = 2 C = C & 2 के समान है
^ = बिटवाइज एक्सक्लूसिव OR एंड असाइनमेंट ऑपरेटर C ^ = 2 C = C ^ 2 के समान है
| = बिटवाइज समावेशी या असाइनमेंट ऑपरेटर C | = 2 C = C के समान है 2

विविध ऑपरेटर्स Oper आकार और त्रिगुट

सहित कुछ अन्य महत्वपूर्ण ऑपरेटर हैं sizeof तथा ? : उद्देश्य-सी भाषा द्वारा समर्थित।

उदाहरण दिखाएं

ऑपरेटर विवरण उदाहरण
इस आकार का() एक चर का आकार देता है। sizeof (a), जहां a पूर्णांक है, 4 पर लौटेगा।
और एक चर का पता देता है। &ए; चर का वास्तविक पता देगा।
* एक चर की ओर इशारा करता है। *ए; एक चर के लिए सूचक होगा।
? : सशर्त अभिव्यक्ति अगर हालत सच है? तब X मान: अन्यथा Y मान

ऑब्जेक्टिव-सी में ऑपरेटर्स प्रिसेंस

ऑपरेटर पूर्वता एक अभिव्यक्ति में शब्दों के समूहीकरण को निर्धारित करता है। यह प्रभावित करता है कि अभिव्यक्ति का मूल्यांकन कैसे किया जाता है। कुछ ऑपरेटरों में अन्य लोगों की तुलना में अधिक पूर्वता है; उदाहरण के लिए, गुणा ऑपरेटर में अतिरिक्त ऑपरेटर की तुलना में अधिक पूर्वता है -

उदाहरण के लिए, x = 7 + 3 * 2; यहाँ, x को 13 नहीं, 20 को सौंपा गया है क्योंकि ऑपरेटर * में + से अधिक पूर्वता है, इसलिए यह पहले 3 * 2 के साथ गुणा हो जाता है और फिर 7 में जुड़ जाता है।

यहां, सबसे अधिक पूर्वता वाले ऑपरेटर तालिका के शीर्ष पर दिखाई देते हैं, जो सबसे निचले तल पर दिखाई देते हैं। एक अभिव्यक्ति के भीतर, उच्च पूर्वता ऑपरेटरों का मूल्यांकन पहले किया जाएगा।

वर्ग  ऑपरेटर  संबद्धता 
पोस्टफ़िक्स  () [] -> ++ - -   बाएं से दाएं 
एकल  + -! ~ ++ - - (प्रकार) * और आकार  दाएं से बाएं 
गुणक   * *%  बाएं से दाएं 
additive   + -  बाएं से दाएं 
खिसक जाना   << >>  बाएं से दाएं 
रिलेशनल   <<=>> =  बाएं से दाएं 
समानता   ==! =  बाएं से दाएं 
बिटवाइज़ XOR  बाएं से दाएं 
बिटवार या  बाएं से दाएं 
तार्किक और  &&  बाएं से दाएं 
तार्किक या  ||  बाएं से दाएं 
सशर्त  ?:  दाएं से बाएं 
असाइनमेंट  = + = - = * = / =% = >> = << = & = ^ = = | = |  दाएं से बाएं 
अल्पविराम  बाएं से दाएं 

एक स्थिति हो सकती है, जब आपको कई बार कोड के ब्लॉक को निष्पादित करने की आवश्यकता होती है। सामान्य तौर पर, बयानों को क्रमिक रूप से निष्पादित किया जाता है: किसी फ़ंक्शन में पहला कथन पहले निष्पादित किया जाता है, उसके बाद दूसरा, और इसी तरह।

प्रोग्रामिंग भाषाएँ विभिन्न नियंत्रण संरचनाएं प्रदान करती हैं जो अधिक जटिल निष्पादन पथों के लिए अनुमति देती हैं।

एक लूप स्टेटमेंट हमें कई बार स्टेटमेंट या स्टेटमेंट को निष्पादित करने की अनुमति देता है और निम्नलिखित प्रोग्रामिंग भाषाओं में से एक लूप स्टेटमेंट का सामान्य रूप है -

ऑब्जेक्टिव-सी प्रोग्रामिंग भाषा लूपिंग आवश्यकताओं को संभालने के लिए निम्न प्रकार के लूप प्रदान करती है। उनके विवरण की जांच करने के लिए निम्नलिखित लिंक पर क्लिक करें।

अनु क्रमांक। लूप प्रकार और विवरण
1 घुमाव के दौरान

एक बयान या बयानों के समूह को दोहराता है जबकि किसी दिए गए शर्त सही है। यह लूप बॉडी को निष्पादित करने से पहले स्थिति का परीक्षण करता है।

2 पाश के लिए

कई बार बयानों का एक क्रम निष्पादित करें और उस कोड को संक्षिप्त करें जो लूप चर का प्रबंधन करता है।

3 करते हैं ... जबकि पाश

थोड़ी देर के बयान की तरह, सिवाय इसके कि यह लूप बॉडी के अंत में स्थिति का परीक्षण करता है।

4 स्थिर फंदा

आप किसी अन्य के अंदर एक या अधिक लूप का उपयोग कर सकते हैं, जबकि या इसके लिए लूप।

लूप नियंत्रण विवरण

लूप नियंत्रण स्टेटमेंट अपने सामान्य अनुक्रम से निष्पादन को बदल देता है। जब निष्पादन एक गुंजाइश छोड़ देता है, तो उस दायरे में बनाए गए सभी स्वचालित ऑब्जेक्ट नष्ट हो जाते हैं।

ऑब्जेक्टिव-सी निम्नलिखित नियंत्रण कथनों का समर्थन करता है। उनके विवरण की जांच करने के लिए निम्नलिखित लिंक पर क्लिक करें।

अनु क्रमांक। नियंत्रण विवरण और विवरण
1 तोड़ बयान

टर्मिनेट करता है loop या switch बयान और लूप या स्विच के तुरंत बाद निष्पादन को बयान में स्थानांतरित करता है।

2 जारी रखें बयान

लूप को उसके शरीर के शेष हिस्से को छोड़ने के लिए कहता है और पुनरावृत्ति करने से पहले तुरंत उसकी स्थिति को फिर से बना देता है।

अनंत लूप

एक लूप अनंत लूप बन जाता है अगर कोई स्थिति कभी झूठी न हो। forइस उद्देश्य के लिए पारंपरिक रूप से लूप का उपयोग किया जाता है। चूँकि लूप के लिए बनने वाले तीन में से कोई भी भाव आवश्यक नहीं है, आप सशर्त अभिव्यक्ति को खाली छोड़कर एक अंतहीन लूप बना सकते हैं।

#import <Foundation/Foundation.h>
 
int main () {

   for( ; ; ) {
      NSLog(@"This loop will run forever.\n");
   }

   return 0;
}

जब सशर्त अभिव्यक्ति अनुपस्थित होती है, तो इसे सच माना जाता है। आपके पास एक आरंभीकरण और वेतन वृद्धि की अभिव्यक्ति हो सकती है, लेकिन उद्देश्य-सी प्रोग्रामर आमतौर पर अनंत लूप को इंगित करने के लिए (;;) निर्माण के लिए अधिक उपयोग करते हैं।

निर्णय लेने की संरचनाओं के लिए आवश्यक है कि प्रोग्रामर एक या एक से अधिक शर्तों को प्रोग्राम द्वारा मूल्यांकन या परीक्षण करने के लिए निर्दिष्ट करे, साथ ही एक स्टेटमेंट या स्टेटमेंट के साथ निष्पादित किया जाए यदि शर्त सही हो, और वैकल्पिक रूप से, अन्य स्टेटमेंट्स को निष्पादित किया जाए, यदि कंडीशन हो झूठा होना तय है।

अधिकांश प्रोग्रामिंग भाषाओं में पाया जाने वाला एक विशिष्ट निर्णय लेने की संरचना का सामान्य रूप निम्नलिखित है -

ऑब्जेक्टिव-सी प्रोग्रामिंग लैंग्वेज किसी को मानती है non-zero तथा non-null मूल्यों के रूप में true, और अगर यह या तो है zero या null, तब इसे मान लिया जाता है false मूल्य।

ऑब्जेक्टिव-सी प्रोग्रामिंग लैंग्वेज निम्नलिखित निर्णय लेने के प्रकार प्रदान करती है। उनके विवरण की जाँच करने के लिए निम्न लिंक पर क्लिक करें -

अनु क्रमांक। विवरण और विवरण
1 अगर बयान

एक if statement एक या अधिक बयानों के बाद एक बूलियन अभिव्यक्ति के होते हैं।

2 अगर ... और बयान

एक if statement एक वैकल्पिक द्वारा पीछा किया जा सकता है else statement, जो निष्पादित करता है जब बूलियन अभिव्यक्ति झूठी होती है।

3 nested if statements

You can use one if or else if statement inside another if or else if statement(s).

4 switch statement

A switch statement allows a variable to be tested for equality against a list of values.

5 nested switch statements

You can use one switch statement inside another switch statement(s).

The ? : Operator

We have covered conditional operator ? : in previous chapter which can be used to replace if...else statements. It has the following general form −

Exp1 ? Exp2 : Exp3;

Where Exp1, Exp2, and Exp3 are expressions. Notice the use and placement of the colon.

The value of a ? expression is determined like this: Exp1 is evaluated. If it is true, then Exp2 is evaluated and becomes the value of the entire ? expression. If Exp1 is false, then Exp3 is evaluated and its value becomes the value of the expression.

A function is a group of statements that together perform a task. Every Objective-C program has one C function, which is main(), and all of the most trivial programs can define additional functions.

You can divide up your code into separate functions. How you divide up your code among different functions is up to you, but logically the division usually is so each function performs a specific task.

A function declaration tells the compiler about a function's name, return type, and parameters. A function definition provides the actual body of the function.

Basically in Objective-C, we call the function as method.

The Objective-C foundation framework provides numerous built-in methods that your program can call. For example, method appendString() to append string to another string.

A method is known with various names like a function or a sub-routine or a procedure, etc.

Defining a Method

The general form of a method definition in Objective-C programming language is as follows −

- (return_type) method_name:( argumentType1 )argumentName1 
joiningArgument2:( argumentType2 )argumentName2 ... 
joiningArgumentn:( argumentTypen )argumentNamen {
   body of the function
}

A method definition in Objective-C programming language consists of a method header and a method body. Here are all the parts of a method −

  • Return Type − A method may return a value. The return_type is the data type of the value the function returns. Some methods perform the desired operations without returning a value. In this case, the return_type is the keyword void.

  • Method Name − This is the actual name of the method. The method name and the parameter list together constitute the method signature.

  • Arguments − A argument is like a placeholder. When a function is invoked, you pass a value to the argument. This value is referred to as actual parameter or argument. The parameter list refers to the type, order, and number of the arguments of a method. Arguments are optional; that is, a method may contain no argument.

  • Joining Argument − A joining argument is to make it easier to read and to make it clear while calling it.

  • Method Body − The method body contains a collection of statements that define what the method does.

उदाहरण

निम्नलिखित विधि के लिए स्रोत कोड कहा जाता है max()। यह विधि दो मापदंडों num1 और num2 लेती है और दोनों के बीच अधिकतम रिटर्न देती है -

/* function returning the max between two numbers */
- (int) max:(int) num1 secondNumber:(int) num2 {
   
   /* local variable declaration */
   int result;
 
   if (num1 > num2) {
      result = num1;
   } else {
      result = num2;
   }
 
   return result; 
}

विधि घोषणाएँ

एक विधी declarationकंपाइलर को फ़ंक्शन नाम और विधि को कॉल करने के तरीके के बारे में बताता है। फ़ंक्शन के वास्तविक निकाय को अलग से परिभाषित किया जा सकता है।

एक विधि घोषणा के निम्नलिखित भाग हैं -

- (return_type) function_name:( argumentType1 )argumentName1 
joiningArgument2:( argumentType2 )argumentName2 ... 
joiningArgumentn:( argumentTypen )argumentNamen;

ऊपर परिभाषित फ़ंक्शन अधिकतम () के लिए, निम्नलिखित विधि घोषणा है -

-(int) max:(int)num1 andNum2:(int)num2;

जब आप एक स्रोत फ़ाइल में किसी विधि को परिभाषित करते हैं और आप किसी अन्य फ़ाइल में उस पद्धति को कॉल करते हैं, तो विधि घोषणा आवश्यक है। ऐसे मामले में आपको फ़ंक्शन को कॉल करने वाली फ़ाइल के शीर्ष पर फ़ंक्शन की घोषणा करनी चाहिए।

एक विधि कहलाना

एक उद्देश्य-सी विधि बनाते समय, आप एक परिभाषा देते हैं कि फ़ंक्शन को क्या करना है। एक विधि का उपयोग करने के लिए, आपको परिभाषित कार्य करने के लिए उस फ़ंक्शन को कॉल करना होगा।

जब कोई प्रोग्राम किसी फ़ंक्शन को कॉल करता है, तो प्रोग्राम कंट्रोल को कॉल किया जाता है। एक विधि कहा जाता है कि परिभाषित कार्य करता है, और जब इसका रिटर्न स्टेटमेंट निष्पादित होता है या जब इसके फ़ंक्शन-एंडिंग समापन ब्रेस तक पहुंच जाता है, तो यह प्रोग्राम नियंत्रण को मुख्य कार्यक्रम में वापस कर देता है।

एक विधि को कॉल करने के लिए, आपको बस विधि नाम के साथ आवश्यक मापदंडों को पारित करने की आवश्यकता है, और यदि विधि एक मान लौटाती है, तो आप लौटे मूल्य को संग्रहीत कर सकते हैं। उदाहरण के लिए -

#import <Foundation/Foundation.h>

@interface SampleClass:NSObject
/* method declaration */
- (int)max:(int)num1 andNum2:(int)num2;
@end

@implementation SampleClass

/* method returning the max between two numbers */
- (int)max:(int)num1 andNum2:(int)num2 {

   /* local variable declaration */
   int result;
 
   if (num1 > num2) {
      result = num1;
   } else {
      result = num2;
   }
 
   return result; 
}

@end

int main () {
   
   /* local variable definition */
   int a = 100;
   int b = 200;
   int ret;
   
   SampleClass *sampleClass = [[SampleClass alloc]init];

   /* calling a method to get max value */
   ret = [sampleClass max:a andNum2:b];
 
   NSLog(@"Max value is : %d\n", ret );
   return 0;
}

मैंने मुख्य () फ़ंक्शन के साथ अधिकतम () फ़ंक्शन रखा और स्रोत कोड का अनुपालन किया। अंतिम निष्पादन योग्य रहते हुए, यह निम्नलिखित परिणाम देगा -

2013-09-07 22:28:45.912 demo[26080] Max value is : 200

कार्य तर्क

यदि कोई फ़ंक्शन तर्कों का उपयोग करना है, तो उसे चर की घोषणा करनी चाहिए जो तर्कों के मूल्यों को स्वीकार करते हैं। इन चर को कहा जाता हैformal parameters समारोह के।

औपचारिक पैरामीटर फ़ंक्शन के अंदर अन्य स्थानीय चर की तरह व्यवहार करते हैं और फ़ंक्शन में प्रवेश करने और बाहर निकलने पर नष्ट हो जाते हैं।

किसी फ़ंक्शन को कॉल करते समय, दो तरीके होते हैं जो किसी फ़ंक्शन को तर्क दे सकते हैं -

अनु क्रमांक। कॉल प्रकार और विवरण
1 मूल्य से बुलाओ

यह विधि फ़ंक्शन के औपचारिक पैरामीटर में एक तर्क के वास्तविक मूल्य की प्रतिलिपि बनाती है। इस स्थिति में, फ़ंक्शन के अंदर पैरामीटर में किए गए परिवर्तनों का तर्क पर कोई प्रभाव नहीं पड़ता है।

2 संदर्भ से कॉल करें

यह विधि औपचारिक पैरामीटर में एक तर्क के पते की प्रतिलिपि बनाती है। फ़ंक्शन के अंदर, पते का उपयोग कॉल में उपयोग किए जाने वाले वास्तविक तर्क तक पहुंचने के लिए किया जाता है। इसका अर्थ है कि पैरामीटर में किए गए परिवर्तन तर्क को प्रभावित करते हैं।

डिफ़ॉल्ट रूप से, Objective-C का उपयोग करता है call by valueतर्क पारित करने के लिए। सामान्य तौर पर, इसका मतलब है कि फ़ंक्शन के भीतर कोड फ़ंक्शन को कॉल करने के लिए उपयोग किए गए तर्कों को बदल नहीं सकता है, और अधिकतम () फ़ंक्शन को कॉल करते समय उपर्युक्त उदाहरण समान विधि का उपयोग करता है।

ऑब्जेक्टिव-सी क्लास एक ऑब्जेक्ट को परिभाषित करता है जो संबंधित व्यवहार के साथ डेटा को जोड़ती है। कभी-कभी, यह केवल एक कार्य या व्यवहार की इकाई का प्रतिनिधित्व करने के लिए समझ में आता है, बजाय तरीकों के संग्रह के।

ब्लॉक सी, ऑब्जेक्टिव-सी और सी ++ में जोड़ा जाने वाला एक भाषा-स्तरीय फीचर है, जो आपको कोड के अलग-अलग सेगमेंट बनाने की अनुमति देता है, जिन्हें मान या फ़ंक्शन के रूप में मानो मान लिया गया हो। ब्लॉक ऑब्जेक्टिव-सी ऑब्जेक्ट हैं जिसका मतलब है कि उन्हें NSArray या NSDictionary जैसे संग्रह में जोड़ा जा सकता है। वे एन्कोडिंग दायरे से मूल्यों को पकड़ने की क्षमता भी रखते हैं, जो उन्हें अन्य प्रोग्रामिंग भाषाओं में क्लोजर या लैम्ब्डा के समान बनाते हैं।

सरल ब्लॉक घोषणा सिंटैक्स

returntype (^blockName)(argumentType);

सरल ब्लॉक कार्यान्वयन

returntype (^blockName)(argumentType)= ^{
};

ये रहा एक सरल उदाहरण

void (^simpleBlock)(void) = ^{
   NSLog(@"This is a block");
};

हम ब्लॉक का उपयोग करके आह्वान कर सकते हैं

simpleBlock();

ब्लॉक तर्क और वापसी मान लें

ब्लॉक भी तरीकों और कार्यों की तरह तर्क और रिटर्न मान ले सकते हैं।

यहाँ तर्क और वापसी मूल्यों के साथ ब्लॉक को लागू करने और लागू करने के लिए एक सरल उदाहरण है।

double (^multiplyTwoValues)(double, double) = 
   ^(double firstValue, double secondValue) {
      return firstValue * secondValue;
   };

double result = multiplyTwoValues(2,4); 
NSLog(@"The result is %f", result);

प्रकार परिभाषा का उपयोग कर ब्लॉक

यहां ब्लॉक में टाइफाइड का उपयोग करके एक सरल उदाहरण दिया गया है। कृपया इस नमूने पर ध्यान देंdoesn't work पर online compilerअभी के लिए। उपयोगXCode उसी को चलाने के लिए।

#import <Foundation/Foundation.h>

typedef void (^CompletionBlock)();
@interface SampleClass:NSObject
- (void)performActionWithCompletion:(CompletionBlock)completionBlock;
@end

@implementation SampleClass

- (void)performActionWithCompletion:(CompletionBlock)completionBlock {

   NSLog(@"Action Performed");
   completionBlock();
}

@end

int main() {
   
   /* my first program in Objective-C */
   SampleClass *sampleClass = [[SampleClass alloc]init];
   [sampleClass performActionWithCompletion:^{
      NSLog(@"Completion is called to intimate action is performed.");
   }];

   return 0;
}

इसे संकलित करें और निष्पादित करें, यह निम्नलिखित परिणाम देगा -

2013-09-10 08:13:57.155 demo[284:303] Action Performed
2013-09-10 08:13:57.157 demo[284:303] Completion is called to intimate action is performed.

आईओएस अनुप्रयोगों और मैक ओएस एक्स में ब्लॉक का उपयोग अधिक किया जाता है। इसलिए ब्लॉक के उपयोग को समझने के लिए इसका अधिक महत्वपूर्ण है।

ऑब्जेक्टिव-सी प्रोग्रामिंग लैंग्वेज में, बेसिक डेटा टाइप्स जैसे इंट, फ्लोट, बूल को ऑब्जेक्ट फॉर्म में सेव करने के लिए,

उद्देश्य-सी NSNumber के साथ काम करने के लिए कई तरीके प्रदान करता है और महत्वपूर्ण लोगों को निम्न तालिका में सूचीबद्ध किया गया है।

अनु क्रमांक। विधि और विवरण
1

+ (NSNumber *)numberWithBool:(BOOL)value

एक NSNumber ऑब्जेक्ट बनाता है और देता है, इसे BOOL के रूप में मानते हुए।

2

+ (NSNumber *)numberWithChar:(char)value

NSNumber ऑब्जेक्ट को दिए गए मान के साथ बनाता है और लौटाता है, इसे एक हस्ताक्षरित चार के रूप में मानता है।

3

+ (NSNumber *)numberWithDouble:(double)value

एक NSNumber ऑब्जेक्ट को एक दिए गए मान के साथ बनाता है और लौटाता है, इसे दोहरा मान लेता है।

4

+ (NSNumber *)numberWithFloat:(float)value

एक NSNumber ऑब्जेक्ट बनाता है और देता है, इसे एक फ्लोट के रूप में मानते हुए।

5

+ (NSNumber *)numberWithInt:(int)value

एक NSNumber ऑब्जेक्ट को दिए गए मान के साथ बनाता है और लौटाता है, इसे एक हस्ताक्षरित इंट के रूप में मानता है।

6

+ (NSNumber *)numberWithInteger:(NSInteger)value

एक NSNumber ऑब्जेक्ट बनाता है और देता है, इसे एक NSInteger के रूप में मानते हुए।

7

- (BOOL)boolValue

BOOL के रूप में रिसीवर का मान लौटाता है।

8

- (char)charValue

रिसीवर के मूल्य को चार के रूप में लौटाता है।

9

- (double)doubleValue

एक डबल के रूप में रिसीवर का मान लौटाता है।

10

- (float)floatValue

एक फ्लोट के रूप में रिसीवर का मान लौटाता है।

1 1

- (NSInteger)integerValue

एक NSInteger के रूप में रिसीवर का मान लौटाता है।

12

- (int)intValue

एक इंट के रूप में रिसीवर का मान लौटाता है।

13

- (NSString *)stringValue

मानव-पठनीय स्ट्रिंग के रूप में रिसीवर का मान लौटाता है।

यहां NSNumber का उपयोग करने के लिए एक सरल उदाहरण है जो दो संख्याओं को गुणा करता है और उत्पाद को वापस करता है।

#import <Foundation/Foundation.h>

@interface SampleClass:NSObject
- (NSNumber *)multiplyA:(NSNumber *)a withB:(NSNumber *)b;
@end

@implementation SampleClass

- (NSNumber *)multiplyA:(NSNumber *)a withB:(NSNumber *)b {
   float number1 = [a floatValue];
   float number2 = [b floatValue];
   float product = number1 * number2;
   NSNumber *result = [NSNumber numberWithFloat:product];
   return result;
}

@end

int main() {
   NSAutoreleasePool * pool = [[NSAutoreleasePool alloc] init];

   SampleClass *sampleClass = [[SampleClass alloc]init];
   NSNumber *a = [NSNumber numberWithFloat:10.5];
   NSNumber *b = [NSNumber numberWithFloat:10.0];   
   NSNumber *result = [sampleClass multiplyA:a withB:b];
   NSString *resultString = [result stringValue];
   NSLog(@"The product is %@",resultString);

   [pool drain];
   return 0;
}

अब जब हम प्रोग्राम को संकलित करते हैं और चलाते हैं, तो हमें निम्नलिखित परिणाम मिलेंगे।

2013-09-14 18:53:40.575 demo[16787] The product is 105

ऑब्जेक्टिव-सी प्रोग्रामिंग भाषा नामक एक डेटा संरचना प्रदान करती है the array, जो एक ही प्रकार के तत्वों के एक निश्चित आकार के अनुक्रमिक संग्रह को संग्रहीत कर सकते हैं। एक सरणी का उपयोग डेटा के संग्रह को संग्रहीत करने के लिए किया जाता है, लेकिन एक सरणी के एक ही प्रकार के संग्रह के रूप में सरणी के बारे में सोचना अक्सर अधिक उपयोगी होता है।

नंबर चर, नंबर 1, ... और नंबर 99 जैसे अलग-अलग चर घोषित करने के बजाय, आप एक सरणी चर जैसे संख्याओं की घोषणा करते हैं और संख्याओं का उपयोग करते हैं [0], संख्याओं [1] और ..., संख्याओं [99] का प्रतिनिधित्व करने के लिए। अलग-अलग चर। एक सरणी में एक विशिष्ट तत्व एक सूचकांक द्वारा पहुँचा जाता है।

सभी सरणियों में सन्निहित स्मृति स्थान शामिल हैं। निम्नतम पता पहले तत्व से मेल खाता है और उच्चतम पता अंतिम तत्व से।

घोषणाएँ

उद्देश्य-सी में एक सरणी घोषित करने के लिए, एक प्रोग्रामर तत्वों के प्रकार और एक सरणी द्वारा आवश्यक तत्वों की संख्या को निर्दिष्ट करता है:

type arrayName [ arraySize ];

इसे एकल-आयामी सरणी कहा जाता है । arraySize शून्य से बड़ा पूर्णांक स्थिरांक होना चाहिए और typeकोई भी मान्य उद्देश्य-सी डेटा प्रकार हो सकता है। उदाहरण के लिए, 10-तत्व सरणी घोषित करने के लिएbalance प्रकार के दोहरे, इस कथन का उपयोग करें -

double balance[10];

अब, संतुलन एक चर सरणी है, जो 10 डबल संख्या तक पकड़ के लिए पर्याप्त है।

आरंभिक सारणी

आप उद्देश्य-सी में एक सारणी को एक-एक करके या एक कथन का उपयोग करके इस प्रकार शुरू कर सकते हैं -

double balance[5] = {1000.0, 2.0, 3.4, 17.0, 50.0};

ब्रेसिज़ {} के बीच मानों की संख्या उन तत्वों की संख्या से अधिक नहीं हो सकती है जिन्हें हम वर्ग कोष्ठक [] के बीच के सरणी के लिए घोषित करते हैं। निम्नलिखित सरणी के एकल तत्व को निर्दिष्ट करने के लिए एक उदाहरण है -

यदि आप सरणी के आकार को छोड़ देते हैं, तो आरंभीकरण धारण करने के लिए केवल एक सरणी काफी बड़ी है। इसलिए, यदि आप लिखते हैं -

double balance[] = {1000.0, 2.0, 3.4, 17.0, 50.0};

आप बिलकुल वही एरे बनाएंगे जैसा आपने पिछले उदाहरण में किया था।

balance[4] = 50.0;

उपरोक्त कथन सरणी में तत्व संख्या 5 को 50.0 के मान के साथ निर्दिष्ट करता है। 4 वें इंडेक्स के साथ ऐरे 5 वें यानी आखिरी तत्व होगा क्योंकि सभी ऐरे में उनके पहले तत्व के इंडेक्स के रूप में 0 होता है जिसे बेस इंडेक्स भी कहा जाता है। ऊपर हमने जिस सरणी की चर्चा की है, उसी चित्र का प्रतिनिधित्व निम्नलिखित है -

ऐरे तत्वों तक पहुँचना

सरणी नाम को अनुक्रमित करके एक तत्व को एक्सेस किया जाता है। यह सरणी के नाम के बाद वर्ग ब्रैकेट के भीतर तत्व के सूचकांक को रखकर किया जाता है। उदाहरण के लिए -

double salary = balance[9];

उपरोक्त कथन सरणी से 10 वें तत्व को ले जाएगा और मान को वेतन चर में निर्दिष्ट करेगा। निम्नलिखित एक उदाहरण है, जो उपरोक्त सभी तीन अवधारणाओं का उपयोग करेगा। घोषणा, असाइनमेंट और पहुँच सरणियाँ -

#import <Foundation/Foundation.h>
 
int main () {
   int n[ 10 ];   /* n is an array of 10 integers */
   int i,j;
 
   /* initialize elements of array n to 0 */         
   for ( i = 0; i < 10; i++ ) {
      n[ i ] = i + 100;    /* set element at location i to i + 100 */
   }
   
   /* output each array element's value */
   for (j = 0; j < 10; j++ ) {
      NSLog(@"Element[%d] = %d\n", j, n[j] );
   }
 
   return 0;
}

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

2013-09-14 01:24:06.669 demo[16508] Element[0] = 100
2013-09-14 01:24:06.669 demo[16508] Element[1] = 101
2013-09-14 01:24:06.669 demo[16508] Element[2] = 102
2013-09-14 01:24:06.669 demo[16508] Element[3] = 103
2013-09-14 01:24:06.669 demo[16508] Element[4] = 104
2013-09-14 01:24:06.669 demo[16508] Element[5] = 105
2013-09-14 01:24:06.669 demo[16508] Element[6] = 106
2013-09-14 01:24:06.669 demo[16508] Element[7] = 107
2013-09-14 01:24:06.669 demo[16508] Element[8] = 108
2013-09-14 01:24:06.669 demo[16508] Element[9] = 109

वस्तुनिष्ठ-सी का विवरण

उद्देश्य-सी के लिए एरे महत्वपूर्ण हैं और बहुत अधिक विवरणों की आवश्यकता है। सरणी से संबंधित कुछ महत्वपूर्ण अवधारणाएँ निम्नलिखित हैं जो एक उद्देश्य-सी प्रोग्रामर के लिए स्पष्ट होनी चाहिए -

अनु क्रमांक। अवधारणा और विवरण
1 बहुआयामी सरणियाँ

उद्देश्य-सी बहुआयामी सरणियों का समर्थन करता है। बहुआयामी सरणी का सबसे सरल रूप द्वि-आयामी सरणी है।

2 कार्यों के लिए सरणियाँ पासिंग

आप एक इंडेक्स के बिना सरणी के नाम को निर्दिष्ट करके एक सरणी के लिए फ़ंक्शन को एक पॉइंटर पास कर सकते हैं।

3 किसी फ़ंक्शन से सरणी लौटाएं

उद्देश्य-सी एक फ़ंक्शन को एक सरणी वापस करने की अनुमति देता है।

4 एक सरणी के लिए सूचक

आप किसी भी सूचकांक के बिना, केवल सरणी नाम निर्दिष्ट करके किसी सरणी के पहले तत्व के लिए एक सूचक उत्पन्न कर सकते हैं।

ऑब्जेक्टिव-सी में पॉइंटर्स सीखने में आसान और मजेदार हैं। कुछ ऑब्जेक्टिव-सी प्रोग्रामिंग कार्य पॉइंटर्स के साथ अधिक आसानी से किए जाते हैं, और अन्य कार्य, जैसे डायनेमिक मेमोरी एलोकेशन, पॉइंटर्स का उपयोग किए बिना नहीं किया जा सकता है। इसलिए एक सही उद्देश्य-सी प्रोग्रामर बनने के लिए पॉइंटर्स सीखना आवश्यक हो जाता है। आइए उन्हें सरल और आसान चरणों में सीखना शुरू करें।

जैसा कि आप जानते हैं, प्रत्येक चर एक मेमोरी लोकेशन है और प्रत्येक मेमोरी लोकेशन में इसका पता परिभाषित होता है जिसे एम्परसेंड (&) ऑपरेटर के उपयोग से एक्सेस किया जा सकता है, जो मेमोरी में एक एड्रेस को दर्शाता है। निम्नलिखित उदाहरण पर विचार करें, जो परिभाषित चर के पते को प्रिंट करेगा -

#import <Foundation/Foundation.h>

int main () {
   int  var1;
   char var2[10];

   NSLog(@"Address of var1 variable: %x\n", &var1 );
   NSLog(@"Address of var2 variable: %x\n", &var2 );

   return 0;
}

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह कुछ इस प्रकार परिणाम उत्पन्न करता है -

2013-09-13 03:18:45.727 demo[17552] Address of var1 variable: 1c0843fc
2013-09-13 03:18:45.728 demo[17552] Address of var2 variable: 1c0843f0

तो, आप समझ गए कि मेमोरी एड्रेस क्या है और इसे कैसे एक्सेस करना है, इसलिए कॉन्सेप्ट का बेस खत्म हो गया है। अब देखते हैं कि पॉइंटर क्या है।

पॉइंटर्स क्या हैं?

pointerएक वैरिएबल जिसका मान किसी दूसरे वैरिएबल का पता होता है, यानी मेमोरी लोकेशन का डायरेक्ट एड्रेस। किसी भी चर या स्थिर की तरह, आपको किसी भी चर पते को संग्रहीत करने के लिए इसका उपयोग करने से पहले एक सूचक घोषित करना होगा। सूचक चर घोषणा का सामान्य रूप है -

type *var-name;

यहाँ, typeसूचक का आधार प्रकार है; यह एक वैध उद्देश्य-सी डेटा प्रकार और होना चाहिएvar-nameसूचक चर का नाम है। तारांकन * आप एक सूचक घोषित करने के लिए इस्तेमाल किया वही तारांकन चिह्न है जिसका उपयोग आप गुणन के लिए करते हैं। हालांकि, इस कथन में तारांकन सूचक के रूप में एक चर को नामित करने के लिए उपयोग किया जा रहा है। मान्य सूचक घोषणा निम्नलिखित हैं -

int    *ip;    /* pointer to an integer */
double *dp;    /* pointer to a double */
float  *fp;    /* pointer to a float */
char   *ch     /* pointer to a character */

सभी बिंदुओं के मूल्य का वास्तविक डेटा प्रकार, चाहे पूर्णांक, फ्लोट, वर्ण, या अन्यथा, एक ही, एक लंबा हेक्साडेसिमल संख्या है जो एक स्मृति पते का प्रतिनिधित्व करता है। अलग-अलग डेटा प्रकारों के पॉइंटर्स के बीच एकमात्र अंतर वैरिएबल या कॉन्टिनेंट का डेटा प्रकार है जो पॉइंटर को इंगित करता है।

पॉइंटर्स का उपयोग कैसे करें?

कुछ महत्वपूर्ण ऑपरेशन हैं, जो हम बहुत बार पॉइंटर्स की मदद से करेंगे। (a) हम एक सूचक चर को परिभाषित करते हैं, (b) एक चर का पता एक सूचक को असाइन करें, और (c)अंत में पॉइंटर चर में उपलब्ध पते पर मूल्य का उपयोग करें। यह अपरेंट्री ऑपरेटर का उपयोग करके किया जाता है*यह उसके ऑपरेंड द्वारा निर्दिष्ट पते पर स्थित चर का मान लौटाता है। निम्नलिखित उदाहरण इन कार्यों का उपयोग करता है -

#import <Foundation/Foundation.h>

int main () {
   int  var = 20;    /* actual variable declaration */
   int  *ip;         /* pointer variable declaration */  
   ip = &var;       /* store address of var in pointer variable*/

   NSLog(@"Address of var variable: %x\n", &var  );

   /* address stored in pointer variable */
   NSLog(@"Address stored in ip variable: %x\n", ip );

   /* access the value using the pointer */
   NSLog(@"Value of *ip variable: %d\n", *ip );

   return 0;
}

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह कुछ इस प्रकार परिणाम उत्पन्न करता है -

2013-09-13 03:20:21.873 demo[24179] Address of var variable: 337ed41c
2013-09-13 03:20:21.873 demo[24179] Address stored in ip variable: 337ed41c
2013-09-13 03:20:21.874 demo[24179] Value of *ip variable: 20

उद्देश्य-सी में पूर्ण बिंदु

आपके द्वारा असाइन किए जाने के लिए सटीक पता नहीं होने की स्थिति में पॉइंटर वैरिएबल पर NULL मान निर्दिष्ट करना हमेशा एक अच्छा अभ्यास होता है। यह चर घोषणा के समय किया जाता है। एक सूचक जिसे NULL सौंपा गया है, a कहलाता हैnull सूचक।

NULL पॉइंटर एक मानक है जिसमें कई मानक पुस्तकालयों में परिभाषित शून्य का मान होता है। निम्नलिखित कार्यक्रम पर विचार करें -

#import <Foundation/Foundation.h>

int main () {
   int  *ptr = NULL;
   NSLog(@"The value of ptr is : %x\n", ptr  );
   return 0;
}

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

2013-09-13 03:21:19.447 demo[28027] The value of ptr is : 0

अधिकांश ऑपरेटिंग सिस्टम पर, प्रोग्रामों को पता 0 पर स्मृति तक पहुंचने की अनुमति नहीं है क्योंकि यह मेमोरी ऑपरेटिंग सिस्टम द्वारा आरक्षित है। हालाँकि, स्मृति पता 0 का विशेष महत्व है; यह इंगित करता है कि सूचक एक सुलभ मेमोरी स्थान को इंगित करने का इरादा नहीं है। लेकिन सम्मेलन द्वारा, यदि एक पॉइंटर में शून्य (शून्य) मान है, तो इसे कुछ भी नहीं माना जाता है।

अशक्त सूचक की जाँच करने के लिए, आप निम्नानुसार एक कथन का उपयोग कर सकते हैं -

if(ptr)     /* succeeds if p is not null */
if(!ptr)    /* succeeds if p is null */

उद्देश्य-सी पॉइंटर्स विस्तार से

पॉइंटर्स के पास कई लेकिन आसान अवधारणाएं हैं और वे ऑब्जेक्टिव-सी प्रोग्रामिंग के लिए बहुत महत्वपूर्ण हैं। कुछ महत्वपूर्ण सूचक अवधारणाएँ निम्नलिखित हैं, जो एक उद्देश्य-सी प्रोग्रामर के लिए स्पष्ट होनी चाहिए -

अनु क्रमांक। अवधारणा और विवरण
1 उद्देश्य-सी - सूचक अंकगणित

चार अंकगणितीय ऑपरेटर हैं जो पॉइंटर्स पर उपयोग किए जा सकते हैं: ++, -, +, -

2 उद्देश्य-सी - संकेत की सरणी

आप कई बिंदुओं को रखने के लिए सरणियों को परिभाषित कर सकते हैं।

3 उद्देश्य-सी - सूचक को इंगित करता है

ऑब्जेक्टिव-सी आपको पॉइंटर वगैरह पर पॉइंटर लगाने की सुविधा देता है।

4 उद्देश्य-सी में कार्यों के लिए पासिंग पॉइंटर्स

एक तर्क को संदर्भ या पते से पास करना दोनों पारित तर्क को बुलाए गए फ़ंक्शन द्वारा कॉलिंग फ़ंक्शन में परिवर्तित करने में सक्षम बनाते हैं।

5 ऑब्जेक्टिव-सी में कार्यों से सूचक लौटें

उद्देश्य-सी एक फ़ंक्शन को स्थानीय चर, स्थिर चर और गतिशील रूप से आवंटित स्मृति के लिए एक सूचक को वापस करने की अनुमति देता है।

ऑब्जेक्टिव-सी प्रोग्रामिंग लैंग्वेज में स्ट्रिंग को NSString का उपयोग करते हुए दर्शाया गया है और इसकी उपवर्ग NSMutableString स्ट्रिंग ऑब्जेक्ट्स बनाने के लिए कई तरीके प्रदान करता है। एक स्ट्रिंग ऑब्जेक्ट बनाने का सबसे सरल तरीका ऑब्जेक्टिव-सी @ "..." निर्माण का उपयोग करना है -

NSString *greeting = @"Hello";

स्ट्रिंग बनाने और प्रिंट करने का एक सरल उदाहरण नीचे दिखाया गया है।

#import <Foundation/Foundation.h>

int main () {
   NSString *greeting = @"Hello";
   NSLog(@"Greeting message: %@\n", greeting );

   return 0;
}

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नानुसार परिणाम उत्पन्न करता है -

2013-09-11 01:21:39.922 demo[23926] Greeting message: Hello

उद्देश्य-सी तार में हेरफेर के लिए कई तरीकों का समर्थन करता है -

अनु क्रमांक। विधि और उद्देश्य
1

- (NSString *)capitalizedString;

रिसीवर का पूंजीकृत प्रतिनिधित्व लौटाता है।

2

- (unichar)characterAtIndex:(NSUInteger)index;

किसी दिए गए सरणी स्थिति में वर्ण लौटाता है।

3

- (double)doubleValue;

रिसीवर के टेक्स्ट के फ्लोटिंग-पॉइंट वैल्यू को डबल के रूप में लौटाता है।

4

- (float)floatValue;

रिसीवर के टेक्स्ट के फ्लोटिंग-पॉइंट वैल्यू को फ्लोट के रूप में लौटाता है।

5

- (BOOL)hasPrefix:(NSString *)aString;

एक बूलियन मान लौटाता है जो इंगित करता है कि क्या दिए गए स्ट्रिंग रिसीवर के शुरुआती वर्णों से मेल खाते हैं।

6

- (BOOL)hasSuffix:(NSString *)aString;

एक बूलियन मान लौटाता है जो इंगित करता है कि क्या दिया गया स्ट्रिंग रिसीवर के अंतिम पात्रों से मेल खाता है।

7

- (id)initWithFormat:(NSString *)format ...;

किसी दिए गए प्रारूप स्ट्रिंग का उपयोग करके एक NSString ऑब्जेक्ट लौटाता है जिसमें एक टेम्पलेट है जिसमें शेष तर्क मान प्रतिस्थापित किए जाते हैं।

8

- (NSInteger)integerValue;

रिसीवर के पाठ का NSInteger मान लौटाता है।

9

- (BOOL)isEqualToString:(NSString *)aString;

एक बूलियन मान लौटाता है जो इंगित करता है कि क्या एक दी गई स्ट्रिंग रिसीवर के बराबर शाब्दिक यूनिकोड-आधारित तुलना का उपयोग कर रही है।

10

- (NSUInteger)length;

रिसीवर में यूनिकोड वर्णों की संख्या देता है।

1 1

- (NSString *)lowercaseString;

रिसीवर के कम प्रतिनिधित्व को लौटाता है।

12

- (NSRange)rangeOfString:(NSString *)aString;

प्राप्त करता है और रिसीवर के भीतर किसी दिए गए स्ट्रिंग की पहली घटना की सीमा देता है।

13

- (NSString *)stringByAppendingFormat:(NSString *)format ...;

रिसीवर को दिए गए तार को दिए गए स्ट्रिंग में दिए गए प्रारूप स्ट्रिंग और निम्न तर्कों से निर्मित स्ट्रिंग देता है।

14

- (NSString *)stringByTrimmingCharactersInSet:(NSCharacterSet *)set;

किसी दिए गए वर्ण सेट में निहित रिसीवर वर्णों के दोनों सिरों को हटाकर बनाई गई एक नई स्ट्रिंग देता है।

15

- (NSString *)substringFromIndex:(NSUInteger)anIndex;

दिए गए इंडेक्स में एक से अंत तक रिसीवर के पात्रों से युक्त एक नया स्ट्रिंग लौटाता है।

उपर्युक्त कार्यों में से कुछ का उदाहरण निम्नलिखित है -

#import <Foundation/Foundation.h>

int main () {
   NSString *str1 = @"Hello";
   NSString *str2 = @"World";
   NSString *str3;
   int  len ;

   NSAutoreleasePool * pool = [[NSAutoreleasePool alloc] init];

   /* uppercase string */
   str3 = [str2 uppercaseString];
   NSLog(@"Uppercase String :  %@\n", str3 );

   /* concatenates str1 and str2 */
   str3 = [str1 stringByAppendingFormat:@"World"];
   NSLog(@"Concatenated string:   %@\n", str3 );

   /* total length of str3 after concatenation */
   len = [str3 length];
   NSLog(@"Length of Str3 :  %d\n", len );

   /* InitWithFormat */
   str3 = [[NSString alloc] initWithFormat:@"%@ %@",str1,str2];	
   NSLog(@"Using initWithFormat:   %@\n", str3 );
   [pool drain];

   return 0;
}

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नानुसार परिणाम उत्पन्न करता है -

2013-09-11 01:15:45.069 demo[30378] Uppercase String :  WORLD
2013-09-11 01:15:45.070 demo[30378] Concatenated string:   HelloWorld
2013-09-11 01:15:45.070 demo[30378] Length of Str3 :  10
2013-09-11 01:15:45.070 demo[30378] Using initWithFormat:   Hello World

आप NSString क्लास संदर्भ में उद्देश्य-सी NSString संबंधित विधियों की एक पूरी सूची पा सकते हैं ।

ऑब्जेक्टिव-सी सरणियां आपको उस प्रकार के चर को परिभाषित करने की अनुमति देती हैं जो एक ही तरह के कई डेटा आइटम पकड़ सकते हैं लेकिन structure एक अन्य उपयोगकर्ता-परिभाषित डेटा प्रकार है जो उद्देश्य-सी प्रोग्रामिंग में उपलब्ध है जो आपको विभिन्न प्रकार के डेटा आइटम को संयोजित करने की अनुमति देता है।

संरचनाएं एक रिकॉर्ड का प्रतिनिधित्व करने के लिए उपयोग की जाती हैं, मान लीजिए कि आप किसी पुस्तकालय में अपनी पुस्तकों का ट्रैक रखना चाहते हैं। आप प्रत्येक पुस्तक के बारे में निम्नलिखित विशेषताओं को ट्रैक करना चाहते हैं -

  • Title
  • Author
  • Subject
  • बुक आईडी

एक संरचना को परिभाषित करना

एक संरचना को परिभाषित करने के लिए, आपको इसका उपयोग करना चाहिए structबयान। स्ट्रक्चर स्टेटमेंट एक नए डेटा प्रकार को परिभाषित करता है, जिसमें आपके प्रोग्राम के लिए एक से अधिक सदस्य होते हैं। संरचना विवरण का प्रारूप यह है -

struct [structure tag] {
   member definition;
   member definition;
   ...
   member definition;
} [one or more structure variables];

structure tagवैकल्पिक है और प्रत्येक सदस्य परिभाषा एक सामान्य चर परिभाषा है, जैसे कि int i; या फ्लोट एफ; या कोई अन्य मान्य चर परिभाषा। संरचना की परिभाषा के अंत में, अंतिम अर्धविराम से पहले, आप एक या एक से अधिक संरचना चर निर्दिष्ट कर सकते हैं लेकिन यह वैकल्पिक है। यहाँ आप बुक संरचना को घोषित करने का तरीका है -

struct Books {
   NSString *title;
   NSString *author;
   NSString *subject;
   int   book_id;
} book;

संरचना सदस्यों तक पहुँचना

किसी संरचना के किसी भी सदस्य तक पहुँचने के लिए, हम इसका उपयोग करते हैं member access operator (.)। सदस्य एक्सेस ऑपरेटर को संरचना चर नाम और संरचना सदस्य के बीच की अवधि के रूप में कोडित किया जाता है जिसे हम एक्सेस करना चाहते हैं। आप उपयोग करेंगेstructकीवर्ड संरचना प्रकार के चर को परिभाषित करने के लिए। निम्नलिखित संरचना के उपयोग की व्याख्या करने के लिए उदाहरण है -

#import <Foundation/Foundation.h>

struct Books {
   NSString *title;
   NSString *author;
   NSString *subject;
   int   book_id;
};
 
int main() {
   struct Books Book1;        /* Declare Book1 of type Book */
   struct Books Book2;        /* Declare Book2 of type Book */
 
   /* book 1 specification */
   Book1.title = @"Objective-C Programming";
   Book1.author = @"Nuha Ali"; 
   Book1.subject = @"Objective-C Programming Tutorial";
   Book1.book_id = 6495407;

   /* book 2 specification */
   Book2.title = @"Telecom Billing";
   Book2.author = @"Zara Ali";
   Book2.subject = @"Telecom Billing Tutorial";
   Book2.book_id = 6495700;
 
   /* print Book1 info */
   NSLog(@"Book 1 title : %@\n", Book1.title);
   NSLog(@"Book 1 author : %@\n", Book1.author);
   NSLog(@"Book 1 subject : %@\n", Book1.subject);
   NSLog(@"Book 1 book_id : %d\n", Book1.book_id);

   /* print Book2 info */
   NSLog(@"Book 2 title : %@\n", Book2.title);
   NSLog(@"Book 2 author : %@\n", Book2.author);
   NSLog(@"Book 2 subject : %@\n", Book2.subject);
   NSLog(@"Book 2 book_id : %d\n", Book2.book_id);

   return 0;
}

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

2013-09-14 04:20:07.947 demo[20591] Book 1 title : Objective-C Programming
2013-09-14 04:20:07.947 demo[20591] Book 1 author : Nuha Ali
2013-09-14 04:20:07.947 demo[20591] Book 1 subject : Objective-C Programming Tutorial
2013-09-14 04:20:07.947 demo[20591] Book 1 book_id : 6495407
2013-09-14 04:20:07.947 demo[20591] Book 2 title : Telecom Billing
2013-09-14 04:20:07.947 demo[20591] Book 2 author : Zara Ali
2013-09-14 04:20:07.947 demo[20591] Book 2 subject : Telecom Billing Tutorial
2013-09-14 04:20:07.947 demo[20591] Book 2 book_id : 6495700

फ़ंक्शन तर्क के रूप में संरचनाएं

आप किसी अन्य चर या पॉइंटर को पास करते हुए एक संरचना को फ़ंक्शन तर्क के रूप में उसी तरह से पारित कर सकते हैं। आप संरचना चर को उसी तरह एक्सेस करेंगे जैसे आपने उपरोक्त उदाहरण में एक्सेस किया है -

#import <Foundation/Foundation.h>

struct Books {
   NSString *title;
   NSString *author;
   NSString *subject;
   int   book_id;
};

@interface SampleClass:NSObject
/* function declaration */
- (void) printBook:( struct Books) book ;
@end

@implementation SampleClass 

- (void) printBook:( struct Books) book {
   NSLog(@"Book title : %@\n", book.title);
   NSLog(@"Book author : %@\n", book.author);
   NSLog(@"Book subject : %@\n", book.subject);
   NSLog(@"Book book_id : %d\n", book.book_id);
}

@end

int main() {
   struct Books Book1;        /* Declare Book1 of type Book */
   struct Books Book2;        /* Declare Book2 of type Book */
 
   /* book 1 specification */
   Book1.title = @"Objective-C Programming";
   Book1.author = @"Nuha Ali"; 
   Book1.subject = @"Objective-C Programming Tutorial";
   Book1.book_id = 6495407;

   /* book 2 specification */
   Book2.title = @"Telecom Billing";
   Book2.author = @"Zara Ali";
   Book2.subject = @"Telecom Billing Tutorial";
   Book2.book_id = 6495700;
 
   SampleClass *sampleClass = [[SampleClass alloc]init];
   /* print Book1 info */
   [sampleClass printBook: Book1];

   /* Print Book2 info */
   [sampleClass printBook: Book2];

   return 0;
}

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

2013-09-14 04:34:45.725 demo[8060] Book title : Objective-C Programming
2013-09-14 04:34:45.725 demo[8060] Book author : Nuha Ali
2013-09-14 04:34:45.725 demo[8060] Book subject : Objective-C Programming Tutorial
2013-09-14 04:34:45.725 demo[8060] Book book_id : 6495407
2013-09-14 04:34:45.725 demo[8060] Book title : Telecom Billing
2013-09-14 04:34:45.725 demo[8060] Book author : Zara Ali
2013-09-14 04:34:45.725 demo[8060] Book subject : Telecom Billing Tutorial
2013-09-14 04:34:45.725 demo[8060] Book book_id : 6495700

पॉइंट टू स्ट्रक्चर्स

आप संरचनाओं को पॉइंटर्स को उसी तरह से परिभाषित कर सकते हैं जैसे आप पॉइंटर को किसी अन्य वैरिएबल के रूप में परिभाषित करते हैं -

struct Books *struct_pointer;

अब, आप उपरोक्त परिभाषित सूचक चर में एक संरचना चर के पते को संग्रहीत कर सकते हैं। संरचना चर का पता खोजने के लिए, संरचना के नाम से पहले & ऑपरेटर को निम्नानुसार रखें -

struct_pointer = &Book1;

एक संरचना के सदस्यों को उस संरचना के लिए एक सूचक का उपयोग करने के लिए, आपको निम्नानुसार -> ऑपरेटर का उपयोग करना होगा -

struct_pointer->title;

आइए हम संरचना सूचक का उपयोग करके उदाहरण के ऊपर फिर से लिखें, आशा है कि यह आपके लिए अवधारणा को समझना आसान होगा -

#import <Foundation/Foundation.h>

struct Books {
   NSString *title;
   NSString *author;
   NSString *subject;
   int   book_id;
};

@interface SampleClass:NSObject
/* function declaration */
- (void) printBook:( struct Books *) book ;
@end

@implementation SampleClass 
- (void) printBook:( struct Books *) book {
   NSLog(@"Book title : %@\n", book->title);
   NSLog(@"Book author : %@\n", book->author);
   NSLog(@"Book subject : %@\n", book->subject);
   NSLog(@"Book book_id : %d\n", book->book_id);
}

@end

int main() {
   struct Books Book1;        /* Declare Book1 of type Book */
   struct Books Book2;        /* Declare Book2 of type Book */
 
   /* book 1 specification */
   Book1.title = @"Objective-C Programming";
   Book1.author = @"Nuha Ali"; 
   Book1.subject = @"Objective-C Programming Tutorial";
   Book1.book_id = 6495407;

   /* book 2 specification */
   Book2.title = @"Telecom Billing";
   Book2.author = @"Zara Ali";
   Book2.subject = @"Telecom Billing Tutorial";
   Book2.book_id = 6495700;
 
   SampleClass *sampleClass = [[SampleClass alloc]init];
   /* print Book1 info by passing address of Book1 */
   [sampleClass printBook:&Book1];

   /* print Book2 info by passing address of Book2 */
   [sampleClass printBook:&Book2];

   return 0;
}

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

2013-09-14 04:38:13.942 demo[20745] Book title : Objective-C Programming
2013-09-14 04:38:13.942 demo[20745] Book author : Nuha Ali
2013-09-14 04:38:13.942 demo[20745] Book subject : Objective-C Programming Tutorial
2013-09-14 04:38:13.942 demo[20745] Book book_id : 6495407
2013-09-14 04:38:13.942 demo[20745] Book title : Telecom Billing
2013-09-14 04:38:13.942 demo[20745] Book author : Zara Ali
2013-09-14 04:38:13.942 demo[20745] Book subject : Telecom Billing Tutorial
2013-09-14 04:38:13.942 demo[20745] Book book_id : 6495700

बिट फील्ड्स

बिट फील्ड्स एक संरचना में डेटा की पैकिंग की अनुमति देता है। यह विशेष रूप से तब उपयोगी होता है जब मेमोरी या डेटा स्टोरेज प्रीमियम पर होता है। विशिष्ट उदाहरण -

  • मशीन शब्द में कई वस्तुओं को पैक करना। उदाहरण के लिए 1 बिट फ़्लैग को कॉम्पैक्ट किया जा सकता है।

  • बाहरी फ़ाइल स्वरूपों को पढ़ना - गैर-मानक फ़ाइल स्वरूपों को पढ़ा जा सकता है। उदा 9 बिट पूर्णांक।

ऑब्जेक्टिव-सी हमें एक संरचना की परिभाषा में ऐसा करने की अनुमति देता है: चर के बाद की लंबाई। उदाहरण के लिए -

struct packed_struct {
   unsigned int f1:1;
   unsigned int f2:1;
   unsigned int f3:1;
   unsigned int f4:1;
   unsigned int type:4;
   unsigned int my_int:9;
} pack;

यहाँ, pack_struct में 6 सदस्य हैं: चार 1 बिट फ़्लैग f1..f3, एक 4 बिट प्रकार और 9 बिट my_int।

ऑब्जेक्टिव-सी स्वचालित रूप से उपरोक्त बिट फ़ील्ड को यथासंभव कॉम्पैक्ट रूप से पैक करता है, बशर्ते कि फ़ील्ड की अधिकतम लंबाई कंप्यूटर के पूर्णांक शब्द की लंबाई से कम या उसके बराबर हो। यदि यह मामला नहीं है, तो कुछ कंपाइलर फ़ील्ड के लिए मेमोरी ओवरलैप की अनुमति दे सकते हैं, जबकि अन्य अगले शब्द में अगले फ़ील्ड को संग्रहीत करेगा।

Objective-C Preprocessorसंकलक का हिस्सा नहीं है, लेकिन संकलन प्रक्रिया में एक अलग कदम है। सरलीकृत शब्दों में, एक ऑब्जेक्टिव-सी प्रीप्रोसेसर केवल एक पाठ प्रतिस्थापन उपकरण है और यह संकलक को वास्तविक संकलन से पहले आवश्यक प्रसंस्करण करने का निर्देश देता है। हम OCPP के रूप में ऑब्जेक्टिव-सी प्रीप्रोसेसर का उल्लेख करेंगे।

सभी प्रीप्रोसेसर कमांड्स पाउंड सिंबल (#) से शुरू होते हैं। यह पहला नॉनब्लैंक चरित्र होना चाहिए, और पठनीयता के लिए, पहले कॉलम में एक प्रीप्रोसेसर निर्देश शुरू होना चाहिए। निम्नलिखित अनुभाग सभी महत्वपूर्ण प्रीप्रोसेसर निर्देशों को सूचीबद्ध करता है -

अनु क्रमांक। निर्देश और विवरण
1

#define

एक प्रीप्रोसेसर मैक्रो को सब्सक्राइब करें

2

#include

किसी अन्य फ़ाइल से एक विशेष हेडर सम्मिलित करता है

3

#undef

एक पूर्वप्रक्रमक मैक्रो को परिभाषित करता है

4

#ifdef

अगर यह मैक्रो परिभाषित है तो सही है

5

#ifndef

यदि यह मैक्रो परिभाषित नहीं है, तो सही है

6

#if

यदि एक संकलन समय की स्थिति सच है तो टेस्ट

7

#else

#If के लिए विकल्प

8

#elif

# एक कथन में एक # शब्द

9

#endif

पूर्वप्रक्रमक सशर्त समाप्त करता है

10

#error

स्टेटर पर प्रिंट त्रुटि संदेश

1 1

#pragma

मानकीकृत विधि का उपयोग करके संकलक को विशेष आदेश जारी करता है

प्रीप्रोसेसर्स उदाहरण

विभिन्न निर्देशों को समझने के लिए निम्नलिखित उदाहरणों का विश्लेषण करें।

#define MAX_ARRAY_LENGTH 20

यह निर्देश OCPP को MAX_ARRAY_LENGTH के उदाहरणों को 20 के साथ बदलने के लिए कहता है। पठनीयता बढ़ाने के लिए स्थिरांक के लिए #define का उपयोग करें ।

#import <Foundation/Foundation.h>
#include "myheader.h"

ये निर्देश OCPP को नींव से प्राप्त करने के लिए कहते हैं Foundation Frameworkऔर पाठ को वर्तमान स्रोत फ़ाइल में जोड़ें। अगली पंक्ति OCPP को बताती हैmyheader.h स्थानीय निर्देशिका से और वर्तमान स्रोत फ़ाइल में सामग्री जोड़ें।

#undef  FILE_SIZE
#define FILE_SIZE 42

यह OCPP को मौजूदा FILE_SIZE को अपरिभाषित करने और इसे 42 के रूप में परिभाषित करने के लिए कहता है।

#ifndef MESSAGE
   #define MESSAGE "You wish!"
#endif

यह OCPP को केवल MESSAGE को परिभाषित करने के लिए बताता है यदि MESSAGE पहले से परिभाषित नहीं है।

#ifdef DEBUG
   /* Your debugging statements here */
#endif

यह OCPP को उस प्रक्रिया को करने के लिए कहता है, जब DEBUG परिभाषित किया जाता है। यदि आप संकलन के समय -DDEBUG ध्वज को gcc संकलक से गुजारें तो यह उपयोगी है । यह डिबग को परिभाषित करेगा, ताकि आप संकलन के दौरान फ्लाई पर डिबगिंग को चालू और बंद कर सकें।

पूर्वनिर्धारित मैक्रों

ANSI C मैक्रोज़ की एक संख्या को परिभाषित करता है। यद्यपि हर एक प्रोग्रामिंग में आपके उपयोग के लिए उपलब्ध है, पूर्वनिर्धारित मैक्रोज़ को सीधे संशोधित नहीं किया जाना चाहिए।

अनु क्रमांक। मैक्रो और विवरण
1

__DATE__

"MMM DD YYYY" प्रारूप में एक चरित्र शाब्दिक के रूप में वर्तमान तिथि

2

__TIME__

"एचएच: एमएम: एसएस" प्रारूप में एक चरित्र शाब्दिक के रूप में वर्तमान समय

3

__FILE__

इसमें एक स्ट्रिंग शाब्दिक के रूप में वर्तमान फ़ाइल नाम शामिल है।

4

__LINE__

इसमें दशमलव स्थिरांक के रूप में वर्तमान लाइन संख्या शामिल है।

5

__STDC__

1 के रूप में परिभाषित किया गया जब कंपाइलर एएनएसआई मानक का अनुपालन करता है।

आइए निम्न उदाहरण का प्रयास करें -

#import <Foundation/Foundation.h>

int main() {
   NSLog(@"File :%s\n", __FILE__ );
   NSLog(@"Date :%s\n", __DATE__ );
   NSLog(@"Time :%s\n", __TIME__ );
   NSLog(@"Line :%d\n", __LINE__ );
   NSLog(@"ANSI :%d\n", __STDC__ );
   
   return 0;
}

जब एक फ़ाइल में उपरोक्त कोड main.m संकलित और निष्पादित किया जाता है, यह निम्नलिखित परिणाम उत्पन्न करता है -

2013-09-14 04:46:14.859 demo[20683] File :main.m
2013-09-14 04:46:14.859 demo[20683] Date :Sep 14 2013
2013-09-14 04:46:14.859 demo[20683] Time :04:46:14
2013-09-14 04:46:14.859 demo[20683] Line :8
2013-09-14 04:46:14.859 demo[20683] ANSI :1

प्रीप्रोसेसर ऑपरेटर

उद्देश्य-सी प्रीप्रोसेसर आपको मैक्रोज़ बनाने में मदद करने के लिए निम्नलिखित ऑपरेटर प्रदान करता है -

मैक्रो कंटिन्यूएशन (\)

एक मैक्रो आमतौर पर एक ही लाइन पर समाहित होना चाहिए। मैक्रो निरंतरता ऑपरेटर का उपयोग मैक्रो को जारी रखने के लिए किया जाता है जो एक पंक्ति के लिए बहुत लंबा है। उदाहरण के लिए -

#define  message_for(a, b)  \
   NSLog(@#a " and " #b ": We love you!\n")

स्ट्रिंग (#)

स्ट्रिंग या नंबर-साइन ऑपरेटर ('#'), जब एक मैक्रो परिभाषा के भीतर उपयोग किया जाता है, तो एक मैक्रो पैरामीटर को एक स्ट्रिंग स्थिरांक में परिवर्तित करता है। इस ऑपरेटर का उपयोग केवल एक मैक्रो में किया जा सकता है जिसमें एक निर्दिष्ट तर्क या पैरामीटर सूची है। उदाहरण के लिए -

#import <Foundation/Foundation.h>

#define  message_for(a, b)  \
   NSLog(@#a " and " #b ": We love you!\n")

int main(void) {
   message_for(Carole, Debra);
   return 0;
}

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

2013-09-14 05:46:14.859 demo[20683] Carole and Debra: We love you!

टोकन पेस्टिंग (##)

एक मैक्रो परिभाषा के भीतर टोकन-पेस्टिंग ऑपरेटर (##) दो तर्कों को जोड़ता है। यह मैक्रो परिभाषा में दो अलग-अलग टोकन को एकल टोकन में शामिल होने की अनुमति देता है। उदाहरण के लिए -

#import <Foundation/Foundation.h>

#define tokenpaster(n) NSLog (@"token" #n " = %d", token##n)

int main(void) {
   int token34 = 40;
   
   tokenpaster(34);
   return 0;
}

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

2013-09-14 05:48:14.859 demo[20683] token34 = 40

यह कैसे हुआ, क्योंकि इस उदाहरण के परिणामस्वरूप प्रीप्रोसेसर से वास्तविक उत्पादन होता है -

NSLog (@"token34 = %d", token34);

यह उदाहरण टोकन ## n के टोकन 34 में समरूपता दिखाता है और यहां हमने दोनों का उपयोग किया है stringize तथा token-pasting

परिभाषित () ऑपरेटर

प्रीप्रोसेसर definedऑपरेटर यह निर्धारित करने के लिए निरंतर अभिव्यक्तियों में उपयोग किया जाता है कि क्या कोई पहचानकर्ता #define का उपयोग करके परिभाषित किया गया है। यदि निर्दिष्ट पहचानकर्ता को परिभाषित किया गया है, तो मान सत्य (गैर-शून्य) है। यदि प्रतीक परिभाषित नहीं है, तो मान गलत है (शून्य)। परिभाषित ऑपरेटर निम्नानुसार है -

#import <Foundation/Foundation.h>

#if !defined (MESSAGE)
   #define MESSAGE "You wish!"
#endif

int main(void) {
   NSLog(@"Here is the message: %s\n", MESSAGE);  
   return 0;
}

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

2013-09-14 05:48:19.859 demo[20683] Here is the message: You wish!

परिमाणित मैक्रोज़

ओसीपीपी के शक्तिशाली कार्यों में से एक पैरामीटर मैक्रोज़ का उपयोग करके कार्यों को अनुकरण करने की क्षमता है। उदाहरण के लिए, हमारे पास कुछ संख्याओं को एक वर्ग के रूप में निम्नानुसार हो सकता है -

int square(int x) {
   return x * x;
}

हम एक मैक्रो का उपयोग करके उपरोक्त कोड को फिर से लिख सकते हैं -

#define square(x) ((x) * (x))

तर्कों के साथ मैक्रोज़ का उपयोग करके परिभाषित किया जाना चाहिए #defineनिर्देश का उपयोग करने से पहले। तर्क सूची को कोष्ठक में संलग्न किया गया है और उसे स्थूल नाम का तुरंत पालन करना चाहिए। स्थूल नाम और खुले कोष्ठक के बीच रिक्त स्थान की अनुमति नहीं है। उदाहरण के लिए -

#import <Foundation/Foundation.h>

#define MAX(x,y) ((x) > (y) ? (x) : (y))

int main(void) {
   NSLog(@"Max between 20 and 10 is %d\n", MAX(10, 20));  
   return 0;
}

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

2013-09-14 05:52:15.859 demo[20683] Max between 20 and 10 is 20

ऑब्जेक्टिव-सी प्रोग्रामिंग लैंग्वेज नामक एक कीवर्ड प्रदान करता है typedef, जिसे आप एक नया नाम देने के लिए उपयोग कर सकते हैं। निम्नलिखित एक शब्द को परिभाषित करने के लिए एक उदाहरण हैBYTE एक-बाइट संख्या के लिए -

typedef unsigned char BYTE;

इस प्रकार की परिभाषा के बाद, पहचानकर्ता BYTE को टाइप के लिए एक संक्षिप्त नाम के रूप में इस्तेमाल किया जा सकता है unsigned char, for example:

BYTE  b1, b2;

कन्वेंशन द्वारा, अपरकेस अक्षरों का उपयोग इन परिभाषाओं के लिए किया जाता है ताकि उपयोगकर्ता को यह याद दिलाया जा सके कि टाइप नाम वास्तव में एक प्रतीकात्मक संक्षिप्त नाम है, लेकिन आप लोअरकेस का उपयोग कर सकते हैं, जैसे कि -

typedef unsigned char byte;

आप उपयोग कर सकते हैं typedefउपयोगकर्ता-परिभाषित डेटा प्रकार को एक नाम देने के लिए। उदाहरण के लिए, आप एक नए डेटा प्रकार को परिभाषित करने के लिए संरचना के साथ टाइपफेड का उपयोग कर सकते हैं और फिर उस डेटा प्रकार का उपयोग सीधे संरचना चर को परिभाषित करने के लिए कर सकते हैं:

#import <Foundation/Foundation.h>

typedef struct Books {
   NSString *title;
   NSString *author;
   NSString *subject;
   int book_id;
} Book;
 
int main() {
   Book book;
   book.title = @"Objective-C Programming";
   book.author = @"TutorialsPoint";
   book.subject = @"Programming tutorial";
   book.book_id = 100;
   
   NSLog( @"Book title : %@\n", book.title);
   NSLog( @"Book author : %@\n", book.author);
   NSLog( @"Book subject : %@\n", book.subject);
   NSLog( @"Book Id : %d\n", book.book_id);

   return 0;
}

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

2013-09-12 12:21:53.745 demo[31183] Book title : Objective-C Programming
2013-09-12 12:21:53.745 demo[31183] Book author : TutorialsPoint
2013-09-12 12:21:53.745 demo[31183] Book subject : Programming tutorial
2013-09-12 12:21:53.745 demo[31183] Book Id : 100

टाइप्डिफ़ बनाम #define

#define एक उद्देश्य-सी निर्देश है, जिसका उपयोग विभिन्न डेटा प्रकारों के लिए उपनामों को परिभाषित करने के लिए भी किया जाता है typedef लेकिन निम्नलिखित मतभेदों के साथ -

  • typedef केवल प्रकारों को प्रतीकात्मक नाम देने तक सीमित है #define मूल्यों के लिए उपनाम को परिभाषित करने के लिए भी इस्तेमाल किया जा सकता है, जैसे कि आप 1 को एक के रूप में परिभाषित कर सकते हैं, आदि।

  • typedef जहां संकलक द्वारा व्याख्या की जाती है #define बयानों को पूर्व-प्रोसेसर द्वारा संसाधित किया जाता है।

#Define का सबसे सरल उपयोग निम्नलिखित है -

#import <Foundation/Foundation.h>
 
#define TRUE  1
#define FALSE 0
 
int main( ) {
   NSLog( @"Value of TRUE : %d\n", TRUE);
   NSLog( @"Value of FALSE : %d\n", FALSE);

   return 0;
}

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

2013-09-12 12:23:37.993 demo[5160] Value of TRUE : 1
2013-09-12 12:23:37.994 demo[5160] Value of FALSE : 0

टाइप कास्टिंग एक चर को एक डेटा प्रकार से दूसरे डेटा प्रकार में बदलने का एक तरीका है। उदाहरण के लिए, यदि आप एक लंबे पूर्णांक को एक साधारण पूर्णांक में संग्रहित करना चाहते हैं तो आप इंट में लंबे कास्ट टाइप कर सकते हैं। आप स्पष्ट रूप से उपयोग करके मूल्यों को एक प्रकार से दूसरे प्रकार में परिवर्तित कर सकते हैंcast operator निम्नानुसार है -

(type_name) expression

ऑब्जेक्टिव-सी में, हम आम तौर पर फ्लोटिंग पॉइंट ऑपरेशन करने के लिए सीजीफ्लोट का उपयोग करते हैं, जो कि 32-बिट के मामले में बुनियादी प्रकार के फ्लोट से उत्पन्न होता है और 64-बिट के मामले में डबल होता है। निम्नलिखित उदाहरण पर विचार करें जहां कास्ट ऑपरेटर एक पूर्णांक चर के विभाजन को एक फ्लोटिंग-पॉइंट ऑपरेशन के रूप में निष्पादित करने का कारण बनता है -

#import <Foundation/Foundation.h>

int main() {
   int sum = 17, count = 5;
   CGFloat mean;

   mean = (CGFloat) sum / count;
   NSLog(@"Value of mean : %f\n", mean );

   return 0;
}

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

2013-09-11 01:35:40.047 demo[20634] Value of mean : 3.400000

यहां यह ध्यान दिया जाना चाहिए कि कास्ट ऑपरेटर की विभाजन पर पूर्वता है, इसलिए इसका मूल्य sum को पहले टाइप किया जाता है double और अंत में यह एक दोहरे मूल्य की गणना द्वारा विभाजित हो जाता है।

टाइप रूपांतरण को निहित किया जा सकता है जो संकलक द्वारा स्वचालित रूप से किया जाता है या इसे के उपयोग के माध्यम से स्पष्ट रूप से निर्दिष्ट किया जा सकता है cast operator। जब भी प्रकार के रूपांतरण आवश्यक होते हैं, कास्ट ऑपरेटर का उपयोग करना अच्छा प्रोग्रामिंग अभ्यास माना जाता है।

पूर्णांक पदोन्नति

पूर्णांक संवर्धन वह प्रक्रिया है जिसके द्वारा पूर्णांक प्रकार के मान "छोटे" की तुलना में अधिक होते हैं int या unsigned int या तो परिवर्तित कर रहे हैं int या unsigned int। एक इंट में एक चरित्र को जोड़ने के एक उदाहरण पर विचार करें -

#import <Foundation/Foundation.h>

int main() {
   int  i = 17;
   char c = 'c';  /* ascii value is 99 */
   int sum;

   sum = i + c;
   NSLog(@"Value of sum : %d\n", sum );

   return 0;
}

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

2013-09-11 01:38:28.492 demo[980] Value of sum : 116

यहां, योग का मान 116 के रूप में आ रहा है क्योंकि संकलक पूर्णांक पदोन्नति कर रहा है और वास्तविक अतिरिक्त संचालन करने से पहले एससीआई के मान को 'सी' में परिवर्तित कर रहा है।

सामान्य अंकगणितीय रूपांतरण

usual arithmetic conversionsअपने मूल्यों को एक सामान्य प्रकार में डालने के लिए अंतर्निहित रूप से प्रदर्शन किया जाता है। कंपाइलर पहले पूर्णांक पदोन्नति करता है , यदि ऑपरेंड में अभी भी विभिन्न प्रकार हैं तो वे उस प्रकार में परिवर्तित हो जाते हैं जो निम्न पदानुक्रम में सबसे अधिक दिखाई देता है - -

सामान्य अंकगणितीय रूपांतरण असाइनमेंट ऑपरेटरों के लिए नहीं किए जाते हैं, न ही तार्किक ऑपरेटरों और& और || आइये अवधारणा को समझने के लिए निम्नलिखित उदाहरण लेते हैं -

#import <Foundation/Foundation.h>

int main() {
   int  i = 17;
   char c = 'c';  /* ascii value is 99 */
   CGFloat sum;

   sum = i + c;
   NSLog(@"Value of sum : %f\n", sum );
   return 0;
}

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

2013-09-11 01:41:39.192 demo[15351] Value of sum : 116.000000

यहाँ, यह समझना आसान है कि पहले c पूर्णांक में परिवर्तित हो जाता है लेकिन क्योंकि अंतिम मान फ्लोट है, इसलिए सामान्य अंकगणितीय रूपांतरण लागू होता है और संकलक i और c को फ्लोट में परिवर्तित करता है और उन्हें फ्लोट परिणाम देता है।

NSLog विधि

लॉग प्रिंट करने के लिए, हम ऑब्जेक्टिव-सी प्रोग्रामिंग भाषा में NSLog पद्धति का उपयोग करते हैं, जिसका उपयोग हमने हैलो वर्ल्ड उदाहरण से किया है।

आइए एक सरल कोड देखें जो "हैलो वर्ल्ड" शब्दों को प्रिंट करेगा -

#import <Foundation/Foundation.h>

int main() {
   NSLog(@"Hello, World! \n");
   return 0;
}

अब, जब हम प्रोग्राम संकलित करते हैं और चलाते हैं, तो हमें निम्नलिखित परिणाम मिलेंगे।

2013-09-16 00:32:50.888 demo[16669] Hello, World!

लाइव ऐप्स में लॉग को अक्षम करना

चूंकि NSLogs का उपयोग हम अपने एप्लिकेशन में करते हैं, इसलिए इसे डिवाइस के लॉग में प्रिंट किया जाएगा और लाइव बिल्ड में लॉग को प्रिंट करना अच्छा नहीं है। इसलिए, हम मुद्रण लॉग के लिए एक प्रकार की परिभाषा का उपयोग करते हैं और हम उन्हें नीचे दिखाए अनुसार उपयोग कर सकते हैं।

#import <Foundation/Foundation.h>

#if DEBUG == 0
#define DebugLog(...)
#elif DEBUG == 1
#define DebugLog(...) NSLog(__VA_ARGS__)
#endif

int main() {
   DebugLog(@"Debug log, our custom addition gets \
   printed during debug only" );
   NSLog(@"NSLog gets printed always" );     
   return 0;
}

अब, जब हम प्रोग्राम को डिबग मोड में संकलित और चलाते हैं, तो हमें निम्न परिणाम मिलेगा।

2013-09-11 02:47:07.723 demo[618] Debug log, our custom addition gets printed during debug only
2013-09-11 02:47:07.723 demo[618] NSLog gets printed always

अब, जब हम प्रोग्राम को रिलीज़ मोड में संकलित करते हैं और चलाते हैं, तो हमें निम्नलिखित परिणाम मिलेंगे।

2013-09-11 02:47:45.248 demo[3158] NSLog gets printed always

ऑब्जेक्टिव-सी प्रोग्रामिंग में, NSError वर्ग के साथ त्रुटि हैंडलिंग उपलब्ध है Foundation framework.

NSError ऑब्जेक्ट केवल एक त्रुटि कोड या त्रुटि स्ट्रिंग का उपयोग करके अमीर और अधिक एक्स्टेंसिबल त्रुटि जानकारी को एन्क्रिप्ट करता है। NSError ऑब्जेक्ट की मुख्य विशेषताएँ एक त्रुटि डोमेन (स्ट्रिंग द्वारा दर्शाया गया), एक डोमेन-विशिष्ट त्रुटि कोड और एप्लिकेशन विशिष्ट जानकारी वाले उपयोगकर्ता जानकारी शब्दकोश हैं।

NSError

उद्देश्य-सी कार्यक्रम NSError ऑब्जेक्ट्स का उपयोग रनटाइम त्रुटियों के बारे में जानकारी देने के लिए करते हैं, जिनके बारे में उपयोगकर्ताओं को सूचित किया जाना चाहिए। ज्यादातर मामलों में, एक प्रोग्राम एक डायलॉग या शीट में इस त्रुटि की जानकारी प्रदर्शित करता है। लेकिन यह जानकारी की व्याख्या भी कर सकता है और या तो उपयोगकर्ता को त्रुटि से उबरने का प्रयास करने या अपने आप त्रुटि को ठीक करने का प्रयास करने के लिए कह सकता है

NSError वस्तु के होते हैं -

  • Domain - त्रुटि डोमेन पूर्वनिर्धारित NSError डोमेन में से एक हो सकता है या कस्टम डोमेन का वर्णन करने वाला एक मनमाना स्ट्रिंग हो सकता है और डोमेन शून्य नहीं होना चाहिए।

  • Code - त्रुटि के लिए त्रुटि कोड।

  • User Info - त्रुटि के लिए userInfo शब्दकोश और userInfo शून्य हो सकता है।

निम्न उदाहरण दिखाता है कि कस्टम त्रुटि कैसे बनाई जाए

NSString *domain = @"com.MyCompany.MyApplication.ErrorDomain";
NSString *desc = NSLocalizedString(@"Unable to complete the process", @"");
NSDictionary *userInfo = @{ NSLocalizedDescriptionKey : desc };
NSError *error = [NSError errorWithDomain:domain code:-101 userInfo:userInfo];

एक संकेतक के संदर्भ में पारित उपरोक्त त्रुटि नमूने का पूरा कोड यहां दिया गया है -

#import <Foundation/Foundation.h>

@interface SampleClass:NSObject
-(NSString *) getEmployeeNameForID:(int) id withError:(NSError **)errorPtr;
@end

@implementation SampleClass

-(NSString *) getEmployeeNameForID:(int) id withError:(NSError **)errorPtr {
   if(id == 1) {
      return @"Employee Test Name";
   } else {
      NSString *domain = @"com.MyCompany.MyApplication.ErrorDomain";
      NSString *desc =@"Unable to complete the process";
      NSDictionary *userInfo = [[NSDictionary alloc] 
      initWithObjectsAndKeys:desc,
      @"NSLocalizedDescriptionKey",NULL];  
      *errorPtr = [NSError errorWithDomain:domain code:-101 
      userInfo:userInfo];
      return @"";
   }
}

@end

int main() {
   NSAutoreleasePool * pool = [[NSAutoreleasePool alloc] init];
   SampleClass *sampleClass = [[SampleClass alloc]init];
   NSError *error = nil;
   NSString *name1 = [sampleClass getEmployeeNameForID:1 withError:&error];
  
   if(error) {
      NSLog(@"Error finding Name1: %@",error);
   } else {
      NSLog(@"Name1: %@",name1);
   }
   
   error = nil;
   NSString *name2 = [sampleClass getEmployeeNameForID:2 withError:&error];

   if(error) {
      NSLog(@"Error finding Name2: %@",error);
   } else {
      NSLog(@"Name2: %@",name2);
   }

   [pool drain];
   return 0; 
}

उपरोक्त उदाहरण में, हम एक नाम वापस करते हैं यदि आईडी 1 है, अन्यथा हम उपयोगकर्ता द्वारा परिभाषित त्रुटि ऑब्जेक्ट सेट करते हैं।

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

2013-09-14 18:01:00.809 demo[27632] Name1: Employee Test Name
2013-09-14 18:01:00.809 demo[27632] Error finding Name2: Unable to complete the process

जब वे निष्पादित हों, तो कमांड लाइन से आपके उद्देश्य-सी कार्यक्रमों तक कुछ मूल्यों को पारित करना संभव है। ये मान कहलाते हैंcommand line arguments और कई बार वे आपके कार्यक्रम के लिए महत्वपूर्ण होते हैं, खासकर जब आप कोड के अंदर उन मूल्यों को कोडित करने के बजाय बाहर से अपने कार्यक्रम को नियंत्रित करना चाहते हैं।

कमांड लाइन तर्कों को मुख्य () फ़ंक्शन तर्कों का उपयोग करके नियंत्रित किया जाता है जहां argc पारित तर्कों की संख्या को संदर्भित करता है, और argv[]एक पॉइंटर ऐरे है, जो प्रोग्राम को दिए गए प्रत्येक तर्क की ओर इशारा करता है। निम्नलिखित एक सरल उदाहरण है, जो यह जांचता है कि क्या कमांड लाइन से आपूर्ति की गई कोई तर्क है और तदनुसार कार्रवाई करें -

#import <Foundation/Foundation.h>

int main( int argc, char *argv[] ) {
   if( argc == 2 ) {
      NSLog(@"The argument supplied is %s\n", argv[1]);
   } else if( argc > 2 ) {
      NSLog(@"Too many arguments supplied.\n");
   } else {
      NSLog(@"One argument expected.\n");
   }
}

जब उपरोक्त कोड को एक ही तर्क के साथ संकलित और निष्पादित किया जाता है, तो "परीक्षण" कहते हैं, यह निम्नलिखित परिणाम उत्पन्न करता है।

2013-09-13 03:01:17.333 demo[7640] The argument supplied is testing

जब उपरोक्त कोड दो तर्कों के साथ संकलित और निष्पादित किया जाता है, तो परीक्षण 1 और परीक्षण 2 कहते हैं, यह निम्नलिखित परिणाम उत्पन्न करता है।

2013-09-13 03:01:18.333 demo[7640] Too many arguments supplied.

जब उपरोक्त कोड को किसी भी तर्क को पारित किए बिना संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है।

2013-09-13 03:01:18.333 demo[7640] One argument expected

इस बात पर ध्यान दिया जाना चाहिए कि argv[0] कार्यक्रम का नाम स्वयं रखता है और argv[1]पहली कमांड-लाइन तर्क के लिए एक संकेतक है, और * argv [n] अंतिम तर्क है। यदि कोई तर्क नहीं दिया जाता है, तो argc एक होगा, अन्यथा यदि आप एक तर्क पास करते हैं, तोargc 2 पर सेट है।

आप सभी कमांड लाइन तर्कों को एक स्थान से अलग कर देते हैं, लेकिन यदि तर्क में एक स्थान होता है, तो आप ऐसे तर्क को दोहरे उद्धरण चिह्नों "" या एकल उद्धरणों '' के अंदर रखकर पारित कर सकते हैं। हम एक बार फिर उदाहरण के ऊपर फिर से लिखते हैं जहां हम प्रोग्राम का नाम प्रिंट करेंगे और हम डबल कोट्स के अंदर डालकर कमांड-लाइन तर्क भी पास करेंगे -

#import <Foundation/Foundation.h>

int main( int argc, char *argv[] ) {
   NSLog(@"Program name %s\n", argv[0]);
 
   if( argc == 2 ) {
      NSLog(@"The argument supplied is %s\n", argv[1]);
   } else if( argc > 2 ) {
      NSLog(@"Too many arguments supplied.\n");
   } else {
      NSLog(@"One argument expected.\n");
   }
   
   return 0;
}

जब उपरोक्त कोड संकलित किया जाता है और इसे अंतरिक्ष द्वारा अलग किए गए एकल तर्क के साथ निष्पादित किया जाता है लेकिन अंदर दोहरे उद्धरण "परीक्षण 1 परीक्षण 2" कहते हैं, तो यह निम्नलिखित परिणाम उत्पन्न करता है।

2017-11-30 06:36:59.081 main[71010] Program name main
2017-11-30 06:36:59.082 main[71010] One argument expected.

ऑब्जेक्टिव-सी प्रोग्रामिंग लैंग्वेज का मुख्य उद्देश्य सी प्रोग्रामिंग लैंग्वेज में ऑब्जेक्ट ओरिएंटेशन को जोड़ना है और क्लास ऑब्जेक्टिव-सी की केंद्रीय विशेषता है जो ऑब्जेक्ट-ओरिएंटेड प्रोग्रामिंग को सपोर्ट करते हैं और जिन्हें अक्सर उपयोगकर्ता-परिभाषित प्रकार कहा जाता है।

एक क्लास का उपयोग ऑब्जेक्ट के रूप को निर्दिष्ट करने के लिए किया जाता है और यह उस डेटा को एक साफ पैकेज में हेरफेर करने के लिए डेटा प्रतिनिधित्व और विधियों को जोड़ती है। किसी वर्ग के भीतर के डेटा और विधियों को कक्षा के सदस्य कहा जाता है।

उद्देश्य-सी विशेषताओं

  • वर्ग को दो अलग-अलग वर्गों में परिभाषित किया गया है @interface तथा @implementation

  • लगभग सब कुछ वस्तुओं के रूप में है।

  • वस्तुओं को संदेश मिलता है और वस्तुओं को अक्सर रिसीवर के रूप में संदर्भित किया जाता है।

  • ऑब्जेक्ट में उदाहरण चर होते हैं।

  • ऑब्जेक्ट और इंस्टेंस चर में गुंजाइश है।

  • कक्षाएं किसी वस्तु के कार्यान्वयन को छिपाती हैं।

  • अन्य वर्गों में वर्ग उदाहरण चर तक पहुंच प्रदान करने के लिए गुणों का उपयोग किया जाता है।

उद्देश्य-सी कक्षा परिभाषाएँ

जब आप एक वर्ग को परिभाषित करते हैं, तो आप डेटा प्रकार के लिए एक खाका परिभाषित करते हैं। यह वास्तव में किसी भी डेटा को परिभाषित नहीं करता है, लेकिन यह परिभाषित करता है कि वर्ग नाम का अर्थ क्या है, अर्थात, कक्षा की कोई वस्तु क्या होगी और इस तरह के ऑब्जेक्ट पर क्या संचालन किया जा सकता है।

एक वर्ग की परिभाषा खोजशब्द से शुरू होती है @interfaceइंटरफ़ेस (वर्ग) नाम के बाद; और वर्ग शरीर, घुंघराले ब्रेसिज़ की एक जोड़ी द्वारा संलग्न। ऑब्जेक्टिव-सी में, सभी वर्गों को बेस क्लास कहा जाता हैNSObject। यह सभी ऑब्जेक्टिव-सी कक्षाओं का सुपरक्लास है। यह स्मृति आवंटन और आरंभीकरण जैसी बुनियादी विधियाँ प्रदान करता है। उदाहरण के लिए, हमने कीवर्ड का उपयोग करके बॉक्स डेटा प्रकार को परिभाषित कियाclass निम्नानुसार है -

@interface Box:NSObject {
   //Instance variables
   double length;    // Length of a box
   double breadth;   // Breadth of a box
}
@property(nonatomic, readwrite) double height;  // Property

@end

उदाहरण चर निजी हैं और केवल कक्षा कार्यान्वयन के अंदर ही सुलभ हैं।

ऑब्जेक्ट-सी ऑब्जेक्ट को आबंटित और प्रारंभ करना

एक वर्ग वस्तुओं के लिए ब्लूप्रिंट प्रदान करता है, इसलिए मूल रूप से एक वर्ग से एक वस्तु बनाई जाती है। हम एक वर्ग की वस्तुओं को उसी तरह की घोषणा के साथ घोषित करते हैं जिसे हम बुनियादी प्रकार के चर घोषित करते हैं। निम्नलिखित कथन कक्षा बॉक्स की दो वस्तुओं की घोषणा करते हैं -

Box box1 = [[Box alloc]init];     // Create box1 object of type Box
Box box2 = [[Box alloc]init];     // Create box2 object of type Box

दोनों ऑब्जेक्ट्स बॉक्स 1 और बॉक्स 2 में डेटा सदस्यों की अपनी प्रति होगी।

डेटा सदस्यों तक पहुँचना

किसी वर्ग की वस्तुओं के गुणों को प्रत्यक्ष सदस्य एक्सेस ऑपरेटर (।) का उपयोग करके एक्सेस किया जा सकता है। हमें चीजों को स्पष्ट करने के लिए निम्न उदाहरण का प्रयास करें -

#import <Foundation/Foundation.h>

@interface Box:NSObject {
   double length;    // Length of a box
   double breadth;   // Breadth of a box
   double height;    // Height of a box
}

@property(nonatomic, readwrite) double height;  // Property
-(double) volume;
@end

@implementation Box

@synthesize height; 

-(id)init {
   self = [super init];
   length = 1.0;
   breadth = 1.0;
   return self;
}

-(double) volume {
   return length*breadth*height;
}

@end

int main() {
   NSAutoreleasePool * pool = [[NSAutoreleasePool alloc] init];    
   Box *box1 = [[Box alloc]init];    // Create box1 object of type Box
   Box *box2 = [[Box alloc]init];    // Create box2 object of type Box

   double volume = 0.0;             // Store the volume of a box here
 
   // box 1 specification
   box1.height = 5.0; 

   // box 2 specification
   box2.height = 10.0;
  
   // volume of box 1
   volume = [box1 volume];
   NSLog(@"Volume of Box1 : %f", volume);
   
   // volume of box 2
   volume = [box2 volume];
   NSLog(@"Volume of Box2 : %f", volume);
   
   [pool drain];
   return 0;
}

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

2013-09-22 21:25:33.314 ClassAndObjects[387:303] Volume of Box1 : 5.000000
2013-09-22 21:25:33.316 ClassAndObjects[387:303] Volume of Box2 : 10.000000

गुण

उद्देश्य-सी में गुण प्रस्तुत किए जाते हैं ताकि यह सुनिश्चित हो सके कि कक्षा के उदाहरण चर को कक्षा के बाहर पहुँचा जा सके।

  • गुणों से शुरू होता है @property, जो एक कीवर्ड है

  • इसका उपयोग एक्सेस स्पेसर्स के साथ किया जाता है, जो कि गैर-परमाणु या परमाणु, रीडराइट या आसानी से और मजबूत, असुरक्षित_अनुरोधी या कमजोर हैं। यह चर के प्रकार के आधार पर भिन्न होता है। किसी भी पॉइंटर प्रकार के लिए, हम मजबूत, असुरक्षित_सुरक्षित या कमजोर का उपयोग कर सकते हैं। इसी प्रकार अन्य प्रकारों के लिए हम रीडराइट या रेडोनली का उपयोग कर सकते हैं।

  • इसके बाद चर का डेटाटाइप होता है।

  • अंत में, हमारे पास एक अर्धविराम द्वारा संपत्ति का नाम है।

  • हम कार्यान्वयन वर्ग में संश्लेषित विवरण जोड़ सकते हैं। लेकिन नवीनतम XCode में, संश्लेषण भाग XCode द्वारा ध्यान रखा जाता है और आपको सिंथेसाइज़ स्टेटमेंट को शामिल करने की आवश्यकता नहीं होती है।

यह केवल उन गुणों के साथ संभव है, जिन्हें हम कक्षा के उदाहरण चर तक पहुँचा सकते हैं। दरअसल, गुणों के लिए आंतरिक रूप से गेटटर और सेटर विधियां बनाई जाती हैं।

उदाहरण के लिए, मान लें कि हमारे पास एक संपत्ति है @property (nonatomic ,readonly ) BOOL isDone। हुड के नीचे, बसे और गेटर्स हैं जो नीचे दिखाए गए हैं।

-(void)setIsDone(BOOL)isDone;
-(BOOL)isDone;

ऑब्जेक्ट-ओरिएंटेड प्रोग्रामिंग में सबसे महत्वपूर्ण अवधारणाओं में से एक विरासत की है। वंशानुक्रम हमें एक वर्ग को दूसरी कक्षा के संदर्भ में परिभाषित करने की अनुमति देता है जिससे एप्लिकेशन बनाना और उसे बनाए रखना आसान हो जाता है। यह कोड कार्यक्षमता और तेजी से कार्यान्वयन समय का पुन: उपयोग करने का अवसर भी प्रदान करता है।

कक्षा बनाते समय, पूरी तरह से नए डेटा सदस्यों और सदस्य कार्यों को लिखने के बजाय, प्रोग्रामर यह नामित कर सकता है कि नए वर्ग को किसी मौजूदा वर्ग के सदस्यों को विरासत में प्राप्त करना चाहिए। इस मौजूदा वर्ग को कहा जाता हैbase वर्ग, और नए वर्ग के रूप में जाना जाता है derived कक्षा।

उत्तराधिकार का विचार लागू करता है is aरिश्ते। उदाहरण के लिए, स्तनपायी IS-A पशु, कुत्ता IS-A स्तनपायी, इसलिए कुत्ता IS-A पशु भी है।

आधार और व्युत्पन्न वर्ग

ऑब्जेक्टिव-सी केवल मल्टीलेवल इनहेरिटेंस की अनुमति देता है, अर्थात, इसमें केवल एक बेस क्लास हो सकता है लेकिन मल्टीलेवल इनहेरिटेंस की अनुमति देता है। ऑब्जेक्टिव-सी में सभी कक्षाएं सुपरक्लास से ली गई हैंNSObject

@interface derived-class: base-class

बेस क्लास पर विचार करें Person और इसका व्युत्पन्न वर्ग Employee निम्नानुसार है -

#import <Foundation/Foundation.h>
 
@interface Person : NSObject {
   NSString *personName;
   NSInteger personAge;
}

- (id)initWithName:(NSString *)name andAge:(NSInteger)age;
- (void)print;

@end

@implementation Person

- (id)initWithName:(NSString *)name andAge:(NSInteger)age {
   personName = name;
   personAge = age;
   return self;
}

- (void)print {
   NSLog(@"Name: %@", personName);
   NSLog(@"Age: %ld", personAge);
}

@end

@interface Employee : Person {
   NSString *employeeEducation;
}

- (id)initWithName:(NSString *)name andAge:(NSInteger)age 
  andEducation:(NSString *)education;
- (void)print;
@end

@implementation Employee

- (id)initWithName:(NSString *)name andAge:(NSInteger)age 
   andEducation: (NSString *)education {
      personName = name;
      personAge = age;
      employeeEducation = education;
      return self;
   }

- (void)print {
   NSLog(@"Name: %@", personName);
   NSLog(@"Age: %ld", personAge);
   NSLog(@"Education: %@", employeeEducation);
}

@end

int main(int argc, const char * argv[]) {
   NSAutoreleasePool * pool = [[NSAutoreleasePool alloc] init];        
   NSLog(@"Base class Person Object");
   Person *person = [[Person alloc]initWithName:@"Raj" andAge:5];
   [person print];
   NSLog(@"Inherited Class Employee Object");
   Employee *employee = [[Employee alloc]initWithName:@"Raj" 
   andAge:5 andEducation:@"MBA"];
   [employee print];        
   [pool drain];
   return 0;
}

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

2013-09-22 21:20:09.842 Inheritance[349:303] Base class Person Object
2013-09-22 21:20:09.844 Inheritance[349:303] Name: Raj
2013-09-22 21:20:09.844 Inheritance[349:303] Age: 5
2013-09-22 21:20:09.845 Inheritance[349:303] Inherited Class Employee Object
2013-09-22 21:20:09.845 Inheritance[349:303] Name: Raj
2013-09-22 21:20:09.846 Inheritance[349:303] Age: 5
2013-09-22 21:20:09.846 Inheritance[349:303] Education: MBA

अभिगम नियंत्रण और विरासत

एक व्युत्पन्न वर्ग अपने आधार वर्ग के सभी निजी सदस्यों को एक्सेस कर सकता है यदि इसे इंटरफ़ेस वर्ग में परिभाषित किया गया है, लेकिन यह कार्यान्वयन फ़ाइल में परिभाषित निजी सदस्यों तक नहीं पहुंच सकता है।

हम अलग-अलग पहुंच प्रकारों को संक्षेप में बता सकते हैं कि कौन उन्हें निम्नलिखित तरीके से एक्सेस कर सकता है -

एक व्युत्पन्न वर्ग को निम्नलिखित अपवादों के साथ सभी आधार वर्ग विधियां और चर प्राप्त होते हैं -

  • एक्सटेंशन की मदद से कार्यान्वयन फ़ाइल में घोषित चर सुलभ नहीं हैं।

  • एक्सटेंशन की मदद से कार्यान्वयन फ़ाइल में घोषित तरीके सुलभ नहीं हैं।

  • आधार वर्ग में विरासत में मिली विधि लागू होने पर, फिर व्युत्पन्न वर्ग में विधि निष्पादित की जाती है।

शब्द polymorphismकई रूपों का मतलब है। आमतौर पर, बहुरूपता तब होती है जब वर्गों का एक पदानुक्रम होता है और वे वंशानुक्रम से संबंधित होते हैं।

ऑब्जेक्टिव-सी बहुरूपता का मतलब है कि एक सदस्य फ़ंक्शन को कॉल करने से फ़ंक्शन के प्रकार के आधार पर किसी भिन्न फ़ंक्शन को निष्पादित किया जाएगा।

उदाहरण पर विचार करें, हमारे पास एक वर्ग आकृति है जो सभी आकारों के लिए मूल इंटरफ़ेस प्रदान करता है। स्क्वायर और रेक्टेंगल को बेस क्लास शेप से लिया गया है।

हमारे पास मेथड प्रिन्ट है जो ओओपी फीचर के बारे में दिखाने जा रहा है polymorphism

#import <Foundation/Foundation.h>

@interface Shape : NSObject {
   CGFloat area;
}

- (void)printArea;
- (void)calculateArea;
@end

@implementation Shape
- (void)printArea {
   NSLog(@"The area is %f", area);
}

- (void)calculateArea {

}

@end

@interface Square : Shape {
   CGFloat length;
}

- (id)initWithSide:(CGFloat)side;
- (void)calculateArea;

@end

@implementation Square
- (id)initWithSide:(CGFloat)side {
   length = side;
   return self;
}

- (void)calculateArea {
   area = length * length;
}

- (void)printArea {
   NSLog(@"The area of square is %f", area);
}

@end

@interface Rectangle : Shape {
   CGFloat length;
   CGFloat breadth;
}

- (id)initWithLength:(CGFloat)rLength andBreadth:(CGFloat)rBreadth;
@end

@implementation Rectangle
- (id)initWithLength:(CGFloat)rLength andBreadth:(CGFloat)rBreadth {
   length = rLength;
   breadth = rBreadth;
   return self;
}

- (void)calculateArea {
   area = length * breadth;
}

@end

int main(int argc, const char * argv[]) {
   NSAutoreleasePool * pool = [[NSAutoreleasePool alloc] init];
   Shape *square = [[Square alloc]initWithSide:10.0];
   [square calculateArea];
   [square printArea];
   Shape *rect = [[Rectangle alloc]
   initWithLength:10.0 andBreadth:5.0];
   [rect calculateArea];
   [rect printArea];        
   [pool drain];
   return 0;
}

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

2013-09-22 21:21:50.785 Polymorphism[358:303] The area of square is 100.000000
2013-09-22 21:21:50.786 Polymorphism[358:303] The area is 50.000000

उपरोक्त गणना में विधि की उपलब्धता के आधार पर गणना करें और प्रिंट करें, या तो आधार वर्ग में विधि या निष्पादित वर्ग।

पॉलीमॉर्फिज़्म दो वर्गों के विधि कार्यान्वयन के आधार पर आधार वर्ग और व्युत्पन्न वर्ग के बीच तरीकों की स्विचिंग को संभालता है।

सभी उद्देश्य-सी कार्यक्रम निम्नलिखित दो मौलिक तत्वों से बने होते हैं -

  • Program statements (code) - यह एक प्रोग्राम का हिस्सा है जो क्रिया करता है और उन्हें विधियाँ कहा जाता है।

  • Program data - डेटा प्रोग्राम की जानकारी है जो प्रोग्राम फ़ंक्शन से प्रभावित होती है।

एनकैप्सुलेशन एक ऑब्जेक्ट-ओरिएंटेड प्रोग्रामिंग अवधारणा है जो डेटा और फ़ंक्शंस को एक साथ बांधता है जो डेटा में हेरफेर करता है और जो बाहरी हस्तक्षेप और दुरुपयोग दोनों से सुरक्षित रखता है। डेटा एनकैप्सुलेशन के महत्वपूर्ण OOP अवधारणा के लिए नेतृत्व कियाdata hiding

Data encapsulation डेटा और उन्हें उपयोग करने वाले फ़ंक्शंस को बंडल करने का एक तंत्र है, और data abstraction केवल इंटरफेस को उजागर करने और उपयोगकर्ता से कार्यान्वयन विवरण को छिपाने का एक तंत्र है।

ऑब्जेक्टिव-सी उपयोगकर्ता-परिभाषित प्रकारों के निर्माण के माध्यम से छुपाए जाने वाले एन्कैप्सुलेशन और डेटा के गुणों का समर्थन करता है, जिन्हें कहा जाता है classes। उदाहरण के लिए -

@interface Adder : NSObject {
   NSInteger total;
}

- (id)initWithInitialNumber:(NSInteger)initialNumber;
- (void)addNumber:(NSInteger)newNumber;
- (NSInteger)getTotal;

@end

चर कुल निजी है और हम कक्षा के बाहर से प्रवेश नहीं कर सकते हैं। इसका मतलब यह है कि उन्हें केवल Adder वर्ग के अन्य सदस्यों द्वारा ही एक्सेस किया जा सकता है, न कि आपके प्रोग्राम के किसी अन्य भाग द्वारा। यह एक तरह से इनकैप्सुलेशन हासिल किया जाता है।

इंटरफ़ेस फ़ाइल के अंदर तरीके सुलभ हैं और दायरे में सार्वजनिक हैं।

निजी तरीके हैं, जिनकी मदद से लिखा जाता है extensions, जो हम आगामी अध्यायों में सीखेंगे।

डेटा एनकैप्सुलेशन उदाहरण

कोई भी उद्देश्य-सी कार्यक्रम जहां आप सार्वजनिक और निजी सदस्यों के चर के साथ एक वर्ग को लागू करते हैं, डेटा एनकैप्सुलेशन और डेटा एब्स्ट्रक्शन का एक उदाहरण है। निम्नलिखित उदाहरण पर विचार करें -

#import <Foundation/Foundation.h>

@interface Adder : NSObject {
   NSInteger total;
}

- (id)initWithInitialNumber:(NSInteger)initialNumber;
- (void)addNumber:(NSInteger)newNumber;
- (NSInteger)getTotal;

@end

@implementation Adder
-(id)initWithInitialNumber:(NSInteger)initialNumber {
   total = initialNumber;
   return self;
}

- (void)addNumber:(NSInteger)newNumber {
   total = total + newNumber;
}

- (NSInteger)getTotal {
   return total;
}

@end

int main(int argc, const char * argv[]) {
   NSAutoreleasePool * pool = [[NSAutoreleasePool alloc] init];        
   Adder *adder = [[Adder alloc]initWithInitialNumber:10];
   [adder addNumber:5];
   [adder addNumber:4];
   
   NSLog(@"The total is %ld",[adder getTotal]);
   [pool drain];
   return 0;
}

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

2013-09-22 21:17:30.485 DataEncapsulation[317:303] The total is 19

ऊपर वर्ग एक साथ संख्या जोड़ता है और राशि लौटाता है। जनता के सदस्यaddNum तथा getTotal बाहरी दुनिया के लिए इंटरफेस हैं और उपयोगकर्ता को कक्षा का उपयोग करने के लिए उन्हें जानना आवश्यक है। निजी सदस्यtotal कुछ ऐसा है जो बाहरी दुनिया से छिपा है, लेकिन कक्षा को ठीक से संचालित करने के लिए आवश्यक है।

डिजाइनिंग रणनीति

जब तक हमें वास्तव में उन्हें उजागर करने की आवश्यकता नहीं है, तब तक हम में से अधिकांश ने कड़वे अनुभव के माध्यम से कक्षा के सदस्यों को निजी बनाना सीखा है। बस यही अच्छा हैencapsulation

चूंकि ऑब्जेक्ट-सी सहित सभी ऑब्जेक्ट-ओरिएंटेड प्रोग्रामिंग (ओओपी) भाषाओं की मुख्य विशेषताओं में से एक यह है कि डेटा एनकैप्सुलेशन को समझना महत्वपूर्ण है।

कभी-कभी, आप पा सकते हैं कि आप एक मौजूदा वर्ग को व्यवहार में जोड़कर उसका विस्तार करना चाहते हैं जो केवल कुछ स्थितियों में उपयोगी होता है। मौजूदा कक्षाओं में इस तरह के विस्तार को जोड़ने के लिए, उद्देश्य-सी प्रदान करता हैcategories तथा extensions

यदि आपको किसी मौजूदा वर्ग में एक विधि जोड़ने की आवश्यकता है, तो शायद, अपने स्वयं के अनुप्रयोग में कुछ करना आसान बनाने के लिए कार्यक्षमता जोड़ना, श्रेणी का उपयोग करने का सबसे आसान तरीका है।

किसी श्रेणी को घोषित करने के लिए सिंटैक्स @interface कीवर्ड का उपयोग करता है, एक मानक उद्देश्य-सी श्रेणी विवरण की तरह, लेकिन उपवर्ग से किसी भी विरासत का संकेत नहीं करता है। इसके बजाय, यह कोष्ठकों में श्रेणी का नाम इस तरह से निर्दिष्ट करता है -

@interface ClassName (CategoryName)

@end

श्रेणी के लक्षण

  • किसी भी वर्ग के लिए एक श्रेणी घोषित की जा सकती है, भले ही आपके पास मूल कार्यान्वयन स्रोत कोड न हो।

  • किसी भी तरीके को आप एक श्रेणी में घोषित करते हैं, जो मूल वर्ग के सभी उदाहरणों के साथ-साथ मूल वर्ग के किसी भी उपवर्ग के लिए उपलब्ध होगा।

  • रनटाइम के दौरान, किसी वर्ग द्वारा जोड़ी गई विधि और मूल वर्ग द्वारा लागू किए गए के बीच कोई अंतर नहीं है।

अब, आइए एक नमूना श्रेणी कार्यान्वयन देखें। चलो कोको वर्ग NSString में एक श्रेणी जोड़ें। यह श्रेणी हमारे लिए एक नई विधि getCopyRightString जोड़ना संभव बनाती है जो कॉपीराइट स्ट्रिंग को वापस करने में हमारी मदद करती है। इसे नीचे दिखाया गया है।

#import <Foundation/Foundation.h>

@interface NSString(MyAdditions)
+(NSString *)getCopyRightString;
@end

@implementation NSString(MyAdditions)

+(NSString *)getCopyRightString {
   return @"Copyright TutorialsPoint.com 2013";
}

@end

int main(int argc, const char * argv[]) {
   NSAutoreleasePool * pool = [[NSAutoreleasePool alloc] init];
   NSString *copyrightString = [NSString getCopyRightString];
   NSLog(@"Accessing Category: %@",copyrightString);
   
   [pool drain];
   return 0;
}

अब जब हम प्रोग्राम को संकलित करते हैं और चलाते हैं, तो हमें निम्नलिखित परिणाम मिलेंगे।

2013-09-22 21:19:12.125 Categories[340:303] Accessing Category: Copyright TutorialsPoint.com 2013

भले ही किसी वर्ग द्वारा जोड़े गए किसी भी तरीके को कक्षा और उसके उपवर्गों के सभी उदाहरणों के लिए उपलब्ध हैं, आपको श्रेणी हेडर फ़ाइल को किसी भी स्रोत कोड फ़ाइल में आयात करना होगा जहां आप अतिरिक्त विधियों का उपयोग करना चाहते हैं, अन्यथा आप में भाग लेंगे संकलक चेतावनी और त्रुटियों।

हमारे उदाहरण में, चूंकि हमारे पास बस एक ही वर्ग है, इसलिए हमने किसी भी हेडर फाइल को शामिल नहीं किया है, ऐसे में हमें ऊपर बताई गई हेडर फाइलों को शामिल करना चाहिए।

ऑब्जेक्टिव-सी में पोज़िंग के बारे में शुरू करने से पहले, मैं आपके ध्यान में लाना चाहूंगा कि पोज़िंग को मैक ओएस एक्स 10.5 में पदावनत घोषित किया गया था और इसके बाद यह उपयोग के लिए उपलब्ध नहीं है। तो उन लोगों के लिए जो इन पदावनत विधियों के बारे में चिंतित नहीं हैं, इस अध्याय को छोड़ सकते हैं।

ऑब्जेक्टिव-सी एक क्लास को पूरी तरह से प्रोग्राम के भीतर दूसरी क्लास बदलने की अनुमति देता है। रिप्लेसमेंट क्लास को लक्ष्य वर्ग के रूप में "पोज़" कहा जाता है।

उन संस्करणों के लिए जो पॉज़िंग का समर्थन करते थे, लक्ष्य वर्ग को भेजे गए सभी संदेश इसके बजाय पॉज़िंग वर्ग द्वारा प्राप्त किए जाते हैं।

NSObject में poseAsClass सम्‍मिलित है: विधि जो हमें ऊपर बताए अनुसार मौजूदा वर्ग को बदलने में सक्षम बनाती है।

पोजिंग में प्रतिबंध

  • एक वर्ग केवल अपने प्रत्यक्ष या अप्रत्यक्ष सुपरक्लेसेस में से एक के रूप में पोज दे सकता है।

  • पॉज़िंग क्लास को किसी भी नए उदाहरण चर को परिभाषित नहीं करना चाहिए जो लक्ष्य वर्ग से अनुपस्थित हैं (हालांकि यह तरीकों को परिभाषित या ओवरराइड कर सकता है)।

  • हो सकता है कि लक्ष्य वर्ग को पोज़िंग से पहले कोई संदेश न मिला हो।

  • एक पॉज़िंग क्लास सुपर के माध्यम से ओवरराइड विधियों को कॉल कर सकता है, इस प्रकार लक्ष्य वर्ग के कार्यान्वयन को शामिल करता है।

  • एक पॉज़िंग क्लास श्रेणियों में परिभाषित तरीकों को ओवरराइड कर सकती है।

#import <Foundation/Foundation.h>

@interface MyString : NSString

@end

@implementation MyString

- (NSString *)stringByReplacingOccurrencesOfString:(NSString *)target
withString:(NSString *)replacement {
   NSLog(@"The Target string is %@",target);
   NSLog(@"The Replacement string is %@",replacement);
}

@end

int main() {
   NSAutoreleasePool * pool = [[NSAutoreleasePool alloc] init];
   [MyString poseAsClass:[NSString class]];
   NSString *string = @"Test";
   [string stringByReplacingOccurrencesOfString:@"a" withString:@"c"];
   
   [pool drain];
   return 0;
}

अब जब हम एक पुराने मैक ओएस एक्स (V_10.5 या पूर्व) में कार्यक्रम को संकलित और चलाते हैं, तो हमें निम्नलिखित परिणाम मिलेगा।

2013-09-22 21:23:46.829 Posing[372:303] The Target string is a
2013-09-22 21:23:46.830 Posing[372:303] The Replacement string is c

उपरोक्त उदाहरण में, हमने अपने कार्यान्वयन के साथ मूल विधि को प्रदूषित कर दिया है और यह उपरोक्त विधि के साथ सभी NSString संचालन में प्रभावित होगा।

एक वर्ग विस्तार एक श्रेणी के लिए कुछ समानता रखता है, लेकिन इसे केवल उस वर्ग में जोड़ा जा सकता है जिसके लिए आपके पास संकलन समय पर स्रोत कोड है (वर्ग विस्तार के रूप में उसी समय संकलित किया जाता है)।

एक क्लास एक्सटेंशन द्वारा घोषित तरीकों को मूल वर्ग के लिए कार्यान्वयन ब्लॉक में लागू किया जाता है, इसलिए आप उदाहरण के लिए, फ्रेमवर्क वर्ग पर एक क्लास एक्सटेंशन की घोषणा नहीं कर सकते हैं, जैसे कि कोको या कोको टच वर्ग जैसे एनएसएसटीरिंग।

एक्सटेंशन वास्तव में श्रेणी के नाम के बिना श्रेणियां हैं। इसे अक्सर कहा जाता हैanonymous categories

किसी एक्सटेंशन को घोषित करने के लिए सिंटैक्स @interface कीवर्ड का उपयोग करता है, एक मानक उद्देश्य-सी वर्ग विवरण की तरह, लेकिन उपवर्ग से किसी भी विरासत को इंगित नहीं करता है। इसके बजाय, यह केवल कोष्ठक जोड़ता है, जैसा कि नीचे दिखाया गया है -

@interface ClassName ()

@end

एक्सटेंशन के लक्षण

  • किसी भी वर्ग के लिए एक एक्सटेंशन घोषित नहीं किया जा सकता है, केवल उन वर्गों के लिए जिनके पास स्रोत कोड का मूल कार्यान्वयन है।

  • एक एक्सटेंशन निजी तरीकों और निजी चर को जोड़ रहा है जो केवल वर्ग के लिए विशिष्ट हैं।

  • एक्सटेंशन के अंदर घोषित कोई भी विधि या चर विरासत में मिली कक्षाओं तक भी पहुंच योग्य नहीं है।

उदाहरण उदाहरण

चलिए एक वर्ग नमूना बनाते हैं जिसमें एक एक्सटेंशन है। विस्तार में, चलो एक निजी चर internalID है।

तो, चलो एक विधि getExternalID है जो internalID को संसाधित करने के बाद externalID देता है।

उदाहरण नीचे दिखाया गया है और यह अभ्यस्त ऑनलाइन कंपाइलर पर काम करता है।

#import <Foundation/Foundation.h>

@interface SampleClass : NSObject {
   NSString *name;
}

- (void)setInternalID;
- (NSString *)getExternalID;

@end

@interface SampleClass() {
   NSString *internalID;
}

@end

@implementation SampleClass

- (void)setInternalID {
   internalID = [NSString stringWithFormat: 
   @"UNIQUEINTERNALKEY%dUNIQUEINTERNALKEY",arc4random()%100];
}

- (NSString *)getExternalID {
   return [internalID stringByReplacingOccurrencesOfString: 
   @"UNIQUEINTERNALKEY" withString:@""];
}

@end

int main(int argc, const char * argv[]) {
   NSAutoreleasePool * pool = [[NSAutoreleasePool alloc] init];
   SampleClass *sampleClass = [[SampleClass alloc]init];
   [sampleClass setInternalID];
   NSLog(@"ExternalID: %@",[sampleClass getExternalID]);        
   [pool drain];
   return 0;
}

अब जब हम प्रोग्राम को संकलित करते हैं और चलाते हैं, तो हमें निम्नलिखित परिणाम मिलेंगे।

2013-09-22 21:18:31.754 Extensions[331:303] ExternalID: 51

उपरोक्त उदाहरण में, हम देख सकते हैं कि इंटरनलिड सीधे नहीं लौटा है। हम यहां UNIQUEINTERNALKEY को हटाते हैं और केवल शेष राशि getExternalID विधि के लिए उपलब्ध कराते हैं।

उपरोक्त उदाहरण सिर्फ एक स्ट्रिंग ऑपरेशन का उपयोग करता है, लेकिन इसमें एन्क्रिप्शन / डिक्रिप्शन और जैसी कई विशेषताएं हो सकती हैं।

उद्देश्य-सी आपको प्रोटोकॉल को परिभाषित करने की अनुमति देता है, जो किसी विशेष स्थिति के लिए उपयोग किए जाने वाले तरीकों की घोषणा करते हैं। प्रोटोकॉल के अनुरूप कक्षाओं में प्रोटोकॉल लागू किए जाते हैं।

एक सरल उदाहरण एक नेटवर्क URL हैंडलिंग क्लास होगा, इसमें एक प्रक्रिया के साथ एक प्रोटोकॉल होगा जैसे कि प्रक्रियापूर्ण प्रतिनिधि विधि जो नेटवर्क URL लाने का ऑपरेशन समाप्त होने पर कॉलिंग क्लास को सूचित करती है।

प्रोटोकॉल का एक सिंटैक्स नीचे दिखाया गया है।

@protocol ProtocolName
@required
// list of required methods
@optional
// list of optional methods
@end

कीवर्ड के तहत तरीके @required उन वर्गों में लागू किया जाना चाहिए जो प्रोटोकॉल और विधियों के अनुरूप हैं @optional कीवर्ड लागू करने के लिए वैकल्पिक हैं।

यहाँ प्रोटोकॉल के अनुरूप वर्ग के लिए वाक्यविन्यास है

@interface MyClass : NSObject <MyProtocol>
...
@end

इसका मतलब यह है कि MyClass का कोई भी उदाहरण न केवल इंटरफ़ेस में विशेष रूप से घोषित तरीकों पर प्रतिक्रिया देगा, बल्कि यह है कि MyClass MyProtocol में आवश्यक विधियों के लिए कार्यान्वयन प्रदान करता है। क्लास इंटरफ़ेस में प्रोटोकॉल विधियों को फिर से बनाने की कोई आवश्यकता नहीं है - प्रोटोकॉल को अपनाना पर्याप्त है।

यदि आपको कई प्रोटोकॉल को अपनाने के लिए एक वर्ग की आवश्यकता है, तो आप उन्हें अल्पविराम से अलग की गई सूची के रूप में निर्दिष्ट कर सकते हैं। हमारे पास एक प्रतिनिधि ऑब्जेक्ट है जो प्रोटोकॉल को लागू करने वाली कॉलिंग ऑब्जेक्ट का संदर्भ रखता है।

एक उदाहरण नीचे दिया गया है।

#import <Foundation/Foundation.h>

@protocol PrintProtocolDelegate
- (void)processCompleted;

@end

@interface PrintClass :NSObject {
   id delegate;
}

- (void) printDetails;
- (void) setDelegate:(id)newDelegate;
@end

@implementation PrintClass
- (void)printDetails {
   NSLog(@"Printing Details");
   [delegate processCompleted];
}

- (void) setDelegate:(id)newDelegate {
   delegate = newDelegate;
}

@end

@interface SampleClass:NSObject<PrintProtocolDelegate>
- (void)startAction;

@end

@implementation SampleClass
- (void)startAction {
   PrintClass *printClass = [[PrintClass alloc]init];
   [printClass setDelegate:self];
   [printClass printDetails];
}

-(void)processCompleted {
   NSLog(@"Printing Process Completed");
}

@end

int main(int argc, const char * argv[]) {
   NSAutoreleasePool * pool = [[NSAutoreleasePool alloc] init];
   SampleClass *sampleClass = [[SampleClass alloc]init];
   [sampleClass startAction];
   [pool drain];
   return 0;
}

अब जब हम प्रोग्राम को संकलित करते हैं और चलाते हैं, तो हमें निम्नलिखित परिणाम मिलेंगे।

2013-09-22 21:15:50.362 Protocols[275:303] Printing Details
2013-09-22 21:15:50.364 Protocols[275:303] Printing Process Completed

उपरोक्त उदाहरण में हमने देखा है कि डेलगेट के तरीकों को कैसे बुलाया जाता है और निष्पादित किया जाता है। इसकी शुरुआत startAction से होती है, एक बार प्रक्रिया पूरी हो जाने के बाद, प्रतिनिधि विधि को पूरा कर लिया जाता है। ऑपरेशन को पूरा करने के लिए कहा जाता है।

किसी भी आईओएस या मैक ऐप में, हमारे पास कभी भी एक प्रतिनिधि के बिना एक कार्यक्रम लागू नहीं होगा। तो इसके महत्वपूर्ण हम प्रतिनिधियों के उपयोग को समझते हैं। डेलिगेट्स ऑब्जेक्ट्स को स्मृति लीक से बचने के लिए असुरक्षित_सक्रिय संपत्ति प्रकार का उपयोग करना चाहिए।

गतिशील बाध्यकारी, संकलन समय के बजाय रनटाइम पर आह्वान करने की विधि निर्धारित कर रहा है। डायनामिक बाइंडिंग को लेट बाइंडिंग के रूप में भी जाना जाता है।

ऑब्जेक्टिव-सी में, सभी विधियों को गतिशील रूप से रनटाइम पर हल किया जाता है। निष्पादित सटीक कोड विधि नाम (चयनकर्ता) और प्राप्त करने वाले ऑब्जेक्ट दोनों द्वारा निर्धारित किया जाता है।

गतिशील बंधन बहुरूपता को सक्षम बनाता है। उदाहरण के लिए, आयत और वर्ग सहित वस्तुओं के संग्रह पर विचार करें। प्रत्येक वस्तु का प्रिंटआरे पद्धति का अपना कार्यान्वयन है।

निम्नलिखित कोड के टुकड़े में, वास्तविक कोड जिसे अभिव्यक्ति द्वारा निष्पादित किया जाना चाहिए [anObject PrintArea] रनटाइम पर निर्धारित किया जाता है। रनटाइम सिस्टम उस विधि के लिए चयनकर्ता का उपयोग करता है जो किसी भी वर्ग के ऑबजेक्ट में उपयुक्त विधि की पहचान करने के लिए चलती है।

आइए हम एक सरल कोड को देखें जो गतिशील बंधन की व्याख्या करेगा।

#import <Foundation/Foundation.h>

@interface Square:NSObject {
   float area;
}

- (void)calculateAreaOfSide:(CGFloat)side;
- (void)printArea;
@end

@implementation Square
- (void)calculateAreaOfSide:(CGFloat)side {
   area = side * side;
}

- (void)printArea {
   NSLog(@"The area of square is %f",area);
}

@end

@interface Rectangle:NSObject {
   float area;
}

- (void)calculateAreaOfLength:(CGFloat)length andBreadth:(CGFloat)breadth;
- (void)printArea;
@end

@implementation  Rectangle

- (void)calculateAreaOfLength:(CGFloat)length andBreadth:(CGFloat)breadth {
   area = length * breadth;
}

- (void)printArea {
   NSLog(@"The area of Rectangle is %f",area);
}

@end

int main() {
   Square *square = [[Square alloc]init];
   [square calculateAreaOfSide:10.0];
   
   Rectangle *rectangle = [[Rectangle alloc]init];
   [rectangle calculateAreaOfLength:10.0 andBreadth:5.0];
   
   NSArray *shapes = [[NSArray alloc]initWithObjects: square, rectangle,nil];
   id object1 = [shapes objectAtIndex:0];
   [object1 printArea];
   
   id object2 = [shapes objectAtIndex:1];
   [object2 printArea];
   
   return 0;
}

अब जब हम प्रोग्राम को संकलित करते हैं और चलाते हैं, तो हमें निम्नलिखित परिणाम मिलेंगे।

2013-09-28 07:42:29.821 demo[4916] The area of square is 100.000000
2013-09-28 07:42:29.821 demo[4916] The area of Rectangle is 50.000000

जैसा कि आप ऊपर के उदाहरण में देख सकते हैं, प्रिंटआरे विधि को गतिशील रूप से रनटाइम में चुना गया है। यह गतिशील बंधन के लिए एक उदाहरण है और कई तरह की वस्तुओं के साथ काम करते समय कई स्थितियों में काफी उपयोगी है।

हम एक वर्ग क्लस्टर के भीतर उपवर्ग बना सकते हैं जो एक वर्ग को परिभाषित करता है जो इसके भीतर एक ऑब्जेक्ट एम्बेड करता है। ये वर्ग वस्तुएँ संयुक्त वस्तुएँ हैं।

तो आप सोच रहे होंगे कि क्लास क्लस्टर क्या है। इसलिए हम पहले देखेंगे कि क्लास क्लस्टर क्या है।

क्लास क्लस्टर्स

क्लास क्लस्टर एक डिज़ाइन पैटर्न है जिसका फाउंडेशन फ्रेमवर्क व्यापक उपयोग करता है। क्लास क्लस्टर एक सार्वजनिक अमूर्त सुपरक्लास के तहत कई निजी ठोस उपवर्गों का समूह बनाते हैं। इस तरह से कक्षाओं का समूहन किसी वस्तु-उन्मुख ढांचे की सार्वजनिक रूप से दिखाई देने वाली वास्तुकला को सरल बनाता है, बिना इसकी कार्यात्मक समृद्धि को कम किए। क्लास क्लस्टर्स एब्सट्रैक्ट फ़ैक्टरी डिज़ाइन पैटर्न पर आधारित होते हैं।

इसे सरल बनाने के लिए, समान कार्यों के लिए कई कक्षाएं बनाने के बजाय, हम एक एकल वर्ग बनाते हैं जो इनपुट के मूल्य के आधार पर इसकी हैंडलिंग का ध्यान रखेगा।

उदाहरण के लिए, NSNumber में हमारे पास char, int, bool इत्यादि जैसी कक्षाओं के कई क्लस्टर हैं। हम उन सभी को एक एकल वर्ग में जोड़ते हैं जो एक ही कक्षा में समान संचालन को संभालने का ध्यान रखते हैं। NSNumber वास्तव में वस्तुओं में इन आदिम प्रकारों के मूल्य को लपेटता है।

तो वास्तव में समग्र वस्तु क्या है?

निजी क्लस्टर ऑब्जेक्ट को अपने स्वयं के डिज़ाइन के ऑब्जेक्ट में एम्बेड करके, हम एक समग्र ऑब्जेक्ट बनाते हैं। यह मिश्रित ऑब्जेक्ट इसकी मूल कार्यक्षमता के लिए क्लस्टर ऑब्जेक्ट पर भरोसा कर सकता है, केवल इंटरसेप्टिंग संदेश जो समग्र ऑब्जेक्ट कुछ विशेष तरीके से संभालना चाहता है। यह आर्किटेक्चर हमारे द्वारा लिखे जाने वाले कोड की मात्रा को कम कर देता है और आपको फाउंडेशन फ्रेमवर्क द्वारा प्रदान किए गए परीक्षण कोड का लाभ उठाने देता है।

यह निम्नलिखित आकृति में समझाया गया है।

समग्र वस्तु को स्वयं को क्लस्टर के सार सुपरक्लास का उपवर्ग घोषित करना चाहिए। एक उपवर्ग के रूप में, यह सुपरक्लास के आदिम तरीकों को ओवरराइड करना चाहिए। यह व्युत्पन्न विधियों को भी ओवरराइड कर सकता है, लेकिन यह आवश्यक नहीं है क्योंकि व्युत्पन्न विधियां आदिम लोगों के माध्यम से काम करती हैं।

NSArray वर्ग की गणना विधि एक उदाहरण है; हस्तक्षेप करने वाली वस्तु के कार्यान्वयन की एक विधि जिसे वह ओवरराइड करता है वह इस प्रकार सरल हो सकती है -

- (unsigned)count  {
   return [embeddedObject count];
}

उपरोक्त उदाहरण में, एम्बेडेड ऑब्जेक्ट वास्तव में प्रकार NSArray है।

एक समग्र वस्तु उदाहरण

अब एक संपूर्ण उदाहरण देखने के लिए, आइए Apple के दस्तावेज़ीकरण से उदाहरण देखें जो नीचे दिया गया है।

#import <Foundation/Foundation.h>

@interface ValidatingArray : NSMutableArray {
   NSMutableArray *embeddedArray;
}

+ validatingArray;
- init;
- (unsigned)count;
- objectAtIndex:(unsigned)index;
- (void)addObject:object;
- (void)replaceObjectAtIndex:(unsigned)index withObject:object;
- (void)removeLastObject;
- (void)insertObject:object atIndex:(unsigned)index;
- (void)removeObjectAtIndex:(unsigned)index;

@end

@implementation ValidatingArray
- init {
   self = [super init];
   if (self) {
      embeddedArray = [[NSMutableArray allocWithZone:[self zone]] init];
   }
   return self;
}

+ validatingArray {
   return [[self alloc] init] ;
}

- (unsigned)count {
   return [embeddedArray count];
}

- objectAtIndex:(unsigned)index {
   return [embeddedArray objectAtIndex:index];
}

- (void)addObject:(id)object {
   if (object != nil) {
      [embeddedArray addObject:object];
   }
}

- (void)replaceObjectAtIndex:(unsigned)index withObject:(id)object; {
   if (index <[embeddedArray count] && object != nil) {
      [embeddedArray replaceObjectAtIndex:index withObject:object];
   }
}

- (void)removeLastObject; {
   if ([embeddedArray count] > 0) {
      [embeddedArray removeLastObject];
   }
}

- (void)insertObject:(id)object atIndex:(unsigned)index; {
   if (object != nil) {
      [embeddedArray insertObject:object atIndex:index];
   }
}

- (void)removeObjectAtIndex:(unsigned)index; {
   if (index <[embeddedArray count]) {
      [embeddedArray removeObjectAtIndex:index];
   }
}

@end

int main() {
   NSAutoreleasePool * pool = [[NSAutoreleasePool alloc] init];
   ValidatingArray *validatingArray = [ValidatingArray validatingArray];
   
   [validatingArray addObject:@"Object1"];
   [validatingArray addObject:@"Object2"];
   [validatingArray addObject:[NSNull null]];
   [validatingArray removeObjectAtIndex:2];
   NSString *aString = [validatingArray objectAtIndex:1];
   NSLog(@"The value at Index 1 is %@",aString);
   [pool drain];
   
   return 0;
}

अब जब हम प्रोग्राम को संकलित करते हैं और चलाते हैं, तो हमें निम्नलिखित परिणाम मिलेंगे।

2013-09-28 22:03:54.294 demo[6247] The value at Index 1 is Object2

उपरोक्त उदाहरण में, हम देख सकते हैं कि सरणी के एक फ़ंक्शन को मान्य करने से अशक्त वस्तुओं को जोड़ने की अनुमति नहीं मिलेगी जो सामान्य परिदृश्य में दुर्घटना का कारण बनेगी। लेकिन हमारा सत्यापन सरणी इसका ख्याल रखती है। इसी प्रकार, सरणी को मान्य करने की प्रत्येक विधि परिचालन के सामान्य अनुक्रम के अलावा मान्य प्रक्रियाओं को जोड़ती है।

यदि आप Apple दस्तावेज़ीकरण का उल्लेख करते हैं, तो आप नीचे दिए गए अनुसार फाउंडेशन फ्रेमवर्क का विवरण देख सकते हैं।

फाउंडेशन फ्रेमवर्क, उद्देश्य-सी कक्षाओं की एक आधार परत को परिभाषित करता है। उपयोगी आदिम वस्तु वर्गों का एक सेट प्रदान करने के अलावा, यह कई प्रतिमानों को प्रस्तुत करता है जो कार्यक्षमता को उद्देश्य-सी भाषा द्वारा कवर नहीं करते हैं। फाउंडेशन फ्रेमवर्क इन लक्ष्यों को ध्यान में रखकर बनाया गया है -

  • बुनियादी उपयोगिता वर्गों का एक छोटा सा सेट प्रदान करें।

  • डीललोकेशन जैसी चीजों के लिए लगातार सम्मेलनों की शुरुआत करके सॉफ्टवेयर विकास को आसान बनाएं।

  • यूनिकोड स्ट्रिंग्स, ऑब्जेक्ट दृढ़ता और ऑब्जेक्ट वितरण का समर्थन करें।

  • पोर्टेबिलिटी बढ़ाने के लिए ओएस स्वतंत्रता का एक स्तर प्रदान करें।

फ्रेमवर्क को NeXTStep द्वारा विकसित किया गया था, जिसे Apple द्वारा अधिग्रहित किया गया था और ये फाउंडेशन क्लास मैक OS X और iOS का हिस्सा बन गए थे।

चूंकि यह NeXTStep द्वारा विकसित किया गया था, इसलिए इसमें "NS" का वर्ग उपसर्ग है।

हमने अपने सभी नमूना कार्यक्रमों में फाउंडेशन फ्रेमवर्क का उपयोग किया है। फाउंडेशन फ्रेमवर्क का उपयोग करना लगभग आवश्यक है।

आम तौर पर, हम कुछ का उपयोग करते हैं #import <Foundation/NSString.h> एक उद्देश्य-सी वर्ग को आयात करने के लिए, लेकिन कई वर्गों के आयात से बचने के लिए, यह सभी में आयात किया जाता है #import <Foundation/Foundation.h>

NSObject नींव किट कक्षाओं सहित सभी वस्तुओं का आधार वर्ग है। यह स्मृति प्रबंधन के लिए तरीके प्रदान करता है। यह रनटाइम सिस्टम और ऑब्जेक्ट-सी ऑब्जेक्ट के रूप में व्यवहार करने की क्षमता के लिए बुनियादी इंटरफ़ेस भी प्रदान करता है। इसका कोई आधार वर्ग नहीं है और सभी वर्गों के लिए मूल है।

फाउंडेशन कक्षाएं कार्यक्षमता पर आधारित हैं

अनु क्रमांक। लूप प्रकार और विवरण
1 आधार सामग्री भंडारण

NSArray, NSDictionary, और NSSet किसी भी वर्ग के उद्देश्य-सी वस्तुओं के लिए भंडारण प्रदान करते हैं।

2 पाठ और तार

NSCharacterSet वर्णों के विभिन्न समूहों का प्रतिनिधित्व करता है जो NSString और NSScanner वर्गों द्वारा उपयोग किए जाते हैं। NSString क्लासेस टेक्स्ट स्ट्रिंग्स का प्रतिनिधित्व करते हैं और स्ट्रिंग्स की खोज, संयोजन और तुलना करने के तरीके प्रदान करते हैं। NSScanner ऑब्जेक्ट का उपयोग NSString ऑब्जेक्ट से संख्याओं और शब्दों को स्कैन करने के लिए किया जाता है।

3 दिनांक और समय

NSDate, NSTimeZone, और NSCalendar क्लासेस समय और तिथियों को संग्रहीत करते हैं और कैलेंडर जानकारी का प्रतिनिधित्व करते हैं। वे तारीख और समय के अंतर की गणना के लिए तरीके प्रदान करते हैं। NSLocale के साथ मिलकर, वे कई स्वरूपों में दिनांक और समय प्रदर्शित करने के लिए और दुनिया में स्थान के आधार पर समय और दिनांक समायोजित करने के लिए तरीके प्रदान करते हैं।

4 उपवाद सम्भालना

अपवाद हैंडलिंग का उपयोग अनपेक्षित स्थितियों को संभालने के लिए किया जाता है और इसे NSException के साथ Objective-C में पेश किया जाता है।

5 फ़ाइल रखरखाव

फाइल की हैंडलिंग क्लास NSFileManager की मदद से की जाती है।

6 URL लोडिंग सिस्टम

कक्षाओं और प्रोटोकॉल का एक सेट जो सामान्य इंटरनेट प्रोटोकॉल तक पहुंच प्रदान करता है।

फास्ट एन्यूमरेशन एक ऑब्जेक्टिव-सी की विशेषता है जो संग्रह के माध्यम से गणना करने में मदद करता है। तो तेजी से गणना के बारे में जानने के लिए, हमें पहले संग्रह के बारे में जानना होगा, जिसे निम्नलिखित अनुभाग में समझाया जाएगा।

उद्देश्य-सी में संग्रह

संग्रह मौलिक निर्माण हैं। इसका उपयोग अन्य वस्तुओं को रखने और प्रबंधित करने के लिए किया जाता है। एक संग्रह का पूरा उद्देश्य यह है कि यह वस्तुओं को कुशलता से संग्रहीत और पुनर्प्राप्त करने का एक सामान्य तरीका प्रदान करता है।

कई अलग-अलग प्रकार के संग्रह हैं। जबकि वे सभी अन्य वस्तुओं को धारण करने में सक्षम होने के समान उद्देश्य को पूरा करते हैं, वे ज्यादातर उसी तरह भिन्न होते हैं जिस तरह से वस्तुओं को पुनः प्राप्त किया जाता है। उद्देश्य-सी में उपयोग किए जाने वाले सबसे आम संग्रह हैं -

  • NSSet
  • NSArray
  • NSDictionary
  • NSMutableSet
  • NSMutableArray
  • NSMutableDictionary

यदि आप इन संरचनाओं के बारे में अधिक जानना चाहते हैं, तो कृपया फाउंडेशन फ्रेमवर्क में डेटा स्टोरेज को देखें ।

तेजी से गणना सिंटैक्स

for (classType variable in collectionObject ) { 
  statements 
}

यहां तेज गणना के लिए एक उदाहरण है।

#import <Foundation/Foundation.h>

int main() {
   NSAutoreleasePool * pool = [[NSAutoreleasePool alloc] init];
   NSArray *array = [[NSArray alloc]
   initWithObjects:@"string1", @"string2",@"string3",nil];
   
   for(NSString *aString in array) {
      NSLog(@"Value: %@",aString);
   }
   
   [pool drain];
   return 0;
}

अब जब हम प्रोग्राम को संकलित करते हैं और चलाते हैं, तो हमें निम्नलिखित परिणाम मिलेंगे।

2013-09-28 06:26:22.835 demo[7426] Value: string1
2013-09-28 06:26:22.836 demo[7426] Value: string2
2013-09-28 06:26:22.836 demo[7426] Value: string3

जैसा कि आप आउटपुट में देख सकते हैं, सरणी में प्रत्येक ऑब्जेक्ट एक क्रम में मुद्रित होता है।

फास्ट एन्यूमरेशन बैकवर्ड

for (classType variable in [collectionObject reverseObjectEnumerator] ) { 
  statements 
}

यहाँ तेजी से गणना में रिवर्स ऑब्जेक्ट इम्प्यूमर के लिए एक उदाहरण है।

#import <Foundation/Foundation.h>

int main() {
   NSAutoreleasePool * pool = [[NSAutoreleasePool alloc] init];
   NSArray *array = [[NSArray alloc]
   initWithObjects:@"string1", @"string2",@"string3",nil];
   
   for(NSString *aString in [array reverseObjectEnumerator]) {
      NSLog(@"Value: %@",aString);
   }
   
   [pool drain];
   return 0;
}

अब जब हम प्रोग्राम को संकलित करते हैं और चलाते हैं, तो हमें निम्नलिखित परिणाम मिलेंगे।

2013-09-28 06:27:51.025 demo[12742] Value: string3
2013-09-28 06:27:51.025 demo[12742] Value: string2
2013-09-28 06:27:51.025 demo[12742] Value: string1

जैसा कि आप आउटपुट में देख सकते हैं, सरणी में प्रत्येक ऑब्जेक्ट मुद्रित होता है, लेकिन सामान्य फास्ट एन्यूमरेशन की तुलना में रिवर्स ऑर्डर में।

मेमोरी प्रबंधन किसी भी प्रोग्रामिंग भाषा में सबसे महत्वपूर्ण प्रक्रिया में से एक है। यह वह प्रक्रिया है जिसके द्वारा वस्तुओं की स्मृति को आवंटित किया जाता है जब उन्हें आवश्यकता होती है और जब उन्हें आवश्यकता नहीं होती है तो उन्हें हटा दिया जाता है।

ऑब्जेक्ट मेमोरी को प्रबंधित करना प्रदर्शन का विषय है; यदि कोई अनुप्रयोग अनावश्यक वस्तुओं को मुक्त नहीं करता है, तो उसकी मेमोरी पदचिह्न बढ़ती है और प्रदर्शन प्रभावित होता है।

उद्देश्य-सी मेमोरी प्रबंधन तकनीकों को मोटे तौर पर दो प्रकारों में वर्गीकृत किया जा सकता है।

  • "मैनुअल रिटेन-रिलीज़" या एमआरआर
  • "स्वचालित संदर्भ गणना" या एआरसी

"मैनुअल रिटेन-रिलीज़" या एमआरआर

एमआरआर में, हम स्पष्ट रूप से अपने आप पर वस्तुओं का ट्रैक रखकर स्मृति का प्रबंधन करते हैं। यह एक मॉडल का उपयोग करके लागू किया जाता है, जिसे संदर्भ गिनती के रूप में जाना जाता है, जो कि फाउंडेशन क्लास NSObject रनटाइम वातावरण के साथ संयोजन के रूप में प्रदान करता है।

एमआरआर और एआरसी के बीच एकमात्र अंतर यह है कि रिटेन और रिलीज को पूर्व में मैन्युअल रूप से हमारे द्वारा नियंत्रित किया जाता है, जबकि बाद में इसका स्वचालित रूप से ध्यान रखा जाता है।

निम्न आंकड़ा एक उदाहरण का प्रतिनिधित्व करता है कि मेमोरी प्रबंधन कैसे उद्देश्य-सी में काम करता है।

कक्षा ए ऑब्जेक्ट का मेमोरी जीवन चक्र उपरोक्त आकृति में दिखाया गया है। जैसा कि आप देख सकते हैं, ऑब्जेक्ट के नीचे रिटेन काउंट दिखाया गया है, जब ऑब्जेक्ट की रिटेन काउंट 0 हो जाता है, तो ऑब्जेक्ट को पूरी तरह से मुक्त कर दिया जाता है और अन्य ऑब्जेक्ट्स का उपयोग करने के लिए इसकी मेमोरी डीलिट की जाती है।

NSObject में उपलब्ध आबंटित / init पद्धति का उपयोग करके क्लास ए ऑब्जेक्ट को पहली बार बनाया गया है। अब, रिटेन काउंट 1 हो गया है।

अब, क्लास बी क्लास ए के ऑब्जेक्ट को बनाए रखता है और क्लास ए की ऑब्जेक्ट की रिटेन गिनती 2 हो जाती है।

फिर, क्लास सी ऑब्जेक्ट की एक कॉपी बनाता है। अब, यह उदाहरण चर के लिए समान मूल्यों के साथ कक्षा ए के एक और उदाहरण के रूप में बनाया गया है। यहां, रिटेन काउंट 1 है और ऑरिजनल ऑब्जेक्ट की रिटेन काउंट नहीं है। यह आंकड़ा में बिंदीदार रेखा द्वारा दर्शाया गया है।

क्लास सी द्वारा रिलीज़ विधि का उपयोग करके कॉपी की गई वस्तु को जारी किया जाता है और रिटेन काउंट 0 हो जाता है और इसलिए ऑब्जेक्ट नष्ट हो जाता है।

प्रारंभिक कक्षा ए ऑब्जेक्ट के मामले में, रिटेन की गिनती 2 है और इसे नष्ट करने के लिए इसे दो बार जारी करना होगा। यह कक्षा ए और क्लास बी के रिलीज स्टेटमेंट द्वारा किया जाता है जो क्रमशः 1 और 0 के लिए गणना की संख्या को घटाता है। अंत में, वस्तु नष्ट हो जाती है।

एमआरआर मूल नियम

  • हमारे द्वारा बनाई गई कोई भी वस्तु हमारे पास है: हम एक ऐसी विधि का उपयोग करते हैं, जिसका नाम "आवंटित", "नया", "प्रतिलिपि", या "उत्परिवर्तनीय" से शुरू होता है

  • हम एक ऑब्जेक्ट का स्वामित्व बनाए रख सकते हैं जिसका उपयोग किया जाता है: एक प्राप्त वस्तु को सामान्य रूप से उस विधि के भीतर वैध बने रहने की गारंटी दी जाती है, और वह विधि सुरक्षित रूप से वस्तु को उसके आक्रमणकारी को वापस कर सकती है। हम दो स्थितियों में रिटेन का उपयोग करते हैं -

    • एक एक्सेसर विधि या एक init विधि के कार्यान्वयन में, एक वस्तु का स्वामित्व लेने के लिए जिसे हम एक संपत्ति मूल्य के रूप में संग्रहीत करना चाहते हैं।

    • किसी वस्तु को किसी अन्य ऑपरेशन के साइड-इफेक्ट के रूप में अमान्य होने से रोकने के लिए।

  • जब हमें इसकी आवश्यकता नहीं रह जाती है, तो हमें अपने पास मौजूद किसी वस्तु के स्वामित्व को त्यागना होगा: किसी वस्तु के स्वामित्व को हम उसे जारी संदेश या ऑटोरेलिज संदेश भेजकर छोड़ देते हैं। कोको शब्दावली में, किसी वस्तु के स्वामित्व को त्याग दिया जाता है, इसलिए उसे आमतौर पर किसी वस्तु को "जारी" कहा जाता है।

  • आपको उस वस्तु के स्वामित्व को त्यागना नहीं चाहिए जो आपके पास नहीं है: यह स्पष्ट रूप से बताए गए पिछले नीति नियमों का केवल सहसंबंध है।

#import <Foundation/Foundation.h>

@interface SampleClass:NSObject
- (void)sampleMethod;
@end

@implementation SampleClass
- (void)sampleMethod {
   NSLog(@"Hello, World! \n");
}

- (void)dealloc  {
  NSLog(@"Object deallocated");
  [super dealloc];
}

@end

int main() {
   
   /* my first program in Objective-C */
   SampleClass *sampleClass = [[SampleClass alloc]init];
   [sampleClass sampleMethod];
   
   NSLog(@"Retain Count after initial allocation: %d", 
   [sampleClass retainCount]);
   [sampleClass retain];
   
   NSLog(@"Retain Count after retain: %d", [sampleClass retainCount]);
   [sampleClass release];
   NSLog(@"Retain Count after release: %d", [sampleClass retainCount]);
   [sampleClass release];
   NSLog(@"SampleClass dealloc will be called before this");
   
   // Should set the object to nil
   sampleClass = nil;
   return 0;
}

जब हम उपरोक्त कार्यक्रम संकलित करते हैं, तो हमें निम्न आउटपुट मिलेगा।

2013-09-28 04:39:52.310 demo[8385] Hello, World!
2013-09-28 04:39:52.311 demo[8385] Retain Count after initial allocation: 1
2013-09-28 04:39:52.311 demo[8385] Retain Count after retain: 2
2013-09-28 04:39:52.311 demo[8385] Retain Count after release: 1
2013-09-28 04:39:52.311 demo[8385] Object deallocated
2013-09-28 04:39:52.311 demo[8385] SampleClass dealloc will be called before this

"स्वचालित संदर्भ गणना" या एआरसी

ऑटोमैटिक रेफरेंस काउंटिंग या एआरसी में, सिस्टम MRR के समान रेफरेंस काउंटिंग सिस्टम का उपयोग करता है, लेकिन यह संकलन-समय पर हमारे लिए उपयुक्त मेमोरी मैनेजमेंट विधि कॉल सम्मिलित करता है। हमें नई परियोजनाओं के लिए एआरसी का उपयोग करने के लिए दृढ़ता से प्रोत्साहित किया जाता है। यदि हम एआरसी का उपयोग करते हैं, तो आमतौर पर इस दस्तावेज़ में वर्णित अंतर्निहित कार्यान्वयन को समझने की कोई आवश्यकता नहीं है, हालांकि यह कुछ स्थितियों में सहायक हो सकता है। ARC के बारे में अधिक जानकारी के लिए, ARC रिलीज़ नोट्स में संक्रमण देखें ।

जैसा कि ऊपर उल्लेख किया गया है, एआरसी में, हमें रिलीज़ और रिटेन विधियों को जोड़ने की आवश्यकता नहीं है क्योंकि कंपाइलर द्वारा ध्यान रखा जाएगा। दरअसल, ऑब्जेक्टिव-सी की अंतर्निहित प्रक्रिया अब भी वही है। यह इन ऑपरेशन के बारे में चिंता किए बिना डेवलपर के लिए आंतरिक रूप से कोड को बनाए रखना और जारी करना आसान बनाता है, जिससे कोड लिखे जाने की मात्रा और मेमोरी लीक होने की संभावना कम हो जाएगी।

कचरा संग्रह नामक एक और सिद्धांत था, जिसका उपयोग एमआरआर के साथ-साथ मैक ओएस-एक्स में किया जाता है, लेकिन ओएस-एक्स माउंटेन लायन में इसके अपवर्तन के बाद से एमआरआर के साथ इसकी चर्चा नहीं की गई है। इसके अलावा, आईओएस ऑब्जेक्ट्स में कभी भी कचरा संग्रह की सुविधा नहीं थी। और एआरसी के साथ, ओएस-एक्स में भी कचरा संग्रह का कोई उपयोग नहीं है।

यहाँ एक सरल एआरसी उदाहरण है। ध्यान दें कि यह ऑनलाइन संकलक पर काम नहीं करेगा क्योंकि यह एआरसी का समर्थन नहीं करता है।

#import <Foundation/Foundation.h>

@interface SampleClass:NSObject
- (void)sampleMethod;
@end

@implementation SampleClass
- (void)sampleMethod {
   NSLog(@"Hello, World! \n");
}

- (void)dealloc  {
  NSLog(@"Object deallocated");
}

@end

int main() {
   /* my first program in Objective-C */
   @autoreleasepool {
      SampleClass *sampleClass = [[SampleClass alloc]init];
      [sampleClass sampleMethod];
      sampleClass = nil;
   }
   return 0;
}

जब हम उपरोक्त कार्यक्रम संकलित करते हैं, तो हमें निम्न आउटपुट मिलेगा।

2013-09-28 04:45:47.310 demo[8385] Hello, World!
2013-09-28 04:45:47.311 demo[8385] Object deallocated