y_nelson 2.1.0 → 2.3.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +4 -4
- data/{Ruby_for_YNelson_Users_in_20_minutes.lyx → Introduction_to_Ruby_for_YNelson_Users.lyx} +1042 -55
- data/Introduction_to_Ruby_for_YNelson_Users.pdf +0 -0
- data/Introduction_to_YNelson.lyx +23 -18
- data/Introduction_to_YNelson.pdf +0 -0
- data/LICENSE.txt +1 -2
- data/Object_model_of_YNelson_and_YPetri.lyx +3806 -0
- data/Object_model_of_YNelson_and_YPetri.pdf +0 -0
- data/README.md +1 -0
- data/Rakefile +1 -0
- data/lib/y_nelson/agent.rb +57 -117
- data/lib/y_nelson/dsl.rb +3 -2
- data/lib/y_nelson/place.rb +0 -1
- data/lib/y_nelson/version.rb +1 -1
- data/lib/y_nelson/zz_point.rb +1 -1
- data/lib/y_nelson.rb +16 -2
- data/y_nelson.gemspec +3 -3
- metadata +10 -15
- data/Ruby_for_YNelson_Users_in_20_minutes.pdf +0 -0
- data/YNelson_&_YPetri_User_Manual.lyx +0 -1037
- data/YNelson_&_YPetri_User_Manual.pdf +0 -0
- data/YNelson_-_Hands-on_Tutorial.lyx +0 -3701
- data/YNelson_-_Hands-on_Tutorial.pdf +0 -0
- data/config/mongoid.yml +0 -6
- data/lib/y_nelson/zz.png +0 -0
- data/test/zz.png +0 -0
data/{Ruby_for_YNelson_Users_in_20_minutes.lyx → Introduction_to_Ruby_for_YNelson_Users.lyx}
RENAMED
@@ -65,7 +65,7 @@
|
|
65
65
|
\begin_body
|
66
66
|
|
67
67
|
\begin_layout Title
|
68
|
-
Ruby for YNelson Users
|
68
|
+
Introduction to Ruby for YNelson Users
|
69
69
|
\end_layout
|
70
70
|
|
71
71
|
\begin_layout Standard
|
@@ -74,29 +74,67 @@ For
|
|
74
74
|
YNelson
|
75
75
|
\family default
|
76
76
|
users, basic Ruby syntax is necessary.
|
77
|
-
This document
|
77
|
+
This document is a primer to Ruby for
|
78
78
|
\family typewriter
|
79
79
|
YNelson
|
80
80
|
\family default
|
81
81
|
users.
|
82
|
-
|
83
|
-
\
|
84
|
-
|
85
|
-
|
82
|
+
This documents should be used in the same way as
|
83
|
+
\family typewriter
|
84
|
+
YNelson
|
85
|
+
\family default
|
86
|
+
tutorial (Introduction to
|
87
|
+
\family typewriter
|
88
|
+
YNelson
|
89
|
+
\family default
|
90
|
+
) – that is, get an
|
91
|
+
\emph on
|
92
|
+
irb
|
93
|
+
\emph default
|
94
|
+
session running, and type all the examples in by yourself.
|
95
|
+
You might also wish to install
|
96
|
+
\family typewriter
|
97
|
+
YNelson
|
98
|
+
\family default
|
99
|
+
by by typing "
|
100
|
+
\family typewriter
|
101
|
+
gem install y_nelson
|
102
|
+
\family default
|
103
|
+
" from your command line.
|
104
|
+
Since many of the code samples here rely heavily on each other, you may
|
105
|
+
consider to do this document in one session.
|
106
|
+
On the other hand, if you happen to be a Ruby hacker, you do not need to
|
107
|
+
continue reading this document at all.
|
108
|
+
This document is not a replacement for a Ruby textbook.
|
109
|
+
For those, who want more thorough introduction to the language, I recommend
|
110
|
+
the document
|
111
|
+
\begin_inset CommandInset href
|
112
|
+
LatexCommand href
|
113
|
+
target "http://www.rubyist.net/~slagell/ruby/index.html"
|
86
114
|
|
87
|
-
\
|
88
|
-
|
89
|
-
|
90
|
-
Those who want more thorough introduction to the language, I recommend
|
91
|
-
http://www.rubyist.net/~slagell/ruby/index.html, or any of the many Ruby textbooks.
|
115
|
+
\end_inset
|
116
|
+
|
117
|
+
, or one of the many Ruby textbooks.
|
92
118
|
\end_layout
|
93
119
|
|
94
|
-
\begin_layout
|
95
|
-
Variables and
|
120
|
+
\begin_layout Part*
|
121
|
+
Variables and constants
|
96
122
|
\end_layout
|
97
123
|
|
98
124
|
\begin_layout Standard
|
99
|
-
In Ruby, everything
|
125
|
+
In Ruby, everything
|
126
|
+
\begin_inset Foot
|
127
|
+
status collapsed
|
128
|
+
|
129
|
+
\begin_layout Plain Layout
|
130
|
+
Almost everything.
|
131
|
+
Non-object include eg.
|
132
|
+
variables or argument fields.
|
133
|
+
\end_layout
|
134
|
+
|
135
|
+
\end_inset
|
136
|
+
|
137
|
+
is an
|
100
138
|
\emph on
|
101
139
|
\color red
|
102
140
|
object
|
@@ -199,7 +237,85 @@ defined? Gamma
|
|
199
237
|
#=> "constant"
|
200
238
|
\end_layout
|
201
239
|
|
202
|
-
\begin_layout
|
240
|
+
\begin_layout Part*
|
241
|
+
Code lines and comments
|
242
|
+
\end_layout
|
243
|
+
|
244
|
+
\begin_layout Standard
|
245
|
+
Ruby lines can be written with or without semicolons:
|
246
|
+
\end_layout
|
247
|
+
|
248
|
+
\begin_layout LyX-Code
|
249
|
+
a = "with";
|
250
|
+
\end_layout
|
251
|
+
|
252
|
+
\begin_layout LyX-Code
|
253
|
+
b = "without"
|
254
|
+
\end_layout
|
255
|
+
|
256
|
+
\begin_layout LyX-Code
|
257
|
+
puts [ a, b ].join " or "
|
258
|
+
\end_layout
|
259
|
+
|
260
|
+
\begin_layout Standard
|
261
|
+
Semicolon is compulsory only when two or more logical lines are crammed
|
262
|
+
together like this:
|
263
|
+
\end_layout
|
264
|
+
|
265
|
+
\begin_layout LyX-Code
|
266
|
+
a =
|
267
|
+
\begin_inset Quotes eld
|
268
|
+
\end_inset
|
269
|
+
|
270
|
+
Hello
|
271
|
+
\begin_inset Quotes eld
|
272
|
+
\end_inset
|
273
|
+
|
274
|
+
; b =
|
275
|
+
\begin_inset Quotes eld
|
276
|
+
\end_inset
|
277
|
+
|
278
|
+
world!
|
279
|
+
\begin_inset Quotes erd
|
280
|
+
\end_inset
|
281
|
+
|
282
|
+
; puts a + b
|
283
|
+
\end_layout
|
284
|
+
|
285
|
+
\begin_layout Standard
|
286
|
+
Comments are denoted by
|
287
|
+
\family typewriter
|
288
|
+
#
|
289
|
+
\family default
|
290
|
+
sign.
|
291
|
+
Anything on the line following the
|
292
|
+
\family typewriter
|
293
|
+
#
|
294
|
+
\family default
|
295
|
+
sign is ignored:
|
296
|
+
\end_layout
|
297
|
+
|
298
|
+
\begin_layout LyX-Code
|
299
|
+
puts
|
300
|
+
\begin_inset Quotes eld
|
301
|
+
\end_inset
|
302
|
+
|
303
|
+
Hello world!
|
304
|
+
\begin_inset Quotes erd
|
305
|
+
\end_inset
|
306
|
+
|
307
|
+
# this line prints the words
|
308
|
+
\begin_inset Quotes eld
|
309
|
+
\end_inset
|
310
|
+
|
311
|
+
Hello world!
|
312
|
+
\begin_inset Quotes erd
|
313
|
+
\end_inset
|
314
|
+
|
315
|
+
|
316
|
+
\end_layout
|
317
|
+
|
318
|
+
\begin_layout Part*
|
203
319
|
Methods
|
204
320
|
\end_layout
|
205
321
|
|
@@ -312,7 +428,7 @@ to_f
|
|
312
428
|
which is not important.
|
313
429
|
\end_layout
|
314
430
|
|
315
|
-
\begin_layout
|
431
|
+
\begin_layout Part*
|
316
432
|
Classes
|
317
433
|
\end_layout
|
318
434
|
|
@@ -404,7 +520,7 @@ end
|
|
404
520
|
\end_layout
|
405
521
|
|
406
522
|
\begin_layout LyX-Code
|
407
|
-
|
523
|
+
Spot = Dog.
|
408
524
|
\color red
|
409
525
|
new
|
410
526
|
\end_layout
|
@@ -414,7 +530,7 @@ new
|
|
414
530
|
\end_layout
|
415
531
|
|
416
532
|
\begin_layout LyX-Code
|
417
|
-
|
533
|
+
Spot.speak!
|
418
534
|
\end_layout
|
419
535
|
|
420
536
|
\begin_layout LyX-Code
|
@@ -446,7 +562,7 @@ end
|
|
446
562
|
\end_layout
|
447
563
|
|
448
564
|
\begin_layout LyX-Code
|
449
|
-
|
565
|
+
Lisa = Cat.new
|
450
566
|
\end_layout
|
451
567
|
|
452
568
|
\begin_layout LyX-Code
|
@@ -454,7 +570,7 @@ Tama = Cat.new
|
|
454
570
|
\end_layout
|
455
571
|
|
456
572
|
\begin_layout LyX-Code
|
457
|
-
|
573
|
+
Lisa.speak!
|
458
574
|
\end_layout
|
459
575
|
|
460
576
|
\begin_layout LyX-Code
|
@@ -474,7 +590,7 @@ puts
|
|
474
590
|
' method, used to simply print characters on the screen.
|
475
591
|
\end_layout
|
476
592
|
|
477
|
-
\begin_layout
|
593
|
+
\begin_layout Part*
|
478
594
|
Strings, Symbols, Arrays and Hashes
|
479
595
|
\end_layout
|
480
596
|
|
@@ -534,7 +650,7 @@ Understanding argument passing and closure writing is essential in using
|
|
534
650
|
YPetri DSL.
|
535
651
|
\end_layout
|
536
652
|
|
537
|
-
\begin_layout
|
653
|
+
\begin_layout Section*
|
538
654
|
Strings
|
539
655
|
\end_layout
|
540
656
|
|
@@ -670,11 +786,11 @@ my_string
|
|
670
786
|
\color red
|
671
787
|
<<
|
672
788
|
\color inherit
|
673
|
-
"
|
789
|
+
"Spot!"
|
674
790
|
\end_layout
|
675
791
|
|
676
792
|
\begin_layout LyX-Code
|
677
|
-
#=> "Hello
|
793
|
+
#=> "Hello Spot!"
|
678
794
|
\end_layout
|
679
795
|
|
680
796
|
\begin_layout LyX-Code
|
@@ -693,7 +809,7 @@ Again,
|
|
693
809
|
operator changed the contents, but the object id remained the same.
|
694
810
|
\end_layout
|
695
811
|
|
696
|
-
\begin_layout
|
812
|
+
\begin_layout Section*
|
697
813
|
Symbols
|
698
814
|
\end_layout
|
699
815
|
|
@@ -709,7 +825,7 @@ Unlike strings, symbols are immutable – they never change.
|
|
709
825
|
\end_layout
|
710
826
|
|
711
827
|
\begin_layout LyX-Code
|
712
|
-
:
|
828
|
+
:Spot.class
|
713
829
|
\end_layout
|
714
830
|
|
715
831
|
\begin_layout LyX-Code
|
@@ -718,7 +834,7 @@ Unlike strings, symbols are immutable – they never change.
|
|
718
834
|
Symbol
|
719
835
|
\end_layout
|
720
836
|
|
721
|
-
\begin_layout
|
837
|
+
\begin_layout Section*
|
722
838
|
Arrays
|
723
839
|
\end_layout
|
724
840
|
|
@@ -733,7 +849,7 @@ As seen earlier, they can be defined with square brackets
|
|
733
849
|
\end_layout
|
734
850
|
|
735
851
|
\begin_layout LyX-Code
|
736
|
-
my_array = [
|
852
|
+
my_array = [ Spot, Lisa ]
|
737
853
|
\end_layout
|
738
854
|
|
739
855
|
\begin_layout LyX-Code
|
@@ -769,7 +885,7 @@ my_array[-2]
|
|
769
885
|
#=> #<Dog:0x9c214ac>
|
770
886
|
\end_layout
|
771
887
|
|
772
|
-
\begin_layout
|
888
|
+
\begin_layout Section*
|
773
889
|
Hashes
|
774
890
|
\end_layout
|
775
891
|
|
@@ -791,7 +907,7 @@ Ruby rocket
|
|
791
907
|
\end_layout
|
792
908
|
|
793
909
|
\begin_layout LyX-Code
|
794
|
-
h1 = {
|
910
|
+
h1 = { Spot => "dog", Lisa => "cat" }
|
795
911
|
\end_layout
|
796
912
|
|
797
913
|
\begin_layout LyX-Code
|
@@ -799,7 +915,7 @@ h1 = { Pochi => "dog", Tama => "cat" }
|
|
799
915
|
\end_layout
|
800
916
|
|
801
917
|
\begin_layout LyX-Code
|
802
|
-
h1[
|
918
|
+
h1[ Lisa ]
|
803
919
|
\end_layout
|
804
920
|
|
805
921
|
\begin_layout LyX-Code
|
@@ -807,7 +923,7 @@ h1[ Tama ]
|
|
807
923
|
\end_layout
|
808
924
|
|
809
925
|
\begin_layout LyX-Code
|
810
|
-
h1[
|
926
|
+
h1[ Spot ]
|
811
927
|
\end_layout
|
812
928
|
|
813
929
|
\begin_layout LyX-Code
|
@@ -820,7 +936,7 @@ The second way is possible only when the keys are symbols.
|
|
820
936
|
\end_layout
|
821
937
|
|
822
938
|
\begin_layout LyX-Code
|
823
|
-
h2 = { dog:
|
939
|
+
h2 = { dog: Spot, cat: Lisa }
|
824
940
|
\end_layout
|
825
941
|
|
826
942
|
\begin_layout LyX-Code
|
@@ -835,7 +951,7 @@ h2[:dog]
|
|
835
951
|
#=> #<Dog:0x9c214ac>
|
836
952
|
\end_layout
|
837
953
|
|
838
|
-
\begin_layout
|
954
|
+
\begin_layout Part*
|
839
955
|
Code blocks and Closures
|
840
956
|
\end_layout
|
841
957
|
|
@@ -946,7 +1062,7 @@ lambda calculus
|
|
946
1062
|
.
|
947
1063
|
\end_layout
|
948
1064
|
|
949
|
-
\begin_layout
|
1065
|
+
\begin_layout Section*
|
950
1066
|
Return values
|
951
1067
|
\end_layout
|
952
1068
|
|
@@ -975,7 +1091,7 @@ Code blocks (and actually, all Ruby statements) have return value.
|
|
975
1091
|
#=> [0, 1, 2, 3]
|
976
1092
|
\end_layout
|
977
1093
|
|
978
|
-
\begin_layout
|
1094
|
+
\begin_layout Section*
|
979
1095
|
Closures
|
980
1096
|
\end_layout
|
981
1097
|
|
@@ -1083,7 +1199,7 @@ my_proc.
|
|
1083
1199
|
\color red
|
1084
1200
|
call
|
1085
1201
|
\color inherit
|
1086
|
-
(
|
1202
|
+
( Spot )
|
1087
1203
|
\end_layout
|
1088
1204
|
|
1089
1205
|
\begin_layout LyX-Code
|
@@ -1091,7 +1207,7 @@ call
|
|
1091
1207
|
\end_layout
|
1092
1208
|
|
1093
1209
|
\begin_layout LyX-Code
|
1094
|
-
my_lambda.call(
|
1210
|
+
my_lambda.call( Lisa )
|
1095
1211
|
\end_layout
|
1096
1212
|
|
1097
1213
|
\begin_layout LyX-Code
|
@@ -1111,7 +1227,7 @@ my_proc
|
|
1111
1227
|
\color red
|
1112
1228
|
.
|
1113
1229
|
\color inherit
|
1114
|
-
(
|
1230
|
+
( Lisa )
|
1115
1231
|
\end_layout
|
1116
1232
|
|
1117
1233
|
\begin_layout LyX-Code
|
@@ -1119,7 +1235,7 @@ my_proc
|
|
1119
1235
|
\end_layout
|
1120
1236
|
|
1121
1237
|
\begin_layout LyX-Code
|
1122
|
-
my_lambda.(
|
1238
|
+
my_lambda.( Spot )
|
1123
1239
|
\end_layout
|
1124
1240
|
|
1125
1241
|
\begin_layout LyX-Code
|
@@ -1152,7 +1268,7 @@ lambda
|
|
1152
1268
|
\end_layout
|
1153
1269
|
|
1154
1270
|
\begin_layout LyX-Code
|
1155
|
-
my_proc.(
|
1271
|
+
my_proc.( Lisa, "garbage" )
|
1156
1272
|
\end_layout
|
1157
1273
|
|
1158
1274
|
\begin_layout LyX-Code
|
@@ -1160,14 +1276,42 @@ my_proc.( Tama, "garbage" )
|
|
1160
1276
|
\end_layout
|
1161
1277
|
|
1162
1278
|
\begin_layout LyX-Code
|
1163
|
-
my_lambda.(
|
1279
|
+
my_lambda.( Lisa, "garbage" )
|
1164
1280
|
\end_layout
|
1165
1281
|
|
1166
1282
|
\begin_layout LyX-Code
|
1167
1283
|
#=> ArgumentError: wrong number of arguments (2 for 1)
|
1168
1284
|
\end_layout
|
1169
1285
|
|
1170
|
-
\begin_layout
|
1286
|
+
\begin_layout Standard
|
1287
|
+
Finally, let us notice the alternative syntax for defining lambdas:
|
1288
|
+
\end_layout
|
1289
|
+
|
1290
|
+
\begin_layout LyX-Code
|
1291
|
+
my_lambda = lambda do |organism| organism.speak! end
|
1292
|
+
\end_layout
|
1293
|
+
|
1294
|
+
\begin_layout LyX-Code
|
1295
|
+
my_lambda = lambda { |organism| oranism.speak! }
|
1296
|
+
\end_layout
|
1297
|
+
|
1298
|
+
\begin_layout LyX-Code
|
1299
|
+
my_lambda =
|
1300
|
+
\color red
|
1301
|
+
->
|
1302
|
+
\color inherit
|
1303
|
+
organism do organism.speak! end
|
1304
|
+
\end_layout
|
1305
|
+
|
1306
|
+
\begin_layout LyX-Code
|
1307
|
+
my_lambda = -> orgnism { organism.speak! }
|
1308
|
+
\end_layout
|
1309
|
+
|
1310
|
+
\begin_layout Standard
|
1311
|
+
All of the four above statements define exactly the same thing.
|
1312
|
+
\end_layout
|
1313
|
+
|
1314
|
+
\begin_layout Section*
|
1171
1315
|
Passing arguments
|
1172
1316
|
\end_layout
|
1173
1317
|
|
@@ -1256,7 +1400,7 @@ YNelson
|
|
1256
1400
|
, this is the language of Ruby closures.
|
1257
1401
|
\end_layout
|
1258
1402
|
|
1259
|
-
\begin_layout
|
1403
|
+
\begin_layout Section*
|
1260
1404
|
Optional arguments
|
1261
1405
|
\end_layout
|
1262
1406
|
|
@@ -1274,7 +1418,7 @@ def average( a, b, c
|
|
1274
1418
|
\color red
|
1275
1419
|
=
|
1276
1420
|
\color inherit
|
1277
|
-
:
|
1421
|
+
:spot )
|
1278
1422
|
\end_layout
|
1279
1423
|
|
1280
1424
|
\begin_layout LyX-Code
|
@@ -1282,7 +1426,7 @@ def average( a, b, c
|
|
1282
1426
|
\color red
|
1283
1427
|
#
|
1284
1428
|
\color inherit
|
1285
|
-
If c argument was not given, :
|
1429
|
+
If c argument was not given, :spot symbol will be assigned
|
1286
1430
|
\end_layout
|
1287
1431
|
|
1288
1432
|
\begin_layout LyX-Code
|
@@ -1298,7 +1442,7 @@ if
|
|
1298
1442
|
\color red
|
1299
1443
|
==
|
1300
1444
|
\color inherit
|
1301
|
-
:
|
1445
|
+
:spot
|
1302
1446
|
\color red
|
1303
1447
|
then
|
1304
1448
|
\color inherit
|
@@ -1431,7 +1575,7 @@ c
|
|
1431
1575
|
All code lines, exception the obvious ones, should have comments.
|
1432
1576
|
\end_layout
|
1433
1577
|
|
1434
|
-
\begin_layout
|
1578
|
+
\begin_layout Section*
|
1435
1579
|
Variable-length argument lists
|
1436
1580
|
\end_layout
|
1437
1581
|
|
@@ -1535,7 +1679,7 @@ examine_arguments( 1 )
|
|
1535
1679
|
\end_layout
|
1536
1680
|
|
1537
1681
|
\begin_layout LyX-Code
|
1538
|
-
examine_arguments( :hello, :
|
1682
|
+
examine_arguments( :hello, :spot, 3, 5, "garbage" )
|
1539
1683
|
\end_layout
|
1540
1684
|
|
1541
1685
|
\begin_layout LyX-Code
|
@@ -1616,7 +1760,7 @@ avg.( 11, 7, 5, 3 )
|
|
1616
1760
|
#=> 6.5
|
1617
1761
|
\end_layout
|
1618
1762
|
|
1619
|
-
\begin_layout
|
1763
|
+
\begin_layout Section*
|
1620
1764
|
Named arguments
|
1621
1765
|
\end_layout
|
1622
1766
|
|
@@ -1717,7 +1861,7 @@ lambda do |arg| ...
|
|
1717
1861
|
) array constructor syntax.
|
1718
1862
|
\end_layout
|
1719
1863
|
|
1720
|
-
\begin_layout
|
1864
|
+
\begin_layout Section*
|
1721
1865
|
Hash-collecting behavior of square brackets
|
1722
1866
|
\end_layout
|
1723
1867
|
|
@@ -1822,7 +1966,7 @@ enzyme
|
|
1822
1966
|
constructor method.
|
1823
1967
|
\end_layout
|
1824
1968
|
|
1825
|
-
\begin_layout
|
1969
|
+
\begin_layout Section*
|
1826
1970
|
Arity
|
1827
1971
|
\end_layout
|
1828
1972
|
|
@@ -2002,7 +2146,7 @@ array_scale.arity
|
|
2002
2146
|
#=> -2
|
2003
2147
|
\end_layout
|
2004
2148
|
|
2005
|
-
\begin_layout
|
2149
|
+
\begin_layout Section*
|
2006
2150
|
Return value
|
2007
2151
|
\end_layout
|
2008
2152
|
|
@@ -2120,12 +2264,28 @@ experimental_lambda.call
|
|
2120
2264
|
#=> 3
|
2121
2265
|
\end_layout
|
2122
2266
|
|
2123
|
-
\begin_layout
|
2267
|
+
\begin_layout Section*
|
2124
2268
|
Return value arity
|
2125
2269
|
\end_layout
|
2126
2270
|
|
2127
2271
|
\begin_layout Standard
|
2128
|
-
It is possible to return more than one value
|
2272
|
+
It is possible to return more than one value
|
2273
|
+
\begin_inset Foot
|
2274
|
+
status open
|
2275
|
+
|
2276
|
+
\begin_layout Plain Layout
|
2277
|
+
Technically, methods and closures always return exactly 1 object – multiple
|
2278
|
+
values are returned via a single array object.
|
2279
|
+
But pragmatically, and especially with respect to
|
2280
|
+
\family typewriter
|
2281
|
+
YPetri
|
2282
|
+
\family default
|
2283
|
+
, the notion of return value arity is useful.
|
2284
|
+
\end_layout
|
2285
|
+
|
2286
|
+
\end_inset
|
2287
|
+
|
2288
|
+
.
|
2129
2289
|
For example:
|
2130
2290
|
\end_layout
|
2131
2291
|
|
@@ -2250,5 +2410,832 @@ z
|
|
2250
2410
|
#=> 3
|
2251
2411
|
\end_layout
|
2252
2412
|
|
2413
|
+
\begin_layout Part*
|
2414
|
+
|
2415
|
+
\family typewriter
|
2416
|
+
YSupport
|
2417
|
+
\family default
|
2418
|
+
library
|
2419
|
+
\end_layout
|
2420
|
+
|
2421
|
+
\begin_layout Standard
|
2422
|
+
Finally, having introduced the basic Ruby syntax, let us mention
|
2423
|
+
\family typewriter
|
2424
|
+
YSupport
|
2425
|
+
\family default
|
2426
|
+
gem (gem = published Ruby library), that collects the assets (modules,
|
2427
|
+
classes, methods...) of general concern in use by
|
2428
|
+
\family typewriter
|
2429
|
+
YPetri
|
2430
|
+
\family default
|
2431
|
+
/
|
2432
|
+
\family typewriter
|
2433
|
+
YNelson
|
2434
|
+
\family default
|
2435
|
+
.
|
2436
|
+
Of these, a particular mention goes to
|
2437
|
+
\family typewriter
|
2438
|
+
NameMagic
|
2439
|
+
\family default
|
2440
|
+
, widely used in
|
2441
|
+
\family typewriter
|
2442
|
+
YPetri
|
2443
|
+
\family default
|
2444
|
+
,
|
2445
|
+
\family typewriter
|
2446
|
+
YNelson
|
2447
|
+
\family default
|
2448
|
+
and
|
2449
|
+
\family typewriter
|
2450
|
+
SY
|
2451
|
+
\family default
|
2452
|
+
(physical units) libraries.
|
2453
|
+
\end_layout
|
2454
|
+
|
2455
|
+
\begin_layout Section*
|
2456
|
+
|
2457
|
+
\family typewriter
|
2458
|
+
NameMagic
|
2459
|
+
\end_layout
|
2460
|
+
|
2461
|
+
\begin_layout Standard
|
2462
|
+
In software engineering,
|
2463
|
+
\emph on
|
2464
|
+
magic
|
2465
|
+
\emph default
|
2466
|
+
is a technical term for irregular side effects of language expressions.
|
2467
|
+
The problem that
|
2468
|
+
\family typewriter
|
2469
|
+
NameMagic
|
2470
|
+
\family default
|
2471
|
+
solves is, that objects (such as chemical species encoded in
|
2472
|
+
\family typewriter
|
2473
|
+
YNelson
|
2474
|
+
\family default
|
2475
|
+
) are frequently named, and naming them is an annoying chore.
|
2476
|
+
Consider a simple case:
|
2477
|
+
\end_layout
|
2478
|
+
|
2479
|
+
\begin_layout LyX-Code
|
2480
|
+
class Student
|
2481
|
+
\end_layout
|
2482
|
+
|
2483
|
+
\begin_layout LyX-Code
|
2484
|
+
attr_accessor :name
|
2485
|
+
\end_layout
|
2486
|
+
|
2487
|
+
\begin_layout LyX-Code
|
2488
|
+
def initialize name: nil
|
2489
|
+
\end_layout
|
2490
|
+
|
2491
|
+
\begin_layout LyX-Code
|
2492
|
+
@name = name
|
2493
|
+
\end_layout
|
2494
|
+
|
2495
|
+
\begin_layout LyX-Code
|
2496
|
+
end
|
2497
|
+
\end_layout
|
2498
|
+
|
2499
|
+
\begin_layout LyX-Code
|
2500
|
+
end
|
2501
|
+
\end_layout
|
2502
|
+
|
2503
|
+
\begin_layout Standard
|
2504
|
+
Now, to create named
|
2505
|
+
\family typewriter
|
2506
|
+
Student
|
2507
|
+
\family default
|
2508
|
+
instances, one has to mention
|
2509
|
+
\family typewriter
|
2510
|
+
:name
|
2511
|
+
\family default
|
2512
|
+
named argument in the constructor, and frequently, the same name has to
|
2513
|
+
be mentioned twice, such as when assigning to constants or variables:
|
2514
|
+
\end_layout
|
2515
|
+
|
2516
|
+
\begin_layout LyX-Code
|
2517
|
+
richard = Student.new( name: "Richard" )
|
2518
|
+
\end_layout
|
2519
|
+
|
2520
|
+
\begin_layout LyX-Code
|
2521
|
+
richard.name
|
2522
|
+
\end_layout
|
2523
|
+
|
2524
|
+
\begin_layout LyX-Code
|
2525
|
+
#=> "Richard"
|
2526
|
+
\end_layout
|
2527
|
+
|
2528
|
+
\begin_layout Standard
|
2529
|
+
In Ruby, we can notice that some objects have built-in capability to be
|
2530
|
+
named simply by constant assignment:
|
2531
|
+
\end_layout
|
2532
|
+
|
2533
|
+
\begin_layout LyX-Code
|
2534
|
+
foo = Class.new
|
2535
|
+
\end_layout
|
2536
|
+
|
2537
|
+
\begin_layout LyX-Code
|
2538
|
+
foo.name
|
2539
|
+
\end_layout
|
2540
|
+
|
2541
|
+
\begin_layout LyX-Code
|
2542
|
+
#=> nil
|
2543
|
+
\end_layout
|
2544
|
+
|
2545
|
+
\begin_layout LyX-Code
|
2546
|
+
Car = foo
|
2547
|
+
\end_layout
|
2548
|
+
|
2549
|
+
\begin_layout LyX-Code
|
2550
|
+
foo.name
|
2551
|
+
\end_layout
|
2552
|
+
|
2553
|
+
\begin_layout LyX-Code
|
2554
|
+
#=> "Car"
|
2555
|
+
\end_layout
|
2556
|
+
|
2557
|
+
\begin_layout Standard
|
2558
|
+
Magically, upon assigning
|
2559
|
+
\family typewriter
|
2560
|
+
Car = foo
|
2561
|
+
\family default
|
2562
|
+
, the object referred to by the
|
2563
|
+
\family typewriter
|
2564
|
+
foo
|
2565
|
+
\family default
|
2566
|
+
variable received an attribute
|
2567
|
+
\family typewriter
|
2568
|
+
name
|
2569
|
+
\family default
|
2570
|
+
, with value set to "
|
2571
|
+
\family typewriter
|
2572
|
+
Car
|
2573
|
+
\family default
|
2574
|
+
".
|
2575
|
+
This standard behavior is termed constant magic.
|
2576
|
+
|
2577
|
+
\family typewriter
|
2578
|
+
NameMagic
|
2579
|
+
\family default
|
2580
|
+
mixin (part of
|
2581
|
+
\family typewriter
|
2582
|
+
YSupport
|
2583
|
+
\family default
|
2584
|
+
) extends this standard behavior to any chosen object, and also takes care
|
2585
|
+
of keeping the instance registry and doing general naming related chores
|
2586
|
+
for its includers:
|
2587
|
+
\end_layout
|
2588
|
+
|
2589
|
+
\begin_layout LyX-Code
|
2590
|
+
require 'y_support/name_magic'
|
2591
|
+
\end_layout
|
2592
|
+
|
2593
|
+
\begin_layout LyX-Code
|
2594
|
+
|
2595
|
+
\end_layout
|
2596
|
+
|
2597
|
+
\begin_layout LyX-Code
|
2598
|
+
class Chemical
|
2599
|
+
\end_layout
|
2600
|
+
|
2601
|
+
\begin_layout LyX-Code
|
2602
|
+
include NameMagic
|
2603
|
+
\end_layout
|
2604
|
+
|
2605
|
+
\begin_layout LyX-Code
|
2606
|
+
end
|
2607
|
+
\end_layout
|
2608
|
+
|
2609
|
+
\begin_layout LyX-Code
|
2610
|
+
|
2611
|
+
\end_layout
|
2612
|
+
|
2613
|
+
\begin_layout LyX-Code
|
2614
|
+
NaCl = Chemical.new
|
2615
|
+
\end_layout
|
2616
|
+
|
2617
|
+
\begin_layout LyX-Code
|
2618
|
+
NaCl.name
|
2619
|
+
\end_layout
|
2620
|
+
|
2621
|
+
\begin_layout LyX-Code
|
2622
|
+
#=> "NaCl"
|
2623
|
+
\end_layout
|
2624
|
+
|
2625
|
+
\begin_layout Standard
|
2626
|
+
It might seem like a small thing, but in a big file of complicated equation,
|
2627
|
+
it really starts to matter whether one writes on each line "
|
2628
|
+
\family typewriter
|
2629
|
+
NaCl = Chemical.new( name: NaCl )
|
2630
|
+
\family default
|
2631
|
+
", or just "
|
2632
|
+
\family typewriter
|
2633
|
+
NaCl = Chemical.new
|
2634
|
+
\family default
|
2635
|
+
".
|
2636
|
+
|
2637
|
+
\family typewriter
|
2638
|
+
NameMagic
|
2639
|
+
\family default
|
2640
|
+
is a part of
|
2641
|
+
\family typewriter
|
2642
|
+
YSupport
|
2643
|
+
\family default
|
2644
|
+
library accompanying
|
2645
|
+
\family typewriter
|
2646
|
+
YPetri
|
2647
|
+
\family default
|
2648
|
+
and
|
2649
|
+
\family typewriter
|
2650
|
+
YNelson
|
2651
|
+
\family default
|
2652
|
+
.
|
2653
|
+
You can install
|
2654
|
+
\family typewriter
|
2655
|
+
YSupport
|
2656
|
+
\family default
|
2657
|
+
from the command line by "
|
2658
|
+
\family typewriter
|
2659
|
+
gem install y_support
|
2660
|
+
\family default
|
2661
|
+
".
|
2662
|
+
\end_layout
|
2663
|
+
|
2664
|
+
\begin_layout Part*
|
2665
|
+
Other essential concepts
|
2666
|
+
\end_layout
|
2667
|
+
|
2668
|
+
\begin_layout Standard
|
2669
|
+
There are a few more essential concepts of Ruby that
|
2670
|
+
\family typewriter
|
2671
|
+
YNelson
|
2672
|
+
\family default
|
2673
|
+
users should be familiar with, such as namespaces and parametrized subclassing.
|
2674
|
+
Code examples in this section are slightly more complicated, and also,
|
2675
|
+
they make use of
|
2676
|
+
\family typewriter
|
2677
|
+
YSupport
|
2678
|
+
\family default
|
2679
|
+
gem.
|
2680
|
+
Install
|
2681
|
+
\family typewriter
|
2682
|
+
YSupport
|
2683
|
+
\family default
|
2684
|
+
by typing
|
2685
|
+
\family typewriter
|
2686
|
+
gem install y_support
|
2687
|
+
\family default
|
2688
|
+
in your command line before studying code examples in this section.
|
2689
|
+
\end_layout
|
2690
|
+
|
2691
|
+
\begin_layout Section*
|
2692
|
+
Namespaces
|
2693
|
+
\end_layout
|
2694
|
+
|
2695
|
+
\begin_layout Standard
|
2696
|
+
In Ruby, namespaces are known as modules (objects of
|
2697
|
+
\family typewriter
|
2698
|
+
Module
|
2699
|
+
\family default
|
2700
|
+
class).
|
2701
|
+
These objects are containers for constants and method definitions.
|
2702
|
+
For example, let us imagine that we want to define constants, classes and
|
2703
|
+
methods related to the game of chess.
|
2704
|
+
We could simply define them in the command line, without any considerations,
|
2705
|
+
We could do it directly, but that way, all of them would be defined in
|
2706
|
+
the root of Ruby namespace – on
|
2707
|
+
\family typewriter
|
2708
|
+
Object
|
2709
|
+
\family default
|
2710
|
+
class.
|
2711
|
+
The reason why this is not a good idea is the same as the reason why it
|
2712
|
+
is not a good idea to put all your files in the root of your filesystem.
|
2713
|
+
Chess-related terms such as
|
2714
|
+
\family typewriter
|
2715
|
+
Field
|
2716
|
+
\family default
|
2717
|
+
or
|
2718
|
+
\family typewriter
|
2719
|
+
Piece
|
2720
|
+
\family default
|
2721
|
+
could collide with concepts from other domains not related to chess.
|
2722
|
+
For that reason, we will collect all the chess-related assets into a single
|
2723
|
+
namespace:
|
2724
|
+
\end_layout
|
2725
|
+
|
2726
|
+
\begin_layout LyX-Code
|
2727
|
+
module Chess
|
2728
|
+
\end_layout
|
2729
|
+
|
2730
|
+
\begin_layout LyX-Code
|
2731
|
+
class Board < Array
|
2732
|
+
\end_layout
|
2733
|
+
|
2734
|
+
\begin_layout LyX-Code
|
2735
|
+
SIZE = 8 # standard chessboard
|
2736
|
+
\end_layout
|
2737
|
+
|
2738
|
+
\begin_layout LyX-Code
|
2739
|
+
|
2740
|
+
\end_layout
|
2741
|
+
|
2742
|
+
\begin_layout LyX-Code
|
2743
|
+
class Field # chessboard field
|
2744
|
+
\end_layout
|
2745
|
+
|
2746
|
+
\begin_layout LyX-Code
|
2747
|
+
attr_accessor :contents
|
2748
|
+
\end_layout
|
2749
|
+
|
2750
|
+
\begin_layout LyX-Code
|
2751
|
+
end
|
2752
|
+
\end_layout
|
2753
|
+
|
2754
|
+
\begin_layout LyX-Code
|
2755
|
+
|
2756
|
+
\end_layout
|
2757
|
+
|
2758
|
+
\begin_layout LyX-Code
|
2759
|
+
def self.new # constructs 8 × 8 array of arrays
|
2760
|
+
\end_layout
|
2761
|
+
|
2762
|
+
\begin_layout LyX-Code
|
2763
|
+
super( SIZE, Array.new( SIZE ) { Field.new } )
|
2764
|
+
\end_layout
|
2765
|
+
|
2766
|
+
\begin_layout LyX-Code
|
2767
|
+
end
|
2768
|
+
\end_layout
|
2769
|
+
|
2770
|
+
\begin_layout LyX-Code
|
2771
|
+
# chessboard is defined here
|
2772
|
+
\end_layout
|
2773
|
+
|
2774
|
+
\begin_layout LyX-Code
|
2775
|
+
end
|
2776
|
+
\end_layout
|
2777
|
+
|
2778
|
+
\begin_layout LyX-Code
|
2779
|
+
|
2780
|
+
\end_layout
|
2781
|
+
|
2782
|
+
\begin_layout LyX-Code
|
2783
|
+
Piece = Class.new # chess piece
|
2784
|
+
\end_layout
|
2785
|
+
|
2786
|
+
\begin_layout LyX-Code
|
2787
|
+
Pawn = Class.new Piece # chess pawn
|
2788
|
+
\end_layout
|
2789
|
+
|
2790
|
+
\begin_layout LyX-Code
|
2791
|
+
Knight = Class.new Piece # chess knight
|
2792
|
+
\end_layout
|
2793
|
+
|
2794
|
+
\begin_layout LyX-Code
|
2795
|
+
Rook = Class.new Piece # chess rook
|
2796
|
+
\end_layout
|
2797
|
+
|
2798
|
+
\begin_layout LyX-Code
|
2799
|
+
# etc.
|
2800
|
+
\end_layout
|
2801
|
+
|
2802
|
+
\begin_layout LyX-Code
|
2803
|
+
end
|
2804
|
+
\end_layout
|
2805
|
+
|
2806
|
+
\begin_layout Standard
|
2807
|
+
We then access the contents of the namespace in the way similar to the way
|
2808
|
+
we address the files in the filesystem:
|
2809
|
+
\end_layout
|
2810
|
+
|
2811
|
+
\begin_layout LyX-Code
|
2812
|
+
Chess::Board # namespace Chess, constant Board
|
2813
|
+
\end_layout
|
2814
|
+
|
2815
|
+
\begin_layout LyX-Code
|
2816
|
+
Chess::Piece # namespace Chess, constant Piece
|
2817
|
+
\end_layout
|
2818
|
+
|
2819
|
+
\begin_layout LyX-Code
|
2820
|
+
Chess::Pawn # namespace Chess, constant Pawn
|
2821
|
+
\end_layout
|
2822
|
+
|
2823
|
+
\begin_layout LyX-Code
|
2824
|
+
Chess::Board::SIZE # namespace Chess::Board, constant SIZE
|
2825
|
+
\end_layout
|
2826
|
+
|
2827
|
+
\begin_layout LyX-Code
|
2828
|
+
Chess::Board::Field # namespace Chess::Board, constant Field
|
2829
|
+
\end_layout
|
2830
|
+
|
2831
|
+
\begin_layout LyX-Code
|
2832
|
+
# etc.
|
2833
|
+
\end_layout
|
2834
|
+
|
2835
|
+
\begin_layout Standard
|
2836
|
+
Let us note that in the above example,
|
2837
|
+
\family typewriter
|
2838
|
+
Board
|
2839
|
+
\family default
|
2840
|
+
,
|
2841
|
+
\family typewriter
|
2842
|
+
Piece
|
2843
|
+
\family default
|
2844
|
+
,
|
2845
|
+
\family typewriter
|
2846
|
+
Pawn
|
2847
|
+
\family default
|
2848
|
+
are merely constants of the namespace
|
2849
|
+
\family typewriter
|
2850
|
+
Chess
|
2851
|
+
\family default
|
2852
|
+
.
|
2853
|
+
Similarly, in
|
2854
|
+
\family typewriter
|
2855
|
+
YPetri
|
2856
|
+
\family default
|
2857
|
+
, when talking about
|
2858
|
+
\family typewriter
|
2859
|
+
YPetri::Place
|
2860
|
+
\family default
|
2861
|
+
,
|
2862
|
+
\family typewriter
|
2863
|
+
YPetri::Transition
|
2864
|
+
\family default
|
2865
|
+
or
|
2866
|
+
\family typewriter
|
2867
|
+
YPetri::Net
|
2868
|
+
\family default
|
2869
|
+
, it means constants
|
2870
|
+
\family typewriter
|
2871
|
+
Place
|
2872
|
+
\family default
|
2873
|
+
,
|
2874
|
+
\family typewriter
|
2875
|
+
Transition
|
2876
|
+
\family default
|
2877
|
+
and
|
2878
|
+
\family typewriter
|
2879
|
+
Net
|
2880
|
+
\family default
|
2881
|
+
belonging to the module
|
2882
|
+
\family typewriter
|
2883
|
+
YPetri
|
2884
|
+
\family default
|
2885
|
+
and containing the relevant class objects.
|
2886
|
+
But each of these classes is a namespace of its own, that can have constants
|
2887
|
+
defined on it.
|
2888
|
+
For example,
|
2889
|
+
\family typewriter
|
2890
|
+
YPetri::Simulation
|
2891
|
+
\family default
|
2892
|
+
has constants
|
2893
|
+
\family typewriter
|
2894
|
+
YPetri::Simulation::PlaceRepresentation
|
2895
|
+
\family default
|
2896
|
+
and
|
2897
|
+
\family typewriter
|
2898
|
+
YPetri::Simulation::TransitionRepresentation
|
2899
|
+
\family default
|
2900
|
+
, representing copies of the net's places and transitions when executed
|
2901
|
+
inside a
|
2902
|
+
\family typewriter
|
2903
|
+
Simulation
|
2904
|
+
\family default
|
2905
|
+
instance.
|
2906
|
+
\end_layout
|
2907
|
+
|
2908
|
+
\begin_layout Section*
|
2909
|
+
Parametrized subclassing
|
2910
|
+
\end_layout
|
2911
|
+
|
2912
|
+
\begin_layout Standard
|
2913
|
+
One of the core techniques used in
|
2914
|
+
\family typewriter
|
2915
|
+
YNelson
|
2916
|
+
\family default
|
2917
|
+
/
|
2918
|
+
\family typewriter
|
2919
|
+
YPetri
|
2920
|
+
\family default
|
2921
|
+
domain model is parametrized subclassing.
|
2922
|
+
Literature on the topic does exist
|
2923
|
+
\begin_inset CommandInset citation
|
2924
|
+
LatexCommand citep
|
2925
|
+
key "Roberts1996efp"
|
2926
|
+
|
2927
|
+
\end_inset
|
2928
|
+
|
2929
|
+
, but again, the concept is best explained on examples:
|
2930
|
+
\end_layout
|
2931
|
+
|
2932
|
+
\begin_layout LyX-Code
|
2933
|
+
require 'y_support/all'
|
2934
|
+
\end_layout
|
2935
|
+
|
2936
|
+
\begin_layout LyX-Code
|
2937
|
+
|
2938
|
+
\end_layout
|
2939
|
+
|
2940
|
+
\begin_layout LyX-Code
|
2941
|
+
class Human
|
2942
|
+
\end_layout
|
2943
|
+
|
2944
|
+
\begin_layout LyX-Code
|
2945
|
+
include NameMagic # allows humans to be named easily
|
2946
|
+
\end_layout
|
2947
|
+
|
2948
|
+
\begin_layout LyX-Code
|
2949
|
+
end
|
2950
|
+
\end_layout
|
2951
|
+
|
2952
|
+
\begin_layout Standard
|
2953
|
+
Humans generally live in settlements.
|
2954
|
+
Let us create class
|
2955
|
+
\family typewriter
|
2956
|
+
Village
|
2957
|
+
\family default
|
2958
|
+
representing settlements.
|
2959
|
+
\end_layout
|
2960
|
+
|
2961
|
+
\begin_layout LyX-Code
|
2962
|
+
class Village
|
2963
|
+
\end_layout
|
2964
|
+
|
2965
|
+
\begin_layout LyX-Code
|
2966
|
+
include NameMagic # allows villages to be named easily
|
2967
|
+
\end_layout
|
2968
|
+
|
2969
|
+
\begin_layout LyX-Code
|
2970
|
+
end
|
2971
|
+
\end_layout
|
2972
|
+
|
2973
|
+
\begin_layout Standard
|
2974
|
+
At this point, we are standing in front of the problem of making humans
|
2975
|
+
associated with their settlements.
|
2976
|
+
One way to do it is to make each
|
2977
|
+
\family typewriter
|
2978
|
+
Human
|
2979
|
+
\family default
|
2980
|
+
instance remember which settlement they belong to.
|
2981
|
+
This approach, which you can certainly imagine well even without demonstration,
|
2982
|
+
is in common use.
|
2983
|
+
But we have a more powerful approach at our disposal – subclassing.
|
2984
|
+
This is how we can define a subclass of humans living in London:
|
2985
|
+
\end_layout
|
2986
|
+
|
2987
|
+
\begin_layout LyX-Code
|
2988
|
+
London = Village.new
|
2989
|
+
\end_layout
|
2990
|
+
|
2991
|
+
\begin_layout LyX-Code
|
2992
|
+
|
2993
|
+
\end_layout
|
2994
|
+
|
2995
|
+
\begin_layout LyX-Code
|
2996
|
+
class Londoner
|
2997
|
+
\color red
|
2998
|
+
<
|
2999
|
+
\color inherit
|
3000
|
+
Human # using < sign
|
3001
|
+
\end_layout
|
3002
|
+
|
3003
|
+
\begin_layout LyX-Code
|
3004
|
+
def self.settlement; London end # let the class know its city
|
3005
|
+
\end_layout
|
3006
|
+
|
3007
|
+
\begin_layout LyX-Code
|
3008
|
+
end
|
3009
|
+
\end_layout
|
3010
|
+
|
3011
|
+
\begin_layout LyX-Code
|
3012
|
+
|
3013
|
+
\end_layout
|
3014
|
+
|
3015
|
+
\begin_layout LyX-Code
|
3016
|
+
John = Londoner.new
|
3017
|
+
\end_layout
|
3018
|
+
|
3019
|
+
\begin_layout LyX-Code
|
3020
|
+
John.class.settlement #=> London
|
3021
|
+
\end_layout
|
3022
|
+
|
3023
|
+
\begin_layout Standard
|
3024
|
+
Alternative syntax for subclassing is this:
|
3025
|
+
\end_layout
|
3026
|
+
|
3027
|
+
\begin_layout LyX-Code
|
3028
|
+
Dublin = Village.new
|
3029
|
+
\end_layout
|
3030
|
+
|
3031
|
+
\begin_layout LyX-Code
|
3032
|
+
|
3033
|
+
\end_layout
|
3034
|
+
|
3035
|
+
\begin_layout LyX-Code
|
3036
|
+
Dubliner = Class.new Human do # using argument to Class.new
|
3037
|
+
\end_layout
|
3038
|
+
|
3039
|
+
\begin_layout LyX-Code
|
3040
|
+
def self.settlement; Dublin end
|
3041
|
+
\end_layout
|
3042
|
+
|
3043
|
+
\begin_layout LyX-Code
|
3044
|
+
end
|
3045
|
+
\end_layout
|
3046
|
+
|
3047
|
+
\begin_layout LyX-Code
|
3048
|
+
|
3049
|
+
\end_layout
|
3050
|
+
|
3051
|
+
\begin_layout LyX-Code
|
3052
|
+
Finnegan = Dubliner.new
|
3053
|
+
\end_layout
|
3054
|
+
|
3055
|
+
\begin_layout LyX-Code
|
3056
|
+
Finnegan.class.settlement #=> Dublin
|
3057
|
+
\end_layout
|
3058
|
+
|
3059
|
+
\begin_layout LyX-Code
|
3060
|
+
|
3061
|
+
\end_layout
|
3062
|
+
|
3063
|
+
\begin_layout Standard
|
3064
|
+
Simply, each settlement has its own class of humans – its inhabitants.
|
3065
|
+
But since there are many settlements, it is inconvenient to manually define
|
3066
|
+
the inhabitant class for each of them.
|
3067
|
+
We therefore make each village automatically construct its own subclass
|
3068
|
+
of
|
3069
|
+
\family typewriter
|
3070
|
+
Human
|
3071
|
+
\family default
|
3072
|
+
and parametrize it with
|
3073
|
+
\family typewriter
|
3074
|
+
settlement
|
3075
|
+
\family default
|
3076
|
+
attribute.
|
3077
|
+
|
3078
|
+
\family typewriter
|
3079
|
+
YSupport
|
3080
|
+
\family default
|
3081
|
+
supports parametrized subclassing with method
|
3082
|
+
\family typewriter
|
3083
|
+
#param_class
|
3084
|
+
\family default
|
3085
|
+
, and makes it easy to construct a PS of
|
3086
|
+
\family typewriter
|
3087
|
+
Human
|
3088
|
+
\family default
|
3089
|
+
for each
|
3090
|
+
\family typewriter
|
3091
|
+
Village
|
3092
|
+
\family default
|
3093
|
+
istance.
|
3094
|
+
\end_layout
|
3095
|
+
|
3096
|
+
\begin_layout LyX-Code
|
3097
|
+
class Village # reopening the class defined earlier
|
3098
|
+
\end_layout
|
3099
|
+
|
3100
|
+
\begin_layout LyX-Code
|
3101
|
+
def initialize # defining a constructor
|
3102
|
+
\end_layout
|
3103
|
+
|
3104
|
+
\begin_layout LyX-Code
|
3105
|
+
param_class( { Human: Human }, with: { settlement: self } )
|
3106
|
+
\end_layout
|
3107
|
+
|
3108
|
+
\begin_layout LyX-Code
|
3109
|
+
end
|
3110
|
+
\end_layout
|
3111
|
+
|
3112
|
+
\begin_layout LyX-Code
|
3113
|
+
end
|
3114
|
+
\end_layout
|
3115
|
+
|
3116
|
+
\begin_layout Standard
|
3117
|
+
Each village has now its own PS of
|
3118
|
+
\family typewriter
|
3119
|
+
Human
|
3120
|
+
\family default
|
3121
|
+
.
|
3122
|
+
\end_layout
|
3123
|
+
|
3124
|
+
\begin_layout LyX-Code
|
3125
|
+
Stockholm, Riga, Canberra = 3.times.map { Village.new }
|
3126
|
+
\end_layout
|
3127
|
+
|
3128
|
+
\begin_layout LyX-Code
|
3129
|
+
Stockholm.Human # class of Stockholm citizens
|
3130
|
+
\end_layout
|
3131
|
+
|
3132
|
+
\begin_layout LyX-Code
|
3133
|
+
Riga.Human # class of Riga citizens
|
3134
|
+
\end_layout
|
3135
|
+
|
3136
|
+
\begin_layout LyX-Code
|
3137
|
+
Canberra.Human # class of Vilnius citizens
|
3138
|
+
\end_layout
|
3139
|
+
|
3140
|
+
\begin_layout LyX-Code
|
3141
|
+
Stockholm.Human == Riga.Human #=> false
|
3142
|
+
\end_layout
|
3143
|
+
|
3144
|
+
\begin_layout LyX-Code
|
3145
|
+
|
3146
|
+
\end_layout
|
3147
|
+
|
3148
|
+
\begin_layout LyX-Code
|
3149
|
+
Fred = Stockholm.Human.new # Stockholm citizen constructor
|
3150
|
+
\end_layout
|
3151
|
+
|
3152
|
+
\begin_layout LyX-Code
|
3153
|
+
Fred.class.settlement #=> Stockholm
|
3154
|
+
\end_layout
|
3155
|
+
|
3156
|
+
\begin_layout Standard
|
3157
|
+
We say that PS of
|
3158
|
+
\family typewriter
|
3159
|
+
Human
|
3160
|
+
\family default
|
3161
|
+
class
|
3162
|
+
\emph on
|
3163
|
+
depends
|
3164
|
+
\emph default
|
3165
|
+
on
|
3166
|
+
\family typewriter
|
3167
|
+
Village
|
3168
|
+
\family default
|
3169
|
+
.
|
3170
|
+
The advantage is that instances of the PS of
|
3171
|
+
\family typewriter
|
3172
|
+
Human
|
3173
|
+
\family default
|
3174
|
+
don't need to be explicitly told which village do they belong to, and have
|
3175
|
+
easy access to the assets of its owner
|
3176
|
+
\family typewriter
|
3177
|
+
Village
|
3178
|
+
\family default
|
3179
|
+
instance.
|
3180
|
+
The concept of parametrized subclassing is actually very simple.
|
3181
|
+
\end_layout
|
3182
|
+
|
3183
|
+
\begin_layout Section*
|
3184
|
+
Convenience methods
|
3185
|
+
\end_layout
|
3186
|
+
|
3187
|
+
\begin_layout Standard
|
3188
|
+
Convenience methods are methods in which the consistency of the behavior
|
3189
|
+
is traded for syntax sweetness.
|
3190
|
+
Convenience methods may do entirely different things for different argument
|
3191
|
+
sets.
|
3192
|
+
For example,
|
3193
|
+
\family typewriter
|
3194
|
+
Place#marking
|
3195
|
+
\family default
|
3196
|
+
without arguments simply returns the place's marking.
|
3197
|
+
But with arguments, it can be used to define a guard.
|
3198
|
+
\end_layout
|
3199
|
+
|
3200
|
+
\begin_layout LyX-Code
|
3201
|
+
A = Place.new marking: 42
|
3202
|
+
\end_layout
|
3203
|
+
|
3204
|
+
\begin_layout LyX-Code
|
3205
|
+
A.marking # Simply returns its marking
|
3206
|
+
\end_layout
|
3207
|
+
|
3208
|
+
\begin_layout LyX-Code
|
3209
|
+
# But with different arguments, same method can be used to
|
3210
|
+
\end_layout
|
3211
|
+
|
3212
|
+
\begin_layout LyX-Code
|
3213
|
+
# define a guard.
|
3214
|
+
\end_layout
|
3215
|
+
|
3216
|
+
\begin_layout LyX-Code
|
3217
|
+
A.marking "must never be negative" do |m| m >= 0 end
|
3218
|
+
\end_layout
|
3219
|
+
|
3220
|
+
\begin_layout Standard
|
3221
|
+
Depending on circumstances, use of convenience methods in reusable code
|
3222
|
+
may or may not be bad practice.
|
3223
|
+
\end_layout
|
3224
|
+
|
3225
|
+
\begin_layout LyX-Code
|
3226
|
+
|
3227
|
+
\end_layout
|
3228
|
+
|
3229
|
+
\begin_layout LyX-Code
|
3230
|
+
\begin_inset CommandInset bibtex
|
3231
|
+
LatexCommand bibtex
|
3232
|
+
bibfiles "/home/boris/b/5ced/ced"
|
3233
|
+
options "plainnat"
|
3234
|
+
|
3235
|
+
\end_inset
|
3236
|
+
|
3237
|
+
|
3238
|
+
\end_layout
|
3239
|
+
|
2253
3240
|
\end_body
|
2254
3241
|
\end_document
|