Copyright © 2017 Oracle America, Inc. Legal Notice.
This document proposes changes to the Java Language Specification to support inference of the types of local variables. See JEP 286 for an overview.
A synthetic type variable is a type variable introduced by the compiler during capture conversion (5.1.10) or inference variable resolution (18.4).
It is sometimes necessary to find a close supertype of a type, where that supertype does not mention certain synthetic type variables. This is achieved with an upward projection applied to the type.
Similarly, a downward projection may be applied to find a close subtype of a type, where that subtype does not mention certain synthetic type variables. Because such a type does not always exist, downward projection is a partial function.
These operations take as input a set of type variables that should no longer be referenced, referred to as the restricted type variables. When the operations recur, the set of restricted type variables is implicitly passed on to the recursive application.
The upward projection of a type T with respect to a set of restricted type variables is defined as follows:
If T does not mention any restricted type variable, then the result is T.
If T is a restricted type variable, then the result is the upward projection of the upper bound of T.
If T is a parameterized class type or a parameterized interface type, G<A1, ..., An>, then the result is G<A1', ..., An'>, where, for 1 ≤ i ≤ n, Ai' is derived from Ai as follows:
If Ai does not mention any restricted type variable, then Ai' = Ai.
Object
, and if either the declared bound of the ith parameter of G, Bi, mentions a type parameter of G, or Bi is not a subtype of U, then Ai' is an upper-bounded wildcard, ? extends
U.? super
L.?
.If Ai is an upper-bounded wildcard that mentions a restricted type variable, then let U be the upward projection of the wildcard bound. Ai' is a wildcard ? extends
U.
If Ai is a lower-bounded wildcard that mentions a restricted type variable, then if the downward projection of the wildcard bound is L, then Ai' is a wildcard ? super
L; if the downward projection of the wildcard bound is undefined, then Ai' is an unbounded wildcard, ?
.
If T is an array type, S[], then the result is an array type whose component type is the upward projection of S.
If T is an intersection type, then the result is an intersection type. For each element, S, of T, the result has as an element the upward projection of S.
The downward projection of a type T with respect to a set of restricted type variables is a partial function, defined as follows:
If T does not mention any restricted type variable, then the result is T.
If T is a restricted type variable, then if T has a lower bound, and if the downward projection of that bound is L, the result is L; if T has no lower bound, or if the downward projection of that bound is undefined, then the result is undefined.
If T is a parameterized class type or a parameterized interface type, G<A1, ..., An>, then the result is G<A1', ..., An'>, if, for 1 ≤ i ≤ n, a type argument Ai' can be derived from Ai as follows; if not, the result is undefined:
If Ai is does not mention a restricted type variable, then Ai' = Ai.
If Ai is a type that mentions a restricted type variable, then Ai' is undefined.
If Ai is an upper-bounded wildcard that mentions a restricted type variable, then if the downward projection of the wildcard bound is U, then Ai' is a wildcard ? extends
U; if the downward projection of the wildcard bound is undefined, then Ai' is undefined.
If Ai is a lower-bounded wildcard that mentions a restricted type variable, then let L be the upward projection of the wildcard bound. Ai' is a wildcard ? super
L.
If T is an array type, S[], then if the downward projection of S is S', the result is S'[]; if the downward projection of S is undefined, then the result is undefined.
If T is an intersection type, then the result is an intersection type, if, for each element of T, an element of the result can be derived as follows. For each element, S, of T, the result has as an element the downward projection of S.
Like lub (4.10.4), upward projection and downward projection may produce infinite types, due to the recursion on type variable bounds.
A local variable declaration statement declares one or more local variable names.
LocalVariableDeclarationStatement:
LocalVariableDeclaration
;
LocalVariableDeclaration:
{ VariableModifier }
UnannTypeLocalVariableType VariableDeclaratorList
LocalVariableType:
UnannType
var
See 8.3 for UnannType. The following productions from 4.3, 8.4.1, and 8.3 are shown here for convenience:
VariableModifier:
(one of)
Annotationfinal
VariableDeclaratorList:
VariableDeclarator {
,
VariableDeclarator }VariableDeclarator:
VariableDeclaratorId [
=
VariableInitializer ]VariableDeclaratorId:
Identifier [ Dims ]
Dims:
{ Annotation }
[
]
{ { Annotation }[
]
}VariableInitializer:
Expression
ArrayInitializer
Every local variable declaration statement is immediately contained by a block. Local variable declaration statements may be intermixed freely with other kinds of statements in the block.
Apart from local variable declaration statements, a local variable declaration can appear in can be declared by the header of a basic for
statement (14.14 14.14.1), an enhanced for
statement (14.14.2) or a try
-with-resources statement (14.20.3). In these cases, it is executed in the same manner as if it were part of a local variable declaration statement.
The try
-with-resources statement doesn't actually use the LocalVariableDeclaration production, which it appears is what this paragraph was originally meant to assert. But anyway, it's useful to point out that not all local variables are introduced by LocalVariableDeclarationStatement, and be specific about it.
The details of execution vary for these different kinds of statements. Better not to assert anything about their execution here.
The rules for annotation modifiers on a local variable declaration are specified in 9.7.4 and 9.7.5.
It is a compile-time error if final
appears more than once as a modifier for a local variable declaration.
It is a compile-time error if the LocalVariableType is var
and any of the following are true:
Example 14.4-1. Local Variables Declared With
var
The following code illustrates these rules restricting the use of
var
:var a = 1; // good var b = 2, c = 3.0; // Error: multiple declarators var d[] = new int[4]; // Error: extra bracket pairs var e; // Error: no initializer var f = { 6 }; // Error: array initializer var g = (g = 7); // Error: reference in initializer
These restrictions help to avoid confusion about the type being represented by
var
.
Each declarator in a local variable declaration declares one local variable, whose name is the Identifier that appears in the declarator.
If the optional keyword final
appears at the start of the declaration, the variable being declared is a final variable (4.12.4).
The declared type of a local variable is denoted by UnannType if no bracket pairs appear in UnannType and VariableDeclaratorId, and is specified by 10.2 otherwise.
The type of a local variable is determined as follows:
If LocalVariableType is an UnannType and no bracket pairs appear in VariableDeclaratorId, then UnannType denotes the type of the local variable.
If LocalVariableType is an UnannType and bracket pairs do appear in VariableDeclaratorId, then the type of the local variable is specified by 10.2.
If LocalVariableType is var
, then let T be the type of the initializer expression when treated as if it did not appear in an assignment context, and were thus a standalone expression (15.2). The type of the local variable is the upward projection of T with respect to all synthetic type variables mentioned by T (4.10.5).
It is a compile-time error if T is the null type.
Because the initializer is treated as if it did not appear in an assignment context, an error occurs if it is a lambda expression (15.27) or a method reference expression (15.13).
Example 14.4.1-1. Type of Local Variables Declared With
var
The following code illustrates the typing of variables declared with
var
:var a = 1; // a has type 'int' var b = java.util.List.of(1, 2); // b has type 'List<Integer>' var c = "x".getClass(); // c has type 'Class<? extends String>' var d = new Object() {}; // d has the type of the anonymous class var e = (CharSequence & Comparable<String>) "x"; // e has type CharSequence & Comparable<String> var f = () -> "hello"; // Error: lambda not in an assignment context var g = null; // Error: null type
Note that some variables declared with
var
cannot be declared with an explicit type, because the type of the variable is not denotable.
Upward projection is applied to the type of the initializer when determining the type of the variable. If the type of the initializer contains capture variables, this maps the type to a supertype that does not contain capture variables.
While it would be possible to allow the variable's type to mention capture variables, by projecting them away we enforce an attractive invariant that the scope of a capture variable is never larger than the statement containing the expression whose type is captured. Informally, capture variables cannot "leak" into subsequent statements.
A local variable of type float
always contains a value that is an element of the float value set (4.2.3); similarly, a local variable of type double
always contains a value that is an element of the double value set. It is not permitted for a local variable of type float
to contain an element of the float-extended-exponent value set that is not also an element of the float value set, nor for a local variable of type double
to contain an element of the double-extended-exponent value set that is not also an element of the double value set.
The scope and shadowing of a local variable declaration is specified in 6.3 and 6.4.
for
StatementThe enhanced for
statement has the form:
EnhancedForStatement:
for
(
{ VariableModifier } LocalVariableType VariableDeclaratorId
:
Expression)
Statement
EnhancedForStatementNoShortIf:
for
(
{ VariableModifier } LocalVariableType VariableDeclaratorId
:
Expression)
StatementNoShortIf
See 8.3 for UnannType.The following productions from4.3, 8.4.1, and 8.34.3, 8.3, 8.4.1, and 14.4 are shown here for convenience:VariableModifier:
(one of)
Annotationfinal
LocalVariableType:
UnannType
var
VariableDeclaratorId:
Identifier [ Dims ]
Dims:
{ Annotation }
[
]
{ { Annotation }[
]
}
The header of the enhanced for
statement declares a local variable, whose name is the identifier given by VariableDeclaratorId.
If the optional keyword final
appears at the start of the declaration, the variable being declared is a final variable (4.12.4).
It is a compile-time error if the LocalVariableType is var
and the VariableDeclaratorId has one or more bracket pairs.
The local variable is initialized, on each iteration of the loop, by the elements of an array or Iterable
produced by Expression. The type of the Expression must be a subtype of raw Iterable
or an array type (10.1), or a compile-time error occurs.
The declared type of the local variable in the header of the enhanced for
statement is denoted by UnannType if no bracket pairs appear in UnannType and VariableDeclaratorId, and is specified by 10.2 otherwise.
The type of the local variable is determined as follows:
If LocalVariableType is an UnannType and no bracket pairs appear in VariableDeclaratorId, then UnannType denotes the type of the local variable.
If LocalVariableType is an UnannType and bracket pairs do appear in VariableDeclaratorId, then the type of the local variable is specified by 10.2.
If LocalVariableType is var
, then let E be derived from the type of the Expression, as follows:
Object
.The type of the local variable is the upward projection of T with respect to all synthetic type variables mentioned by T (4.10.5).
The scope and shadowing of the local variable declared in the header of an enhanced is specified in 6.3 and 6.4.for
statement
...
The var
keyword can also be used in a basic for
statement (14.14.1), but in that case the grammar is described in terms of LocalVariableDeclaration, so no spec changes are necessary.
try
-With-ResourcesA try
-with-resources statement is parameterized with local variables (known as resources) that are initialized before execution of the try
block and closed automatically, in the reverse order from which they were initialized, after execution of the try
block. catch
clauses and a finally
clause are often unnecessary when resources are closed automatically.
TryWithResourcesStatement:
try
ResourceSpecification Block [ Catches ] [ Finally ]
ResourceSpecification:
(
ResourceList [;
])
ResourceList:
Resource {
;
Resource }
Resource:
{ VariableModifier } LocalVariableType
VariableDeclaratorIdIdentifier=
Expression
See 8.3 for UnannType.The following productions from4.3, 8.4.1, and 8.38.4.1 and 14.4 are shown here for convenience:VariableModifier:
(one of)
Annotationfinal
LocalVariableType:
UnannType
var
VariableDeclaratorId:
Identifier [ Dims ]
Dims:
{ Annotation }[
]
{ { Annotation }[
]
}
A resource's type must be a subtype of AutoCloseable
, so cannot be an array type. Thus, there's no need for the grammar to support bracket pairs following the resource's name. That simplifies some of the text below.
A resource specification declares one or more local variables with initializer expressions to act as resources for the try
statement.
It is a compile-time error for a resource specification to declare two variables with the same name.
It is a compile-time error if final
appears more than once as a modifier for each any variable declared in a resource specification.
A variable declared in a resource specification is implicitly declared final
(4.12.4) if it is not explicitly declared final
.
It is a compile-time error if the LocalVariableType of a variable declared in a resource specification is var
and the initializer of the VariableDeclarator contains a reference to the variable.
The type of a variable declared in a resource specification is determined as follows:
If LocalVariableType is an UnannType, then UnannType denotes the type of the local variable.
If LocalVariableType is var
, then let T be the type of the initializer expression when treated as if it did not appear in an assignment context, and were thus a standalone expression (15.2). The type of the local variable is the upward projection of T with respect to all synthetic type variables mentioned by T (4.10.5).
It is a compile-time error if T is the null type.
The type of a variable declared in a resource specification each variable must be a subtype of AutoCloseable
, or a compile-time error occurs.
The scope and shadowing of a variable declared in a resource specification is specified in 6.3 and 6.4.
Resources are initialized in left-to-right order. If a resource fails to initialize (that is, its initializer expression throws an exception), then all resources initialized so far by the try
-with-resources statement are closed. If all resources initialize successfully, the try
block executes as normal and then all non-null resources of the try
-with-resources statement are closed.
Resources are closed in the reverse order from that in which they were initialized. A resource is closed only if it initialized to a non-null value. An exception from the closing of one resource does not prevent the closing of other resources. Such an exception is suppressed if an exception was thrown previously by an initializer, the try
block, or the closing of a resource.
A try
-with-resources statement whose resource specification declares multiple resources is treated as if it were multiple try
-with-resources statements, each of which has a resource specification that declares a single resource. When a try
-with-resources statement with n resources (n > 1) is translated, the result is a try
-with-resources statement with n-1 resources. After n such translations, there are n nested try
-catch
-finally
statements, and the overall translation is complete.
...
It is possible for an annotation to appear at a syntactic location in a program where it could plausibly apply to a declaration, or a type, or both. This can happen in any of the five declaration contexts where modifiers immediately precede the type of the declared entity:
for
statements and resource variables of try
-with-resources statements)The grammar of the Java programming language unambiguously treats annotations at these locations as modifiers for a declaration (8.3), but that is purely a syntactic matter. Whether an annotation applies to a declaration or to the type of the declared entity—and thus, whether the annotation is a declaration annotation or a type annotation—depends on the applicability of the annotation's type:
If the annotation's type is applicable in the declaration context corresponding to the declaration, and not in type contexts, then the annotation is deemed to apply only to the declaration.
If the annotation's type is applicable in type contexts, and not in the declaration context corresponding to the declaration, then the annotation is deemed to apply only to the type which is closest to the annotation.
If the annotation's type is applicable in the declaration context corresponding to the declaration and in type contexts, then the annotation is deemed to apply to both the declaration and the type which is closest to the annotation.
In the second and third cases above, the type which is closest to the annotation is the type written in source code for the declared entity; if that type is an array type, then the element type is deemed to be closest to the annotation. determined as follows:
If the annotation appears before a void
method declaration or a local variable declared with var
(14.4, 14.14.2, 14.20.3), then there is no closest type. If the annotation's type is applicable in type contexts, and not in the declaration context corresponding to the declaration, a compile-time error occurs.
If the annotation appears before a constructor declaration, then the "closest" type is the type of the newly constructed object. The type of the newly constructed object is the fully qualified name of the type immediately enclosing the constructor declaration. Within that fully qualified name, the annotation applies to the simple type name indicated by the constructor declaration.
In all other cases, the closest type is the type written in source code for the declared entity; if that type is an array type, then the element type is deemed to be closest to the annotation.
For example, in the field declaration
@Foo public static String f;
, the type which is closest to@Foo
isString
. (If the type of the field declaration had been written asjava.lang.String
, thenjava.lang.String
would be the type closest to@Foo
, and later rules would prohibit a type annotation from applying to the package namejava
.) In the generic method declaration@Foo <T> int[] m() {...}
, the type written for the declared entity isint[]
, so@Foo
applies to the element typeint
.
Local variable declarations are similar to formal parameter declarations of lambda expressions, in that both allow declaration annotations and type annotations in source code, but only the type annotations can be stored in the
class
file.
There are two special cases involving method/constructor declarations:
If an annotation appears before a constructor declaration and is deemed to apply to the type which is closest to the annotation, that type is the type of the newly constructed object. The type of the newly constructed object is the fully qualified name of the type immediately enclosing the constructor declaration. Within that fully qualified name, the annotation applies to the simple type name indicated by the constructor declaration.
If an annotation appears before a void
method declaration and is deemed to apply only to the type which is closest to the annotation, a compile-time error occurs.
For the purpose of type annotations, we treat var
the same as void
, its closest analog. In addition, some restructuring here helps to clarify what constitutes a "closest type".
...
The following sections make grammatical changes to the declaration and use of types. These changes make it impossible to declare or reference a type named var
.
An alternative design might allow declarations while prohibiting references, or even allow all references except the unqualified use of var
in a local variable declaration. However, we believe a straightforward, blanket prohibition of types named var
will be easier to understand and better-suited to future language evolution.
As a practical matter, the widespread convention of capitalizing the names of types in Java means these grammatical changes should have very little impact. Existing code that does not follow the convention and declares a class or interface named var
will have to make a binary-incompatible change to the name of the type.
An identifier is an unlimited-length sequence of Java letters and Java digits, the first of which must be a Java letter.
Identifier:
IdentifierChars but not a Keyword or BooleanLiteral or NullLiteral
IdentifierChars:
JavaLetter { JavaLetterOrDigit }
JavaLetter:
any Unicode character that is a "Java letter"
JavaLetterOrDigit:
any Unicode character that is a "Java letter-or-digit"
A "Java letter" is a character for which the method Character.isJavaIdentifierStart(int)
returns true
.
A "Java letter-or-digit" is a character for which the method Character.isJavaIdentifierPart(int)
returns true
.
The "Java letters" include uppercase and lowercase ASCII Latin letters
A
-Z
(\u0041
-\u005a
), anda
-z
(\u0061
-\u007a
), and, for historical reasons, the ASCII underscore (_
, or\u005f
) and dollar sign ($
, or\u0024
). The$
sign should be used only in mechanically generated source code or, rarely, to access pre-existing names on legacy systems.
The "Java digits" include the ASCII digits
0
-9
(\u0030
-\u0039
).
Letters and digits may be drawn from the entire Unicode character set, which supports most writing scripts in use in the world today, including the large sets for Chinese, Japanese, and Korean. This allows programmers to use identifiers in their programs that are written in their native languages.
An identifier cannot have the same spelling (Unicode character sequence) as a keyword (3.9), boolean literal (3.10.3), or the null literal (3.10.7), or a compile-time error occurs.
Two identifiers are the same only if they are identical, that is, have the same Unicode character for each letter or digit. Identifiers that have the same external appearance may yet be different.
For example, the identifiers consisting of the single letters LATIN CAPITAL LETTER A (
A
,\u0041
), LATIN SMALL LETTER A (a
,\u0061
), GREEK CAPITAL LETTER ALPHA (A
,\u0391
), CYRILLIC SMALL LETTER A (a
,\u0430
) and MATHEMATICAL BOLD ITALIC SMALL A (a
,\ud835\udc82
) are all different.
Unicode composite characters are different from their canonical equivalent decomposed characters. For example, a LATIN CAPITAL LETTER A ACUTE (
Á
,\u00c1
) is different from a LATIN CAPITAL LETTER A (A
, 041) immediately followed by a NON-SPACING ACUTE (́
,\u0301
) in identifiers. See The Unicode Standard, Section 3.11 "Normalization Forms".
A type identifier is an identifier that is not the reserved word var
.
TypeIdentifier:
Identifier but not
var
This restricted form of identifier is used in certain contexts involving the declaration or use of types. For example, the name of a class must be a TypeIdentifier, so it is illegal to declare a class named
var
(8.1).
Examples of identifiers are:
String
i3
αρετη
MAX_VALUE
isLetterOrDigit
50 character sequences, formed from ASCII letters, are reserved for use as keywords and cannot be used as identifiers (3.8).
Keyword:
(one of)
abstract continue for new switch assert default if package synchronized boolean do goto private this break double implements protected throw byte else import public throws case enum instanceof return transient catch extends int short try char final interface static void class finally long strictfp volatile const float native super while
The keywords
const
andgoto
are reserved, even though they are not currently used. This may allow a Java compiler to produce better error messages if these C++ keywords incorrectly appear in programs.
While
true
andfalse
might appear to be keywords, they are technically boolean literals (3.10.3). Similarly, whilenull
might appear to be a keyword, it is technically the null literal (3.10.7).
The character sequence
var
is generally treated as an identifier, but in certain special cases acts as if it were a keyword instead. Syntactic ambiguities are avoided by making use of the TypeIdentifier production (3.8).
...
ClassType:
{ Annotation } TypeIdentifier [ TypeArguments ]
PackageName.
{ Annotation } TypeIdentifier [ TypeArguments ]
ClassOrInterfaceType.
{ Annotation } TypeIdentifier [ TypeArguments ]
InterfaceType:
ClassType
TypeVariable
{ Annotation } TypeIdentifier
...
This section needs some cleanup, a separate effort. For now, we want to force TypeIdentifiers in the right places, but don't want to prohibit qualified names like var.Foo
. A lightweight workaround is to add an ambiguous variant pointing to PackageName.
A type variable is an unqualified identifier used as a type in class, interface, method, and constructor bodies.
A type variable is introduced by the declaration of a type parameter of a generic class, interface, method, or constructor (8.1.2, 9.1.2, 8.4.4, 8.8.4).
TypeParameter:
{ TypeParameterModifier } TypeIdentifier [ TypeBound ]
TypeParameterModifier:
Annotation
TypeBound:
extends
TypeVariable
extends
ClassOrInterfaceType { AdditionalBound }
AdditionalBound:
&
InterfaceType
...
A declaration introduces an entity into a program and includes an identifier (3.8) that can be used in a name to refer to this entity. The identifier is constrained to be a type identifier when the entity being introduced is a class, interface, or type parameter.
A declared entity is one of the following:
...
The meaning of a name depends on the context in which it is used. The determination of the meaning of a name requires three steps:
First, context causes a name syntactically to fall into one of six categories: PackageName, TypeName, ExpressionName, MethodName, PackageOrTypeName, or AmbiguousName.
Second, a name that is initially classified by its context as an AmbiguousName or as a PackageOrTypeName is then reclassified to be a PackageName, TypeName, or ExpressionName.
Third, the resulting category then dictates the final determination of the meaning of the name (or a compile-time error if the name has no meaning).
PackageName:
Identifier
PackageName.
Identifier
TypeName:
TypeIdentifier
PackageOrTypeName.
TypeIdentifier
PackageOrTypeName:
Identifier
PackageOrTypeName.
Identifier
ExpressionName:
Identifier
AmbiguousName.
Identifier
MethodName:
Identifier
AmbiguousName:
Identifier AmbiguousName
.
Identifier
A TypeName has a restricted form: the identifier must be a TypeIdentifier, which excludes the reserved word var
(3.8).
The use of context helps to minimize name conflicts between entities of different kinds. Such conflicts will be rare if the naming conventions described in 6.1 are followed. Nevertheless, conflicts may arise unintentionally as types developed by different programmers or different organizations evolve. For example, types, methods, and fields may have the same name. It is always possible to distinguish between a method and a field with the same name, since the context of a use always tells whether a method is intended.
An AmbiguousName is then reclassified as follows.
If the AmbiguousName is a simple name, consisting of a single Identifier:
If the Identifier appears within the scope (6.3) of a local variable declaration (14.4) or parameter declaration (8.4.1, 8.8.1, 14.20) or field declaration (8.3) with that name, then the AmbiguousName is reclassified as an ExpressionName.
Otherwise, if a field of that name is declared in the compilation unit (7.3) containing the Identifier by a single-static-import declaration (7.5.3), or by a static-import-on-demand declaration (7.5.4) then the AmbiguousName is reclassified as an ExpressionName.
Otherwise, if the Identifier is a valid TypeIdentifier and appears within the scope (6.3) of a top level class (8) or interface type declaration (9), a local class declaration (14.3) or member type declaration (8.5, 9.5) with that name, then the AmbiguousName is reclassified as a TypeName.
Otherwise, if the Identifier is a valid TypeIdentifier and a type of that name is declared in the compilation unit (7.3) containing the Identifier, either by a single-type-import declaration (7.5.1), or by a type-import-on-demand declaration (7.5.2), or by a single-static-import declaration (7.5.3), or by a static-import-on-demand declaration (7.5.4), then the AmbiguousName is reclassified as a TypeName.
Otherwise, the AmbiguousName is reclassified as a PackageName. A later step determines whether or not a package of that name actually exists.
If the AmbiguousName is a qualified name, consisting of a name, a ".
", and an Identifier, then the name to the left of the ".
" is first reclassified, for it is itself an AmbiguousName. There is then a choice:
If the name to the left of the ".
" is reclassified as a PackageName, then:
.
", and that package contains a declaration of a type whose name is the same as the Identifier, then this AmbiguousName is reclassified as a TypeName.– Otherwise, this AmbiguousName is reclassified as a PackageName. A later step determines whether or not a package of that name actually exists.
If the name to the left of the ".
" is reclassified as a TypeName, then:
– If the Identifier is the name of a method or field of the type denoted by TypeName, this AmbiguousName is reclassified as an ExpressionName.
– Otherwise, if the Identifier is a valid TypeIdentifier and is the name of a member type of the type denoted by TypeName, this AmbiguousName is reclassified as a TypeName.
– Otherwise, a compile-time error occurs.
If the name to the left of the ".
" is reclassified as an ExpressionName, then let T be the type of the expression denoted by ExpressionName.
– If the Identifier is the name of a method or field of the type denoted by T, this AmbiguousName is reclassified as an ExpressionName.
Otherwise, if the Identifier is the name of a member type (8.5, 9.5) of the type denoted by T, then this AmbiguousName is reclassified as a TypeName.
This is a bug: referring to a member type qualified by a field name is not allowed.
– Otherwise, a compile-time error occurs.
...
The requirement that a potential type name be "a valid TypeIdentifier" prevents treating var
as a type name. It is usually redundant, because the rules for declarations already prevent the introduction of types named var
. However, in some cases, a compiler may find a binary class named var
, and we want to be clear that such classes can never be named. The simplest solution is to consistently check for a valid TypeIdentifier.
If the PackageOrTypeName, Q, is a valid TypeIdentifier and occurs in the scope of a type named Q, then the PackageOrTypeName is reclassified as a TypeName.
Otherwise, the PackageOrTypeName is reclassified as a PackageName. The meaning of the PackageOrTypeName is the meaning of the reclassified name.
Given a qualified PackageOrTypeName of the form Q.
Id, if Id is a valid TypeIdentifier and the type or package denoted by Q has a member type named Id, then the qualified PackageOrTypeName name is reclassified as a TypeName.
Otherwise, it is reclassified as a PackageName. The meaning of the qualified PackageOrTypeName is the meaning of the reclassified name.
A class declaration specifies a new named reference type.
There are two kinds of class declarations: normal class declarations and enum declarations.
ClassDeclaration:
NormalClassDeclaration
EnumDeclaration
NormalClassDeclaration:
{ ClassModifier }
class
TypeIdentifier [ TypeParameters ][ Superclass ] [ Superinterfaces ] ClassBody
The rules in this section apply to all class declarations, including enum declarations. However, special rules apply to enum declarations with regard to class modifiers, inner classes, and superclasses; these rules are stated in 8.9.
The TypeIdentifier in a class declaration specifies the name of the class.
It is a compile-time error if a class has the same simple name as any of its enclosing classes or interfaces.
The scope and shadowing of a class declaration is specified in 6.3 and 6.4.
...
UnannClassType:
TypeIdentifier [ TypeArguments ]
PackageName.
{ Annotation } TypeIdentifier [ TypeArguments ]
UnannClassOrInterfaceType.
{ Annotation } TypeIdentifier [ TypeArguments ]
UnannInterfaceType:
UnannClassType
UnannTypeVariable:
TypeIdentifier
...
A constructor is used in the creation of an object that is an instance of a class (12.5, 15.9).
ConstructorDeclaration:
{ ConstructorModifier } ConstructorDeclarator [ Throws ] ConstructorBody
ConstructorDeclarator:
[ TypeParameters ]
SimpleTypeNameTypeIdentifier(
[ FormalParameterList ])
SimpleTypeName:
Identifier
The rules in this section apply to constructors in all class declarations, including enum declarations. However, special rules apply to enum declarations with regard to constructor modifiers, constructor bodies, and default constructors; these rules are stated in 8.9.2.
The SimpleTypeName TypeIdentifier in the ConstructorDeclarator must be the simple name of the class that contains the constructor declaration, or a compile-time error occurs.
In all other respects, a constructor declaration looks just like a method declaration that has no result (8.4.5).
...
The SimpleTypeName production seems to serve no purpose, and is unused outside of this section. It's possible the intent was to treat this identifier as a restricted form of TypeName. But identifiers used in declarations are not usually considered names (see 6.2).
An enum declaration specifies a new enum type, a special kind of class type.
EnumDeclaration:
{ ClassModifier }
enum
TypeIdentifier [ Superinterfaces ] EnumBody
...
An interface declaration specifies a new named reference type. There are two kinds of interface declarations—normal interface declarations and annotation type declarations (9.6).
InterfaceDeclaration:
NormalInterfaceDeclaration
AnnotationTypeDeclaration
NormalInterfaceDeclaration:
{ InterfaceModifier }
interface
TypeIdentifier [ TypeParameters ][ ExtendsInterfaces ] InterfaceBody
The TypeIdentifier in an interface declaration specifies the name of the interface.
It is a compile-time error if an interface has the same simple name as any of its enclosing classes or interfaces.
The scope and shadowing of an interface declaration is specified in 6.3 and 6.4.
An annotation type declaration specifies a new annotation type, a special kind of interface type. To distinguish an annotation type declaration from a normal interface declaration, the keyword interface
is preceded by an at-sign (@
).
AnnotationTypeDeclaration:
{ InterfaceModifier }
@
interface
TypeIdentifier AnnotationTypeBody
Note that the at-sign (
@
) and the keywordinterface
are distinct tokens. It is possible to separate them with whitespace, but this is discouraged as a matter of style.
The rules for annotation modifiers on an annotation type declaration are specified in 9.7.4 and 9.7.5.
The TypeIdentifier in an annotation type declaration specifies the name of the annotation type.
...
Copyright © 2017 Oracle America, Inc. 4150 Network Circle, Santa Clara, California 95054, U.S.A. All rights reserved.
The Specification is protected by copyright and the information described therein may be protected by one or more U.S. patents, foreign patents, or pending applications. Except as provided under the following license, no part of the Specification may be reproduced in any form by any means without the prior written authorization of Oracle America, Inc. ("Oracle") and its licensors, if any. Any use of the Specification and the information described therein will be governed by the terms and conditions of this Agreement.
Subject to the terms and conditions of this license, including your compliance with Paragraphs 1 and 2 below, Oracle hereby grants you a fully-paid, non-exclusive, non-transferable, limited license (without the right to sublicense) under Oracle's intellectual property rights to:
Review the Specification for the purposes of evaluation. This includes: (i) developing implementations of the Specification for your internal, non-commercial use; (ii) discussing the Specification with any third party; and (iii) excerpting brief portions of the Specification in oral or written communications which discuss the Specification provided that such excerpts do not in the aggregate constitute a significant portion of the Technology.
Distribute implementations of the Specification to third parties for their testing and evaluation use, provided that any such implementation:
does not modify, subset, superset or otherwise extend the Licensor Name Space, or include any public or protected packages, classes, Java interfaces, fields or methods within the Licensor Name Space other than those required/authorized by the Specification or Specifications being implemented;
is clearly and prominently marked with the word "UNTESTED" or "EARLY ACCESS" or "INCOMPATIBLE" or "UNSTABLE" or "BETA" in any list of available builds and in proximity to every link initiating its download, where the list or link is under Licensee's control; and
includes the following notice: "This is an implementation of an early-draft specification developed under the Java Community Process (JCP) and is made available for testing and evaluation purposes only. The code is not compatible with any specification of the JCP."
The grant set forth above concerning your distribution of implementations of the specification is contingent upon your agreement to terminate development and distribution of your "early draft" implementation as soon as feasible following final completion of the specification. If you fail to do so, the foregoing grant shall be considered null and void.
No provision of this Agreement shall be understood to restrict your ability to make and distribute to third parties applications written to the Specification.
Other than this limited license, you acquire no right, title or interest in or to the Specification or any other Oracle intellectual property, and the Specification may only be used in accordance with the license terms set forth herein. This license will expire on the earlier of: (a) two (2) years from the date of Release listed above; (b) the date on which the final version of the Specification is publicly released; or (c) the date on which the Java Specification Request (JSR) to which the Specification corresponds is withdrawn. In addition, this license will terminate immediately without notice from Oracle if you fail to comply with any provision of this license. Upon termination, you must cease use of or destroy the Specification.
"Licensor Name Space" means the public class or interface declarations whose names begin with "java", "javax", "com.oracle" or their equivalents in any subsequent naming convention adopted by Oracle through the Java Community Process, or any recognized successors or replacements thereof.
No right, title, or interest in or to any trademarks, service marks, or trade names of Oracle or Oracle's licensors is granted hereunder. Oracle, the Oracle logo, Java are trademarks or registered trademarks of Oracle USA, Inc. in the U.S. and other countries.
THE SPECIFICATION IS PROVIDED "AS IS" AND IS EXPERIMENTAL AND MAY CONTAIN DEFECTS OR DEFICIENCIES WHICH CANNOT OR WILL NOT BE CORRECTED BY ORACLE. ORACLE MAKES NO REPRESENTATIONS OR WARRANTIES, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO, WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE, OR NON-INFRINGEMENT THAT THE CONTENTS OF THE SPECIFICATION ARE SUITABLE FOR ANY PURPOSE OR THAT ANY PRACTICE OR IMPLEMENTATION OF SUCH CONTENTS WILL NOT INFRINGE ANY THIRD PARTY PATENTS, COPYRIGHTS, TRADE SECRETS OR OTHER RIGHTS. This document does not represent any commitment to release or implement any portion of the Specification in any product.
THE SPECIFICATION COULD INCLUDE TECHNICAL INACCURACIES OR TYPOGRAPHICAL ERRORS. CHANGES ARE PERIODICALLY ADDED TO THE INFORMATION THEREIN; THESE CHANGES WILL BE INCORPORATED INTO NEW VERSIONS OF THE SPECIFICATION, IF ANY. ORACLE MAY MAKE IMPROVEMENTS AND/OR CHANGES TO THE PRODUCT(S) AND/OR THE PROGRAM(S) DESCRIBED IN THE SPECIFICATION AT ANY TIME. Any use of such changes in the Specification will be governed by the then-current license for the applicable version of the Specification.
TO THE EXTENT NOT PROHIBITED BY LAW, IN NO EVENT WILL ORACLE OR ITS LICENSORS BE LIABLE FOR ANY DAMAGES, INCLUDING WITHOUT LIMITATION, LOST REVENUE, PROFITS OR DATA, OR FOR SPECIAL, INDIRECT, CONSEQUENTIAL, INCIDENTAL OR PUNITIVE DAMAGES, HOWEVER CAUSED AND REGARDLESS OF THE THEORY OF LIABILITY, ARISING OUT OF OR RELATED TO ANY FURNISHING, PRACTICING, MODIFYING OR ANY USE OF THE SPECIFICATION, EVEN IF ORACLE AND/OR ITS LICENSORS HAVE BEEN ADVISED OF THE POSSIBILITY OF SUCH DAMAGES.
You will hold Oracle (and its licensors) harmless from any claims based on your use of the Specification for any purposes other than the limited right of evaluation as described above, and from any claims that later versions or releases of any Specification furnished to you are incompatible with the Specification provided to you under this license.
If this Software is being acquired by or on behalf of the U.S. Government or by a U.S. Government prime contractor or subcontractor (at any tier), then the Government's rights in the Software and accompanying documentation shall be only as set forth in this license; this is in accordance with 48 C.F.R. 227.7201 through 227.7202-4 (for Department of Defense (DoD) acquisitions) and with 48 C.F.R. 2.101 and 12.212 (for non-DoD acquisitions).
You may wish to report any ambiguities, inconsistencies or inaccuracies you may find in connection with your evaluation of the Specification ("Feedback"). To the extent that you provide Oracle with any Feedback, you hereby: (i) agree that such Feedback is provided on a non-proprietary and non-confidential basis, and (ii) grant Oracle a perpetual, non-exclusive, worldwide, fully paid-up, irrevocable license, with the right to sublicense through multiple levels of sublicensees, to incorporate, disclose, and use without limitation the Feedback for any purpose related to the Specification and future versions, implementations, and test suites thereof.
Any action related to this Agreement will be governed by California law and controlling U.S. federal law. The U.N. Convention for the International Sale of Goods and the choice of law rules of any jurisdiction will not apply.
The Specification is subject to U.S. export control laws and may be subject to export or import regulations in other countries. Licensee agrees to comply strictly with all such laws and regulations and acknowledges that it has the responsibility to obtain such licenses to export, re-export or import as may be required after delivery to Licensee.
This Agreement is the parties' entire agreement relating to its subject matter. It supersedes all prior or contemporaneous oral or written communications, proposals, conditions, representations and warranties and prevails over any conflicting or additional terms of any quote, order, acknowledgment, or other communication between the parties relating to its subject matter during the term of this Agreement. No modification to this Agreement will be binding, unless in writing and signed by an authorized representative of each party.