forked from dlang/dlang.org
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathoverview.dd
1192 lines (930 loc) · 36.5 KB
/
overview.dd
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
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
Ddoc
$(D_S Overview,
$(DIVC page-contents quickindex,
$(DIVC page-contents-header,
$(H3 Contents)
)
$(OL
$(LI $(RELATIVE_LINK2 what_is_d, What is D?))
$(LI $(RELATIVE_LINK2 why_d, Why D?)$(OL
$(LI $(RELATIVE_LINK2 goals, Major Design Goals of D))
$(LI $(RELATIVE_LINK2 features_to_keep, Transitioning To D))
$(LI $(RELATIVE_LINK2 features_to_drop, Features To Leave Behind))
$(LI $(RELATIVE_LINK2 for_d, Who D is For?))
$(LI $(RELATIVE_LINK2 not_for_d, Who D is Not For?))
))
$(LI $(RELATIVE_LINK2 major_features, Major Features of D)$(OL
$(LI $(RELATIVE_LINK2 OOP, Object Oriented Programming))
$(LI $(RELATIVE_LINK2 functional, Functional Programming))
$(LI $(RELATIVE_LINK2 productivity, Productivity))
$(LI $(RELATIVE_LINK2 functions, Functions))
$(LI $(RELATIVE_LINK2 arrays, Arrays))
$(LI $(RELATIVE_LINK2 ranges, Ranges))
$(LI $(RELATIVE_LINK2 resource, Resource Management))
$(LI $(RELATIVE_LINK2 performance, Performance))
$(LI $(RELATIVE_LINK2 reliability, Reliability))
$(LI $(RELATIVE_LINK2 compatibility, Compatibility))
$(LI $(RELATIVE_LINK2 management, Project Management))
))
$(LI $(RELATIVE_LINK2 sample, Sample D Program))
)
)
$(SECTION2 $(LNAME2 what_is_d, What is D?),
<img src="images/d3.png" border="0" align="left" alt="D Man">
$(P D is a general purpose systems and applications programming language.
It is a high level language, but retains the ability
to write high performance code and interface directly with the
operating system
<acronym title="Application Programming Interface">API</acronym>'s
and with hardware.
D is well suited to writing medium to large scale
million line programs with teams of developers. D is easy
to learn, provides many capabilities to aid the programmer,
and is well suited to aggressive compiler optimization technology.
)
$(P D is not a scripting language, nor an interpreted language.
It doesn't
come with a <acronym title="Virtual Machine">VM</acronym>,
a religion, or an overriding
philosophy. It's a practical language for practical programmers
who need to get the job done quickly, reliably, and leave behind
maintainable, easy to understand code.
)
$(P D is the culmination of decades of experience implementing
compilers for many diverse languages, and attempting to construct
large projects using those languages. D draws inspiration from
those other languages (most especially C++) and tempers it with
experience and real world practicality.
)
)
$(SECTION2 $(LNAME2 why_d, Why D?),
$(P Why, indeed. Who needs another programming language?
)
$(P The software industry is continuously evolving at a breakneck pace.
New ideas appear, and older ideas are either validated or discarded.
What programmers need and want out of a programming language changes.
Available memory and computing power have increased by orders of
magnitude, as well as the scale of programs being developed.
)
$(P Compilers are no longer terribly constrained by available computing
resources, and so are able to do much more for the programmer.
Much more powerful language features have become practical.
These features can be difficult to retrofit into existing languages,
and when there are enough of these, a new language is justified.
)
$(SECTION3 $(LNAME2 goals, Major Design Goals of D),
$(P Everything in designing a language is a tradeoff. Keeping some
principles in mind will help to make the right decisions.)
$(OL
$(LI Enable writing fast, effective code in a straightforward manner.)
$(LI Make it easier to write code that is portable from compiler
to compiler, machine to machine, and operating system to operating
system. Eliminate undefined and implementation defined behaviors
as much as practical.)
$(LI Provide syntactic and semantic constructs that eliminate or
at least reduce common mistakes. Reduce or even eliminate the need
for third party static code checkers.)
$(LI Support memory safe programming.)
$(LI Support multi-paradigm programming, i.e. at a minimum support
imperative, structured, object oriented, generic and even
functional programming paradigms.)
$(LI Make doing things the right way easier than the wrong way.)
$(LI Have a short learning curve for programmers comfortable with
programming in C, C++ or Java.)
$(LI Provide low level bare metal access as required. Provide a means
for the advanced programmer to escape checking as necessary.)
$(LI Be compatible with the local C application binary interface.)
$(LI Where D code looks the same as C code, have it either behave the
same or issue an error.)
$(LI Have a context-free grammar. Successful parsing must not require
semantic analysis.)
$(LI Easily support writing internationalized applications - Unicode everywhere.)
$(LI Incorporate Contract Programming and unit testing methodology.)
$(LI Be able to build lightweight, standalone programs.)
$(LI Reduce the costs of creating documentation.)
$(LI Provide sufficient semantics to enable advances in compiler optimization
technology.)
$(LI Cater to the needs of numerical analysis programmers.)
$(LI Obviously, sometimes these goals will conflict. Resolution will be
in favor of usability.)
)
)
$(SECTION3 $(LNAME2 features_to_keep, Transitioning to D),
$(P The general look of D is like C and C++. This makes it easier to learn
and port code to D. Transitioning from C/C++ to D should feel natural.
The
programmer will not have to learn an entirely new way of doing things.
)
$(P Using D will not mean that the programmer will become restricted to a
specialized runtime vm (virtual machine) like the Java vm or the
Smalltalk vm.
There is no D vm, it's a straightforward compiler that generates
linkable object files.
D connects to the operating system just like C does.
The usual familiar tools like $(B make) will fit right in with
D development.
)
$(UL
$(LI The general $(B look and feel) of C/C++ is adopted.
It uses the same algebraic syntax, most of the same expression
and statement forms, and the general layout.
)
$(LI D programs can be written in familiar paradigms,
$(B function-and-data),
$(B object-oriented),
$(B template metaprogramming),
$(B functional),
or any mix of them.
)
$(LI The $(B compile/link/debug) development model is
carried forward from languages like C,
although nothing precludes D from being compiled into bytecode
and interpreted.
)
$(LI $(B Exception handling).
More and more experience with exception handling shows it to be a
superior way to handle errors than error codes.
)
$(LI $(B Runtime Type Identification).
Fully supporting RTTI it enables better garbage
collection, better debugger support, more automated persistence, etc.
)
$(LI D maintains function link compatibility with the $(B C calling
conventions). This makes
it possible for D programs to access operating system APIs directly.
Programmers' knowledge and experience with existing programming APIs
and paradigms can be carried forward to D with minimal effort.
)
$(LI Limited support for function link compatibility with $(B C++
calling conventions). In particular, there is compatibility with
C++ name mangling, namespaces, templates, and exceptions.
)
$(LI $(B Operator overloading).
D programs can overload operators enabling
extension of the basic types with user-defined types.
)
$(LI $(B Template Metaprogramming).
Parametric polymorphism (aka template metaprogramming) is straightforward
in D.
)
$(LI <acronym title="Resource Acquisition Is Initialization">$(B RAII)</acronym>
(Resource Acquisition Is Initialization).
RAII techniques are an essential component of writing reliable
software.
)
$(LI $(B Custom memory management).
There are times in systems programming where developers need
alternatives to garbage collection. D also allows for manual memory
management, techniques such as reference counting, and the use of
specialized memory allocators.
)
$(LI $(B Down and dirty programming). D retains the ability to
do down-and-dirty programming without resorting to referring to
external modules compiled in a different language. Sometimes,
it's just necessary to coerce a pointer or dip into assembly
when doing systems work. D's goal is not to $(I prevent) down
and dirty programming, but to minimize the need for it in
solving routine coding tasks.
)
)
)
$(SECTION3 $(LNAME2 features_to_drop, Features To Leave Behind),
$(UL
$(LI Support for 16 bit computers.
No consideration is given in D for mixed near/far pointers and all the
machinations necessary to generate good 16 bit code. The D language
design assumes at least a 32 bit flat memory space and supports 64 bit as
well.
)
$(LI Mutual dependence of compiler passes.
Having a clean separation between the lexer, parser, and semantic
passes makes for an easier to implement and understand language.
This means no user-defined tokens and no user-defined syntax.
)
$(LI Macro systems.
Macros are an easy way for users to add very powerful features.
Unfortunately, the complexity is pushed off on to the user trying
to understand the macro usage, and the result is rarely worth
it and not justifiable.
)
$(LI C/C++ source code compatibility.
The use of the preprocessor has made this difficult to achieve without
inevitably requiring manual touch up. It's best to leave this
to external tools.
)
$(LI Multiple inheritance. It's a complex
feature of debatable value. It's very difficult to implement in an
efficient manner, and compilers are prone to many bugs in implementing
it. Nearly all the value of
<acronym title="multiple inheritance">MI</acronym> can be handled with
single inheritance
coupled with interfaces and aggregation. What's left does not
justify the weight of MI implementation.
)
)
)
$(SECTION3 $(LNAME2 for_d, Who is D For?),
$(UL
$(LI Programmers who routinely use lint or similar code analysis tools
to eliminate bugs before the code is even compiled.
)
$(LI People who compile with maximum warning levels turned on and who
instruct the compiler to treat warnings as errors.
)
$(LI Programming managers who are forced to rely on programming style
guidelines to avoid common bugs.
)
$(LI Projects that need built-in testing and verification.
)
$(LI Teams who write apps with a million lines of code in it.
)
$(LI Programmers who think the language should provide enough
features to obviate
the continual necessity to manipulate pointers directly.
)
$(LI Programmers who write half their application in scripting
languages the other half in native langauges to
speed up the bottlenecks.
D has productivity features that make using such hybrid approaches
unnecessary.
)
$(LI D's lexical analyzer and parser are totally independent of each other and of the
semantic analyzer. This means it is easy to write simple tools to manipulate D source
perfectly without having to build a full compiler. It also means that source code can be
transmitted in tokenized form for specialized applications.
)
)
)
$(SECTION3 $(LNAME2 not_for_d, Who D is Not For?),
$(UL
$(LI Realistically, nobody is going to convert million line legacy
programs into D.
D, however, can connect directly to any code that exposes a C ABI interface
or (to a limited extent) a C++ ABI interface.
)
$(LI As a first programming language - Python or JavaScript is more suitable
for beginners. D makes an excellent second language for intermediate
to advanced programmers.
)
$(LI Language purists. D is a practical language, and each feature
of it is evaluated in that light, rather than by an ideal.
For example, D has constructs and semantics that virtually eliminate
the need for pointers for ordinary tasks. But pointers are still
there, because sometimes the rules need to be broken.
Similarly, casts are still there for those times when the typing
system needs to be overridden.
)
)
)
)
$(SECTION2 $(LNAME2 major_features, Major Features of D),
$(P This section lists some of the more interesting features of D
in various categories.
)
$(SECTION3 $(LNAME2 OOP, Object Oriented Programming),
$(SECTION4 Classes,
$(P D's object oriented nature comes from classes.
The inheritance model is single inheritance enhanced
with interfaces. The class Object sits at the root
of the inheritance hierarchy, so all classes implement
a common set of functionality.
Classes are instantiated
by reference, and so complex code to clean up after exceptions
is not required.
)
)
$(P See the $(LINK2 spec/class.html, Classes) page for more
information.
)
$(SECTION4 Operator Overloading,
$(P Classes can be crafted that work with existing operators to extend
the type system to support new types. An example would be creating
a bignumber class and then overloading the +, -, * and / operators
to enable using ordinary algebraic syntax with them.
)
)
$(P See the $(LINK2 spec/operatoroverloading.html, Operator Overloading)
page for more information.
)
)
$(SECTION3 $(LNAME2 functional, Functional Programming),
$(P Functional programming has a lot to offer in terms of encapsulation,
concurrent programming, memory safety, and composition. D's support for
functional style programming include:
)
$(UL
$(LI Pure functions)
$(LI Immutable types and data structures)
$(LI Lambda functions and closures)
)
)
$(SECTION3 $(LNAME2 productivity, Productivity),
$(SECTION4 Modules,
$(P Source files have a one-to-one correspondence with modules.
)
)
$(P See the $(LINK2 spec/module.html, Module)
page for more information.
)
$(SECTION4 Declaration vs Definition,
$(P Functions and classes are defined once. There is no need
for declarations when they are forward referenced.
A module can be imported, and all its public declarations
become available to the importer.
)
$(P Example:
)
-----------------------
class ABC
{
int func() { return 7; }
static int z = 7;
}
int q;
-----------------------
$(P All members are defined in the class or struct, not separately.
)
-----------------------
class Foo
{
int foo(Bar c) { return c.bar; }
}
class Bar
{
int bar() { return 3; }
}
-----------------------
$(P Whether a D function is inlined or not is determined by the
optimizer settings.
)
)
$(SECTION4 Templates,
$(P D templates offer a clean way to support generic programming while
offering the power of partial specialization.
Template classes and template functions are available, along
with variadic template arguments and tuples.
)
)
$(P See the $(LINK2 spec/template.html, Templates)
page for more information.
)
$(SECTION4 Associative Arrays,
$(P Associative arrays are arrays with an arbitrary data type as
the index rather than being limited to an integer index.
In essence, associated arrays are hash tables. Associative
arrays make it easy to build fast, efficient, bug-free symbol
tables.
)
)
$(P See the $(LINK2 spec/hash-map.html, Associative Arrays)
page for more information.
)
$(SECTION4 Documentation,
$(P Documentation has traditionally been done twice - first there
are comments documenting what a function does, and then this gets
rewritten into a separate html or man page.
And naturally, over time, they'll tend to diverge as the code
gets updated and the separate documentation doesn't.
Being able to generate the requisite polished documentation directly
from the comments embedded in the source will not only cut the time
in half needed to prepare documentation, it will make it much easier
to keep the documentation in sync with the code.
$(LINK2 spec/ddoc.html, Ddoc) is the specification for the D
documentation generator. This page was generated by Ddoc, too.
)
$(P Although third party tools exist to do this for other languages.
they have some serious shortcomings:
$(UL
$(LI It is difficult to parse the language correctly.
Third party tools tend to
parse only a subset of a language correctly, so their use will constrain
the source code to that subset.)
$(LI Different compilers support different versions of a language and have
different extensions to it. Third party tools have a problem matching
all these variations.)
$(LI Third party tools may not be available for all the desired
platforms, and they're necessarily on a different upgrade cycle
from the compilers.)
$(LI Having it builtin to the compiler means it is standardized across
all D implementations. Having a default one ready to go at all times
means it is far more likely to be used.)
)
)
)
)
$(SECTION3 $(LNAME2 functions, Functions),
$(P D has the expected support for ordinary functions including
global functions, overloaded functions, inlining of functions,
member functions, virtual functions, function pointers, etc.
In addition:
)
$(SECTION4 Nested Functions,
$(P Functions can be nested within other functions.
This is highly useful for code factoring, locality, and
function closure techniques.
)
)
$(SECTION4 Function Literals,
$(P Anonymous functions can be embedded directly into an expression.
)
)
$(SECTION4 Dynamic Closures,
$(P Nested functions and class member functions can be referenced
with closures (also called delegates), making generic programming
much easier and type safe.
)
)
$(SECTION4 In$(COMMA) Out$(COMMA) and Ref Parameters,
$(P Not only does specifying this help make functions more
self-documenting, it eliminates much of the necessity for pointers
without sacrificing anything, and it opens up possibilities
for more compiler help in finding coding problems.
)
$(P Such makes it possible for D to directly interface to a
wider variety of foreign APIs. There would be no need for
workarounds like "Interface Definition Languages".
)
)
)
$(SECTION3 $(LNAME2 arrays, Arrays),
$(P C arrays have several faults that can be corrected:
)
$(UL
$(LI Dimension information is not carried around with
the array, and so has to be stored and passed separately.
The classic example of this are the argc and argv
parameters to $(D main(int $(D_PARAM argc), char *$(D_PARAM argv)[])).
(In D, main is declared as $(D main(string[] $(D_PARAM args))).)
)
$(LI Arrays are not first class objects. When an array is passed to a function, it is
converted to a pointer, even though the prototype confusingly says it's an
array. When this conversion happens, all array type information
gets lost.
)
$(LI C arrays cannot be resized. This means that even simple aggregates like a stack
need to be constructed as a complex class.)
$(LI C arrays cannot be bounds checked, because they don't know
what the array bounds are.)
$(LI Arrays are declared with the [] after the identifier. This leads to
very clumsy
syntax to declare things like a pointer to an array:
$(CCODE
int (*array)[3];
)
$(P In D, the [] for the array go on the left:
)
-----------------------
int[3]* array; // declares a pointer to an array of 3 ints
long[] func(int x); // declares a function returning an array of longs
-----------------------
$(P which is much simpler to understand.
)
)
)
$(P D arrays come in several varieties: pointers, static arrays, dynamic
arrays, and associative arrays.
)
$(P See the $(LINK2 spec/arrays.html, Arrays) page for more information.
)
$(SECTION4 Strings,
$(P String manipulation
needs direct support in the language. Modern languages handle
string concatenation, copying, etc., and so does D. Strings are
a direct consequence of improved array handling.
)
)
)
$(SECTION3 $(LNAME2 ranges, Ranges),
$(P D uses the concept of a range in lieu of iterators or generators found in
other languages. A range is any type that provides a common interface to a
sequence of values. The purpose of a range is to allow for a simpler way to
write code that works on arbitrary data, thereby making it reusable.
)
$(P The most basic type of range is called an input range,
which provides three methods.
)
-----------------------
struct MyRange
{
auto front()
{
// return the next value in the sequence
}
void popFront()
{
// move the front of the sequence to the next value
}
bool empty()
{
// true if the range has no more values to return
}
}
-----------------------
$(P To understand the power of this simple interface, let's run through an
example. Say we wanted to write a program that took all of the employees in
a company, took out the ones younger than 40, and grouped the remaining into
an array of arrays by their organization.
)
-----------------------
struct Employee
{
uint id;
uint organization_id;
string name;
uint age;
}
struct Employees
{
Employee[] data;
this(Employee[] employees)
{
data = employees;
}
Employee front()
{
return data[0];
}
void popFront()
{
data = data[1 .. $];
}
bool empty()
{
return data.length == 0;
}
}
-----------------------
$(P Here the data is coming from a constructor as a simple example, but it
could come from any source, like a CSV or a database.
)
$(P Please note that this code should not be used in actual code, because in D,
the basic dynamic array also acts as a range, so any algorithm that accepts
ranges also accepts arrays. However, static arrays are not considered ranges,
as the operation $(D popFront) is based on mutating the length of the range,
which is impossible with static arrays. To get a range out of a static array,
create a slice containing all of its elements, like so:
)
-----------------------
int[4] array = [1, 2, 3, 4]; // not a range
array[]; // valid range
-----------------------
$(P Now that the range is defined, we can populate it and write the filtering code.
)
-----------------------
void main()
{
import std.algorithm.iteration : filter, chunkBy;
Employees employees = Employees([
Employee(1, 1, "George", 50),
Employee(2, 3, "John", 65),
Employee(3, 2, "David", 40),
Employee(4, 1, "Eli", 40),
Employee(5, 2, "Hal", 35)
]);
auto older_employees = employees
.filter!(a => a.age > 40) // lambdas in D use the => syntax
.chunkBy!((a,b) => a.organization_id == b.organization_id);
}
-----------------------
$(P All of the algorithms in std.algorithm work with ranges to avoid the
problem of rewriting common functionality for every project. std.algorithm
implements sorts, filters, maps, reductions, and more.
)
$(P Because our Employees struct conforms to the input range definition,
it can also be used in $(D foreach) loops, which automatically detects if the
value passed is an input range.
)
-----------------------
foreach(employee; employees)
{
writeln(employee);
}
-----------------------
$(P which is equivalent to)
-----------------------
for(; !employees.empty; employees.popFront())
{
writeln(employees.front);
}
-----------------------
$(SECTION4 Types of Ranges,
$(P The input range is just the most basic form of range, there are also
)
$(UL
$(LI forward ranges)
$(LI bidirectional ranges)
$(LI random access ranges)
$(LI output ranges)
)
$(P Each of these ranges represents a distinct way of accessing the
underlying data. Or, in the case of the output range, a way to send data
to another source. To learn more about each type of range, see the
$(MREF_ALTTEXT range primitives, std, range, primitives)
page in the standard library documentation.
)
$(P Each of these types of ranges give you access to different algorithms
in the standard library. For example, a filter can be run on an input
range, but not a sort, which requires a random access range with the
slice operator overload defined. The requirements for each function can
be seen in the function signature in the documentation in the template
constraints. See the $(LINK2 spec/template.html, template)
page and the range primitives link above for more details on template
constraints.
)
$(P As stated before, dynamic arrays and associative arrays in D act
as ranges. Specifically, they are random access ranges, so any of the
functions in std.algorithm work with these basic types.
)
)
$(SECTION4 Advanced Range Code,
$(P The following example uses input ranges and an output range to take
data from stdin, take only the unique lines, sort them, and then print
the result to stdout.
)
-----------------------
// Sort lines
import std.stdio;
import std.array;
import std.algorithm;
void main()
{
stdin
.byLine(KeepTerminator.yes)
.uniq
.map!(a => a.idup)
.array
.sort
.copy(stdout.lockingTextWriter());
-----------------------
$(P For more examples of range based code, see the
$(LINK2 http://ddili.org/ders/d.en/ranges.html, Ranges) chapter in Ali
Çehreli's book "Programming In D". Also, see the article
$(LINK2 https://wiki.dlang.org/Component_programming_with_ranges,
Component programming with ranges) by H. S. Teoh.
)
)
)
$(SECTION3 $(LNAME2 resource, Resource Management),
$(SECTION4 Automatic Memory Management,
$(P D memory allocation is fully garbage collected.
With garbage collection, programming gets much simpler.
Garbage collection eliminates the need for tedious, error prone memory
allocation tracking code. This not only means much
faster development time and lower maintenance costs,
but the resulting program frequently runs
faster.
)
$(P For a fuller discussion of this, see
$(LINK2 spec/garbage.html, garbage collection).
)
)
$(SECTION4 Explicit Memory Management,
$(P Despite D being a garbage collected language, the new and delete
operations can be overridden for particular classes so that
a custom allocator can be used.
)
)
$(SECTION4 RAII,
$(P RAII is a modern software development technique to manage resource
allocation and deallocation. D supports RAII in a controlled,
predictable manner that is independent of the garbage collection
cycle.
)
)
)
$(SECTION3 $(LNAME2 performance, Performance),
$(SECTION4 Lightweight Aggregates,
$(P D supports simple C style structs, both for compatibility with
C data structures and because they're useful when the full power
of classes is overkill.
)
)
$(SECTION4 Inline Assembler,
$(P Device drivers, high performance system applications, embedded systems,
and specialized code sometimes need to dip into assembly language
to get the job done. While D implementations are not required
to implement the inline assembler, it is defined and part of the
language. Most assembly code needs can be handled with it,
obviating the need for separate assemblers or DLLs.
)
$(P Many D implementations will also support intrinsic functions
analogously to C's support of intrinsics for I/O port manipulation,
direct access to special floating point operations, etc.
)
)
$(P See the $(LINK2 spec/iasm.html, Inline Assembler)
page for more information.
)
)
$(SECTION3 $(LNAME2 reliability, Reliability),
$(P A modern language should do all it can to help the programmer flush
out bugs in the code. Help can come in many forms;
from making it easy to use more robust techniques,
to compiler flagging of obviously incorrect code, to runtime checking.
)
$(SECTION4 Contracts,
$(P $(LINK2 https://en.wikipedia.org/wiki/Design_by_contract, Contract Programming)
(invented by Dr. Bertrand Meyer) is a
technique
to aid in ensuring the correctness of programs. D's version of
Contracts includes function preconditions, function postconditions, class
invariants, and assert contracts.
See $(LINK2 spec/contracts.html, Contracts) for D's implementation.
)
)
$(SECTION4 Unit Tests,
$(P Unit tests can be added to a class, such that they are automatically
run upon program startup. This aids in verifying, in every build,
that class implementations weren't inadvertently broken. The unit
tests form part of the source code for a class. Creating them
becomes a natural part of the class development process, as opposed
to throwing the finished code over the wall to the testing group.
)
$(P Unit tests can be done in other languages, but the result is kludgy
and the languages just aren't accommodating of the concept.
Unit testing is a main feature of D. For library functions it works
out great, serving both to guarantee that the functions
actually work and to illustrate how to use the functions.
)
$(P Consider the many library and application code bases out there for
download on the web. How much of it comes with *any* verification
tests at all, let alone unit testing? The usual practice
is if it compiles, we assume it works. And we wonder if the warnings
the compiler spits out in the process are real bugs or just nattering
about nits.
)
$(P Along with Contract Programming, unit testing makes D far and away
the best language for writing reliable, robust systems applications.
Unit testing also gives us a quick-and-dirty estimate of the quality
of some unknown piece of D code dropped in our laps - if it has no
unit tests and no contracts, it's unacceptable.
)
)
$(P See the $(LINK2 spec/unittest.html, Unit Tests)
page for more information.
)
$(SECTION4 Debug Attributes and Statements,
$(P Now debug is part of the syntax of the language.
The code can be enabled or disabled at compile time, without the
use of macros or preprocessing commands. The debug syntax enables
a consistent, portable, and understandable recognition that real
source code needs to be able to generate both debug compilations and
release compilations.
)
)
$(SECTION4 Exception Handling,
$(P The superior $(I try-catch-finally) model is used rather than just
try-catch. There's no need to create dummy objects just to have
the destructor implement the $(I finally) semantics.
)
)
$(SECTION4 Synchronization,
$(P Multithreaded programming is becoming more and more mainstream,
and D provides primitives to build multithreaded programs with.
Synchronization can be done at either the method or the object level.
)
-----------------------
synchronized int func() { ... }
-----------------------
$(P Synchronized functions allow only one thread at a time to be
executing that function.
)
$(P The synchronize statement puts a mutex around a block of statements,