home *** CD-ROM | disk | FTP | other *** search
/ Peanuts NeXT Software Archives / Peanuts-1.iso / CDROM / FAQs / Objective-C / sample < prev   
Encoding:
Text File  |  1996-10-17  |  24.5 KB  |  972 lines

  1. Path: informatik.tu-muenchen.de!lrz-muenchen.de!uni-erlangen.de!uniol!fu-berlin.de!news.mathworks.com!howland.erols.net!surfnet.nl!news.tue.nl!usenet
  2. From: tiggr@es.ele.tue.nl (Tiggr)
  3. Newsgroups: comp.lang.objective-c,comp.answers,news.answers
  4. Subject: comp.lang.objective-c FAQ, part 3/3: A Sample Program
  5. Supersedes: <sample_842976002@es.ele.tue.nl>
  6. Followup-To: comp.lang.objective-c
  7. Date: 17 Oct 1996 16:00:07 GMT
  8. Organization: Eindhoven University of Technology
  9. Lines: 953
  10. Approved: news-answers-request@mit.edu
  11. Expires: 26 Nov 1996 16:00:01 GMT
  12. Message-ID: <sample_845568001@es.ele.tue.nl>
  13. Reply-To: tiggr@es.ele.tue.nl (Tiggr)
  14. NNTP-Posting-Host: krait.es.ele.tue.nl
  15. Summary: This third part of the comp.lang.objective-c FAQ postings
  16.         presents a simple sample program written in Objective-C.
  17. Xref: informatik.tu-muenchen.de comp.lang.objective-c:5927 comp.answers:21751 news.answers:84433
  18.  
  19. Archive-name: Objective-C/sample
  20. Version: $Id: sample.preamble,v 3.7 1996/03/20 11:38:17 tiggr Exp $
  21.  
  22.  
  23.         A simple sample
  24.         Objective-C program
  25.  
  26.  
  27. This is the third part in a series of three informational postings
  28. concerning comp.lang.objective-c.  This article presents a simple program
  29. written in Objective-C.  The program consist of several files which are
  30. contained in a shar file (see instructions below on how to unpack).  [Note,
  31. from version 2.3 of this file, the sample has been changed in order to
  32. reduce the number of compiler warnings (down to 1, which is in there for
  33. explanatory purposes) and to reflect the use of `+alloc' and `-init' instead
  34. of `+new'.]
  35.  
  36. The early version of this FAQ was compiled by Bill Shirley, with the aid of
  37. many people.  The current version is being maintained by Tiggr, aided by a
  38. lot of people, including Paul Sanchez and Bill Shirley.
  39.  
  40. A Japanese language version of this FAQ is maintained by Norihiro Itoh
  41. <nito@argotechnos.co.jp>.  It is posted to fj.archives.answers
  42. regularly.  A hypertext version is maintained by Toru Sato
  43. <www-admin@cnds.canon.co.jp> and available at:
  44. http://www.cnds.canon.co.jp/Japanese_EUC/Contribution/FAQ_Objective-C/objc_faq_J.html
  45.  
  46. A World Wide Web hypertext version of this FAQ is maintained by Brian Harvey
  47. <theharv@csld.ucr.edu>.  It is http://csld.ucr.edu/NeXTSTEP/objc_faq.html.
  48. Another WWW version of this FAQ is maintained by Steve Dekorte
  49. <dekorte@suite.com> at http://www.batech.com/~dekorte/Objective-C/objc.html
  50.  
  51. Send your suggestions, additions, bug reports, comments and fixes to
  52. `tiggr@es.ele.tue.nl'.
  53.  
  54. #---------------------------------- cut here ----------------------------------
  55. # This is a shell archive.  Remove anything before this line,
  56. # then unpack it by saving it in a file and typing "sh file".
  57. #
  58. # Wrapped by Pieter Schoenmakers <tiggr@cobra> on Thu Oct 10 11:27:05 1996
  59. #
  60. # This archive contains:
  61. #    objc-sample    
  62. #
  63. # Existing files will not be overwritten.
  64. # Error checking via wc(1) will be performed.
  65.  
  66. LANG=""; export LANG
  67. PATH=/bin:/usr/bin:$PATH; export PATH
  68.  
  69. echo mkdir - objc-sample
  70. mkdir objc-sample
  71.  
  72. if test -f objc-sample/Char.h
  73. then
  74.     echo Ok to overwrite existing file objc-sample/Char.h\?
  75.     read answer
  76.     case "$answer" in
  77.     [yY]*)    echo Proceeding;;
  78.     *)    echo Aborting; exit 1;;
  79.     esac
  80.     rm -f objc-sample/Char.h
  81.     if test -f objc-sample/Char.h
  82.     then
  83.         echo Error: could not remove objc-sample/Char.h, aborting
  84.         exit 1
  85.     fi
  86. fi
  87. echo x - objc-sample/Char.h
  88. cat >objc-sample/Char.h <<'__EOF__'
  89. #import <objc/Object.h>
  90.  
  91. @interface Char: Object
  92. {
  93.   int value;
  94. }
  95.  
  96. -init: (int) x;
  97. -report;
  98.  
  99. @end
  100. __EOF__
  101. set `wc -lwc <objc-sample/Char.h`
  102. if test $1$2$3 != 111498
  103. then
  104.     echo ERROR: wc results of objc-sample/Char.h are $* should be 11 14 98
  105. fi
  106.  
  107. chmod 644 objc-sample/Char.h
  108.  
  109. if test -f objc-sample/Char.m
  110. then
  111.     echo Ok to overwrite existing file objc-sample/Char.m\?
  112.     read answer
  113.     case "$answer" in
  114.     [yY]*)    echo Proceeding;;
  115.     *)    echo Aborting; exit 1;;
  116.     esac
  117.     rm -f objc-sample/Char.m
  118.     if test -f objc-sample/Char.m
  119.     then
  120.         echo Error: could not remove objc-sample/Char.m, aborting
  121.         exit 1
  122.     fi
  123. fi
  124. echo x - objc-sample/Char.m
  125. cat >objc-sample/Char.m <<'__EOF__'
  126. #import <stdio.h>
  127. #import "Char.h"
  128.  
  129. @implementation Char
  130. {
  131.   int value;
  132. }
  133.  
  134. - init: (int) x
  135. {
  136.   [super init];        // In case the parent class is doing
  137.               // something special in its init...
  138.   value = x;
  139.   return self;
  140. }
  141.  
  142. - report
  143. {
  144.   printf("   %c", value);
  145.   return self;
  146. }
  147.  
  148. @end
  149. __EOF__
  150. set `wc -lwc <objc-sample/Char.m`
  151. if test $1$2$3 != 2347279
  152. then
  153.     echo ERROR: wc results of objc-sample/Char.m are $* should be 23 47 279
  154. fi
  155.  
  156. chmod 644 objc-sample/Char.m
  157.  
  158. if test -f objc-sample/Float.h
  159. then
  160.     echo Ok to overwrite existing file objc-sample/Float.h\?
  161.     read answer
  162.     case "$answer" in
  163.     [yY]*)    echo Proceeding;;
  164.     *)    echo Aborting; exit 1;;
  165.     esac
  166.     rm -f objc-sample/Float.h
  167.     if test -f objc-sample/Float.h
  168.     then
  169.         echo Error: could not remove objc-sample/Float.h, aborting
  170.         exit 1
  171.     fi
  172. fi
  173. echo x - objc-sample/Float.h
  174. cat >objc-sample/Float.h <<'__EOF__'
  175. #import <objc/Object.h>
  176.  
  177. @interface Float: Object
  178. {
  179.   float value;
  180. }
  181.  
  182. -initFloatValue: (float) x;
  183. -report;
  184.  
  185. @end
  186. __EOF__
  187. set `wc -lwc <objc-sample/Float.h`
  188. if test $1$2$3 != 1114113
  189. then
  190.     echo ERROR: wc results of objc-sample/Float.h are $* should be 11 14 113
  191. fi
  192.  
  193. chmod 644 objc-sample/Float.h
  194.  
  195. if test -f objc-sample/Float.m
  196. then
  197.     echo Ok to overwrite existing file objc-sample/Float.m\?
  198.     read answer
  199.     case "$answer" in
  200.     [yY]*)    echo Proceeding;;
  201.     *)    echo Aborting; exit 1;;
  202.     esac
  203.     rm -f objc-sample/Float.m
  204.     if test -f objc-sample/Float.m
  205.     then
  206.         echo Error: could not remove objc-sample/Float.m, aborting
  207.         exit 1
  208.     fi
  209. fi
  210. echo x - objc-sample/Float.m
  211. cat >objc-sample/Float.m <<'__EOF__'
  212. #import <stdio.h>
  213. #import "Float.h"
  214.  
  215. @implementation Float
  216. {
  217.   float value;
  218. }
  219.  
  220. -initFloatValue: (float) x
  221. {
  222.   [super init];
  223.   value = x;
  224.   return self;
  225. }
  226.  
  227. -report
  228. {
  229.   printf ("%4.1f", value);
  230.   return self;
  231. }
  232.  
  233. @end
  234. __EOF__
  235. set `wc -lwc <objc-sample/Float.m`
  236. if test $1$2$3 != 2231215
  237. then
  238.     echo ERROR: wc results of objc-sample/Float.m are $* should be 22 31 215
  239. fi
  240.  
  241. chmod 644 objc-sample/Float.m
  242.  
  243. if test -f objc-sample/Makefile
  244. then
  245.     echo Ok to overwrite existing file objc-sample/Makefile\?
  246.     read answer
  247.     case "$answer" in
  248.     [yY]*)    echo Proceeding;;
  249.     *)    echo Aborting; exit 1;;
  250.     esac
  251.     rm -f objc-sample/Makefile
  252.     if test -f objc-sample/Makefile
  253.     then
  254.         echo Error: could not remove objc-sample/Makefile, aborting
  255.         exit 1
  256.     fi
  257. fi
  258. echo x - objc-sample/Makefile
  259. cat >objc-sample/Makefile <<'__EOF__'
  260. # This Makefile assumes you have GNU gcc 2.3 or better and a suitable
  261. # runtime library with object support.  It also works on a NeXT.
  262. # Don't know about Stepstone.
  263.  
  264. ..SUFFIXES: .o .m
  265. ..m.o:
  266.     $(CC) -c $(CFLAGS) $<
  267.  
  268. # Use this on a NeXT
  269. #CC=        cc
  270. #LIBS=        
  271. # Use this with GNU CC on a non-NeXT,
  272. # and avoid the GCC moaning on using #import.
  273. CC=        gcc -Wno-import
  274. LIBS=        -lobjc
  275. LDFLAGS=    -L/usr/local/lib -L/usr/gnu/lib
  276.  
  277. CFLAGS=        -Wall -g
  278. OFILES=        main.o Node.o Queue.o Stack.o Float.o Char.o
  279.  
  280. demo: $(OFILES)
  281.     $(CC) $(CFLAGS) $(LDFLAGS) -o demo $(OFILES) $(LIBS)
  282.  
  283. clean:
  284.     rm -f $(OFILES) demo
  285.     
  286. Char.o : Char.m Char.h 
  287.  
  288. Float.o : Float.m Float.h 
  289.  
  290. Node.o : Node.m Node.h 
  291.  
  292. Queue.o : Queue.m Queue.h Node.h 
  293.  
  294. Stack.o : Stack.m Stack.h Node.h 
  295.  
  296. main.o : main.m Queue.h Node.h Stack.h Float.h
  297. __EOF__
  298. set `wc -lwc <objc-sample/Makefile`
  299. if test $1$2$3 != 37127783
  300. then
  301.     echo ERROR: wc results of objc-sample/Makefile are $* should be 37 127 783
  302. fi
  303.  
  304. chmod 644 objc-sample/Makefile
  305.  
  306. if test -f objc-sample/Node.h
  307. then
  308.     echo Ok to overwrite existing file objc-sample/Node.h\?
  309.     read answer
  310.     case "$answer" in
  311.     [yY]*)    echo Proceeding;;
  312.     *)    echo Aborting; exit 1;;
  313.     esac
  314.     rm -f objc-sample/Node.h
  315.     if test -f objc-sample/Node.h
  316.     then
  317.         echo Error: could not remove objc-sample/Node.h, aborting
  318.         exit 1
  319.     fi
  320. fi
  321. echo x - objc-sample/Node.h
  322. cat >objc-sample/Node.h <<'__EOF__'
  323. #import <objc/Object.h>
  324.  
  325. @interface Node : Object
  326. {
  327.   id next;
  328.   id data;
  329. }
  330.  
  331. -init: anItem;        // create a Node and store anItem in it
  332. -free;            // free a Node and return the item in it
  333. -next;            // report the id of the next node after this one
  334. -setNext: aNode;    // make the next node be aNode
  335.  
  336. @end
  337. __EOF__
  338. set `wc -lwc <objc-sample/Node.h`
  339. if test $1$2$3 != 1456295
  340. then
  341.     echo ERROR: wc results of objc-sample/Node.h are $* should be 14 56 295
  342. fi
  343.  
  344. chmod 644 objc-sample/Node.h
  345.  
  346. if test -f objc-sample/Node.m
  347. then
  348.     echo Ok to overwrite existing file objc-sample/Node.m\?
  349.     read answer
  350.     case "$answer" in
  351.     [yY]*)    echo Proceeding;;
  352.     *)    echo Aborting; exit 1;;
  353.     esac
  354.     rm -f objc-sample/Node.m
  355.     if test -f objc-sample/Node.m
  356.     then
  357.         echo Error: could not remove objc-sample/Node.m, aborting
  358.         exit 1
  359.     fi
  360. fi
  361. echo x - objc-sample/Node.m
  362. cat >objc-sample/Node.m <<'__EOF__'
  363. #import <objc/Object.h>
  364. #import "Node.h"
  365.  
  366. @implementation    Node: Object
  367.  
  368. -init: anItem
  369. {
  370.   self = [super init];
  371.   next = 0;
  372.   data = anItem;
  373.   return self;
  374. }
  375.  
  376. -free
  377. {
  378.   id tmp = data;
  379.   [super free];
  380.   return tmp;
  381. }
  382.  
  383. -next
  384. {
  385.   return next;
  386. }
  387.  
  388. -setNext: aNode
  389. {
  390.   next = aNode;
  391.   return self;
  392. }
  393.  
  394. @end
  395. __EOF__
  396. set `wc -lwc <objc-sample/Node.m`
  397. if test $1$2$3 != 3249299
  398. then
  399.     echo ERROR: wc results of objc-sample/Node.m are $* should be 32 49 299
  400. fi
  401.  
  402. chmod 644 objc-sample/Node.m
  403.  
  404. if test -f objc-sample/Queue.h
  405. then
  406.     echo Ok to overwrite existing file objc-sample/Queue.h\?
  407.     read answer
  408.     case "$answer" in
  409.     [yY]*)    echo Proceeding;;
  410.     *)    echo Aborting; exit 1;;
  411.     esac
  412.     rm -f objc-sample/Queue.h
  413.     if test -f objc-sample/Queue.h
  414.     then
  415.         echo Error: could not remove objc-sample/Queue.h, aborting
  416.         exit 1
  417.     fi
  418. fi
  419. echo x - objc-sample/Queue.h
  420. cat >objc-sample/Queue.h <<'__EOF__'
  421. #import <objc/Object.h>
  422. #import "Node.h"
  423.  
  424. @interface Queue: Object
  425. {
  426.   id head;
  427.   id tail;
  428.   unsigned qsize;
  429. }
  430.  
  431. -empty;            // clear out all contents of the Queue
  432. -put: anItem;        // put anItem on the Queue
  433. -get;            // return the item on top of the Queue
  434. -(unsigned int) size;    // tell us the current size of the Queue
  435.  
  436. @end
  437. __EOF__
  438. set `wc -lwc <objc-sample/Queue.h`
  439. if test $1$2$3 != 1655319
  440. then
  441.     echo ERROR: wc results of objc-sample/Queue.h are $* should be 16 55 319
  442. fi
  443.  
  444. chmod 644 objc-sample/Queue.h
  445.  
  446. if test -f objc-sample/Queue.m
  447. then
  448.     echo Ok to overwrite existing file objc-sample/Queue.m\?
  449.     read answer
  450.     case "$answer" in
  451.     [yY]*)    echo Proceeding;;
  452.     *)    echo Aborting; exit 1;;
  453.     esac
  454.     rm -f objc-sample/Queue.m
  455.     if test -f objc-sample/Queue.m
  456.     then
  457.         echo Error: could not remove objc-sample/Queue.m, aborting
  458.         exit 1
  459.     fi
  460. fi
  461. echo x - objc-sample/Queue.m
  462. cat >objc-sample/Queue.m <<'__EOF__'
  463. #import "Queue.h"
  464.  
  465. @implementation    Queue
  466.  
  467. -empty
  468. {
  469.   while([self size])
  470.     [[self get] free];
  471.   return self;
  472. }
  473.  
  474. -put: anItem
  475. {
  476.   if (tail)
  477.     tail = [[tail setNext : [[Node alloc] init: anItem]] next];
  478.   else
  479.     head = tail = [[Node alloc] init: anItem];
  480.   ++qsize;
  481.   return self;
  482. }
  483.  
  484. -get
  485. {
  486.   id contents;
  487.   id old_head = head;
  488.  
  489.   head = [head next];
  490.   contents = [old_head free];
  491.   if (--qsize == 0)
  492.     tail = head;
  493.   return contents;
  494. }
  495.  
  496. -(unsigned) size
  497. {
  498.   return qsize;
  499. }
  500.  
  501. @end
  502. __EOF__
  503. set `wc -lwc <objc-sample/Queue.m`
  504. if test $1$2$3 != 3975486
  505. then
  506.     echo ERROR: wc results of objc-sample/Queue.m are $* should be 39 75 486
  507. fi
  508.  
  509. chmod 644 objc-sample/Queue.m
  510.  
  511. if test -f objc-sample/README
  512. then
  513.     echo Ok to overwrite existing file objc-sample/README\?
  514.     read answer
  515.     case "$answer" in
  516.     [yY]*)    echo Proceeding;;
  517.     *)    echo Aborting; exit 1;;
  518.     esac
  519.     rm -f objc-sample/README
  520.     if test -f objc-sample/README
  521.     then
  522.         echo Error: could not remove objc-sample/README, aborting
  523.         exit 1
  524.     fi
  525. fi
  526. echo x - objc-sample/README
  527. cat >objc-sample/README <<'__EOF__'
  528. This directory contains the complete code for the "Simple Sample Objective-C
  529. program" described in the comp.lang.objective-c FAQ.  If you have a suitable
  530. compiler, use the supplied Makefile.  Otherwise, program output can be found
  531. in the file "output".
  532.  
  533. You should probably read "main.m" first.  It is very heavily annotated.
  534.  
  535. Also note and read the file COPYRIGHT.
  536. __EOF__
  537. set `wc -lwc <objc-sample/README`
  538. if test $1$2$3 != 855366
  539. then
  540.     echo ERROR: wc results of objc-sample/README are $* should be 8 55 366
  541. fi
  542.  
  543. chmod 644 objc-sample/README
  544.  
  545. if test -f objc-sample/Stack.h
  546. then
  547.     echo Ok to overwrite existing file objc-sample/Stack.h\?
  548.     read answer
  549.     case "$answer" in
  550.     [yY]*)    echo Proceeding;;
  551.     *)    echo Aborting; exit 1;;
  552.     esac
  553.     rm -f objc-sample/Stack.h
  554.     if test -f objc-sample/Stack.h
  555.     then
  556.         echo Error: could not remove objc-sample/Stack.h, aborting
  557.         exit 1
  558.     fi
  559. fi
  560. echo x - objc-sample/Stack.h
  561. cat >objc-sample/Stack.h <<'__EOF__'
  562. #import <objc/Object.h>
  563. #import "Node.h"
  564.  
  565. @interface Stack: Object
  566. {
  567.   id stack;
  568.   unsigned int stack_size;
  569. }
  570.  
  571. -empty;                // clear out all contents of the Stack
  572. -put: anItem;            // put anItem on the Stack
  573. -get;                // return the item on top of the Stack
  574. -(unsigned) size;        // tell us the current size of the Stack
  575.  
  576. @end
  577. __EOF__
  578. set `wc -lwc <objc-sample/Stack.h`
  579. if test $1$2$3 != 1553318
  580. then
  581.     echo ERROR: wc results of objc-sample/Stack.h are $* should be 15 53 318
  582. fi
  583.  
  584. chmod 644 objc-sample/Stack.h
  585.  
  586. if test -f objc-sample/Stack.m
  587. then
  588.     echo Ok to overwrite existing file objc-sample/Stack.m\?
  589.     read answer
  590.     case "$answer" in
  591.     [yY]*)    echo Proceeding;;
  592.     *)    echo Aborting; exit 1;;
  593.     esac
  594.     rm -f objc-sample/Stack.m
  595.     if test -f objc-sample/Stack.m
  596.     then
  597.         echo Error: could not remove objc-sample/Stack.m, aborting
  598.         exit 1
  599.     fi
  600. fi
  601. echo x - objc-sample/Stack.m
  602. cat >objc-sample/Stack.m <<'__EOF__'
  603. #import "Stack.h"
  604.  
  605. @implementation    Stack
  606.  
  607. -empty
  608. {
  609.   while([self size])
  610.     [[self get] free];
  611.   return self;
  612. }
  613.  
  614. -put: anItem
  615. {
  616.   stack = [[[Node alloc] init: anItem] setNext : stack];
  617.   ++stack_size;
  618.   return self;
  619. }
  620.  
  621. -get
  622. {
  623.   id contents;
  624.   id old_stack = stack;
  625.  
  626.   stack = [stack next];
  627.   contents = [old_stack free];
  628.   --stack_size;
  629.   return contents;
  630. }
  631.  
  632. -(unsigned) size
  633. {
  634.   return stack_size;
  635. }
  636.  
  637. @end
  638. __EOF__
  639. set `wc -lwc <objc-sample/Stack.m`
  640. if test $1$2$3 != 3557407
  641. then
  642.     echo ERROR: wc results of objc-sample/Stack.m are $* should be 35 57 407
  643. fi
  644.  
  645. chmod 644 objc-sample/Stack.m
  646.  
  647. if test -f objc-sample/main.m
  648. then
  649.     echo Ok to overwrite existing file objc-sample/main.m\?
  650.     read answer
  651.     case "$answer" in
  652.     [yY]*)    echo Proceeding;;
  653.     *)    echo Aborting; exit 1;;
  654.     esac
  655.     rm -f objc-sample/main.m
  656.     if test -f objc-sample/main.m
  657.     then
  658.         echo Error: could not remove objc-sample/main.m, aborting
  659.         exit 1
  660.     fi
  661. fi
  662. echo x - objc-sample/main.m
  663. cat >objc-sample/main.m <<'__EOF__'
  664. /* main.m - comp.lang.objective-c simple sample Objective-C program.  */
  665.  
  666. // This is a comment, just like the previous line.  Everything to the right
  667. // of a double slash is ignored.
  668.  
  669. /* Classes are the one real extension which Objective-C adds to C.  A class
  670.    is a description of a collection of data, like a C structure, and the
  671.    methods by which that data may be accessed or manipulated.  Instances of
  672.    a class are called objects, and methods are invoked by sending messages
  673.    to either the class itself, to produce objects, or to those objects.  The
  674.    recipient of a message is called a "receiver".  The form of a message is:
  675.  
  676.     [receiver method andMaybeSomeArguments]
  677.  
  678.    the receiver and method components are mandatory, as are the square
  679.    brackets surrounding the message.  Additional arguments may or may not be
  680.    present, depending upon the method definition.  Messages may appear
  681.    anywhere a statement is allowed in C.
  682.  
  683.    The first thing we do is bring in some include files, as in C.  On the
  684.    NeXT, it is customary to use the "import" statement which guarantees that
  685.    the file isn't included more than once.  Using GNU CC this is not all
  686.    that nice sinds it generates a huge warning for every file being
  687.    compiled.  So, since it does not really matter, we'll stick to
  688.    `#include'.  */
  689.  
  690. #import <stdio.h>
  691. #import <objc/Object.h>
  692. #import "Queue.h"
  693. #import "Stack.h"
  694.  
  695. /* That brought in class definitions for Objects, Queues, and Stacks.  The
  696.    Object class is the basis for all other classes, which is why it gets
  697.    brought in first.  It provides basic functional behavior which is
  698.    inherited by all derived classes.  All user created classes normally have
  699.    Object somewhere in their ancestry.
  700.  
  701.    Queue and Stack are classes of our own construction, and provide FIFO and
  702.    LIFO storage capabilities, respectively.  I'm not going to go into
  703.    implementation details here.  It's irrelevant how they work, all that is
  704.    important is that they both respond to 'put:' and 'get'.  If you want to
  705.    inspect them, look into the Queue.m, Stack.m, Queue.h and Stack.h files.
  706.  
  707.    A simple Class definition follows.  It inherits directly from the base
  708.    class "Object".  This gives it lots of nice properties, not the least of
  709.    which is the ability to be referenced by any pointer of the generic
  710.    object type "id".  All objects can be pointed to by any id variable, and
  711.    the default return type from methods is id.  This allows messages to be
  712.    embedded in other messages, either as receivers or arguments.
  713.  
  714.    An Int object allocates space for a single integer.  The "report" message
  715.    causes it to report its value.  Everything between the @implementation
  716.    and the @end is part of the class definition.
  717.  
  718.    Note - It is *highly* unusual to have a class implementation in your main
  719.    program.  Since the object is fully defined before it gets used, no
  720.    interface description is required.  There is nothing illegal about doing
  721.    things this way, but it is so unusual that the compiler will produce a
  722.    warning for this class.  The Int class implementation is here solely for
  723.    expository purposes.  */
  724.  
  725. @implementation Int: Object    // Int is derived from Object
  726. {
  727.     int value;        // This is the data portion.  Like a struct.
  728. }
  729.  
  730. /* The following are the method definitions.  A `+' prefix means it is a
  731.    factory method, i.e., how to manufacture instances of the class.  The
  732.    body of the method is between braces, like a C function.
  733.  
  734.    This class doesn't define any factory methods.  It relies on the +alloc
  735.    method defined in class Object.  For examples of factory methods, look at
  736.    the +new method defined in the Stack or Queue implementations.
  737.  
  738.    Self is a special variable, which refers to the object currently being
  739.    manipulated.  Super refers to the parent class of self.  The following
  740.    method asks the parent class (Object) to hand us a new instance, which
  741.    becomes self.  Then we update the instance variables and return a pointer
  742.    to the new object.
  743.  
  744.    It is standard for methods that do not need to return any special value
  745.    to instead return self.  This allows for a nested syntax of method calls.
  746.  
  747.    The "-" in front of init means that it's an instance method, i.e.,
  748.    something a particular object should respond to.  */
  749.  
  750. -init: (int) i
  751. {
  752.   /* Have our superclass initialize its part of us.  After that,
  753.      initialize the part of us introduced by this class (Int).  */
  754.   [super init];
  755.   value = i;
  756.   return self;
  757. }
  758.  
  759. -report
  760. {
  761.   printf ("%4d", value);
  762.   return self;
  763. }
  764.  
  765. @end
  766.  
  767. /* We have implemented Float and Char classes more traditionally, using
  768.    separate files for the interface (.h) and implementation (.m).  The Float
  769.    and Char objects are like the Int object, but with the obvious difference
  770.    that they work with floats and characters.  We include the interface
  771.    definitions at this point.  */
  772.  
  773. #import "Float.h"
  774. #import "Char.h"
  775.  
  776. /* If you inspect those files, note polymorphism -- methods have same
  777.    names as in the Int class.  */
  778.  
  779. int main (void)
  780. {
  781.     /* First create instances of "Stack" and "Queue" data structures.  */
  782.     id queue = [[Queue alloc] init];
  783.     id stack = [[Stack alloc] init];
  784.     int i, reply;
  785.     
  786.     fprintf (stderr, "Include the Char class in the demo? (y/n): ");
  787.  
  788.     /* Anything not matching `y.*' means no.  */
  789.     reply = getchar ();
  790.  
  791.     for (i = 5; i > -6; --i)
  792.       {
  793.     /* Depending on which version of the demo we're running, we
  794.        alternately put Ints and Floats onto the queue and stack, or
  795.        Ints, Floats, and Chars.  */
  796.     if (reply == 'y')
  797.       {
  798.         /* If I is odd we put an Int on the queue and a Char on the
  799.            stack.  If I is even we put an Char on the queue and a Float
  800.            on the stack.
  801.  
  802.            Since there is more than one method `-init:' and since
  803.            `+alloc' returns a plain, typeless, `id', the compiler
  804.            doesn't know the type of the object returned by alloc.  An
  805.            explicit cast (i.e. static type indication) ensures that the
  806.            compiler knows which `init:' is invoked---the one accepting a
  807.            char or the other one accepting an int.
  808.  
  809.            Another solution, which avoids the static type indication, is
  810.            to put typing information on the method in the method's name.
  811.            This is done for the Float class.  */
  812.         id my_char = [(Char *) [Char alloc] init: 'm' + i];
  813.  
  814.         if (i & 1)
  815.           {
  816.         [queue put: [(Int *) [Int alloc] init: i]];
  817.         [stack put: my_char];
  818.           }
  819.         else
  820.           {
  821.         [queue put: my_char];
  822.         [stack put: [[Float alloc] initFloatValue: i]];
  823.           }
  824.           }
  825.     else
  826.       {
  827.         /* If I is odd we put an Int on the queue and a Float on the
  828.            stack.  If I is even we put a Float on the queue and an Int
  829.            on the stack.  */
  830.             [queue put: ((i & 1)
  831.              ? [(Int *) [Int alloc] init: i]
  832.              : [[Float alloc] initFloatValue: i])];
  833.             [stack put: ((i & 1)
  834.              ? [[Float alloc] initFloatValue: i]
  835.              : [(Int*) [Int alloc] init: i])];
  836.     }
  837.     }
  838.  
  839.     while ([queue size] && [stack size])
  840.       {
  841.     /* The following illustrates run-time binding.  Will report be
  842.        invoked for a Float object or an Int object?  Did the user elect
  843.        for Char objects at run time?  We don't know ahead of time, but
  844.        with run-time binding and polymorphism it works properly.  The
  845.        burden is on the class implementer rather than the class user.
  846.  
  847.        Note that the following lines remain unchanged, whether we are
  848.        using the Char class or not.  The queue and stack hand us the
  849.        next object, it reports itself regardless of its type, and then
  850.        it frees itself.  */
  851.  
  852.     printf ("queue:");
  853.     [[[queue get] report] free];
  854.     printf (", stack:");
  855.     [[[stack get] report] free];
  856.     putchar('\n');
  857.       }
  858.   return 0;
  859. }
  860. __EOF__
  861. set `wc -lwc <objc-sample/main.m`
  862. if test $1$2$3 != 19612587741
  863. then
  864.     echo ERROR: wc results of objc-sample/main.m are $* should be 196 1258 7741
  865. fi
  866.  
  867. chmod 644 objc-sample/main.m
  868.  
  869. if test -f objc-sample/output
  870. then
  871.     echo Ok to overwrite existing file objc-sample/output\?
  872.     read answer
  873.     case "$answer" in
  874.     [yY]*)    echo Proceeding;;
  875.     *)    echo Aborting; exit 1;;
  876.     esac
  877.     rm -f objc-sample/output
  878.     if test -f objc-sample/output
  879.     then
  880.         echo Error: could not remove objc-sample/output, aborting
  881.         exit 1
  882.     fi
  883. fi
  884. echo x - objc-sample/output
  885. cat >objc-sample/output <<'__EOF__'
  886. Output from demo, excluding Char class:
  887.  
  888. Include the Char class in the demo? (y/n): n
  889. queue:   5, stack:-5.0
  890. queue: 4.0, stack:  -4
  891. queue:   3, stack:-3.0
  892. queue: 2.0, stack:  -2
  893. queue:   1, stack:-1.0
  894. queue: 0.0, stack:   0
  895. queue:  -1, stack: 1.0
  896. queue:-2.0, stack:   2
  897. queue:  -3, stack: 3.0
  898. queue:-4.0, stack:   4
  899. queue:  -5, stack: 5.0
  900.  
  901. Output from demo, including Char class:
  902.  
  903. Include the Char class in the demo? (y/n): y
  904. queue:   5, stack:   h
  905. queue:   q, stack:-4.0
  906. queue:   3, stack:   j
  907. queue:   o, stack:-2.0
  908. queue:   1, stack:   l
  909. queue:   m, stack: 0.0
  910. queue:  -1, stack:   n
  911. queue:   k, stack: 2.0
  912. queue:  -3, stack:   p
  913. queue:   i, stack: 4.0
  914. queue:  -5, stack:   r
  915. __EOF__
  916. set `wc -lwc <objc-sample/output`
  917. if test $1$2$3 != 29111679
  918. then
  919.     echo ERROR: wc results of objc-sample/output are $* should be 29 111 679
  920. fi
  921.  
  922. chmod 644 objc-sample/output
  923.  
  924. if test -f objc-sample/COPYRIGHT
  925. then
  926.     echo Ok to overwrite existing file objc-sample/COPYRIGHT\?
  927.     read answer
  928.     case "$answer" in
  929.     [yY]*)    echo Proceeding;;
  930.     *)    echo Aborting; exit 1;;
  931.     esac
  932.     rm -f objc-sample/COPYRIGHT
  933.     if test -f objc-sample/COPYRIGHT
  934.     then
  935.         echo Error: could not remove objc-sample/COPYRIGHT, aborting
  936.         exit 1
  937.     fi
  938. fi
  939. echo x - objc-sample/COPYRIGHT
  940. cat >objc-sample/COPYRIGHT <<'__EOF__'
  941. This copyright notice applies to all source files distributed in the
  942. comp.lang.objective-c FAQ: `A Simple Sample Objective-C program'.
  943.  
  944. Copyright (C) 1993 Paul J. Sanchez and Bill Shirley
  945. Copyright (C) 1994, 1996 Pieter J. Schoenmakers
  946.  
  947. The `simple sample Objective-C program' is free software; you can
  948. redistribute it and/or modify it under the terms of the GNU General Public
  949. License as published by the Free Software Foundation; either version 2, or
  950. (at your option) any later version.
  951.  
  952. The `simple sample Objective-C program' is distributed in the hope that it
  953. will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty
  954. of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General
  955. Public License for more details.
  956.  
  957. You should have received a copy of the GNU General Public License along
  958. with GNU Emacs; see the file COPYING.  If not, write to the Free Software
  959. Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
  960. __EOF__
  961. set `wc -lwc <objc-sample/COPYRIGHT`
  962. if test $1$2$3 != 19155973
  963. then
  964.     echo ERROR: wc results of objc-sample/COPYRIGHT are $* should be 19 155 973
  965. fi
  966.  
  967. chmod 644 objc-sample/COPYRIGHT
  968.  
  969. chmod 755 objc-sample
  970.  
  971. exit 0
  972.