In common with other aspects of PM design, the type system is designed to facilitate the use of flexible programming constructs, such as polymorphism and dynamic dispatch, while emphasising the generation of fast, static, code.  In common with many other modern languages, PM also attempts to combine the safety of static typing with the expressivity of dynamically typed languages such as Python. 
The type of any PM expression may usually be determined by a static examination of the code. Variables take their types from an initialising expression using the ‘:=’ declaration syntax popularised by Go. 
 a:= 100  ! Declare ‘a’ as an integer (int) variable  
Variables do not change type during the course of their lifetime – polymorphic programming requires the use of special values, described later. Composite values such as structures are generated using specialised expressions rather than through the invocation of type-specific creator functions:
 b:= struct var_descriptor { num_values=a; var_name=”Pressure” }  
Of course, the creation of complex composite values may (and typically will) be encapsulated in a procedure:
 proc var_descriptor(n,name) = struct var_descriptor { 
   num_values=n
   var_name=name
}   
PM interprets all names starting with an underscore as being local to the module in which they are declared. Using this mechanism with structure tags and/or member names enables values to be created with an opaque type whose composition is unknown to other modules.
 proc make(x,y) = struct _my_type { x = x; _y=y }    
 ! Member x is public, _y is private to module  
All PM procedure declarations are regarded as generic templates, in the sense that separate code is generated (at least theoretically) for every combination of argument types occurring in a given program.
 proc r2(x,y)=x**2+y**2  
 a:= r2(3,4)  
 b:= r2(3.0,4.0)  
Of course, it will often be necessary to employ different procedure definitions for different combinations of argument types. This is achieved by applying type constraints to formal parameters.
 proc area(x: struct triangle{base,height}) = x.base*x.height/2  
 proc area(x: struct rectangle{base,height} = x.base*x.height  
The first procedure definition will only apply to structured values with the “triangle” tag and two members named “base” and “height”. The constraint does not specify types for these two members, although this could be done if required.
 proc area(x: struct triangle{base:real,height:real})= x.base*x.height/2  
 type triangle is struct triangle{base,height}  
 type rectangle is struct rectangle{base,height}  
 proc area(x: triangle) = x.base*x.height/2 
 proc area(x: rectangle) = x.base*x.height  
 type shape is triangle, rectangle  
 proc base(x:shape)=x.base  
 type shape includes triangle, rectangle  
 …  
 type polygon is struct polygon { points: point[] }  
 type shape also includes polygon  
 type polygon in shape is struct polygon { points: point[]}  
 p:= <shape>triangle(1.0,3.0)  
 p=<shape>rectangle(5.0,5.0)  
 shapes := <shape>triangle(0.0,0.0) dim N  
 …  
 this_shape := *shape[i] 
 this_shape_area := area(this_shape)  
Comments
Post a Comment