I can't seem to grasp exactly what name equivalence is. I'm pretty sure I have structural down though. An example my professor gave was this:
Type TI=integer
Type TTI=TI
a=integer
b=TTI
f= ref float
g= ref float
a and b are both structural and name equivalent, while f and g are just structural equivalent.I don't understand why a and b would be name equivalent, but f and g aren't.
a and b are alies thats why they are name equivalent. f and g are not so they are not name equivalent.
Consider the two definitions bellow.
In the example above, variables x and y will be considered to have different types under name equivalence: x uses the type declared at line 1; y uses the type declared at line 4. Name equivalence is based on the assumption that if the programmer goes to the effort of writing two type definitions, then those definitions are probably meant to represent different types. (I'm not sure about the example you have given)
Reference: Programming Languages Pragmatics, by M.L. Scott
In a name type equivalence, two variables have the same type if they are defined in the same declaration or in declarations that use the same type name. Therefore, the variables
'f'
and'g'
in your example are equivalents. However, the variables'a'
and'b'
are not equivalents, because they have different type names. Furthermore, under structure type equivalence, two variables have the same type if they have identical structures. Thus, the variables'a'
and'b'
are equivalents and also the variables'f'
and'g'
are equivalents because, obviously, types with the same name have the same structure.Reference: Sebesta, Concepts of Programming Languages, 10th ed.
Type Equality
The meaning of basic operations such as assignment (denoted by = in C) is specified in a language definition. Thus, for example, the meaning of statements such as
here the value of object
y
is copied into the memory locations for variablex
.However, before an operation such as an assignment can be accepted by the translator, usually the types of the two operands must be the same (or perhaps compatible in some other specified way).
Thus a language translator must decide whether two types are equal in some cases. We now consider what it means to say that two types are "equal" (or equivalent).
There are two standard ways to determine whether two types are considered the same: name equivalence and structural equivalence.
Name equivalence is the most straightforward: two types are equal if, and only if, they have the same name. Thus, for example, in the code (using C syntax)
if name equivalence is used in the language then
x
andy
would be of the same type andr
ands
would be of the same type, but the type ofx
ory
would not be equivalent to the type ofr
ors
. This means that statements such aswould be valid, but statements such as
would not be valid (i.e., would not be accepted by a translator).
Using structural equivalence:, two types are equal if, and only if, they have the same "structure", which can be interpreted in different ways.
A strict interpretation would be that the names and types of each component of the two types must be the same and must be listed in the same order in the type definition.
A less stringent requirement would be that the component types must be the same and in the same order in the two types, but the names of the components could be different.
Again looking at the example above, using structural equivalence the two types
Stack
andSet
would be considered equivalent, which means that a translator would accept statements such as(Note that
C
doesn't support structural equivalence and will give error for above assignment.)The notion of name equivalence makes the most sense if you consider the internal data structures a compiler might use to represent types. Suppose that types are represented as pointers to data structures. Furthermore, suppose that I implement type equivalence checking as a simple pointer comparison (e.g. name equivalence). Primitive types like
integer
andfloat
would be stored in some global environment, since there are only a finite number of them. Furthermore, if I compareinteger
withinteger
, they’re guaranteed to be equivalent because they point to the same structure, by virtue of this global environment.However, since
ref
is not a type constructor, not an atomic type, I can use it to create infinitely many types (e.g.ref float
,ref ref float
, etc). So we can’t store them all in a global environment. One easy strategy the compiler can adopt for managing these types is allocate a new structure whenever we encounter a type constructor, we allocate a new data structure for that type. So the instance ofref float
would result in one new data structure, and the other instance ofref float
would result in a completely new, different data structure. The pointer comparison fails, and so they fail to be name equivalent.There’s one more piece to the puzzle, which is what the semantics of your assignment operator are. This type aliasing is a simple pointer copy in the compiler, so if I write
A=B
,A
is always name equivalent toB
. But, to reiterate,F A
is not name equivalent to another instance ofF A
!