forked from SquareBracketAssociates/UpdatedPharoByExample
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathBasicClasses.pier
846 lines (647 loc) · 30.5 KB
/
BasicClasses.pier
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
!Basic Classes
@cha:basicClasses
Most of the magic of Pharo is not in the language but in the class
libraries. To program effectively in it, you will need to learn how the
class libraries support the language and environment. The class libraries are
entirely written in Pharo, and can easily be extended. (Recall that a package may add
new functionality to a class even if it does not define this class.)
Our goal here is not to present in tedious detail the whole of the Pharo class
library, but rather to point out the key classes and methods that you will need
to use (or subclass/override) to program effectively. In this chapter, we will cover the basic
classes that you will need for nearly every application: ==Object==, ==Number==
and its subclasses, ==Character==, ==String==, ==Symbol== and ==Boolean==.
!!Object
For all intents and purposes, ==Object== is the root of the inheritance
hierarchy. Actually, in Pharo the true root of the hierarchy is ==ProtoObject==,
which is used to define minimal entities that masquerade as objects, but we can
ignore this point for the time being.
==Object== can be found in the ==Kernel== package. There are almost 400 methods
to be found here (in other words, every class that you define will automatically
provide all those methods). ''Note:'' You can count the number of methods in
a class, yourself, like so:
[[[
Object selectors size "Count the instance methods in Object"
Object class selectors size "Count the class-side methods"
]]]
The class comment for the ==Object== states:
''==Object== is the root class for almost all of the other classes in the class hierarchy.
The exceptions are ==ProtoObject== (the superclass of ==Object==) and its subclasses.''
''Class ==Object== provides default behaviour common to all normal objects, such
as access, copying, comparison, error handling, message sending, and reflection.
Also utility messages that all objects should respond to are defined here.
==Object== has no instance variables, nor should any be added. This is due to
several classes of objects that inherit from ==Object== that have special
implementations (==SmallInteger== and ==UndefinedObject== for example) that the VM
knows about and depends on the structure and layout of certain standard classes.''
If we begin to browse the method protocols on the instance side of ==Object==
we will start to see some of the key behaviour it provides.
!!!Printing
Every object in Smalltalk can return a printed form of itself. You can select
any expression in a workspace and select the ==Print it== menu: this executes
the expression and asks the returned object to print itself. In fact this sends
the message ==printString== to the returned object. The method ==printString==,
which is a template method, at its core sends the message ==printOn:== to its
receiver. The message ==printOn:== is a hook that can be specialized.
==Object>>printOn:== is very likely one of the methods that you will most
frequently override. This method takes as its argument a ==Stream== on which a
==String== representation of the object will be written. The default
implementation simply writes the class name preceded by ''==a=='' or ''==an==''.
==Object>>printString== returns the ==String== that is written.
For example, the class ==Browser== does not redefine the method ==printOn:== and
sending the message ==printString== to an instance executes the methods defined in
==Object==.
[[[
Browser new printString --> 'a Browser'
]]]
The class ==Color== shows an example of ==printOn:== specialization. It prints
the name of the class followed by the name of the class method used to generate
that color.
[[[caption=printOn: redefinition.
Color>>>printOn: aStream
| name |
(name := self name) ifNotNil:
[ ^ aStream
nextPutAll: 'Color ';
nextPutAll: name ].
self storeOn: aStream
]]]
[[[
Color red printString --> 'Color red'
]]]
Note that the message ==printOn:== is not the same as ==storeOn:==. The message
==storeOn:== writes to its argument stream an expression that can be used to
recreate the receiver. This expression is evaluated when the stream is read
using the message ==readFrom:==. ==printOn:== just returns a textual version of
the receiver. Of course, it may happen that this textual representation may
represent the receiver as a self-evaluating expression.
!!!!!A word about representation and self-evaluating representation.
In functional programming, expressions return values when executed. In
Smalltalk, messages (expressions) return objects (values). Some objects have the
nice properties that their value is themselves. For example, the value of the
object ==true== is itself i.e., the object ==true==. We call such objects
''self-evaluating objects''. You can see a ''printed'' version of an object
value when you print the object in a workspace. Here are some examples of such
self-evaluating expressions.
[[[
true --> true
3@4 --> (3@4)
$a --> $a
#(1 2 3) --> #(1 2 3)
Color red --> Color red
]]]
Note that some objects such as arrays are self-evaluating or not depending on
the objects they contain. For example, an array of booleans is self-evaluating,
whereas an array of persons is not. The following example shows that a dynamic
array is self-evaluating only if its elements are:
[[[
{10@10. 100@100} --> {(10@10). (100@100)}
{Browser new . 100@100} --> an Array(a Browser (100@100))
]]]
Remember that literal arrays can only contain literals. Hence the following
array does not contain two points but rather six literal elements.
[[[
#(10@10 100@100) --> #(10 #@ 10 100 #@ 100)
]]]
Lots of ==printOn:== method specializations implement self-evaluating behavior.
The implementations of ==Point>>printOn:== and ==Interval>>printOn:== are
self-evaluating.
[[[caption=Self-evaluation of ==Point==
Point>>>printOn: aStream
"The receiver prints on aStream in terms of infix notation."
aStream nextPut: $(.
x printOn: aStream.
aStream nextPut: $@.
(y notNil and: [y negative])
ifTrue: [
"Avoid ambiguous @- construct"
aStream space].
y printOn: aStream.
aStream nextPut: $).
]]]
[[[caption=Self-evaluation of ==Interval==
Interval>>>printOn: aStream
aStream nextPut: $(;
print: start;
nextPutAll: ' to: ';
print: stop.
step ~= 1 ifTrue: [aStream nextPutAll: ' by: '; print: step].
aStream nextPut: $)
]]]
[[[
1 to: 10 --> (1 to: 10) "intervals are self-evaluating"
]]]
!!!Identity and equality
In Smalltalk, the message ==\=== tests object ''equality'' (i.e., whether
two objects represent the same value) whereas ==\=\=== tests object ''identity''
(whether two expressions represent the same object).
The default implementation of object equality is to test for object identity:
[[[caption=Object equality
Object>>>= anObject
"Answer whether the receiver and the argument represent the same object.
If = is redefined in any subclass, consider also redefining the message hash."
^ self == anObject
]]]
This is a method that you will frequently want to override. Consider the case of
==Complex== numbers:
[[[
(1 + 2 i) = (1 + 2 i) --> true "same value"
(1 + 2 i) == (1 + 2 i) --> false "but different objects"
]]]
This works because ==Complex== overrides ===== as follows:
[[[caption=Equality for complex numbers
Complex>>>= anObject
anObject isComplex
ifTrue: [^ (real = anObject real) & (imaginary = anObject imaginary)]
ifFalse: [^ anObject adaptToComplex: self andSend: #=]
]]]
The default implementation of ==Object>>~=== (a test for inequality) simply negates ==Object>>===, and
should not normally need to be changed.
[[[
(1 + 2 i) ~= (1 + 4 i) --> true
]]]
If you override =====, you should consider overriding ==hash==. If instances of
your class are ever used as keys in a ==Dictionary==, then you should make sure
that instances that are considered to be equal have the same hash value:
[[[caption=Hash must be reimplemented for complex numbers
Complex>>>hash
"Hash is reimplemented because = is implemented."
^ real hash bitXor: imaginary hash.
]]]
Although you should override ==\=== and ==hash== together, you should ''never''
override ==\=\===. (The semantics of object identity is the same for all
classes.) ==\=\=== is a primitive method of ==ProtoObject==.
Note that Pharo has some strange equality behaviour compared to other Smalltalks. For
example a symbol and a string can be equal. (We consider this to be a bug, not a
feature.)
[[[
#'lulu' = 'lulu' --> true
'lulu' = #'lulu' --> true
]]]
!!!Class membership
Several methods allow you to query the class of an object.
!!!!!==class==
You can ask any object about its class using the message ==class==.
[[[
1 class --> SmallInteger
]]]
!!!!!==isMemberOf:==
Conversely, you can ask if an object is an instance of a specific class:
[[[
1 isMemberOf: SmallInteger --> true "must be precisely this class"
1 isMemberOf: Integer --> false
1 isMemberOf: Number --> false
1 isMemberOf: Object --> false
]]]
Since Pharo is written in itself, you can really navigate through its
structure using the right combination of superclass and class messages (see
Chapter *: Classes and Metaclasses>../Metaclasses/Metaclasses.pillar@cha:metaclasses*).
!!!!!==isKindOf:==
==Object>>isKindOf:== answers whether the receiver's class is either the same
as, or a subclass of the argument class.
[[[
1 isKindOf: SmallInteger --> true
1 isKindOf: Integer --> true
1 isKindOf: Number --> true
1 isKindOf: Object --> true
1 isKindOf: String --> false
1/3 isKindOf: Number --> true
1/3 isKindOf: Integer --> false
]]]
==1/3== which is a ==Fraction== is a kind of ==Number==, since the class
==Number== is a superclass of the class ==Fraction==, but ==1/3== is not a
==Integer==.
!!!!!==respondsTo:==
==Object>>respondsTo:== answers whether the receiver understands the message
selector given as an argument.
[[[
1 respondsTo: #, --> false
]]]
A note on the usage of ==respondsTo:==. Normally it is a bad idea to query an
object for its class, or to ask it which messages it understands. Instead of
making decisions based on the class of object, you should simply send a message
to the object and let it decide (on the basis of its class) how it should
behave. (This concept is sometimes referred to as ''duck typing'').
!!!Copying
Copying objects introduces some subtle issues. Since instance variables are
accessed by reference, a ''shallow copy'' of an object would share its
references to instance variables with the original object:
[[[
a1 := { { 'harry' } }.
a1 --> #(#('harry'))
a2 := a1 shallowCopy.
a2 --> #(#('harry'))
(a1 at: 1) at: 1 put: 'sally'.
a1 --> #(#('sally'))
a2 --> #(#('sally')) "the subarray is shared!"
]]]
==Object>>shallowCopy== is a primitive method that creates a shallow copy of an
object. Since ==a2== is only a shallow copy of ==a1==, the two arrays share a
reference to the nested ==Array== that they contain.
==Object>>shallowCopy== is the ''public interface'' to ==Object>>copy== and
should be overridden if instances are unique. This is the case, for example,
with the classes ==Boolean==, ==Character==, ==SmallInteger==, ==Symbol== and
==UndefinedObject==.
==Object>>copyTwoLevel== does the obvious thing when a simple shallow copy does
not suffice:
[[[
a1 := { { 'harry' } } .
a2 := a1 copyTwoLevel.
(a1 at: 1) at: 1 put: 'sally'.
a1 --> #(#('sally'))
a2 --> #(#('harry')) "fully independent state"
]]]
==Object>>deepCopy== makes an arbitrarily deep copy of an object.
[[[
a1 := { { { 'harry' } } } .
a2 := a1 deepCopy.
(a1 at: 1) at: 1 put: 'sally'.
a1 --> #(#('sally'))
a2 --> #(#(#('harry')))
]]]
The problem with ==deepCopy== is that it will not terminate when applied to a
mutually recursive structure:
[[[
a1 := { 'harry' }.
a2 := { a1 }.
a1 at: 1 put: a2.
a1 deepCopy --> !''... does not terminate!''!
]]]
Although it is possible to override ==deepCopy== to do the right thing,
==Object>>copy== offers a better solution:
[[[caption=Copying objects as a template method
Object>>>copy
"Answer another instance just like the receiver.
Subclasses typically override postCopy;
they typically do not override shallowCopy."
^ self shallowCopy postCopy
]]]
You should override ==postCopy== to copy any instance variables that should not
be shared. ==postCopy== should always do a ==super postCopy==.
!!!Debugging
The most important method here is ==halt==. In order to set a breakpoint in a
method, simply insert the message send ==self halt.== at some point in the body
of the method. When this message is sent, execution will be interrupted and a
debugger will open to this point in your program. (See
Chapter *: The Pharo Environment>../Environment/Environment.pillar@cha:env* for more details about the debugger.)
The next most important message is ==assert:==, which takes a block as its
argument. If the block evaluates to ==true==, execution continues. Otherwise an
==AssertionFailure== exception will be raised. If this exception is not
otherwise caught, the debugger will open to this point in the execution.
==assert:== is especially useful to support ''design by contract''. The most
typical usage is to check non-trivial pre-conditions to public methods of
objects. ==Stack>>pop== could easily have been implemented as follows:
[[[caption=Checking a pre-condition}
Stack>>>pop
"Return the first element and remove it from the stack."
self assert: [ self isEmpty not ].
^self linkedList removeFirst element
]]]
Do not confuse ==Object>>assert:== with ==TestCase>>assert:==, which occurs in
the SUnit testing framework (see Chapter *: SUnit>../SUnit/SUnit.pillar@cha:sunit*). While the former expects
a block as its argument (actually, it will take any argument that understands
==value==, including a ==Boolean==), the latter expects a ==Boolean==. Although
both are useful for debugging, they each serve a very different purpose.
!!!Error handling
This protocol contains several methods useful for signaling run-time errors.
Sending ==self deprecated:== signals that the current method should no longer be
used, if deprecation has been turned on in the Debugging section of the
Settings browser. The ==String== argument should offer an alternative.
[[[
1 doIfNotNil: [ :arg | arg printString, ' is not nil' ]
--> !''SmallInteger(Object)>>doIfNotNil: has been deprecated. use ifNotNilDo:''!
]]]
==doesNotUnderstand:== (commonly abbreviated in discussions as DNU or MNU)
is sent whenever message lookup fails. The default
implementation, i.e., ==Object>>doesNotUnderstand:== will trigger the
debugger at this point. It may be useful to override ==doesNotUnderstand:== to
provide some other behaviour.
==Object>>error== and ==Object>>error:== are generic methods that can be used to
raise exceptions. (Generally it is better to raise your own custom exceptions,
so you can distinguish errors arising from your code from those coming from
kernel classes.)
Abstract methods in Smalltalk are implemented by convention with the body ==self
subclassResponsibility==. Should an abstract class be instantiated by accident,
then calls to abstract methods will result in ==Object>>subclassResponsibility==
being evaluated.
[[[caption=Signaling that a method is abstract
Object>>>subclassResponsibility
"This message sets up a framework for the behavior of the class' subclasses.
Announce that the subclass should have implemented this message."
self error: 'My subclass should have overridden ', thisContext sender selector printString
]]]
==Magnitude==, ==Number== and ==Boolean== are classical examples of abstract
classes that we shall see shortly in this chapter.
[[[
Number new + 1 --> !''Error: My subclass should have overridden #+''!
]]]
==self shouldNotImplement== is sent by convention to signal that an inherited
method is not appropriate for this subclass. This is generally a sign that
something is not quite right with the design of the class hierarchy. Due to the
limitations of single inheritance, however, sometimes it is very hard to avoid
such workarounds.
A typical example is ==Collection>>remove:== which is inherited by
==Dictionary== but flagged as not implemented. (A ==Dictionary== provides
==removeKey:== instead.)
!!!Testing
The ==testing== methods have nothing to do with SUnit testing! A testing method
is one that lets you ask a question about the state of the receiver and returns
a ==Boolean==.
Numerous testing methods are provided by ==Object==. We have already seen
==isComplex==. Others include ==isArray==, ==isBoolean==, ==isBlock==,
==isCollection== and so on. Generally such methods are to be avoided since
querying an object for its class is a form of violation of encapsulation.
Instead of testing an object for its class, one should simply send a request and
let the object decide how to handle it.
Nevertheless some of these testing methods are undeniably useful. The most
useful are probably ==ProtoObject>>isNil== and ==Object>>notNil== (though the
Null Object design pattern can obviate the need for even these methods).
!!!Initialize
A final key method that occurs not in ==Object== but in ==ProtoObject== is
==initialize==.
[[[caption= ==initialize== as an empty hook method
ProtoObject>>>initialize
"Subclasses should redefine this method to perform initializations on instance creation"
]]]
The reason this is important is that in Pharo, the default ==new== method
defined for every class in the system will send ==initialize== to newly created
instances.
[[[caption= ==new== as a class-side template method
Behavior>>>new
"Answer a new initialized instance of the receiver (which is a class) with no indexable
variables. Fail if the class is indexable."
^ self basicNew initialize
]]]
This means that simply by overriding the ==initialize== hook method, new
instances of your class will automatically be initialized. The ==initialize==
method should normally perform a ==super initialize== to establish the class
invariant for any inherited instance variables. (Note that this is ''not'' the
standard behaviour of other Smalltalks.)
!!Numbers
@sec:Numbers
Numbers in Pharo are not primitive data values but true objects.
Of course numbers are implemented efficiently in the virtual machine, but the
==Number== hierarchy is as perfectly accessible and extensible as any other
portion of the Smalltalk class hierarchy.
+The Number Hierarchy>file://figures/NumberHierarchy.png+
Numbers are found in the ==Kernel-Numbers== package. The abstract root of this
hierarchy is ==Magnitude==, which represents all kinds of classes supporting
comparison operators. ==Number== adds various arithmetic and other operators as
mostly abstract methods. ==Float== and ==Fraction== represent, respectively,
floating point numbers and fractional values. ==Integer== is also abstract,
thus distinguishing between subclasses ==SmallInteger==,
==LargePositiveInteger== and ==LargeNegativeInteger==. For the most part users
do not need to be aware of the difference between the three ==Integer== classes,
as values are automatically converted as needed.
!!!Magnitude
==Magnitude== is the parent not only of the ==Number== classes, but also of
other classes supporting comparison operations, such as ==Character==,
==Duration== and ==Timespan==. (==Complex== numbers are not comparable, and so
do not inherit from ==Number==.)
Methods ==<== and ===== are abstract. The remaining operators are generically
defined. For example:
[[[caption=Abstract comparison methods
Magnitude>>> < aMagnitude
"Answer whether the receiver is less than the argument."
^ self subclassResponsibility
Magnitude>>> > aMagnitude
"Answer whether the receiver is greater than the argument."
^ aMagnitude < self
]]]
!!!Number
Similarly, ==Number== defines ==+==, ==-==, ==*== and ==/== to be abstract, but
all other arithmetic operators are generically defined.
All ==Number== objects support various ''converting'' operators, such as
==asFloat== and ==asInteger==. There are also numerous ''shortcut constructor
methods'', such as ==i==, which converts a ==Number== to an instance of
==Complex== with a zero real component, and others which generate ==Duration==s,
such as ==hour==, ==day== and ==week==.
==Numbers== directly support common ''math functions'' such as ==sin==, ==log==,
==raiseTo:==, ==squared==, ==sqrt== and so on.
==Number>>printOn:== is implemented in terms of the abstract method
==Number>>printOn:base:==. (The default base is 10.)
Testing methods include ==even==, ==odd==, ==positive== and ==negative==.
Unsurprisingly ==Number== overrides ==isNumber==. More interestingly,
==isInfinite== is defined to return ==false==.
''Truncation'' methods include ==floor==, ==ceiling==, ==integerPart==,
==fractionPart== and so on.
[[[
1 + 2.5 --> 3.5 "Addition of two numbers"
3.4 * 5 --> 17.0 "Multiplication of two numbers"
8 / 2 --> 4 "Division of two numbers"
10 - 8.3 --> 1.7 "Subtraction of two numbers"
12 = 11 --> false "Equality between two numbers"
12 ~= 11 --> true "Test if two numbers are different"
12 > 9 --> true "Greater than"
12 >= 10 --> true "Greater or equal than"
12 < 10 --> false "Smaller than"
100@10 --> 100@10 "Point creation"
]]]
The following example works surprisingly well in Pharo:
[[[
1000 factorial / 999 factorial --> 1000
]]]
Note that ==1000 factorial== is really calculated, which in many other languages
can be quite difficult to compute. This is an excellent example of automatic
coercion and exact handling of a number.
@@todo Try to display the result of ==1000 factorial==. It takes more time to display it than to calculate it!
!!!Float
==Float== implements the abstract ==Number== methods for floating point numbers.
More interestingly, ==Float class== (i.e., the class-side of ==Float==)
provides methods to return the following ''constants'': ==e==, ==infinity==,
==nan== and ==pi==.
[[[
Float pi --> 3.141592653589793
Float infinity --> Infinity
Float infinity isInfinite --> true
]]]
!!!Fraction
==Fraction==s are represented by instance variables for the numerator and
denominator, which should be ==Integer==s. ==Fraction==s are normally created by
==Integer== division (rather than using the constructor method
==Fraction>>numerator:denominator:==):
[[[
6/8 --> (3/4)
(6/8) class --> Fraction
]]]
Multiplying a ==Fraction== by an ==Integer== or another ==Fraction== may yield
an ==Integer==:
[[[
6/8 * 4 --> 3
]]]
!!!Integer
==Integer== is the abstract parent of three concrete integer implementations. In
addition to providing concrete implementations of many abstract ==Number==
methods, it also adds a few methods specific to integers, such as ==factorial==,
==atRandom==, ==isPrime==, ==gcd:== and many others.
==SmallInteger== is special in that its instances are represented compactly —
instead of being stored as a reference, a ==SmallInteger== is represented
directly using the bits that would otherwise be used to hold a reference. The
first bit of an object reference indicates whether the object is a
==SmallInteger== or not.
The class methods ==minVal== and ==maxVal== tell us the range of a
==SmallInteger==:
[[[
SmallInteger maxVal = ((2 raisedTo: 30) - 1) --> true
SmallInteger minVal = (2 raisedTo: 30) negated --> true
]]]
When a ==SmallInteger== goes out of this range, it is automatically converted to
a ==LargePositiveInteger== or a ==LargeNegativeInteger==, as needed:
[[[
(SmallInteger maxVal + 1) class --> LargePositiveInteger
(SmallInteger minVal - 1) class --> LargeNegativeInteger
]]]
Large integers are similarly converted back to small integers when appropriate.
As in most programming languages, integers can be useful for specifying
iterative behaviour. There is a dedicated method ==timesRepeat:== for
evaluating a block repeatedly. We have already seen a similar example in
Chapter *: Syntax in a Nutshell>../SyntaxNutshell/SyntaxNutshell.pillar@cha:syntax*.
[[[
n := 2.
3 timesRepeat: [ n := n * n ].
n --> 256
]]]
!!Characters
==Character== is defined in the ==Kernel-BasicObjects== package as a subclass
of ==Magnitude==. Printable characters are represented in Pharo as ==$<char>==.
For example:
[[[
$a < $b --> true
]]]
Non-printing characters can be generated by various class methods. ==Character
class>>value:== takes the Unicode (or ASCII) integer value as argument and
returns the corresponding character. The protocol ==accessing untypeable
characters== contains a number of convenience constructor methods such as
==backspace==, ==cr==, ==escape==, ==euro==, ==space==, ==tab==, and so on.
[[[
Character space = (Character value: Character space asciiValue) --> true
]]]
The ==printOn:== method is clever enough to know which of the three ways to
generate characters offers the most appropriate representation:
[[[
Character value: 1 --> Character home
Character value: 2 --> Character value: 2
Character value: 32 --> Character space
Character value: 97 --> $a
]]]
Various convenient ''testing'' methods are built in: ==isAlphaNumeric==,
==isCharacter==, ==isDigit==, ==isLowercase==, ==isVowel==, and so on.
To convert a ==Character== to the string containing just that character, send
==asString==. In this case ==asString== and ==printString== yield different
results:
[[[
$a asString --> 'a'
$a --> $a
$a printString --> '$a'
]]]
Every ascii ==Character== is a unique instance, stored in the class variable
==CharacterTable==:
[[[
(Character value: 97) == $a --> true
]]]
==Characters== outside the range 0 to 255 are not unique, however:
[[[
Character characterTable size --> 256
(Character value: 500) == (Character value: 500) --> false
]]]
!!Strings
The ==String== class is defined in the package ==Collections-Strings==. A
==String== is an indexed ==Collection== that holds only ==Character==s.
+The String Hierarchy>file://figures/StringHierarchy.png|label=fig:strings+
In fact, ==String== is abstract and Pharo strings are actually instances of
the concrete class ==ByteString==.
[[[
'hello world' class --> ByteString
]]]
The other important subclass of ==String== is ==Symbol==. The key difference is
that there is only ever a single instance of ==Symbol== with a given value.
(This is sometimes called ''the unique instance property''). In contrast, two
separately constructed ==String==s that happen to contain the same sequence of
characters will often be different objects.
[[[
'hel','lo' == 'hello' --> false
]]]
[[[
('hel','lo') asSymbol == #hello --> true
]]]
Another important difference is that a ==String== is mutable, whereas a
==Symbol== is immutable.
[[[
'hello' at: 2 put: $u; yourself --> 'hullo'
]]]
[[[
#hello at: 2 put: $u --> error!
]]]
It is easy to forget that since strings are collections, they understand the
same messages that other collections do:
[[[
#hello indexOf: $o --> 5
]]]
Although ==String== does not inherit from ==Magnitude==, it does support the
usual ==comparing== methods, ==<==, ===== and so on. In addition,
==String>>match:== is useful for some basic glob-style pattern-matching:
[[[
'*or*' match: 'zorro' --> true
]]]
Regular expressions will be discussed in more detail in
Chapter *: Regular Expressions in Pharo>../Regex/Regex.pier@cha:regex*.
Strings support a rather large number of conversion methods. Many of these are
shortcut constructor methods for other classes, such as ==asDate==,
==asInteger== and so on. There are also a number of useful methods for
converting a string to another string, such as ==capitalized== and
==translateToLowercase==.
For more on strings and collections, see Chapter *: Collections>../Collections/Collections.pier@cha:collections*.
!!Booleans
The class ==Boolean== offers a fascinating insight into how much of the
Pharo language has been pushed into the class library. ==Boolean== is the
abstract superclass of the ==Singleton== classes ==True== and ==False==.
+The Boolean Hierarchy>file://figures/BooleanHierarchy.png|label=fig:booleans+
Most of the behaviour of ==Boolean==s can be understood by considering the
method ==ifTrue:ifFalse:==, which takes two ==Blocks== as arguments.
[[[
(4 factorial > 20) ifTrue: [ 'bigger' ] ifFalse: [ 'smaller' ] --> 'bigger'
]]]
The method is abstract in ==Boolean==. The implementations in its concrete
subclasses are both trivial:
[[[caption=Implementations of ==ifTrue:ifFalse:==
True>>>ifTrue: trueAlternativeBlock ifFalse: falseAlternativeBlock
^trueAlternativeBlock value
False>>>ifTrue: trueAlternativeBlock ifFalse: falseAlternativeBlock
^falseAlternativeBlock value
]]]
In fact, this is the essence of OOP: when a message is sent to an object, the
object itself determines which method will be used to respond. In this case an
instance of ==True== simply evaluates the ''true'' alternative, while an
instance of ==False== evaluates the ''false'' alternative. All the abstract
==Boolean== methods are implemented in this way for ==True== and ==False==. For
example:
[[[caption=Implementing negation
True>>>not
"Negation--answer false since the receiver is true."
^false
]]]
==Boolean==s offer several useful convenience methods, such as ==ifTrue:==,
==ifFalse:==, and ==ifFalse:ifTrue==. You also have the choice between eager and
lazy conjunctions and disjunctions.
[[[
(1>2) & (3<4) --> false "Eager, must evaluate both sides"
(1>2) and: [ 3<4 ] --> false "Lazy, only evaluate receiver"
(1>2) and: [ (1/0) > 0 ] --> false "argument block is never evaluated, so no exception"
]]]
In the first example, both ==Boolean== subexpressions are evaluated, since ==&==
takes a ==Boolean== argument. In the second and third examples, only the first
is evaluated, since ==and:== expects a ==Block== as its argument. The ==Block==
is evaluated only if the first argument is ==true==.
@@todo Try to imagine how ==and:== and ==or:== are implemented. Check the implementations in ==Boolean==, ==True== and ==False==.
!!Chapter summary
- If you override ===== then you should override ==hash== as well.
- Override ==postCopy== to correctly implement copying for your objects.
- Use ==self halt.== to set a breakpoint.
- Return ==self subclassResponsibility== to make a method abstract.
- To give an object a ==String== representation you should override ==printOn:==.
- Override the hook method ==initialize== to properly initialize instances.
- ==Number== methods automatically convert between ==Float==s, ==Fraction==s and ==Integer==s.
- ==Fraction==s truly represent rational numbers rather than floats.
- Ascii ==Character==s are unique instances.
- ==String==s are mutable; ==Symbol==s are not. Take care not to mutate string literals, however!
- ==Symbol==s are unique; ==String==s are not.
- ==String==s and ==Symbol==s are ==Collection==s and therefore support the usual ==Collection== methods.