Visual Basic Language Specification 10.0
Visual Basic Language Specification 10.0
Version 10.0
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved. Please send corrections, comments, and other feedback to basic@microsoft.com.
Filename: 88281906.docx 2
The information contained in this document represents the current view of Microsoft Corporation on the issues discussed as of the date of publication. Because Microsoft must respond to changing market conditions, it should not be interpreted to be a commitment on the part of Microsoft, and Microsoft cannot guarantee the accuracy of any information presented after the date of publication. This Language Specification is for informational purposes only. MICROSOFT MAKES NO WARRANTIES, EXPRESS, IMPLIED OR STATUTORY, AS TO THE INFORMATION IN THIS DOCUMENT. Complying with all applicable copyright laws is the responsibility of the user. Without limiting the rights under copyright, no part of this document may be reproduced, stored in or introduced into a retrieval system, or transmitted in any form or by any means (electronic, mechanical, photocopying, recording, or otherwise), or for any purpose, without the express written permission of Microsoft Corporation. Microsoft may have patents, patent applications, trademarks, copyrights, or other intellectual property rights covering subject matter in this document. Except as expressly provided in any written license agreement from Microsoft, the furnishing of this document does not give you any license to these patents, trademarks, copyrights, or other intellectual property. Unless otherwise noted, the example companies, organizations, products, domain names, e-mail addresses, logos, people, places and events depicted herein are fictitious, and no association with any real company, organization, product, domain name, email address, logo, person, place or event is intended or should be inferred. 2010Microsoft Corporation. All rights reserved. Microsoft, MS-DOS, Visual Basic, Windows 2000, Windows 95, Windows 98, Windows ME, Windows NT, Windows XP, Windows Vista and Windows are either registered trademarks or trademarks of Microsoft Corporation in the United States and/or other countries.
Filename: 88281906.docx 3
The names of actual companies and products mentioned herein may be the trademarks of their respective owners.
Table of Contents
The Microsoft Visual Basic Language Specification....................................................................................1 Version 10.0....................................................................................................................1 Paul Vick, Lucian Wischik................................................................................................1 Microsoft Corporation......................................................................................................1 Introduction....................................................................................................1 Introduction....................................................................................................1 1.1 Grammar Notation..................................................................................................... 1 1.2 Compatibility.............................................................................................................. 2 Lexical Grammar..............................................................................................7 Lexical Grammar..............................................................................................7 1.3 1.4 1.5 1.6 1.7 1.8 Characters and Lines..................................................................................................7 Identifiers.................................................................................................................11 Keywords.................................................................................................................14 Literals.....................................................................................................................16 Separators............................................................................................................... 23 Operator Characters.................................................................................................23
Preprocessing Directives................................................................................24 Preprocessing Directives................................................................................24 1.9 Conditional Compilation...........................................................................................24 1.10 External Source Directives.....................................................................................28 1.11 Region Directives...................................................................................................29 1.12 External Checksum Directives................................................................................30 General Concepts...........................................................................................32 General Concepts...........................................................................................32 1.13 Declarations...........................................................................................................32 1.14 Scope..................................................................................................................... 35 1.15 Inheritance.............................................................................................................37
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
Table of Contents
1.16 Implementation......................................................................................................52 1.17 Polymorphism........................................................................................................ 59 1.18 Accessibility........................................................................................................... 67 1.19 Type and Namespace Names.................................................................................72 1.20 Variables................................................................................................................ 77 1.21 Generic Types and Methods...................................................................................77 Attributes......................................................................................................94 Attributes......................................................................................................94 1.22 Attribute Classes....................................................................................................95 1.23 Attribute Blocks......................................................................................................98 Source Files and Namespaces.......................................................................105 Source Files and Namespaces.......................................................................105 1.24 Program Startup and Termination........................................................................106 1.25 Compilation Options.............................................................................................106 1.26 Imports Statement...............................................................................................110 1.27 Namespaces.........................................................................................................118 Types..........................................................................................................122 Types..........................................................................................................122 1.28 Value Types and Reference Types........................................................................122 1.29 Interface Implementation.....................................................................................125 1.30 Primitive Types.....................................................................................................128 1.31 Enumerations....................................................................................................... 129 1.32 Classes................................................................................................................. 133 1.33 Structures............................................................................................................ 137 1.34 Standard Modules................................................................................................ 140 1.35 Interfaces............................................................................................................. 142 1.36 Arrays.................................................................................................................. 147 1.37 Delegates............................................................................................................. 150 1.38 Partial types......................................................................................................... 152 Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved. ii
Table of Contents
1.39 Constructed Types............................................................................................... 155 1.40 Special Types....................................................................................................... 158 Conversions.................................................................................................159 Conversions.................................................................................................159 1.41 Implicit and Explicit Conversions..........................................................................159 1.42 Boolean Conversions............................................................................................160 1.43 Numeric Conversions............................................................................................160 1.44 Reference Conversions.........................................................................................162 1.45 Array Conversions................................................................................................165 1.46 Value Type Conversions.......................................................................................167 1.47 String Conversions...............................................................................................174 1.48 Widening Conversions..........................................................................................175 1.49 Narrowing Conversions.........................................................................................178 1.50 Type Parameter Conversions................................................................................181 1.51 User-Defined Conversions....................................................................................182 1.52 Native Conversions.............................................................................................. 187 1.53 Dominant Type.....................................................................................................187 Type Members.............................................................................................188 Type Members.............................................................................................188 1.54 Interface Method Implementation.........................................................................188 1.55 Methods............................................................................................................... 192 1.56 Constructors.........................................................................................................221 1.57 Events..................................................................................................................228 1.58 Constants............................................................................................................. 235 1.59 Instance and Shared Variables.............................................................................237 1.60 Properties............................................................................................................. 251 1.61 Operators............................................................................................................. 267 Statements..................................................................................................276 Statements..................................................................................................276
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
iii
Table of Contents
1.62 Blocks and Labels.................................................................................................276 1.63 Local Declaration Statements...............................................................................281 1.64 With Statement.................................................................................................... 286 1.65 SyncLock Statement.............................................................................................287 1.66 Event Statements.................................................................................................289 1.67 Assignment Statements.......................................................................................292 1.68 Invocation Statements.........................................................................................298 1.69 Conditional Statements........................................................................................298 1.70 Loop Statements..................................................................................................302 1.71 Exception-Handling Statements...........................................................................312 1.72 Branch Statements...............................................................................................320 1.73 Array-Handling Statements..................................................................................321 1.74 Using statement...................................................................................................324 Expressions.................................................................................................327 Expressions.................................................................................................327 1.75 Expression Classifications.....................................................................................327 1.76 Constant Expressions...........................................................................................333 1.77 Late-Bound Expressions.......................................................................................334 1.78 Simple Expressions.............................................................................................. 337 1.79 Type Expressions.................................................................................................343 1.80 Member Access Expressions.................................................................................347 1.81 Dictionary Member Access Expressions................................................................364 1.82 Invocation Expressions.........................................................................................364 1.83 Index Expressions................................................................................................387 1.84 New Expressions.................................................................................................. 388 1.85 Cast Expressions..................................................................................................406 1.86 Operator Expressions...........................................................................................408 1.87 Arithmetic Operators............................................................................................414 1.88 Relational Operators.............................................................................................426 1.89 Like Operator....................................................................................................... 429 1.90 Concatenation Operator.......................................................................................431
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
iv
Table of Contents
1.91 Logical Operators.................................................................................................433 1.92 Shift Operators.....................................................................................................439 1.93 Boolean Expressions............................................................................................440 1.94 Lambda Expressions.............................................................................................442 1.95 Query Expressions................................................................................................450 1.96 Conditional Expressions........................................................................................475 1.97 XML Literal Expressions........................................................................................476 1.98 XML Member Access Expressions.........................................................................489 Documentation Comments............................................................................492 Documentation Comments............................................................................492 1.99 Documentation Comment Format........................................................................492 1.100 Recommended tags...........................................................................................493 1.101 ID Strings...........................................................................................................504 1.102 Documentation comments example...................................................................510 Grammar Summary......................................................................................520 Grammar Summary......................................................................................520 1.103 Lexical Grammar................................................................................................520 1.104 Preprocessing Directives....................................................................................525 1.105 Syntactic Grammar............................................................................................528 Change List..................................................................................................568 Change List..................................................................................................568 1.106 Version 7.1 to Version 8.0..................................................................................568 1.107 Version 8.0 to Version 8.0 (2nd Edition)..............................................................575 1.108 Version 8.0 (2nd Edition) to Version 9.0..............................................................577 1.109 Version 9.0 to Version 10.0................................................................................580
Introduction
The Microsoft Visual Basic programming language is a high-level programming language for the Microsoft .NET Framework. Although it is designed to be an approachable and easyto-learn language, it is also powerful enough to satisfy the needs of experienced programmers. The Visual Basic programming language has a syntax that is similar to English, which promotes the clarity and readability of Visual Basic code. Wherever possible, meaningful words or phrases are used instead of abbreviations, acronyms, or special characters. Extraneous or unneeded syntax is generally allowed but not required. The Visual Basic programming language can be either a strongly typed or a loosely typed language. Loose typing defers much of the burden of type checking until a program is already running. This includes not only type checking of conversions but also of method calls, meaning that the binding of a method call can be deferred until run-time. This is useful when building prototypes or other programs in which speed of development is more important than execution speed. The Visual Basic programming language also provides strongly typed semantics that performs all type checking at compile-time and disallows run-time binding of method calls. This guarantees maximum performance and helps ensure that type conversions are correct. This is useful when building production applications in which speed of execution and execution correctness is important. This document describes the Visual Basic language. It is meant to be a complete language description rather than a language tutorial or a user's reference manual.
4.3
Inheritance
Case is unimportant in Visual Basic programs. For simplicity, all terminals will be given in standard casing, but any casing will match them. Terminals that are printable elements of the ASCII character set are represented by their corresponding ASCII characters. Visual Basic is also width insensitive when matching terminals, allowing full-width Unicode characters to match their half-width Unicode equivalents, but only on a whole-token basis. A token will not match if it contains mixed half-width and full-width characters. A set of productions begins with the name of a nonterminal, followed by two colons and an equal sign. The right side contains a terminal or nonterminal production. A nonterminal may have multiple productions that are separated by the vertical-bar metasymbol (|). Items included in square-bracket metasymbols ([]) are optional. A plus metasymbol (+) following an item means the item may occur one or more times. Line breaks and indentation may be added for readability and are not part of the production.
1.2 Compatibility
An important feature of a programming language is compatibility between different versions of the language. If a newer version of a language does not accept the same code as a previous version of the language, or interprets it differently than the previous version, then a burden can be placed on a programmer when upgrading his code from one version of the language to another. As such, compatibility between versions must be preserved except when the benefit to language consumers is of a clear and overwhelming nature. The following policy governs changes to the Visual Basic language between versions. The term language, when used in this context, refers only to the syntactic and semantic aspects of the Visual Basic language itself and does not include any .NET Framework classes included as a part of the Microsof t .V i sua lBas i c namespace (and sub-namespaces). All classes in the .NET Framework are covered by a separate versioning and compatibility policy outside the scope of this document. 1.2.1 Kinds of compatibility breaks In an ideal world, compatibility would be 100% between the existing version of Visual Basic and all future versions of Visual Basic. However, there may be situations where the need
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
4.3
Inheritance
for a compatibility break may outweigh the cost it may impose on programmers. Such situations are: New warnings. Introducing a new warning is not, per se, a compatibility break. However, because many developers compile with treat warnings as errors turned on, extra care must be taken when introducing warnings. New keywords. Introducing new keywords may be necessary when introducing new language features. Reasonable efforts will be made to choose keywords that minimize the possibility of collision with users identifiers and to use existing keywords where it makes sense. Help will be provided to upgrade projects from previous versions and escape any new keywords. Compiler bugs. When the compilers behavior is at odds with a documented behavior in the language specification, fixing the compiler behavior to match the documented behavior may be necessary. Specification bug. When the compiler is consistent with the language specification but the language specification is clearly wrong, changing the language specification and the compiler behavior may be necessary. The phrase clearly wrong means that the documented behavior runs counter to what a clear and unambiguous majority of users would expect and produces highly undesirable behavior for users. Specification ambiguity. When the language specification should spell out what happens in a particular situation but doesnt, and the compiler handles the situation in a way that is either inconsistent or clearly wrong (using the same definition from the previous point), clarifying the specification and correcting the compiler behavior may be necessary. In other words, when the specification covers cases a, b, d and e, but omits any mention of what happens in case c, and the compiler behaves incorrectly in case c, it may be necessary to document what happens in case c and change the behavior of the compiler to match. (Note that if the specification was ambiguous as to what happens in a situation and the compiler behaves in a manner that is not clearly wrong, the compiler behavior becomes the de facto specification.)
4.3
Inheritance
Making run-time errors into compile-time errors. In a situation where code is 100% guaranteed to fail at runtime (i.e. the user code has an unambiguous bug in it), it may be desirable to add a compile-time error that catches the situation. Specification omission. When the language specification does not specifically allow or disallow a particular situation and the compiler handles the situation in a way that is undesirable (if the compiler behavior was clearly wrong, it would a specification bug, not a specification omission), it may be necessary to clarify the specification and change the compiler behavior. In addition to the usual impact analysis, changes of this kind are further restricted to cases where the impact of the change is considered to be extremely minimal and the benefit to developers is very high. New features. In general, introducing new features should not change existing parts of the language specification or the existing behavior of the compiler. In the situation where introducing a new feature requires changing the existing language specification, such a compatibility break is reasonable only if the impact would be extremely minimal and the benefit of the feature is high. Security. In extraordinary situations, security concerns may necessitate a compatibility break, such as removing or modifying a feature that is inherently insecure and poses a clear security risk for users. The following situations are not acceptable reasons for introducing compatibility breaks: Undesirable or regrettable behavior. Language design or compiler behavior which is reasonable but considered undesirable or regrettable in retrospect is not a justification for breaking backward compatibility. The language deprecation process, covered below, must be used instead. Anything else. Otherwise, compiler behavior remains backwards compatible. 1.2.2 Impact Criteria When considering whether a compatibility break might be acceptable, several criteria are used to determine what the impact of the change might be. The greater the impact, the higher the bar for accepting the compatibility breaks. The criteria are:
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
4.3
Inheritance
What is the scope of the change? In other words, how many programs are likely to be affected? How many users are likely to be affected? How common will it be to write code that is affected by the change? Do any workarounds exist to get the same behavior prior to the change? How obvious is the change? Will users get immediate feedback that something has changed, or will their programs just execute differently? Can the change be reasonably addressed during upgrade? Is it possible to write a tool that can find the situation in which the change occurs with perfect accuracy and change the code to work around the change? What is the community feedback on the change? 1.2.3 Language deprecation Over time, parts of the language or compiler may become deprecated. As discussed previously, it is not acceptable to break compatibility to remove such deprecated features. Instead, the following steps must be followed: Given a feature that exists in version A of Visual Studio, feedback must be solicited from the user community on deprecation of the feature and full notice given before any final deprecation decision is made. The deprecation process may be reversed or abandoned at any point based on user community feedback. A full version (i.e. not a point release) B of Visual Studio must be released with compiler warnings that warn of deprecated usage. The warnings must be on by default and can be turned off. The deprecations must be clearly documented in the product documentation and on the web. A full version C of Visual Studio must be released with compiler warnings that cannot be turned off. A full version D of Visual Studio must subsequently be released with the deprecated compiler warnings converted into compiler errors. The release of D must occur after the end of the Mainstream Support Phase (5 years as of this writing) of release A. Finally, a version E of Visual Studio may be released that removes the compiler errors.
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
4.3
Inheritance
Changes that cannot be handled within this deprecation framework will not be allowed.
Lexical Grammar
Compilation of a Visual Basic program first involves translating the raw stream of Unicode characters into an ordered set of lexical tokens. Because the Visual Basic language is not free-format, the set of tokens is then further divided into a series of logical lines. A logical line spans from either the start of the stream or a line terminator through to the next line terminator that is not preceded by a line continuation or through to the end of the stream. Note With the introduction of XML literal expressions in version 9.0 of the language, Visual Basic no longer has a distinct lexical grammar in the sense that Visual Basic code can be tokenized without regard to the syntactic context. This is due to the fact that XML and Visual Basic have different lexical rules and the set of lexical rules in use at any particular time depends on what syntactic construct is being processed at that moment. This specification retains this lexical grammar section as a guide to the lexical rules of regular Visual Basic code. However, long term the lexical rules will likely be merged into the syntactic rules. Start ::= [ LogicalLine+ ] LogicalLine ::= [ LogicalLineElement+ ] [ Comment ] LineTerminator LogicalLineElement ::= WhiteSpace | LineContinuation | Token Token ::= Identifier | Keyword | Literal | Separator | Operator
4.3
Inheritance
text line. A line continuation allows a logical line to span more than one physical line. Line continuations are treated as if they were white space, even though they are not. The following program shows some line continuations:
Module Test Sub Print( _ Param1 As Integer, _ Param2 As Integer ) If (Param1 < Param2) Or _ (Param1 > Param2) Then Console.WriteLine("Not equal") End If End Function End Module
Some places in the syntactic grammar allow for implicit line continuations. When a line terminator is encountered: after a comma (,), open parenthesis ((), open curly brace ({), or open embedded expression (<%=) after a member qualifier (. or .@ or ...), provided that something is being qualified (i.e. is not using an implicit With context) before a close parenthesis ()), close curly brace (}), or close embedded expression (%>) after a less-than (<) in an attribute context before a greater-than (>) in an attribute context after a greater-than (>) in a non-file-level attribute context before and after query operators (Where, Order, Select, etc.) after binary operators (+, -, /, *, etc.) in an expression context after assignment operators (=, :=, +=, -=, etc.) in any context.
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
4.3
Inheritance
the line terminator is treated as if it was a line continuation. For example, the previous example could also be written as:
Module Test Sub Print( Param1 As Integer, Param2 As Integer) If (Param1 < Param2) Or (Param1 > Param2) Then Console.WriteLine("Not equal") End If End Function End Module
Implicit line continuations will only ever be inferred directly before or after the specified token. They will not be inferred before or after a line continuation. For example:
Dim y = 10 ' Error: Expression expected for assignment to x Dim x = _ y
Line continuations will not be inferred in conditional compilation contexts. Annotation This last restriction is required because text in conditional compilation blocks that are not compiled do not have to be syntactically valid. Thus, text in the block might accidentally get picked up by the conditional compilation statement, especially as the language gets extended in the future. LineContinuation ::= WhiteSpace _ [ WhiteSpace+ ] LineTerminator Comma ::= , [ LineTerminator ] Period ::= . [ LineTerminator ] OpenParenthesis ::= ( [ LineTerminator ]
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
4.3
Inheritance
CloseParenthesis ::= [ LineTerminator ] ) OpenCurlyBrace ::= { [ LineTerminator ] CloseCurlyBrace ::= [ LineTerminator ] } Equals ::= = [ LineTerminator ] ColonEquals ::= : = [ LineTerminator ] 1.3.3 White Space White space serves only to separate tokens and is otherwise ignored. Logical lines containing only white space are ignored. Note Line terminators are not considered white space. WhiteSpace ::= < Unicode blank characters (class Zs) > | < Unicode tab character (0x0009) > 1.3.4 Comments A comment begins with a single-quote character or the keyword REM . A single-quote character is either an ASCII single-quote character, a Unicode left single-quote character, or a Unicode right single-quote character. Comments can begin anywhere on a source line, and the end of the physical line ends the comment. The compiler ignores the characters between the beginning of the comment and the line terminator. Consequently, comments cannot extend across multiple lines by using line continuations. Comment ::= CommentMarker [ Character+ ] CommentMarker ::= SingleQuoteCharacter | REM SingleQuoteCharacter ::= ' | < Unicode left single-quote character (0x2018) > | < Unicode right single-quote character (0x2019) >
10
4.3
Inheritance
1.4 Identifiers
An identifier is a name. Visual Basic identifiers conform to the Unicode Standard Annex 15 with one exception: identifiers may begin with an underscore (connector) character. If an identifier begins with an underscore, it must contain at least one other valid identifier character to disambiguate it from a line continuation. Regular identifiers may not match keywords, but escaped identifiers or identifiers with a type character can. An escaped identifier is an identifier delimited by square brackets. Escaped identifiers follow the same rules as regular identifiers except that they may match keywords and may not have type characters. This example defines a class named classwith a shared method named shared that takes a parameter named boolean and then calls the method.
Class [c lass ] Shared Sub [shared]( [boolean] As Boolean) I f [boolean] Then Console .Wri teL ine (" t rue" ) Else Console .Wri teL ine (" fa l se " ) End I f End Sub End Class Module [module] Sub Main() [c lass ] . [ shared] (True) End Sub End Module
Identifiers are case insensitive, so two identifiers are considered to be the same identifier if they differ only in case. Note The Unicode Standard one-to-one case mappings are used when comparing identifiers and any locale-specific case mappings are ignored. 11
4.3
Inheritance
Identifier ::= NonEscapedIdentifier [ TypeCharacter ] | Keyword TypeCharacter | EscapedIdentifier NonEscapedIdentifier ::= < IdentifierName but not Keyword > EscapedIdentifier ::= [ IdentifierName ] IdentifierName ::= IdentifierStart [ IdentifierCharacter+ ] IdentifierStart ::= AlphaCharacter | UnderscoreCharacter IdentifierCharacter IdentifierCharacter ::= UnderscoreCharacter | AlphaCharacter | NumericCharacter | CombiningCharacter | FormattingCharacter AlphaCharacter ::= < Unicode alphabetic character (classes Lu, Ll, Lt, Lm, Lo, Nl) > NumericCharacter ::= < Unicode decimal digit character (class Nd) > CombiningCharacter ::= < Unicode combining character (classes Mn, Mc) > FormattingCharacter ::= < Unicode formatting character (class Cf) > UnderscoreCharacter ::= < Unicode connection character (class Pc) > IdentifierOrKeyword ::= Identifier | Keyword 1.4.1 Type Characters A type character denotes the type of the preceding identifier. The type character is not considered part of the identifier. If a declaration includes a type character, the type character must agree with the type specified in the declaration itself; otherwise, a compileConfidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
12
4.3
Inheritance
time error occurs. If the declaration omits the type (for example, if it does not specify an As clause), the type character is implicitly substituted as the type of the declaration. No white space may come between an identifier and its type character. There are no type characters for Byte, SByte, UShort, Short, UInteger or ULong, due to a lack of suitable characters. Appending a type character to an identifier that conceptually does not have a type (for example, a namespace name) or to an identifier whose type disagrees with the type of the type character causes a compile-time error. The following example shows the use of type characters:
' The fo l l ow l i ne wil l Module Test1# End Module Module Test2 ' This funct ion takes a Long parameter and returns a Str ing . Funct ion Func$(Param&) ' The fo l l ow ing l i ne causes an error because the type character ' conf l i c t s with the declared type of Func and Param. Func# = CStr(Param@) ' The fo l l ow ing l i ne i s val id . Func$ = CStr(Param&) End Funct ion End Module cause an error : standard modules have no type.
The type character ! presents a special problem in that it can be used both as a type character and as a separator in the language. To remove ambiguity, a ! character is a type character as long as the character that follows it cannot start an identifier. If it can, then the ! character is a separator, not a type character. TypeCharacter ::= IntegerTypeCharacter |
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
13
4.3
Inheritance
LongTypeCharacter | DecimalTypeCharacter | SingleTypeCharacter | DoubleTypeCharacter | StringTypeCharacter IntegerTypeCharacter ::= % LongTypeCharacter ::= & DecimalTypeCharacter ::= @ SingleTypeCharacter ::= ! DoubleTypeCharacter ::= # StringTypeCharacter ::= $
1.5 Keywords
A keyword is a word that has special meaning in a language construct. All keywords are reserved by the language and may not be used as identifiers unless the identifiers are escaped. Note EndIf, GoSub , Let, Variant, and Wend are retained as keywords, although they are no longer used in Visual Basic. Keyword ::= < member of keyword table >
AddHandler AndAlso Byte Catch CDate CInt Const AddressOf As ByVal CBool CDbl Class Continue Alias Boolean Call CByte CDec CLng CSByte And ByRef Case CChar Char CObj CShort
14
4.3
Inheritance
CSng CULng Declare DirectCast Else Enum Exit Friend GetXmlNamespace Handles In Is Like Mod MyBase New NotInheritable On Or Overrides Property ReadOnly
CStr CUShort Default Do ElseIf Erase False Function Global If Inherits IsNot Long Module MyClass Next NotOverridable Operator OrElse ParamArray Protected ReDim
CType Date Delegate Double End Error Finally Get GoSub Implements Integer Let Loop MustInherit Namespace Not Object Option Overloads Partial Public REM
CUInt Decimal Dim Each EndIf Event For GetType GoTo Imports Interface Lib Me MustOverride Narrowing Nothing Of Optional Overridable Private RaiseEvent RemoveHandler
15
4.3
Inheritance
1.6 Literals
A literal is a textual representation of a particular value of a type. Literal types include Boolean, integer, floating point, string, character, and date. Literal ::= BooleanLiteral | IntegerLiteral | FloatingPointLiteral | StringLiteral | CharacterLiteral | DateLiteral | Nothing 1.6.1 Boolean Literals
True and False are literals of the Boolean type that map to the true and false state,
16
4.3
Inheritance
1.6.2 Integer Literals Integer literals can be decimal (base 10), hexadecimal (base 16), or octal (base 8). A decimal integer literal is a string of decimal digits (0-9). A hexadecimal literal is &H followed by a string of hexadecimal digits (0-9, A-F). An octal literal is &O followed by a string of octal digits (0-7). Decimal literals directly represent the decimal value of the integral literal, whereas octal and hexadecimal literals represent the binary value of the integer literal (thus, &H8000S is 32768, not an overflow error). The type of a literal is determined by its value or by the following type character. If no type character is specified, values in the range of the Integer type are typed as Integer; values outside the range for Integer are typed as Long. If an integer literal's type is of insufficient size to hold the integer literal, a compile-time error results. Annotation There isnt a type character for Byte because the most natural character would be B, which is a legal character in a hexadecimal literal. IntegerLiteral ::= IntegralLiteralValue [ IntegralTypeCharacter ] IntegralLiteralValue ::= IntLiteral | HexLiteral | OctalLiteral IntegralTypeCharacter ::= ShortCharacter | UnsignedShortCharacter | IntegerCharacter | UnsignedIntegerCharacter | LongCharacter | UnsignedLongCharacter | IntegerTypeCharacter | LongTypeCharacter ShortCharacter ::= S UnsignedShortCharacter ::= US
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
17
4.3
Inheritance
IntegerCharacter ::= I UnsignedIntegerCharacter ::= UI LongCharacter ::= L UnsignedLongCharacter ::= UL IntLiteral ::= Digit+ HexLiteral ::= & H HexDigit+ OctalLiteral ::= & O OctalDigit+ Digit ::= 0 | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 HexDigit ::= 0 | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | A | B | C | D | E | F OctalDigit ::= 0 | 1 | 2 | 3 | 4 | 5 | 6 | 7 1.6.3 Floating-Point Literals A floating-point literal is an integer literal followed by an optional decimal point (the ASCII period character) and mantissa, and an optional base 10 exponent. By default, a floatingpoint literal is of type Double. If the Single, Double, or Decimal type character is specified, the literal is of that type. If a floating-point literal's type is of insufficient size to hold the floating-point literal, a compile-time error results. Annotation: It is worth noting that the Decimal data type can encode trailing zeros in a value. The specification currently makes no comment about whether trailing zeros in a Decimal literal should be honored by a compiler. FloatingPointLiteral ::= FloatingPointLiteralValue [ FloatingPointTypeCharacter ] | IntLiteral FloatingPointTypeCharacter
18
4.3
Inheritance
FloatingPointTypeCharacter ::= SingleCharacter | DoubleCharacter | DecimalCharacter | SingleTypeCharacter | DoubleTypeCharacter | DecimalTypeCharacter SingleCharacter ::= F DoubleCharacter ::= R DecimalCharacter ::= D FloatingPointLiteralValue ::= IntLiteral . IntLiteral [ Exponent ] | . IntLiteral [ Exponent ] | IntLiteral Exponent Exponent ::= E [ Sign ] IntLiteral Sign ::= + | 1.6.4 String Literals A string literal is a sequence of zero or more Unicode characters beginning and ending with an ASCII double-quote character, a Unicode left double-quote character, or a Unicode right double-quote character. Within a string, a sequence of two double-quote characters is an escape sequence representing a double quote in the string. A string constant is of the Str ingtype.
Module Test Sub Main() ' This pr ints out: " . Console .Wri teL ine (" " " " ) ' This pr ints out: a"b.
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
19
4.3
Inheritance
Console.WriteLine("a""b") ' This causes a compile error due to mismatched double-quotes. Console.WriteLine("a"b") End Sub End Module
The compiler is allowed to replace a constant string expression with a string literal. Each string literal does not necessarily result in a new string instance. When two or more string literals that are equivalent according to the string equality operator using binary comparison semantics appear in the same program, these string literals may refer to the same string instance. For instance, the output of the following program may return True because the two literals may refer to the same string instance. Module Test Sub Main() Dim a As Object = "he" & "llo" Dim b As Object = "hello" Console.WriteLine(a Is b) End Sub End Module StringLiteral ::= DoubleQuoteCharacter [ StringCharacter+ ] DoubleQuoteCharacter DoubleQuoteCharacter ::= " | < Unicode left double-quote character (0x201C) > | < Unicode right double-quote character (0x201D) > StringCharacter ::= < Character except for DoubleQuoteCharacter > | DoubleQuoteCharacter DoubleQuoteCharacter
20
4.3
Inheritance
1.6.5 Character Literals A character literal represents a single Unicode character of the Char type. Two doublequote characters is an escape sequence representing the double-quote character.
Module Test Sub Main() ' This pr ints out: a. Console .Wri teL ine ("a"c ) ' This pr ints out: " . Console .Wri teL ine (" " " " c ) End Sub End Module
CharacterLiteral ::= DoubleQuoteCharacter StringCharacter DoubleQuoteCharacter C 1.6.6 Date Literals A date literal represents a particular moment in time expressed as a value of the Date type. The literal may specify both a date and a time, just a date, or just a time. If the date value is omitted, then January 1 of the year 1 in the Gregorian calendar is assumed. If the time value is omitted, then 12:00:00 AM is assumed. To avoid problems with interpreting the year value in a date value, the year value cannot be two digits. When expressing a date in the first century AD/CE, leading zeros must be specified. A time value may be specified either using a 24-hour value or a 12-hour value; time values that omit an AM or PM are assumed to be 24-hour values. If a time value omits the minutes, the literal 0 is used by default. If a time value omits the seconds, the literal 0 is used by default. If both minutes and second are omitted, then AM or PM must be specified. If the date value specified is outside the range of the Date type, a compile-time error occurs. The following example contains several date literals.
Dim d As Date
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
21
4.3
Inheritance
d d d d d d d
= = = = = = =
# # # # # # #
value: 8/23/1970 12:00:00AM. value: 1/1/1 3:45:39AM. value: 1/1/1 3:45:39AM. value: 1/1/1 1:45:39PM. value: 1/1/1 1:00:00AM. date value is not valid.
DateLiteral ::= # [ Whitespace+ ] DateOrTime [ Whitespace+ ] # DateOrTime ::= DateValue Whitespace+ TimeValue | DateValue | TimeValue DateValue ::= MonthValue / DayValue / YearValue | MonthValue DayValue - YearValue TimeValue ::= HourValue : MinuteValue [ : SecondValue ] [ WhiteSpace+ ] [ AMPM ] | HourValue [ WhiteSpace+ ] AMPM MonthValue ::= IntLiteral DayValue ::= IntLiteral YearValue ::= IntLiteral HourValue ::= IntLiteral MinuteValue ::= IntLiteral SecondValue ::= IntLiteral AMPM ::= AM | PM ElseIf ::= ElseIf | Else If
22
4.3
Inheritance
1.6.7 Nothing
Nothing is a special literal; it does not have a type and is convertible to all types in the type
system, including type parameters. When converted to a particular type, it is the equivalent of the default value of that type. Nothing ::= Nothing
1.7 Separators
The following ASCII characters are separators: Separator ::= ( | ) | { | } | ! | # | , | . | : | ?
23
Preprocessing Directives
Once a file has been lexically analyzed, several kinds of source preprocessing occur. The most important, conditional compilation, determines which source is processed by the syntactic grammar; two other types of directives external source directives and region directives provide meta-information about the source but have no effect on compilation.
24
4.3
Inheritance
The constant expressions allowed in conditional compilation directives are a subset of general constant expressions. The preprocessor allows whitespace and explicit line continuations before and after every token. Start ::= [ CCStatement+ ] CCStatement ::= CCConstantDeclaration | CCIfGroup | LogicalLine CCExpression ::= LiteralExpression | CCParenthesizedExpression | CCSimpleNameExpression | CCCastExpression | CCOperatorExpression | CCConditionalExpression CCParenthesizedExpression ::= ( CCExpression ) CCSimpleNameExpression ::= Identifier CCCastExpression ::= DirectCast ( CCExpression , TypeName ) | TryCast ( CCExpression , TypeName ) | CType ( CCExpression , TypeName ) | CastTarget ( CCExpression ) 25
4.3
Inheritance
CCOperatorExpression ::= CCUnaryOperator CCExpression CCExpression CCBinaryOperator CCExpression CCUnaryOperator ::= + | - | Not CCBinaryOperator ::= + | - | * | / | \ | Mod | ^ | = | < > | < | > | < = | > = | & | And | Or | Xor | AndAlso | OrElse | < < | > > CCConditionalExpression ::= If ( CCExpression , CCExpression , CCExpression ) | If ( CCExpression , CCExpression ) 1.9.1 Conditional Constant Directives Conditional constant statements define constants that exist in a separate conditional compilation declaration space scoped to the source file. The declaration space is special in that no explicit declaration of conditional compilation constants is necessary conditional constants can be implicitly defined in a conditional compilation directive. Prior to being assigned a value, a conditional compilation constant has the value Nothing. When a conditional compilation constant is assigned a value, which must be a constant expression, the type of the constant becomes the type of the value being assigned to it. A conditional compilation constant may be redefined multiple times throughout a source file. For example, the following code prints only the string about to print value and the value of Test.
Module M1 Sub Pr intVa lue(Test As Integer ) #Const DebugCode = True #If DebugCode Then Console .Wri teL ine ("about to pr int value" ) #End I f #Const DebugCode = False
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
26
4.3
Inheritance
Console.WriteLine(Test) #If DebugCode Then Console.WriteLine("printed value") #End If End Sub End Module
The compilation environment may also define conditional constants in a conditional compilation declaration space. CCConstantDeclaration ::= # Const Identifier = CCExpression LineTerminator 1.9.2 Conditional Compilation Directives Conditional compilation directives control conditional compilation and can only reference constant expressions and conditional compilation constants. Each of the constant expressions within a single conditional compilation group is evaluated and converted to the Boolean type in textual order from first to last until one of the conditional expressions evaluates to True. If an expression is not convertible to Boolean, a compile-time error results. Permissive semantics and binary string comparisons are always used when evaluating conditional compilation constant expressions, regardless of any Option directives or compilation environment settings. All lines enclosed by the group, including nested conditional compilation directives, are disabled except for lines between the statement containing the True expression and the next conditional statement of the group, or lines between the Else statement and the End If statement if an Else appears in the group and all of the expressions evaluate to False. In this example, the call to WriteToLog in the Trace conditional compilation directive is not processed because the surrounding Debug conditional compilation directive evaluates to False.
#Const Debug = False #Const Trace = True ' Debugging off ' Tracing on
27
4.3
Inheritance
Class PurchaseTransaction Sub Commit() #If Debug Then CheckConsistency() #If Trace Then WriteToLog(Me.ToString()) #End If #End If ... End Sub End Class
CCIfGroup ::= # I f CCExpression [ Then ] LineTerminator [ CCStatement+ ] [ CCElseIfGroup+ ] [ CCElseGroup ] # End If LineTerminator CCElseIfGroup ::= # ElseIf CCExpression [ Then ] LineTerminator [ CCStatement+ ] CCElseGroup ::= # Else LineTerminator [ CCStatement+ ]
28
4.3
Inheritance
#ExternalSource("c:\wwwroot\inetpub\test.aspx", 30) Console.WriteLine("In test.aspx") #End ExternalSource End Sub End Module
[ LogicalLine+ ]
# End ExternalSource LineTerminator
29
4.3
Inheritance
Start ::= [ RegionStatement+ ] RegionStatement ::= RegionGroup | LogicalLine RegionGroup ::= # Region StringLiteral LineTerminator [ RegionStatement+ ] # End Region LineTerminator
30
4.3
Inheritance
Start ::= [ ExternalChecksumStatement+ ] ExternalChecksumStatement ::= # ExternalChecksum ( StringLiteral , StringLiteral , StringLiteral ) LineTerminator
31
General Concepts
This chapter covers a number of concepts that are required to understand the semantics of the Microsoft Visual Basic language. Many of the concepts should be familiar to Visual Basic programmers or C/C++ programmers, but their precise definitions may differ.
1.13 Declarations
A Visual Basic program is made up of named entities. These entities are introduced through declarations and represent the "meaning" of the program. At a top level, namespaces are entities that organize other entities, such as nested namespaces and types. Types are entities that describe values and define executable code. Types may contain nested types and type members. Type members are constants, variables, methods, operators, properties, events, enumeration values, and constructors. An entity that can contain other entities defines a declaration space. Entities are introduced into a declaration space either through declarations or inheritance; the containing declaration space is called the entities' declaration context. Declaring an entity in a declaration space in turn defines a new declaration space that can contain further nested entity declarations; thus, the declarations in a program form a hierarchy of declaration spaces. Except in the case of overloaded type members, it is invalid for declarations to introduce identically named entities of the same kind into the same declaration context. Additionally, a declaration space may never contain different kinds of entities with the same name; for example, a declaration space may never contain a variable and a method by the same name. Annotation It may be possible in other languages to create a declaration space that contains different kinds of entities with the same name (for example, if the language is case sensitive and allows different declarations based on casing). In that situation, the most accessible entity is considered bound to that name; if more than one type of entity is most accessible then the name is ambiguous. Publ i c is more accessible than Protected Fr iend , Protected Friend is more accessible than Protected or Friend, and Protected or Friend is more accessible than Private. The declaration space of a namespace is "open ended," so two namespace declarations with the same fully qualified name contribute to the same declaration space. In the example below, the two namespace declarations contribute to the same declaration space,
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
32
4.3
Inheritance
in this case declaring two classes with the fully qualified names Data.Customer and
Data.Order : Namespace Data Class Customer End Class End Namespace Namespace Data Class Order End Class End Namespace
Because the two declarations contribute to the same declaration space, a compile-time error would occur if each contained a declaration of a class with the same name. 1.13.1 Overloading and Signatures The only way to declare identically named entities of the same kind in a declaration space is through overloading. Only methods, operators, instance constructors, and properties may be overloaded. Overloaded type members must possess unique signatures. The signature of a type member consists of the name of the type member, the number of type parameters, and the number and types of the member's parameters. Conversion operators also include the return type of the operator in the signature. The following are not part of a member's signature, and hence cannot be overloaded on: Modifiers to a type member (for example, Shared or Pr ivate ). Modifiers to a parameter (for example, ByVal or ByRef). The names of the parameters. The return type of a method or operator (except for conversion operators) or the element type of a property. Constraints on a type parameter.
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
33
4.3
Inheritance
The following example shows a set of overloaded method declarations along with their signatures. This declaration would not be valid since several of the method declarations have identical signatures.
Interface ITest Sub F1() Sub F2(x As Integer) Sub F3(ByRef x As Integer) Sub F4(x As Integer, y As Integer) Function F5(s As String) As Integer Function F6(x As Integer) As Integer Sub F7(a() As String) Sub F8(ParamArray a() As String) Sub F9(Of T)() Sub F10(Of T, U)(x As T, y As U) Sub F11(Of U, T)(x As T, y As U) Sub F12(Of T)(x As T) Sub F13(Of T As IDisposable)(x As T) End Interface ' ' ' ' ' ' ' ' ' ' ' ' ' Signature is F1(). Signature is F2(Integer). Signature is F3(Integer). Signature is F4(Integer, Integer). Signature is F5(String). Signature is F6(Integer). Signature is F7(String()). Signature is F8(String()). Signature is F9!1(). Signature is F10!2(!1, !2) Signature is F11!2(!2, !1) Signature is F12!1(!1) Signature is F13!1(!1)
A method with optional parameters is considered to have multiple signatures, one for each set of parameters that can be passed in by the caller. For example, the following method has three corresponding signatures:
Sub F(x As Short, _ Optional y As Integer = 10, _ Optional z As Long = 20)
It is valid to define a generic type that may contain members with identical signatures based on the type arguments supplied. Overload resolution rules are used to try and
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
34
4.3
Inheritance
disambiguate between such overloads, although there may be situations in which it is impossible to disambiguate. For example:
Class C(Of T) Sub F(x As Integer) End Sub Sub F(x As T) End Sub Sub G(Of U)(x As T, y As U) End Sub Sub G(Of U)(x As U, y As T) End Sub End Class Module Test Sub Main() Dim x As New C(Of Integer) x.F(10) ' Calls C(Of T).F(Integer) x.G(Of Integer)(10, 10) ' Error: Can't choose between overloads End Sub End Module
1.14 Scope
The scope of an entity's name is the set of all declaration spaces within which it is possible to refer to that name without qualification. In general, the scope of an entity's name is its entire declaration context; however, an entity's declaration may contain nested declarations of entities with the same name. In that case, the nested entity shadows, or hides, the outer entity, and access to the shadowed entity is only possible through qualification. Shadowing through nesting occurs in namespaces or types nested within namespaces, in types nested within other types, and in the bodies of type members. Shadowing through the nesting of declarations always occurs implicitly; no explicit syntax is required.
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
35
4.3
Inheritance
In the following example, within the F method, the instance variable i is shadowed by the local variable i, but within the G method, i still refers to the instance variable.
Class Test Pr ivate i As Integer = 0 Sub F() Dim i As Integer = 1 End Sub Sub G() i =1 End Sub End Class
When a name in an inner scope hides a name in an outer scope, it shadows all overloaded occurrences of that name. In the following example, the call F(1) invokes the F declared in Inner because all outer occurrences of F are hidden by the inner declaration. For the same reason, the call F("Hello") is in error.
Class Outer Shared Sub F( i As Integer ) End Sub Shared Sub F(s As Str ing) End Sub Class Inner Shared Sub F( l As Long) End Sub Sub G() F(1) ' Invokes Outer . Inner .F . F("Hel l o " ) ' Error . End Sub End Class End Class
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
36
4.3
Inheritance
1.15 Inheritance
An inheritance relationship is one in which one type (the derived type) derives from another (the base type), such that the derived type's declaration space implicitly contains the accessible non-constructor type members and nested types of its base type. In the following example, class A is the base class of B, and B is derived from A.
Class A End Class Class B Inher i t s A End Class
Since A does not explicitly specify a base class, its base class is implicitly Object. The following are important aspects of inheritance: Inheritance is transitive. If type C is derived from type B, and type B is derived from type A, type C inherits the type members declared in type B as well as the type members declared in type A. A derived type extends, but cannot narrow, its base type. A derived type can add new type members, and it can shadow inherited type members, but it cannot remove the definition of an inherited type member. Because an instance of a type contains all of the type members of its base type, a conversion always exists from a derived type to its base type. All types must have a base type, except for the type Object. Thus, Object is the ultimate base type of all types, and all types can be converted to it. Circularity in derivation is not permitted. That is, when a type B derives from a type A, it is an error for type A to derive directly or indirectly from type B. A type may not directly or indirectly derive from a type nested within it. The following example produces a compile-time error because the classes circularly depend on each other.
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
37
4.3
Inheritance
Class A Inherits B End Class Class B Inherits C End Class Class C Inherits A End Class
The following example also produces a compile-time error because B indirectly derives from its nested class C through class A.
Class A Inher i t s B.C End Class Class B Inher i t s A Publ i c Class C End Class End Class
The next example does not produce an error because class A does not derive from class B.
Class A Class B Inher i t s A End Class End Class
1.15.1 MustInherit and NotInheritable Classes A MustInherit class is an incomplete type that can act only as a base type. A MustInherit class cannot be instantiated, so it is an error to use the New operator on one. It is valid to declare
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
38
4.3
Inheritance
variables of MustInher i tclasses; such variables can only be assigned Nothing or a value that is of a class derived from the MustInherit class. When a regular class is derived from a MustInherit class, the regular class must override all inherited MustOverride members. For example:
MustInher i t Class A Publ i c MustOverr ide Sub F() End Class MustInher i t Class B Inher i t s A Publ i c Sub G() End Sub End Class Class C Inher i t s B Publ i c Overr ides Sub F() End Sub End Class
The MustInherit class A introduces a MustOverride method F. Class B introduces an additional method G, but does not provide an implementation of F. Class B must therefore also be declared MustInherit. Class C overrides F and provides an actual implementation. Since there are no outstanding MustOverride members in class C, it is not required to be MustInherit. A NotInheritable class is a class from which another class cannot be derived. NotInheritable classes are primarily used to prevent unintended derivation. In this example, class B is in error because it attempts to derive from the NotInheritable class A. A class cannot be marked both MustInherit and NotInheritable.
NotInher i tab le Class A End Class
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
39
4.3
Inheritance
Class B ' Error, a class cannot derive from a NotInheritable class. Inherits A End Class
1.15.2 Interfaces and Multiple Inheritance Unlike other types, which only derive from a single base type, an interface may derive from multiple base interfaces. Because of this, an interface can inherit an identically named type member from different base interfaces. In such a case, the multiply-inherited name is not available in the derived interface, and referring to any of those type members through the derived interface causes a compile-time error, regardless of signatures or overloading. Instead, conflicting type members must be referenced through a base interface name. In the following example, the first two statements cause compile-time errors because the multiply-inherited member Count is not available in interface IL i s tCounter :
Inter face IL i s t Property Count() As Integer End Inter face Inter face ICounter Sub Count( i As Integer ) End Inter face Inter face IL i s tCounter Inher i t s IL i s t Inher i t s ICounter End Inter face Module Test Sub F(x As IL i s tCounter ) x.Count(1) x.Count = 1 CType(x, IL i s t ) . Count = 1 CType(x, ICounter ) .Count(1 )
Error , Count i s not avai lab le . Error , Count i s not avai lab le . Ok, invokes IL i s t . Count . Ok, invokes ICounter .Count.
40
4.3
Inheritance
As illustrated by the example, the ambiguity is resolved by casting x to the appropriate base interface type. Such casts have no run-time costs; they merely consist of viewing the instance as a less-derived type at compile time. When a single type member is inherited from the same base interface through multiple paths, the type member is treated as if it were only inherited once. In other words, the derived interface only contains one instance of each type member inherited from a particular base interface. For example:
Inter face IBase Sub F( i As Integer ) End Inter face Inter face ILe f t Inher i t s IBase End Inter face Inter face IR ight Inher i t s IBase End Inter face Inter face IDer ived Inher i t s ILe f t , End Inter face IR ight
Class Derived Implements IDer ived ' Only have to implement F once. Sub F( i As Integer ) Implements IDer ived.F End Sub End Class
41
4.3
Inheritance
If a type member name is shadowed in one path through the inheritance hierarchy, then the name is shadowed in all paths. In the following example, the IBase.Fmember is shadowed by the ILe f t . F member, but is not shadowed in IRight:
Inter face IBase Sub F( i As Integer ) End Inter face Inter face ILe f t Inher i t s IBase Shadows Sub F( i As Integer ) End Inter face Inter face IR ight Inher i t s IBase Sub G() End Inter face Inter face IDer ived Inher i t s ILe f t , End Inter face IR ight
Class Test Sub H(d As IDer ived) d.F(1) CType(d, IBase) . F (1 ) CType(d, ILe f t ) . F (1 ) CType(d, IR ight ) . F (1 ) End Sub End Class
The invocation d.F(1) selects ILeft.F, even though IBase.F appears to not be shadowed in the access path that leads through IRight. Because the access path from IDerived to ILeft to 42
4.3
Inheritance
IBase shadows IBase.F the member is also shadowed in the access path from IDerived to , IRight to IBase.
1.15.3 Shadowing A derived type shadows the name of an inherited type member by re-declaring it. Shadowing a name does not remove the inherited type members with that name; it merely makes all of the inherited type members with that name unavailable in the derived class. The shadowing declaration may be any type of entity. Entities than can be overloaded can choose one of two forms of shadowing. Shadowing by name is specified using the Shadows keyword. An entity that shadows by name hides everything by that name in the base class, including all overloads. Shadowing by name and signature is specified using the Overloads keyword. An entity that shadows by name and signature hides everything by that name with the same signature as the entity. For example:
Class Base Sub F() End Sub Sub F( i As Integer ) End Sub Sub G() End Sub Sub G(i As Integer ) End Sub End Class Class Derived Inher i t s Base ' Only hides F( Integer ) . Overloads Sub F( i As Integer ) End Sub
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
43
4.3
Inheritance
' Hides G() and G(Integer). Shadows Sub G(i As Integer) End Sub End Class Module Test Sub Main() Dim x As New Derived() x.F() ' Calls Base.F(). x.G() ' Error: Missing parameter. End Sub End Module
Shadowing a method with a ParamArray argument by name and signature hides only the individual signature, not all possible expanded signatures. This is true even if the signature of the shadowing method matches the unexpanded signature of the shadowed method. The following example:
Class Base Sub F(ParamArray x() As Integer ) Console .Wri teL ine ("Base" ) End Sub End Class Class Derived Inher i t s Base Overloads Sub F(x( ) As Integer ) Console .Wri teL ine ("Der ived" ) End Sub End Class Module Test Sub Main Dim d As New Derived()
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
44
4.3
Inheritance
prints Base, even though Derived.Fhas the same signature as the unexpanded form of Base.F. Conversely, a method with a ParamArray argument only shadows methods with the same signature, not all possible expanded signatures. The following example:
Class Base Sub F(x As Integer ) Console .Wri teL ine ("Base" ) End Sub End Class Class Derived Inher i t s Base Overloads Sub F(ParamArray x() As Integer ) Console .Wri teL ine ("Der ived" ) End Sub End Class Module Test Sub Main() Dim d As New Derived() d.F(10) End Sub End Module
prints Base, even though Derived.F has an expanded form that has the same signature as Base.F. A shadowing method or property that does not specify Shadows or Overloads assumes Overloads if the method or property is declared Overrides, Shadows otherwise. If one member of a set of overloaded entities specifies the Shadows or Overloads keyword, they all must specify it. The Shadows and Overloads keywords cannot be specified at the same time. Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved. 45
4.3
Inheritance
Neither Shadows nor Overloads can be specified in a standard module; members in a standard module implicitly shadow members inherited from Object. It is valid to shadow the name of a type member that has been multiply-inherited through interface inheritance (and which is thereby unavailable), thus making the name available in the derived interface. For example:
Inter face ILe f t Sub F() End Inter face Inter face IR ight Sub F() End Inter face Inter face ILe f tR ight Inher i t s ILe f t , IR ight Shadows Sub F() End Inter face Module Test Sub G(i As ILe f tR ight ) i . F ( ) ' Cal l s ILe f tR ight . F . CType(i , ILe f t ) . F ( ) ' Cal l s ILe f t . F . CType(i , IR ight ) . F ( ) ' Cal l s IR ight . F . End Sub End Module
Because methods are allowed to shadow inherited methods, it is possible for a class to contain several Overridable methods with the same signature. This does not present an ambiguity problem, since only the most-derived method is visible. In the following example, the C and D classes contain two Overridable methods with the same signature:
Class A Publ i c Overr idable Sub F()
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
46
4.3
Inheritance
Console.WriteLine("A.F") End Sub End Class Class B Inherits A Public Overrides Sub F() Console.WriteLine("B.F") End Sub End Class Class C Inherits B Public Shadows Overridable Sub F() Console.WriteLine("C.F") End Sub End Class Class D Inherits C Public Overrides Sub F() Console.WriteLine("D.F") End Sub End Class Module Test Sub Main() Dim d As Dim a As Dim b As Dim c As a.F() b.F()
New A = B = C =
D() d d d
47
4.3
Inheritance
There are two Overr idablemethods here: one introduced by class A and the one introduced by class C. The method introduced by class C hides the method inherited from class A. Thus, the Overrides declaration in class D overrides the method introduced by class C, and it is not possible for class D to override the method introduced by class A. The example produces the output:
B.F B.F D.F D.F
It is possible to invoke the hidden Overridable method by accessing an instance of class D through a less-derived type in which the method is not hidden. It is not valid to shadow a MustOverride method, because in most cases this would make the class unusable. For example:
MustInher i t Class Base Publ i c MustOverr ide Sub F() End Class MustInher i t Class Derived Inher i t s Base Publ i c Shadows Sub F() End Sub End Class Class MoreDerived Inher i t s Derived ' Error : MustOverr ide method Base.F i s not overr idden. End Class
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
48
4.3
Inheritance
In this case, the class MoreDerived is required to override the MustOverr ide method Base.F, but because the class Derived shadows Base.F, this is not possible. There is no way to declare a valid descendent of Derived. In contrast to shadowing a name from an outer scope, shadowing an accessible name from an inherited scope causes a warning to be reported, as in the following example:
Class Base Publ i c Sub F() End Sub Pr ivate Sub G() End Sub End Class Class Derived Inher i t s Base Publ i c Sub F() ' Warning: shadowing an inher i ted name. End Sub Publ i c Sub G() ' No warning, Base.G i s not accessib le here. End Sub End Class
The declaration of method F in class Derived causes a warning to be reported. Shadowing an inherited name is specifically not an error, since that would preclude separate evolution of base classes. For example, the above situation might have come about because a later version of class Base introduced a method F that was not present in an earlier version of the class. Had the above situation been an error, any change made to a base class in a separately versioned class library could potentially cause derived classes to become invalid. The warning caused by shadowing an inherited name can be eliminated through use of the Shadows or Overloads modifier: 49
4.3
Inheritance
Class Base Public Sub F() End Sub End Class Class Derived Inherits Base Public Shadows Sub F() 'OK. End Sub End Class
The Shadows modifier indicates the intention to shadow the inherited member. It is not an error to specify the Shadows or Overloads modifier if there is no type member name to shadow. A declaration of a new member shadows an inherited member only within the scope of the new member, as in the following example:
Class Base Publ i c Shared Sub F() End Sub End Class Class Derived Inher i t s Base Pr ivate Shared Shadows Sub F() ' Shadows Base.F in class Derived only . End Sub End Class Class MoreDerived Inher i t s Derived Shared Sub G() F() ' Invokes Base.F .
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
50
4.3
Inheritance
In the example above, the declaration of method F in class Derived shadows the method F that was inherited from class Base, but since the new method F in class Derived has Private access, its scope does not extend to class MoreDerived. Thus, the call F() in MoreDerived.G is valid and will invoke Base.F. In the case of overloaded type members, the entire set of overloaded type members is treated as if they all had the most permissive access for the purposes of shadowing.
Class Base Publ i c Sub F() End Sub End Class Class Derived Inher i t s Base Pr ivate Shadows Sub F() End Sub Publ i c Shadows Sub F( i As Integer ) End Sub End Class Class MoreDerived Inher i t s Derived Publ i c Sub G() F() ' Error . No accessib le member with th is signature . End Sub End Class
In this example, even though the declaration of F() in Derived is declared with Private access, the overloaded F(Integer) is declared with Public access. Therefore, for the purpose of shadowing, the name F in Derived is treated as if it was Public, so both methods shadow F in Base. Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved. 51
4.3
Inheritance
1.16 Implementation
An implementation relationship exists when a type declares that it implements an interface and the type implements all the type members of the interface. A type that implements a particular interface is convertible to that interface. Interfaces cannot be instantiated, but it is valid to declare variables of interfaces; such variables can only be assigned a value that is of a class that implements the interface. For example:
Interface ITestable Function Test(value As Byte) As Boolean End Interface Class TestableClass Implements ITestable Function Test(value As Byte) As Boolean Implements ITestable.Test Return value > 128 End Function End Class Module Test Sub F() Dim x As ITestable = New TestableClass Dim b As Boolean b = x.Test(34) End Sub End Module
A type implementing an interface with multiply-inherited type members must still implement those methods, even though they cannot be accessed directly from the derived interface being implemented. For example:
Interface ILeft Sub Test() End Interface
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
52
4.3
Inheritance
Interface IRight Sub Test() End Interface Interface ILeftRight Inherits ILeft, IRight End Interface Class LeftRight Implements ILeftRight ' Has to reference ILeft explicitly. Sub TestLeft() Implements ILeft.Test End Sub ' Has to reference IRight explicitly. Sub TestRight() Implements IRight.Test End Sub ' Error: Test is not available in ILeftRight. Sub TestLeftRight() Implements ILeftRight.Test End Sub End Class
Even MustInher i tclasses must provide implementations of all the members of implemented interfaces; however, they can defer implementation of these methods by declaring them as MustOverr ide For example: .
Inter face ITest Sub Test1( ) Sub Test2() End Inter face MustInher i t Class TestBase Implements ITest ' Provides an implementat ion.
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
53
4.3
Inheritance
Sub Test1() Implements ITest.Test1 End Sub ' Defers implementation. MustOverride Sub Test2() Implements ITest.Test2 End Class Class TestDerived Inherits TestBase ' Have to implement MustOverride method. Overrides Sub Test2() End Sub End Class
A type may choose to re-implement an interface that its base type implements. To reimplement the interface, the type must explicitly state that it implements the interface. A type re-implementing an interface may choose to re-implement only some, but not all, of the members of the interface any members not re-implemented continue to use the base types implementation. For example:
Class TestBase Implements ITest Sub Test1() Implements ITest.Test1 Console.WriteLine("TestBase.Test1") End Sub Sub Test2() Implements ITest.Test2 Console.WriteLine("TestBase.Test2") End Sub End Class Class TestDerived Inherits TestBase Implements ITest ' Required to re-implement
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
54
4.3
Inheritance
Sub DerivedTest1() Implements ITest.Test1 Console.WriteLine("TestDerived.DerivedTest1") End Sub End Class Module Test Sub Main() Dim Test As ITest = New TestDerived() Test.Test1() Test.Test2() End Sub End Module
When a derived type implements an interface whose base interfaces are implemented by the derived type's base types, the derived type can choose to only implement the interface's type members that are not already implemented by the base types. For example:
Interface IBase Sub Base() End Interface Interface IDerived Inherits IBase Sub Derived() End Interface Class Base Implements IBase Public Sub Base() Implements IBase.Base End Sub
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
55
4.3
Inheritance
End Class Class Derived Inherits Base Implements IDerived ' Required: IDerived.Derived not implemented by Base. Public Sub Derived() Implements IDerived.Derived End Sub End Class
An interface method can also be implemented using an overridable method in a base type. In that case, a derived type may also override the overridable method and alter the implementation of the interface. For example:
Class Base Implements ITest Public Sub Test1() Implements ITest.Test1 Console.WriteLine("TestBase.Test1") End Sub Public Overridable Sub Test2() Implements ITest.Test2 Console.WriteLine("TestBase.Test2") End Sub End Class Class Derived Inherits Base ' Overrides base implementation. Public Overrides Sub Test2() Console.WriteLine("TestDerived.Test2") End Sub End Class
56
4.3
Inheritance
1.16.1 Implementing Methods A type implements a type member of an implemented interface by supplying a method with an Implements clause. The two type members must have the same number of parameters, all of the types and modifiers of the parameters must match, including the default value of optional parameters, the return type must match, and all of the constraints on method parameters must match. For example:
Inter face ITest Sub F(ByRef x As Integer ) Sub G(Optional y As Integer = 20) Sub H(Paramarray z() As Integer ) End Inter face Class Test Implements ITest ' Error : ByRef/ByVal mismatch. Sub F(x As Integer ) Implements ITest . F End Sub ' Error : Defaul ts do not match. Sub G(Optional y As Integer = 10) Implements ITest .G End Sub ' Error : Paramarray does not match. Sub H(z() As Integer ) Implements ITest .H End Sub End Class
A single method may implement any number of interface type members if they all meet the above criteria. For example:
Inter face ITest Sub F( i As Integer ) Sub G(i As Integer ) End Inter face
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
57
4.3
Inheritance
Class Test Implements ITest Sub F(i As Integer) Implements ITest.F, ITest.G End Sub End Class
When implementing a method in a generic interface, the implementing method must supply the type arguments that correspond to the interfaces type parameters. For example:
Interface I1(Of U, V) Sub M(x As U, y As List(Of V)) End Interface Class C1(Of W, X) Implements I1(Of W, X) ' W corresponds to U and X corresponds to V Public Sub M(x As W, y As List(Of X)) Implements I1(Of W, X).M End Sub End Class Class C2 Implements I1(Of String, Integer) ' String corresponds to U and Integer corresponds to V Public Sub M(x As String, y As List(Of Integer)) _ Implements I1(Of String, Integer).M End Sub End Class
Note that it is possible that a generic interface may not be implementable for some set of type arguments. 58
4.3
Inheritance
Interface I1(Of T, U) Sub S1(x As T) Sub S1(y As U) End Interface Class C1 ' Unable to implement because I1.S1 has two identical signatures Implements I1(Of Integer, Integer) End Class
1.17 Polymorphism
Polymorphism provides the ability to vary the implementation of a method or property. With polymorphism, the same method or property can perform different actions depending on the run-time type of the instance that invokes it. Methods or properties that are polymorphic are called overridable. By contrast, the implementation of a non-overridable method or property is invariant; the implementation is the same whether the method or property is invoked on an instance of the class in which it is declared or an instance of a derived class. When a non-overridable method or property is invoked, the compile-time type of the instance is the determining factor. For example:
Class Base Public Overridable Property X() As Integer Get End Get Set End Set End Property End Class Class Derived Inherits Base Public Overrides Property X() As Integer Get End Get
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
59
4.3
Inheritance
Set End Set End Property End Class Module Test Sub F() Dim Z As Base Z = Z.X Z = Z.X End Sub End Module New Base() = 10 New Derived() = 10 ' Calls Base.X ' Calls Derived.X
An overridable method may also be MustOverr ide which means that it provides no method , body and must be overridden. MustOverr ide methods are only allowed in MustInherit classes. In the following example, the class Shape defines the abstract notion of a geometrical shape object that can paint itself:
MustInher i t Publ i c Class Shape Publ i c MustOverr ide Sub Paint (g As Graphics , r As Rectangle) End Class Publ i c Class El l i pse Inher i t s Shape Publ i c Overr ides Sub Paint (g As Graphics , r As Rectangle) g.drawEl l i p se ( r ) End Sub End Class Publ i c Class Box Inher i t s Shape
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
60
4.3
Inheritance
Public Overrides Sub Paint(g As Graphics, r As Rectangle) g.drawRect(r) End Sub End Class
The Paint method is MustOverr ide because there is no meaningful default implementation. The Ellipse and Box classes are concrete Shape implementations. Because these classes are not MustInherit, they are required to override the Paint method and provide an actual implementation. It is an error for a base access to reference a MustOverride method, as the following example demonstrates:
MustInher i t Class A Publ i c MustOverr ide Sub F() End Class Class B Inher i t s A Publ i c Overr ides Sub F() MyBase.F( ) ' Error , MyBase.F i s MustOverr ide . End Sub End Class
An error is reported for the MyBase.F() invocation because it references a MustOverride method. 1.17.1 Overriding Methods A type may override an inherited overridable method by declaring a method with the same name and , signature, and marking the declaration with the Overrides modifier.There are additional requirements on overriding methods, listed below. Whereas an Overridable method declaration introduces a new method, an Overrides method declaration replaces the inherited implementation of the method. 61
4.3
Inheritance
An overriding method may be declared NotOverr idable which prevents any further , overriding of the method in derived types. In effect, NotOverr idable methods become nonoverridable in any further derived classes. Consider the following example:
Class A Publ i c Overr idable Sub F() Console .Wri teL ine ("A.F" ) End Sub Publ i c Overr idable Sub G() Console .Wri teL ine ("A.G" ) End Sub End Class Class B Inher i t s A Publ i c Overr ides NotOverr idable Sub F() Console .Wri teL ine ("B .F" ) End Sub Publ i c Overr ides Sub G() Console .Wri teL ine ("B .G" ) End Sub End Class Class C Inher i t s B Publ i c Overr ides Sub G() Console .Wri teL ine ("C .G" ) End Sub End Class
62
4.3
Inheritance
In the example, class B provides two Overr idesmethods: a method F that has the NotOverridable modifier and a method G that does not. Use of the NotOverridable modifier prevents class C from further overriding method F. An overriding method may also be declared MustOverride, even if the method that it is overriding is not declared MustOverride. This requires that the containing class be declared MustInherit and that any further derived classes that are not declared MustInherit must override the method. For example:
Class A Publ i c Overr idable Sub F() Console .Wri teL ine ("A.F" ) End Sub End Class MustInher i t Class B Inher i t s A Publ i c Overr ides MustOverr ide Sub F() End Class
In the example, class B overrides A.F with a MustOverride method. This means that any classes derived from B will have to override F, unless they are declared MustInherit as well. A compile-time error occurs unless all of the following are true of an overriding method: 1. The declaration context contains a single accessible inherited method with the same signature and return type (if any) as the overriding method. 2. The inherited method being overridden is overridable. In other words, the inherited method being overridden is not Shared or NotOverridable. 3. The accessibility domain of the method being declared is the same as the accessibility domain of the inherited method being overridden. There is one exception: a Protected Friend method must be overridden by a Protected method if the other method is in another assembly that the overriding method does not have Friend access to. 63
4.3
Inheritance
4. The parameters of the overriding method match the overridden method's parameters in regards to usage of the ByVal, ByRef, ParamArray, and Optional modifiers, including the values provided for optional parameters. 5. The type parameters of the overriding method match the overridden methods type parameters in regards to type constraints. When overriding a method in a base generic type, the overriding method must supply the type arguments that correspond to the base type parameters. For example:
Class Base(Of U, V) Publ i c Overr idable Sub M(x As U, y As Lis t (Of V)) End Sub End Class Class Derived(Of W, X) Inher i t s Base(Of W, X) ' W corresponds to U and X corresponds to V Publ i c Overr ides Sub M(x As W, y As Lis t (Of X)) End Sub End Class Class MoreDerived Inher i t s Derived(Of Str ing , Integer ) ' Str ing corresponds to U and Integer corresponds to V Publ i c Overr ides Sub M(x As Str ing , y As Lis t (Of Integer ) ) End Sub End Class
Note that it is possible that an overridable method in a generic class may not be able to be overridden for some sets of type arguments. If the method is declared MustOverride, this means that some inheritance chains may not be possible. For example:
MustInher i t Class Base(Of T, U) Publ i c MustOverr ide Sub S1(x As T) Publ i c MustOverr ide Sub S1(y As U)
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
64
4.3
Inheritance
End Class Class Derived Inherits Base(Of Integer, Integer) ' Error: Can't override both S1's at once Public Overrides Sub S1(x As Integer) End Sub End Class
An override declaration can access the overridden base method using a base access, as in the following example:
Class Base Private x As Integer Public Overridable Sub PrintVariables() Console.WriteLine("x = " & x) End Sub End Class Class Derived Inherits Base Private y As Integer Public Overrides Sub PrintVariables() MyBase.PrintVariables() Console.WriteLine("y = " & y) End Sub End Class
In the example, the invocation of MyBase.Pr in tVar iab lesin class Derived invokes the () PrintVariables method declared in class Base. A base access disables the overridable invocation mechanism and simply treats the base method as a non-overridable method. Had the invocation in Derived been written CType(Me, Base).PrintVariables(), it would
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
65
4.3
Inheritance
recursively invoke the Pr intVar iab les method declared in Derived not the one declared in , Base. Only when it includes an Overrides modifier can a method override another method. In all other cases, a method with the same signature as an inherited method simply shadows the inherited method, as in the example below:
Class Base Publ i c Overr idable Sub F() End Sub End Class Class Derived Inher i t s Base Publ i c Overr idable Sub F() ' Warning, shadowing inher i ted F() . End Sub End Class
In the example, the method F in class Derived does not include an Overrides modifier and therefore does not override method F in class Base. Rather, method F in class Derived shadows the method in class Base, and a warning is reported because the declaration does not include a Shadows or Overloads modifier. In the following example, method F in class Derived shadows the overridable method F inherited from class Base:
Class Base Publ i c Overr idable Sub F() End Sub End Class Class Derived Inher i t s Base Pr ivate Shadows Sub F() ' Shadows Base.F with in Derived. End Sub End Class
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
66
4.3
Inheritance
Class MoreDerived Inherits Derived Public Overrides Sub F() ' Ok, overrides Base.F. End Sub End Class
Since the new method F in class Derived has Private access, its scope only includes the class body of Derived and does not extend to class MoreDerived. The declaration of method F in class MoreDerived is therefore permitted to override the method F inherited from class Base. When an Overridable method is invoked, the most derived implementation of the instance method is called, based on the type of the instance, regardless of whether the call is to the method in the base class or the derived class. The most derived implementation of an Overridable method M with respect to a class R is determined as follows: If R contains the introducing Overridable declaration of M, this is the most derived implementation of M. Otherwise, if R contains an override of M, this is the most derived implementation of M. Otherwise, the most derived implementation of M is the same as that of the direct base class of R.
1.18 Accessibility
A declaration specifies the accessibility of the entity it declares. An entity's accessibility does not change the scope of an entity's name. The accessibility domain of a declaration is the set of all declaration spaces in which the declared entity is accessible. The five access types are Public, Protected, Friend, Protected Friend, and Private. Public is the most permissive access type, and the four other types are all subsets of Public. The least permissive access type is Private, and the four other access types are all supersets of Private. The access type for a declaration is specified via an optional access modifier, which can be Public, Protected, Friend, Private, or the combination of Protected and Friend. If no access
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
67
4.3
Inheritance
modifier is specified, the default access type depends on the declaration context; the permitted access types also depend on the declaration context. Entities declared with the Publ i c modifier have Publ i c access. There are no restrictions on the use of Public entities. Entities declared with the Protected modifier have Protected access. Protected access can only be specified on members of classes (both regular type members and nested classes) or on Overridable members of standard modules and structures (which must, by definition, be inherited from System.Object or System.ValueType). A Protected member is accessible to a derived class, provided that either the member is not an instance member, or the access takes place through an instance of the derived class. Protected access is not a superset of Friend access. Entities declared with the Friend modifier have Friend access. An entity with Friend access is accessible only within the program that contains the entity declaration or any assemblies that have been given Friend access through the System.Runtime.CompilerServices.InternalsVisibleToAttribute attribute. Entities declared with the Protected Friend modifiers have the union of Protected and Friend access. Entities declared with the Private modifier have Private access. A Private entity is accessible only within its declaration context, including any nested entities. The accessibility in a declaration does not depend on the accessibility of the declaration context. For example, a type declared with Private access may contain a type member with Public access. The following code demonstrates various accessibility domains:
Publ i c Class A Publ i c Shared X As Integer Fr iend Shared Y As Integer Pr ivate Shared Z As Integer End Class Fr iend Class B
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
68
4.3
Inheritance
Public Shared X As Integer Friend Shared Y As Integer Private Shared Z As Integer Public Class C Public Shared X As Integer Friend Shared Y As Integer Private Shared Z As Integer End Class Private Class D Public Shared X As Integer Friend Shared Y As Integer Private Shared Z As Integer End Class End Class
The classes and members have the following accessibility domains: The accessibility domain of A and A.X is unlimited. The accessibility domain of A.Y, B, B.X, B.Y, B.C, B.C.X, and B.C.Y is the containing program. The accessibility domain of A.Z is A. The accessibility domain of B.Z, B.D, B.D.X, and B.D.Y is B, including B.C and B.D. The accessibility domain of B.C.Z is B.C. The accessibility domain of B.D.Z is B.D. As the example illustrates, the accessibility domain of a member is never larger than that of a containing type. For example, even though all X members have Public declared accessibility, all but A.X have accessibility domains that are constrained by a containing type. Access to Protected instance members must be through an instance of the derived type so that unrelated types cannot gain access to each other's protected members. For example:
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
69
4.3
Inheritance
Class User Protected Password As String End Class Class Employee Inherits User End Class Class Guest Inherits User Public Function GetPassword(u As User) As String ' Error: protected access has to go through derived type. Return U.Password End Function End Class
In the above example, the class Guest only has access to the protected Password field if it is qualified with an instance of Guest. This prevents Guest from gaining access to the Password field of an Employee object simply by casting it to User. For the purposes of Protected member access in generic types, the declaration context includes type parameters. This means that a derived type with one set of type arguments does not have access to the Protected members of a derived type with a different set of type arguments. For example:
Class Base(Of T) Protected x As T End Class Class Derived(Of T) Inher i t s Base(Of T) Publ i c Sub F(y As Derived(Of Str ing) ) ' Error : Derived(Of T) cannot access Derived(Of Str ing) ' s ' protected members y.x = "a"
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
70
4.3
Inheritance
Annotation The C# language (and possibly other languages) allows a generic type to access Protected members regardless of what type arguments are supplied. This should be kept in mind when designing generic classes that contain Protectedmembers. AccessModifier ::= Publ i c | Protected | Friend | Private | Protected Friend 1.18.1 Constituent Types The constituent types of a declaration are the types that are referenced by the declaration. For example, the type of a constant, the return type of a method and the parameter types of a constructor are all constituent types. The accessibility domain of a constituent type of a declaration must be the same as or a superset of the accessibility domain of the declaration itself. For example:
Publ i c Class X Pr ivate Class Y End Class ' Error : Exposing pr ivate class Y outs ide of X. Publ i c Funct ion Z() As Y End Funct ion ' Val id : Not exposing outs ide of X. Pr ivate Funct ion A() As Y End Funct ion End Class Fr iend Class B Pr ivate Class C End Class
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
71
4.3
Inheritance
' Error: Exposing private class Y outside of B. Public Function D() As C End Function End Class
' X.Y.
72
4.3
Inheritance
' X.Y.E.
In some situations, a qualified name may begin with the keyword Global. The keyword represents the unnamed outermost namespace, which is useful in situations where a declaration shadows an enclosing namespace. The Global keyword allows escaping out to the outermost namespace in that situation. For example:
Namespace NS1 Class System End Class Module Test Sub Main() ' Error : Class System does not contain Int32 Dim x As System.Int32 ' Legal , binds to System in outermost namespace Dim y As Global .System. Int32 End Sub End Module End Namespace
In the above example, the first method call is invalid because the identifier System binds to the class System, not the namespace System. The only way to access the System namespace is to use Global to escape out to the outermost namespace. Global cannot be used in an Imports statement or Namespace declaration. Because other languages may introduce types and namespaces that match keywords in the language, Visual Basic recognizes keywords to be part of a qualified name as long as they follow a period. Keywords used in this way are treated as identifiers. For example, the qualified identifier X.Default.Class is a valid qualified identifier, while Default.Class is not.
73
4.3
Inheritance
1.19.1 Qualified Name Resolution Given a qualified namespace or type name of the form N.R(Of A) where R is the rightmost , identifier in the qualified name and A is an optional type argument list, the following steps describe how to determine to which namespace or type the qualified name refers: 1. Resolve N, which may be either a qualified or unqualified name. 2. If resolution of N fails, resolves to a type parameter, does not resolve to a namespace or type, or does not resolve to a type that has the same number of type parameters as type arguments, if any, a compile-time error occurs. If R matches the name of a namespace or accessible type in N, then the qualified name refers to that namespace or type. 3. If N contains one or more standard modules, and R matches the name of an accessible type with the same number of type parameters as type arguments, if any, in exactly one standard module, then the qualified name refers to that type. If R matches the name of accessible types with the same number of type parameters as type argument, if any, in more than one standard module, a compile-time error occurs. 4. Otherwise, a compile-time error occurs. Note An implication of this resolution process is that type members do not shadow namespaces or types when resolving namespace or type names. 1.19.2 Unqualified Name Resolution Given an unqualified name R(Of A), where A is an optional type argument list, the following steps describe how to determine to which namespace or type an unqualified name refers: 1. For each nested type containing the name reference, starting from the innermost type and going to the outermost: 1.1. If R matches the name of an accessible nested type with the same number of type parameters as type arguments, if any, then the unqualified name refers to that type.
74
4.3
Inheritance
1.2. Otherwise, if R matches the name of a type parameter in the current type and no type arguments were supplied, then the unqualified name refers to that type parameter. 2. For each nested namespace containing the name reference, starting from the innermost namespace and going to the outermost namespace: 2.1. If R matches the name of a nested namespace in the current namespace and no type argument list is supplied, then the unqualified name refers to that nested namespace. 2.2. Otherwise, if R matches the name of an accessible type with the same number of type parameters as type arguments, if any, in the current namespace, then the unqualified name refers to that type. 2.3. Otherwise, if the namespace contains one or more accessible standard modules, and R matches the name of an accessible nested type with the same number of type parameters as type arguments, if any, in exactly one standard module, then the unqualified name refers to that nested type. If R matches the name of accessible nested types with the same number of type parameters as type arguments, if any, in more than one standard module, a compile-time error occurs. 3. If the source file has one or more import aliases, and R matches the name of one of them, then the unqualified name refers to that import alias. If a type argument list is supplied, a compile-time error occurs. 4. If the source file containing the name reference has one or more imports: 4.1. If R matches the name of an accessible type with the same number of type parameters as type arguments, if any, in exactly one import, then the unqualified name refers to that type. If R matches the name of an accessible type with the same number of type parameters as type arguments, if any, in more than one import and all are not the same type, a compile-time error occurs. 4.2. Otherwise, if no type argument list was supplied and R matches the name of a namespace with accessible types in exactly one import, then the unqualified name refers to that namespace. If no type argument list was supplied and R matches the
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
75
4.3
Inheritance
name of a namespace with accessible types in more than one import and all are not the same namespace, a compile-time error occurs. 4.3. Otherwise, if the imports contain one or more accessible standard modules, and R matches the name of an accessible nested type with the same number of type parameters as type arguments, if any, in exactly one standard module, then the unqualified name refers to that type. If R matches the name of accessible nested types with the same number of type parameters as type arguments, if any, in more than one standard module, a compile-time error occurs. 5. If the compilation environment defines one or more import aliases, and R matches the name of one of them, then the unqualified name refers to that import alias. If a type argument list is supplied, a compile-time error occurs. 6. If the compilation environment defines one or more imports: 6.1. If R matches the name of an accessible type with the same number of type parameters as type arguments, if any, in exactly one import, then the unqualified name refers to that type. If R matches the name of an accessible type with the same number of type parameters as type arguments, if any, in more than one import, a compile-time error occurs. 6.2. Otherwise, if no type argument list was supplied and R matches the name of a namespace with accessible types in exactly one import, then the unqualified name refers to that namespace. If no type argument list was supplied and R matches the name of a namespace with accessible types in more than one import, a compiletime error occurs. 6.3. Otherwise, if the imports contain one or more accessible standard modules, and R matches the name of an accessible nested type with the same number of type parameters as type arguments, if any, in exactly one standard module, then the unqualified name refers to that type. If R matches the name of accessible nested types with the same number of type parameters as type arguments, if any, in more than one standard module, a compile-time error occurs. 7. Otherwise, a compile-time error occurs.
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
76
4.3
Inheritance
Note An implication of this resolution process is that type members do not shadow namespaces or types when resolving namespace or type names. Normally, a name can only occur once in a particular namespace. However, because namespaces can be declared across multiple .NET assemblies, it is possible to have a situation where two assemblies define a type with the same fully qualified name. In that case, a type declared in the current set of source files is preferred over a type declared in an external .NET assembly. Otherwise, the name is ambiguous and there is no way to disambiguate the name.
1.20 Variables
A variable represents a storage location. Every variable has a type that determines what values can be stored in the variable. Because Visual Basic is a type-safe language, every variable in a program has a type and the language guarantees that values stored in variables are always of the appropriate type. Variables are always initialized to the default value of their type before any reference to the variable can be made. It is not possible to access uninitialized memory.
77
4.3
Inheritance
which the type or method is used. For example, a generic stack class could be implemented as:
Public Class Stack(Of ItemType) Protected Items(0 To 99) As ItemType Protected CurrentIndex As Integer = 0 Public Sub Push(data As ItemType) If CurrentIndex = 100 Then Throw New ArgumentException("Stack is full.") End If Items(CurrentIndex) = Data CurrentIndex += 1 End Sub Public Function Pop() As ItemType If CurrentIndex = 0 Then Throw New ArgumentException("Stack is empty.") End If CurrentIndex -= 1 Return Items(CurrentIndex + 1) End Function End Class
Declarations that use the Stack(Of I temType) class must supply a type argument for the type parameter I temType. This type is then filled in wherever ItemType is used within the class:
Option Str i c t On Module Test Sub Main() Dim s1 As New Stack(Of Integer ) ( ) Dim s2 As New Stack(Of Double)( ) s1.Push(10.10) ' Error : Stack(Of Integer ) . Push takes an Integer
78
4.3
Inheritance
s2.Push(10.10) ' OK: Stack(Of Double).Push takes a Double Console.WriteLine(s2.Pop().GetType().ToString()) ' Prints: Double End Sub End Module
1.21.1 Type Parameters Type parameters may be supplied on type or method declarations. Each type parameter is an identifier which is a place-holder for a type argument that is supplied to create a constructed type or method. By contrast, a type argument is the actual type that is substituted for the type parameter when a generic type or method is used. Each type parameter in a type or method declaration defines a name in the declaration space of that type or method. Thus, it cannot have the same name as another type parameter, a type member, a method parameter, or a local variable. The scope of a type parameter on a type or method is the entire type or method. Because type parameters are scoped to the entire type declaration, nested types can use outer type parameters. This also means that type parameters must always be specified when accessing types nested inside generic types:
Public Class Outer(Of T) Public Class Inner Public Sub F(x As T) ... End Sub End Class End Class Module Test Sub Main() Dim x As New Outer(Of Integer).Inner() ... End Sub End Module
Unlike other members of a class, type parameters are not inherited. Type parameters in a type can only be referred to by their simple name; in other words, they cannot be qualified
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
79
4.3
Inheritance
with the containing type name. Although it is bad programming style, the type parameters in a nested type can hide a member or type parameter declared in the outer type:
Class Outer(Of T) Class Inner(Of T) Public t1 As T End Class End Class
Types and methods may be overloaded based on the number of type parameters (or arity) that the types or methods declare. For example, the following declarations are legal:
Module C Sub M() End Sub Sub M(Of T)() End Sub Sub M(Of T, U)() End Sub End Module Structure C(Of T) Dim x As T End Structure Class C(Of T, U) End Class
In the case of types, overloads are always matched against the number of type arguments specified. This is useful when using both generic and non-generic classes together in the same program:
Class Queue End Class Class Queue(Of T)
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
80
4.3
Inheritance
End Class Class X Dim q1 As Queue Dim q2 As Queue(Of Integer) End Class ' Non-generic queue ' Generic queue
Rules for methods overloaded on type parameters are covered in the section on method overload resolution. Within the containing declaration, type parameters are considered full types. Since a type parameter can be instantiated with many different actual type arguments, type parameters have slightly different operations and restrictions than other types as described below: A type parameter cannot be used directly to declare a base class or interface. The rules for member lookup on type parameters depend on the constraints, if any, applied to the type parameter. The available conversions for a type parameter depend on the constraints, if any, applied to the type parameters. In the absence of a Structureconstraint, a value with a type represented by a type parameter can be compared with Nothing using Is and IsNot. A type parameter can only be used in a New expression if the type parameter is constrained by a New or a Structure constraint. A type parameter cannot be used anywhere within an attribute exception within a GetType expression. Type parameters can be used as type arguments to other generic types and parameters. The following example is a generic type that extends the Stack(Of ItemType) class:
Class MyStack(Of I temType) Inher i t s Stack(Of I temType) Publ i c ReadOnly Property Size( ) As Integer
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
81
4.3
Inheritance
When a declaration supplies a type argument to MyStack , the same type argument will be applied to Stack as well. As a type, type parameters are purely a compile-time construct. At run-time, each type parameter is bound to a run-time type that was specified by supplying a type argument to the generic declaration. Thus, the type of a variable declared with a type parameter will, at run-time, be a non-generic type or a specific constructed type. The run-time execution of all statements and expressions involving type parameters uses the actual type that was supplied as the type argument for that parameter. TypeParameterList ::= OpenParenthesis Of TypeParameters CloseParenthesis TypeParameters ::= TypeParameter | TypeParameters Comma TypeParameter TypeParameter ::= [ VarianceModifier ] Identifier [ TypeParameterConstraints ] VarianceModifier ::= In | Out 1.21.2 Type Constraints Because a type argument can be any type in the type system, a generic type or method cannot make any assumptions about a type parameter. Thus, the members of a type parameter are considered to be the members of the type Object, since all types derive from Object. In the case of a collection like Stack(Of I temType) , this fact may not be a particularly important restriction, but there may be cases where a generic type may wish to make an
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
82
4.3
Inheritance
assumption about the types that will be supplied as type arguments. Type constraints can be placed on type parameters that restrict which types can be supplied as a type parameter and allow generic types or methods to assume more about type parameters.
Public Class DisposableStack(Of ItemType As IDisposable) Implements IDisposable Private _items(0 To 99) As ItemType Private _currentIndex As Integer = 0 Public Sub Push(data As ItemType) ... End Sub Public Function Pop() As ItemType ... End Function Private Sub Dispose() Implements IDisposable.Dispose For Each item As IDisposable In _items If item IsNot Nothing Then item.Dispose() End If Next item End Sub End Class
In this example, the DisposableStack(Of I temType) constrains its type parameter to only types that implement the interface System.IDisposableAs a result, it can implement a . Dispose method that disposes any objects still left in the queue. A type constraint must be one of the special constraints Class, Structure, or New, or it must be a type T where:
T is a class, an interface, or a type parameter. T is not NotInheritable.
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
83
4.3
Inheritance
T is not one of, or a type inherited from one of, the following special types: System.Array , System.Delegate, System.MulticastDelegate, System.Enum, or System.ValueType. T is not Object. Since all types derive from Object, such a constraint would have no effect if
it were permitted.
T must be at least as accessible as the generic type or method being declared.
Multiple type constraints can be specified for a single type parameter by enclosing the type constraints in curly braces ({}).. Only one type constraint for a given type parameter can be a class. It is an error to combine a Structure special constraint with a named class constraint or the Class special constraint.
Class ControlFactory(Of T As {Control , ... End Class New})
Type constraints can use the containing types or any of the containing types type parameters. In the following example, the constraint requires that the type argument supplied implements a generic interface using itself as a type argument:
Class Sorter (Of V As IComparable(Of V)) ... End Class
The special type constraint Class constrains the supplied type argument to any reference type. Annotation The special type constraint Class can be satisfied by an interface. And a structure can implement an interface. Therefore, the constraint (Of T As U, U As Class) might be satisfied with T a structure (which does not satisfy the Class special constraint), and U an interface that it implements (which does satisfy the Class special constraint). The special type constraint Structure constrains the supplied type argument to any value type except System.Nullable(Of T).
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
84
4.3
Inheritance
Annotation Structure constraints do not allow System.Nul lab le (Of so that it is not possible to supply T) System.Nul lab le (Of as a type argument to itself. T) The special type constraint New requires that the supplied type argument must have an accessible parameterless constructor and cannot be declared MustInher i. For example: t
Class Factory(Of T As New) Funct ion Create Instance() As T Return New T() End Funct ion End Class
A class type constraint requires that the supplied type argument must either be that type as or inherit from it. An interface type constraint requires that the supplied type argument must implement that interface. A type parameter constraint requires that the supplied type argument must derive from or implement all of the bounds given for the matching type parameter. For example:
Class Lis t (Of T) Sub AddRange(Of S As T)(co l l e c t i on As IEnumerable(Of S)) ... End Sub End Class
In this example, the type parameter S on AddRange is constrained to the type parameter T of List. This means that a List(Of Control) would constrain AddRanges type parameter to any type that is or inherits from Control. A type parameter constraint "Of S As T" is resolved by transitively adding all of T's constraints onto S, other than the special constraints (Class, Structure, New). It is an error to have circular constraints (e.g. Of S As T, T As S). It is an error to have a type parameter constraint which itself has the Structure constraint. After adding constraints, it is possible that a number of special situations may occur:
85
4.3
Inheritance
If multiple class constraints exist, the most derived class is considered to be the constraint. If one or more class constraints have no inheritance relationship, the constraint is unsatisfiable and it is an error. If a type parameter combines a Structurespecial constraint with a named class constraint or the Class special constraint, it is an error.A class constraint may be NotInheritable, in which case no derived types of that constraint are accepted and it is an error. The type may be one of, or a type inherited from, the following special types: System.Array, System.Delegate, System.MulticastDelegate, System.Enum, or System.ValueType. In that case, only the type, or a type inherited from it, is accepted. A type parameter constrained to one of these types can only use the conversions allowed by the DirectCast operator. For example:
MustInherit Class Base(Of T) MustOverride Sub S1(Of U As T)(x As U) End Class Class Derived Inherits Base(Of Integer) ' The constraint of U must be Integer, which is normally not allowed. Overrides Sub S1(Of U As Integer)(x As U) Dim y As Integer = x ' OK Dim z As Long = x ' Error: Can't convert End Sub End Class
Additionally, a type parameter constrained to a value type due to one of the above relaxations cannot call any methods defined on that value type. For example:
Class C1(Of T) Overridable Sub F(Of G As T)(x As G) End Sub End Class Class C2
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
86
4.3
Inheritance
Inherits C1(Of IntPtr) Overrides Sub F(Of G As IntPtr)(ByVal x As G) ' Error: Cannot access structure members x.ToInt32() End Sub End Class
If the constraint, after substitution, ends up as an array type, any covariant array type is allowed as well. For example:
Module Test Class B End Class Class D Inher i t s B End Class Funct ion F(Of T, U As T)(x As U) As T Return x End Funct ion Sub Main() Dim a(9) As B Dim b(9) As D a = F(Of B() , D()) (b ) End Sub End Module
A type parameter with a class or interface constraint is considered to have the same members as that class or interface constraint. If a type parameter has multiple constraints, then the type parameter is considered to have the union of all the members of the constraints. If there are members with the same name in more than one constraint, then members are hidden in the following order: the class constraint hides members in interface constraints, which hide members in System.ValueType(if Structureconstraint is specified),
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
87
4.3
Inheritance
which hides members in Object. If a member with the same name appears in more than one interface constraint the member is unavailable (as in multiple interface inheritance) and the type parameter must be cast to the desired interface. For example:
Class C1 Sub S1(x As Integer ) End Sub End Class Inter face I1 Sub S1(x As Integer ) End Inter face Inter face I2 Sub S1(y As Double) End Inter face Module Test Sub T1(Of T As {C1, I1 , I2}) ( ) Dim a As T a.S1(10) ' Cal l s C1.S1, which i s prefer red a.S1(10.10) ' Also cal l s C1.S1, class i s st i l l End Sub
prefer red
Sub T2(Of T As {I1 , I2}) ( ) Dim a As T a.S1(10) ' Error : Cal l i s ambiguous between I1 .S1 , I2 .S1 End Sub End Module
When supplying type parameters as type arguments, the type parameters must satisfy the constraints of the matching type parameters.
Class Base(Of T As Class) End Class Class Derived(Of V)
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
88
4.3
Inheritance
' Error: V does not satisfy the constraints of T Inherits Base(Of V) End Class
Values of a constrained type parameter can be used to access the instance members, including instance methods, specified in the constraint.
Interface IPrintable Sub Print() End Interface Class Printer(Of V As IPrintable) Sub PrintOne(v1 As V) V1.Print() End Sub End Class
TypeParameterConstraints ::= As Constraint | As OpenCurlyBrace ConstraintList CloseCurlyBrace ConstraintList ::= ConstraintList Comma Constraint | Constraint Constraint ::= TypeName | New | Structure | Class 1.21.3 Type Parameter Variance A type parameter in an interface or a delegate type declaration can optionally specify a variance modifier. Type parameters with variance modifiers restrict how the type parameter can be used in the interface or delegate type but allow a generic interface or delegate type to be converted to another generic type with variant compatible type arguments. For example:
Class Base End Class Class Derived
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
89
4.3
Inheritance
Inherits Base End Class Module Test Sub Main() Dim x As IEnumerable(Of Derived) = ... ' OK, as IEnumerable(Of Base) is variant compatible ' with IEnumerable(Of Derived) Dim y As IEnumerable(Of Base) = x End Sub End Module
Generic interfaces that have type parameters with variance modifiers have several restrictions: They cannot contain an event declaration that specifies a parameter list (but a custom event declaration or an event declaration with a delegate type is allowed). They cannot contain a nested class, structure, or enumerated type. Annotation These restrictions are due to the fact that types nested in generic types implicitly copy the generic parameters of their parent. In the case of nested classes, structures, or enumerated types, those kinds of types cannot have variance modifiers on their type parameters. In the case of an event declaration with a parameter list, the generated nested delegate class could have confusing errors when a type that appears to be used in an In position (i.e. a parameter type) is actually used in an Out position (i.e. the type of the event). A type parameter that is declared with the Out modifier is covariant. Informally, a covariant type parameter can only be used in an output positioni.e. a value that is being returned from the interface or delegate typeand cannot be used in an input position. A type T is considered to be valid covariantly if:
T is a class, structure, or enumerated type.
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
90
4.3
Inheritance
T is non-generic delegate or interface type. T is an array type whose element type is valid covariantly. T is a type parameter which was not declared as an Out type parameter. T is a constructed interface or delegate type X(Of P,,Pn) with type arguments A1,,An such 1
that:
If Pi was declared as an Out type parameter then Ai is valid covariantly. If Pi was declared as an In type parameter then Ai is valid contravariantly. The following must be valid covariantly in an interface or delegate type: The base interface of an interface. The return type of a function or the delegate type. The type of a property if there is a Get accessor. The type of any ByRef parameter. For example:
Delegate Funct ion D(Of Out T, U)(x As U) As T Inter face I1 (Of Out T) End Inter face Inter face I2(Of Out T) Inher i t s I1 (Of T) ' OK, T i s only used in an Out posi t i on Funct ion M1(x As I1 (Of T)) As T ' Error : T i s used in an In posi t i on Funct ion M2(x As T) As T End Inter face
91
4.3
Inheritance
A type parameter that is declared with the In modifier is contravariant. Informally, a contravariant type parameter can only be used in an input positioni.e. a value that is being passed in to the interface or delegate typeand cannot be use in an output position. A type T is considered to be valid contravariantly if:
T is a class, structure, or enumerated type. T is a non-generic delegate or interface type. T is an array type whose element type is valid contravariantly. T is a type parameter which was not declared as an In type parameter. T is a constructed interface or delegate type X(Of P,,Pn) with type arguments A1,,An such 1
that:
If Pi was declared as an Out type parameter then Ai is valid contravariantly. If Pi was declared as an In type parameter then Ai is valid covariantly. The following must be valid contravariantly in an interface or delegate type: The type of a parameter. A type constraint on a method type parameter. The type of a property if it has a Set accessor. The type of an event. For example:
Delegate Funct ion D(Of T, In U)(x As U) As T Inter face I1 (Of In T) End Inter face Inter face I2(Of In T) ' OK, T i s only used in an In posi t i on Sub M1(x As I1 (Of T))
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
92
4.3
Inheritance
In the case where a type must be valid be contravariantly and covariantly (such as a property with both a Get and Set accessor or a ByRef parameter), a variant type parameter cannot be used. Note Out is not a reserved word.
93
Attributes
The Visual Basic language enables the programmer to specify modifiers on declarations, which represent information about the entities being declared. For example, affixing a class method with the modifiers Publ i c Protected Friend, Protected Friend, or Private specifies , , its accessibility. In addition to the modifiers defined by the language, Visual Basic also enables programmers to create new modifiers, called attributes, and to use them when declaring new entities. These new modifiers, which are defined through the declaration of attribute classes, are then assigned to entities through attribute blocks. Note Attributes may be retrieved at run time through the .NET Framework's reflection APIs. These APIs are outside the scope of this specification. For instance, a framework might define a Help attribute that can be placed on program elements such as classes and methods to provide a mapping from program elements to documentation, as the following example demonstrates:
<Attr ibuteUsage(Att r i bu teTargets .A l l )> _ Publ i c Class HelpAttr i bu te Inher i t s Attr i bute Publ i c Sub New(urlVa lue As Str ing) Me.UrlVa lue = ur lVa lue End Sub Publ i c Topic As Str ing Pr ivate UrlValue As Str ing Publ i c ReadOnly Property Url ( ) Get Return UrlValue End Get End Property End Class As Str ing
The example defines an attribute class named HelpAttribute, or Help for short, that has one positional parameter (UrlValue) and one named argument (Topic). The next example shows several uses of the attribute:
<Help("ht tp : / /www.example.com/. . . / C l a ss1 .htm")> _ Publ i c Class Class1
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
94
4.3
Inheritance
The next example checks to see if Class1 has a Help attribute, and writes out the associated Topic and Url values if the attribute is present.
Module Test Sub Main() Dim type As Type = GetType(Class1) Dim arr ( ) As Object = _ type.GetCustomAttr ibutes(GetType(HelpAtt r i bu te ) ,
True)
I f arr . Length = 0 Then Console .Wri teL ine ("C lass1 has no Help att r i bute . " ) Else Dim ha As HelpAttr i bu te = CType(arr (0 ) , HelpAttr i bu te ) Console .Wri teL ine ("Ur l = " & ha.Ur l & " , Topic = " & ha.Topic ) End I f End Sub End Module
95
4.3
Inheritance
The next example shows a few uses of the Simple attribute. Although the attribute class is named SimpleAtt r i bute , uses of this attribute may omit the Attribute suffix, thus shortening the name to Simple:
<Simple> Class Class1 End Class <Simple> Inter face Inter face1 End Inter face
If the attribute lacks a System.AttributeUsage, then the attribute can be placed onany target (equivalent to AttributeTargets.All). The System.AttributeUsage attribute has a variable initializer, AllowMultiple, which specifies whether the indicated attribute can be specified more than once for a given declaration. If AllowMultiple for an attribute is True, it is a multiple-use attribute class, and can be specified more than once on a declaration. If AllowMultiple for an attribute is False or unspecified for an attribute, it is a single-use attribute class, and can be specified at most once on a declaration. The following example defines a multiple-use attribute class named AuthorAttribute:
<Attr ibuteUsage(Att r i bu teTargets .C lass , Al lowMult ip l e :=True)> _ Publ i c Class AuthorAtt r i bu te Inher i t s System.Attr i bu te Pr ivate _Value As Str ing Publ i c Sub New(value As Str ing) Me._Value = value End Sub Publ i c ReadOnly Property Value() As Str ing Get Return _Value End Get End Property End Class
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
96
4.3
Inheritance
The example shows a class declaration with two uses of the Author attribute:
<Author("Mar ia Hammond"), Author("Ramesh Meyyappan")> _ Class Class1 End Class
The System.Att r i buteUsage attribute has a public instance variable, Inher i tedthat specifies , whether the attribute, when specified on a base type, is also inherited by types that derive from this base type. If the Inherited public instance variable is not initialized, a default value of False is used. Properties and events do not inherit attributes, although the methods defined by properties and events do. Interfaces do not inherit attributes. If a single-use attribute is both inherited and specified on a derived type, the attribute specified on the derived type overrides the inherited attribute. If a multiple-use attribute is both inherited and specified on a derived type, both attributes are specified on the derived type. For example:
<Attr ibuteUsage(Att r i bu teTargets .C lass , Al lowMult ip l e :=True, _ Inher i ted :=True) > _ Class Mult iUseAtt r i bute Inher i t s System.Attr i bu te Publ i c Sub New(value As Boolean) End Sub End Class <Attr ibuteUsage(Att r i bu teTargets .C lass , Inher i ted :=True)> _ Class SingleUseAttr i bute Inher i t s Attr i bute Publ i c Sub New(value As Boolean) End Sub End Class <SingleUse(True) , Mult iUse(True)> Class Base End Class
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
97
4.3
Inheritance
' Derived has three attributes defined on it: SingleUse(False), ' MultiUse(True) and MultiUse(False) <SingleUse(False), MultiUse(False)> _ Class Derived Inherits Base End Class
The positional parameters of the attribute are defined by the parameters of the public constructors of the attribute class. Positional parameters must be ByVal and may not specify ByRef. Public instance variables and properties are defined by public read-write properties or instance variables of the attribute class. The types that can be used in positional parameters and public instance variables and properties are restricted to attribute types. A type is an attribute type if it is one of the following: Any primitive type except for Date and Decimal. The type Object. The type System.Type. An enumerated type, provided that it and the types in which it is nested (if any) have Publ i c accessibility. A one-dimensional array of one of the previous types in this list.
98
4.3
Inheritance
<AttributeUsage(AttributeTargets.Class)> _ Public Class HelpStringAttribute Inherits System.Attribute Private InternalValue As String Public Sub New(value As String) Me.InternalValue = value End Sub Public ReadOnly Property Value() As String Get Return InternalValue End Get End Property End Class ' Error: HelpString only applies to classes. <HelpString("Description of Interface1")> _ Interface Interface1 Sub Sub1() End Interface ' Error: HelpString is single-use. <HelpString("Description of Class1"), _ HelpString("Another description of Class1")> _ Public Class Class1 End Class
An attribute consists of an optional attribute modifier, an attribute name, an optional list of positional arguments, and variable/property initializers. If there are no parameters or initializers, the parentheses may be omitted. If an attribute has a modifier, it must be in an attribute block at the top of a source file.
99
4.3
Inheritance
If a source file contains an attribute block at the top of the file that specifies attributes for the assembly or module that will contain the source file, each attribute in the attribute block must be prefixed by both the Assembly or Module modifier and a colon. Attributes ::= AttributeBlock | Attributes AttributeBlock AttributeBlock ::= [ LineTerminator ] < AttributeList [ LineTerminator ] > [ LineTerminator ] AttributeList ::= Attribute | AttributeList Comma Attribute Attribute ::= [ AttributeModifier : ] SimpleTypeName [ OpenParenthesis [ AttributeArguments ] CloseParenthesis ] AttributeModifier ::= Assembly | Module 1.23.1 Attribute Names The name of an attribute specifies an attribute class. By convention, attribute classes are named with the suffix Attr i buteUses of an attribute may either include or omit this suffix. . Consequently the name of an attribute class that corresponds to an attribute identifier is either the identifier itself or the concatenation of the qualified identifier and Attr i buteWhen . the compiler resolves an attribute name, it appends Attribute to the name and tries the lookup. If that lookup fails, the compiler tries the lookup without the suffix. For example, uses of an attribute class SimpleAttribute may omit the Attribute suffix, thus shortening the name to Simple:
<Simple> Class Class1 End Class <Simple> Inter face Inter face1 End Inter face
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
100
4.3
Inheritance
In general, attributes named with the suffix Attr i bute are preferred. The following example shows two attribute classes named T and TAttribute.
<Attr ibuteUsage(Att r i bu teTargets .A l l )> _ Publ i c Class T Inher i t s System.Attr i bu te End Class <Attr ibuteUsage(Att r i bu teTargets .A l l )> _ Publ i c Class TAttr i bute Inher i t s System.Attr i bu te End Class ' Refers to TAttr i bute . <T> Class Class1 End Class ' Refers to TAttr i bute . <TAttr ibute> Class Class2 End Class
Both the attribute block <T> and the attribute block <TAttribute> refer to the attribute class named TAttribute. It is not possible to use T as an attribute until you remove the declaration for class TAttribute. 1.23.2 Attribute Arguments Arguments to an attribute may take two forms: positional arguments and instance variable/property initializers. Any positional arguments to the attribute must precede the instance variable/property initializers. A positional argument consists of a constant
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
101
4.3
Inheritance
expression, a one-dimensional array-creation expression or a GetType expression. An instance variable/property initializer consists of an identifier, which can match keywords, followed by a colon and equal sign, and terminated by a constant expression or a GetType expression. Given an attribute with attribute class T, positional argument list P, and instance variable/property initializer list N, these steps determine whether the arguments are valid: Follow the compile-time processing steps for compiling an expression of the form New T(P). This either results in a compile-time error or determines a constructor on T that is most applicable to the argument list. If the constructor determined in step 1 has parameters that are not attribute types or is inaccessible at the declaration site, a compile-time error occurs. For each instance variable/property initializer Arg in N: Let Name be the identifier of the instance variable/property initializer Arg.
Name must identify a non-Shared, writeable, Public instance variable or parameterless property on T whose type is an attribute type. If T has no such instance variable or
102
4.3
Inheritance
Get End Get Set End Set End Property End Class ' Calls the first constructor. <General(10, z:=30, y:=GetType(Integer))> _ Class C1 End Class ' Calls the second constructor. <General(10.5, z:=10)> _ Class C2 End Class
Type parameters cannot be used anywhere in attribute arguments. However, constructed types may be used:
<AttributeUsage(AttributeTargets.All)> _ Class A Inherits System.Attribute Public Sub New(t As Type) End Sub End Class Class List(Of T) ' Error: attribute argument cannot use type parameter <A(GetType(T))> Dim t1 As T ' OK: closed type <A(GetType(List(Of Integer)))> Dim y As Integer End Class
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
103
4.3
Inheritance
AttributeArguments ::= AttributePositionalArgumentList | AttributePositionalArgumentList Comma VariablePropertyInitializerList | VariablePropertyInitializerList AttributePositionalArgumentList ::= [ AttributeArgumentExpression ] | AttributePositionalArgumentList Comma [ AttributeArgumentExpression ] VariablePropertyInitializerList ::= VariablePropertyInitializer | VariablePropertyInitializerList Comma VariablePropertyInitializer VariablePropertyInitializer ::= IdentifierOrKeyword ColonEquals AttributeArgumentExpression AttributeArgumentExpression ::= ConstantExpression | GetTypeExpression | ArrayExpression
104
FileB.vb:
Class B End Class
The two source files contribute to the global namespace, in this case declaring two classes with the fully qualified names A and B. Because the two source files contribute to the same declaration space, it would have been an error if each contained a declaration of a member with the same name. Note The compilation environment may override the namespace declarations into which a source file is implicitly placed. Except where noted, statements within a Visual Basic program can be terminated either by a line terminator or by a colon. Start ::= [ OptionStatement+ ] [ ImportsStatement+ ] [ AttributesStatement+ ] [ NamespaceMemberDeclaration+ ] StatementTerminator ::= LineTerminator | : AttributesStatement ::= Attributes StatementTerminator
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
105
4.3
Inheritance
The accessibility of the entry point method is irrelevant. If a program contains more than one suitable entry point, the compilation environment must designate one as the entry point. Otherwise, a compile-time error occurs. The compilation environment may also create an entry point method if one does not exist. When a program begins, if the entry point has a parameter, the argument supplied by the execution environment contains the command-line arguments to the program represented as strings. If the entry point has a return type of Integer then the value returned from the , function is returned to the execution environment as the result of the program. In all other respects, entry point methods behave in the same manner as other methods. When execution leaves the invocation of the entry point method made by the execution environment, the program terminates.
' Not al lowed, Option Str i c t i s al ready speci f i ed . ' Not al lowed, Option Compare i s al ready speci f i ed .
106
4.3
Inheritance
There are four compilation options: strict type semantics, explicit declaration semantics, comparison semantics, and local variable type inference semantics. If a source file does not include a particular Option statement, then the compilation environment determines which particular set of semantics will be used. There is also a fifth compilation option, integer overflow checks, which can only be specified through the compilation environment. OptionStatement ::= OptionExplicitStatement | OptionStrictStatement | OptionCompareStatement | OptionInferStatement 1.25.1 Option Explicit Statement The Option Expl i c i t statement determines whether local variables may be implicitly declared. The keywords On or Off may follow the statement; if neither is specified, the default is On. If no statement is specified in a file, the compilation environment determines which will be used. Note Explicit and Off are not reserved words.
Option Expl i c i t Off
Module Test Sub Main() x = 5 ' Val id because Option Expl i c i t End Sub End Module
i s off .
In this example, the local variable x is implicitly declared by assigning to it. The type of x is Object. OptionExplicitStatement ::= Option Explicit [ OnOff ] StatementTerminator OnOff ::= On | Off
107
4.3
Inheritance
1.25.2 Option Strict Statement The Option Str i c tstatement determines whether conversions and operations on Object are governed by strict or permissive type semantics and whether types are implicitly typed as Object if no As clause is specified. The statement may be followed by the keywords On or Off; if neither is specified, the default is On. If no statement is specified in a file, the compilation environment determines which will be used. Note Strict and Off are not reserved words.
Option Str i c t On Module Test Sub Main() Dim x ' Error , no type speci f i ed . Dim o As Object Dim b As Byte = o ' Error , narrowing convers ion . o.F( ) ' Error , la te binding disa l l owed. o = o + 1 ' Error , addit i on i s not def ined on Object . End Sub End Module
Under strict semantics, the following are disallowed: Narrowing conversions without an explicit cast operator. Late binding. Operations on type Object other than TypeOf...Is, Is, and IsNot. Omitting the As clause in a declaration that does not have an inferred type. OptionStrictStatement ::= Option Strict [ OnOff ] StatementTerminator 1.25.3 Option Compare Statement The Option Compare statement determines the semantics of string comparisons. String comparisons are carried out either using binary comparisons (in which the binary Unicode value of each character is compared) or text comparisons (in which the lexical meaning of
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
108
4.3
Inheritance
each character is compared using the current culture). If no statement is specified in a file, the compilation environment controls which type of comparison will be used. Note Compare , Binary and Text are not reserved words. ,
Option Compare Text Module Test Sub Main() Console .Wri teL ine ("a" = "A") End Sub End Module
In this case, the string comparison is done using a text comparison that ignores case differences. If Option Compare Binary had been specified, then this would have printed False. OptionCompareStatement ::= Option Compare CompareOption StatementTerminator CompareOption ::= Binary | Text 1.25.4 Integer Overflow Checks Integer operations can either be checked or not checked for overflow conditions at run time. If overflow conditions are checked and an integer operation overflows, a System.OverflowException exception is thrown. If overflow conditions are not checked, integer operation overflows do not throw an exception. The compilation environment determines whether this option is on or off. 1.25.5 Option Infer Statement The Option Infer statement determines whether local variable declarations that have no As clause have an inferred type or use Object. The statement may be followed by the keywords On or Off; if neither is specified, the default is On. If no statement is specified in a file, the compilation environment determines which will be used. Note Infer and Off are not reserved words.
Option Infer On Module Test
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
109
4.3
Inheritance
Sub Main() ' The type of x is Integer Dim x = 10 ' The type of y is String Dim y = "abc" End Sub End Module
referenced without qualification, or import a namespace for use in XML expressions. Within member declarations in a source file that contains an Imports statement, the types contained in the given namespace can be referenced directly, as seen in the following example:
Imports N1.N2 Namespace N1.N2 Class A End Class End Namespace Namespace N3 Class B Inher i t s A End Class End Namespace
Here, within the source file, the type members of namespace N1.N2 are directly available, and thus class N3.B derives from class N1.N2.A.
Imports statements must appear after any Option statements but before any type declarations. The compilation environment may also define implicit Imports statements.
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
110
4.3
Inheritance
Imports statements make names available in a source file, but do not declare anything in the global namespace's declaration space. The scope of the names imported by an Imports
statement extends over the namespace member declarations contained in the source file. The scope of an Imports statement specifically does not include other Imports statements, nor does it include other source files. Imports statements may not refer to one another. In this example, the last Imports statement is in error because it is not affected by the first import alias.
Imports R1 = N1 ' OK. Imports R2 = N1.N2 ' OK. Imports R3 = R1.N2 ' Error : Can't refer to R1. Namespace N1.N2 End Namespace
Note The namespace or type names that appear in Imports statements are always treated as if they are fully qualified. That is, the leftmost identifier in a namespace or type name always resolves in the global namespace and the rest of the resolution proceeds according to normal name resolution rules. This is the only place in the language that applies such a rule; the rule ensures that a name cannot be completely hidden from qualification. Without the rule, if a name in the global namespace were hidden in a particular source file, it would be impossible to specify any names from that namespace in a qualified way. In this example, the Imports statement always refers to the global System namespace, and not the class in the source file.
Imports System Class System End Class ' Imports the namespace, not the class .
ImportsStatement ::= Imports ImportsClauses StatementTerminator ImportsClauses ::= ImportsClause | ImportsClauses Comma ImportsClause
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
111
4.3
Inheritance
ImportsClause ::= AliasImportsClause | MembersImportsClause | XMLNamespaceImportsClause 1.26.1 Import Aliases An import alias defines an alias for a namespace or type.
Imports A = N1.N2.A Namespace N1.N2 Class A End Class End Namespace Namespace N3 Class B Inherits A End Class End Namespace
Here, within the source file, A is an alias for N1.N2.A, and thus class N3.B derives from class N1.N2.A. The same effect can be obtained by creating an alias R for N1.N2 and then referencing R.A:
Imports R = N1.N2 Namespace N3 Class B Inher i t s R.A End Class End Namespace
The identifier of an import alias must be unique within the declaration space of the global namespace (not just the global namespace declaration in the source file in which the import alias is defined), even though it does not declare a name in the global namespace's declaration space. Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved. 112
4.3
Inheritance
Annotation Declarations in a module do not introduce names into the containing declaration space. Thus, it is valid for a declaration in a module to have the same name as an import alias, even though the declarations name will be accessible in the containing declaration space.
' Error: Alias A conflicts with typename A Imports A = N3.A Class A End Class Namespace N3 Class A End Class End Namespace
Here, the global namespace already contains a member A, so it is an error for an import alias to use that identifier. It is likewise an error for two or more import aliases in the same source file to declare aliases by the same name. An import alias can create an alias for any namespace or type. Accessing a namespace or type through an alias yields exactly the same result as accessing the namespace or type through its declared name.
Imports R1 = N1 Imports R2 = N1.N2 Namespace N1.N2 Class A End Class End Namespace Namespace N3 Class B Pr ivate a As N1.N2.A Pr ivate b As R1.N2.A Pr ivate c As R2.A
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
113
4.3
Inheritance
Here, the names N1.N2.A, R1.N2.A, and R2.A are equivalent, and all refer to the class whose fully qualified name is N1.N2.A. Declarations in the source file may shadow the import alias name.
Imports R = N1.N2 Namespace N1.N2 Class A End Class End Namespace Namespace N3 Class R End Class Class B Inher i t s R.A End Class End Namespace ' Error , R has no member A
In the preceding example the reference to R.A in the declaration of B causes an error because R refers to N3.R, not N1.N2. An import alias makes an alias available within a particular source file, but it does not contribute any new members to the underlying declaration space. In other words, an import alias is not transitive, but rather affects only the source file in which it occurs. File1.vb:
Imports R = N1.N2 Namespace N1.N2 Class A End Class End Namespace
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
114
4.3
Inheritance
File2.vb:
Class B Inherits R.A ' Error, R unknown. End Class
In the above example, because the scope of the import alias that introduces R only extends to declarations in the source file in which it is contained, R is unknown in the second source file. AliasImportsClause ::= Identifier Equals TypeName 1.26.2 Namespace Imports A namespace import imports all of the members of a namespace or type, allowing the identifier of each member of the namespace or type to be used without qualification. In the case of types, a namespace import only allows access to the shared members of the type without requiring qualification of the class name. In particular, it allows the members of enumerated types to be used without qualification. For example:
Imports Colors Enum Colors Red Green Blue End Enum Module M1 Sub Main() Dim c As Colors = Red End Sub End Module
Unlike an import alias, a namespace import has no restrictions on the names it imports and may import namespaces and types whose identifiers are already declared within the global Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved. 115
4.3
Inheritance
namespace. The names imported by a regular import are shadowed by import aliases and declarations in the source file. In the following example, A refers to N3.A rather than N1.N2.A within member declarations in the N3 namespace.
Imports N1.N2 Namespace N1.N2 Class A End Class End Namespace Namespace N3 Class A End Class Class B Inher i t s A End Class End Namespace
When more than one imported namespace contains members by the same name (and that name is not otherwise shadowed by an import alias or declaration), a reference to that name is ambiguous and causes a compile-time error.
Imports N1 Imports N2 Namespace N1 Class A End Class End Namespace Namespace N2 Class A End Class
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
116
4.3
Inheritance
End Namespace Namespace N3 Class B Inherits A ' Error, A is ambiguous. End Class End Namespace
In the above example, both N1 and N2 contain a member A. Because N3 imports both, referencing A in N3 causes a compile-time error. In this situation, the conflict can be resolved either through qualification of references to A, or by introducing an import alias that picks a particular A, as in the following example:
Imports N1 Imports N2 Imports A = N1.A Namespace N3 Class B Inher i t s A ' A means N1.A. End Class End Namespace
Only namespaces, classes, structures, enumerated types, and standard modules may be imported. MembersImportsClause ::= TypeName 1.26.3 XML Namespace Imports An XML namespace import defines a namespace or the default namespace for unqualified XML expressions contained within the compilation unit. For example:
Imports <xmlns:db="http: / / example.org /database"> Module Test Sub Main() ' db namespace i s "http : / / example.org /database"
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
117
4.3
Inheritance
An XML namespace, including the default namespace, can only be defined once for a particular set of imports. For example:
Imports <xmlns:db="http://example.org/database-one"> ' Error: namespace db is already defined Imports <xmlns:db="http://example.org/database-two">
XMLNamespaceImportsClause ::= < XMLNamespaceAttributeName [ XMLWhitespace ] Equals [ XMLWhitespace ] XMLNamespaceValue > XMLNamespaceValue ::= DoubleQuoteCharacter [ XMLAttributeDoubleQuoteValueCharacter+ ] DoubleQuoteCharacter | SingleQuoteCharacter [ XMLAttributeSingleQuoteValueCharacter+ ] SingleQuoteCharacter
1.27 Namespaces
Visual Basic programs are organized using namespaces. Namespaces both internally organize a program as well as organize the way program elements are exposed to other programs. Unlike other entities, namespaces are open-ended, and may be declared multiple times within the same program and across many programs, with each declaration contributing members to the same namespace. In the following example, the two namespace declarations contribute to the same declaration space, declaring two classes with the fully qualified names N1.N2.A and N1.N2.B.
Namespace N1.N2 Class A End Class
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
118
4.3
Inheritance
Because the two declarations contribute to the same declaration space, it would be an error if each contained a declaration of a member with the same name. There is a global namespace that has no name and whose nested namespaces and types can always be accessed without qualification. The scope of a namespace member declared in the global namespace is the entire program text. Otherwise, the scope of a type or namespace declared in a namespace whose fully qualified name is N is the program text of each namespace whose corresponding namespace's fully qualified name begins with N or is N itself. (Note that a compiler can choose to put declarations in a particular namespace by default. This does not alter the fact that there is still a global, unnamed namespace.) In this example, the class B can see the class A because B's namespace N1.N2.N3 is conceptually nested within the namespace N1.N2.
Namespace N1.N2 Class A End Class End Namespace Namespace N1.N2.N3 Class B Inher i t s A End Class End Namespace
1.27.1 Namespace Declarations A namespace declaration consists of the keyword Namespace followed by a qualified identifier and optional namespace member declarations. If the namespace name is qualified, the namespace declaration is treated as if it is lexically nested within namespace
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
119
4.3
Inheritance
declarations corresponding to each name in the qualified name. For example, the following two namespaces are semantically equivalent:
Namespace N1.N2 Class A End Class Class B End Class End Namespace Namespace N1 Namespace N2 Class A End Class Class B End Class End Namespace End Namespace
When dealing with the members of a namespace, it is not important where a particular member is declared. If two programs define an entity with the same name in the same namespace, attempting to resolve the name in the namespace causes an ambiguity error. Namespaces are by definition Publ i c so a namespace declaration cannot include any , access modifiers. NamespaceDeclaration ::= Namespace NamespaceName StatementTerminator [ NamespaceMemberDeclaration+ ] End Namespace StatementTerminator NamespaceName ::= Identifier | NamespaceName Period IdentifierOrKeyword
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
120
4.3
Inheritance
1.27.2 Namespace Members Namespace members can only be namespace declarations and type declarations. Type declarations may have Publ i c or Fr iendaccess. The default access for types is Friend access. NamespaceMemberDeclaration ::= NamespaceDeclaration | TypeDeclaration TypeDeclaration ::= ModuleDeclaration | NonModuleDeclaration NonModuleDeclaration ::= EnumDeclaration | StructureDeclaration | InterfaceDeclaration | ClassDeclaration | DelegateDeclaration
121
Types
The two fundamental categories of types in Visual Basic are value types and reference types. Primitive types (except strings), enumerations, and structures are value types. Classes, strings, standard modules, interfaces, arrays, and delegates are reference types. Every type has a default value, which is the value that is assigned to variables of that type upon initialization. TypeName ::= ArrayTypeName | NonArrayTypeName NonArrayTypeName ::= SimpleTypeName | NullableTypeName SimpleTypeName ::= QualifiedTypeName | BuiltInTypeName QualifiedTypeName ::= Identifier [ TypeArguments ] | Global Period IdentifierOrKeyword [ TypeArguments ] | QualifiedTypeName Period IdentifierOrKeyword [ TypeArguments ] TypeArguments ::= OpenParenthesis Of TypeArgumentList CloseParenthesis TypeArgumentList ::= TypeName | TypeArgumentList Comma TypeName BuiltInTypeName ::= Object | PrimitiveTypeName TypeModifier ::= AccessModifier | Shadows IdentifierModifiers ::= [ NullableNameModifier ] [ ArrayNameModifier ]
122
4.3
Inheritance
references, their lifetime is managed by the .NET Framework. Outstanding references to a particular instance are tracked and the instance is destroyed only when no more references remain. A variable of reference type contains a reference to a value of that type, a value of a more derived type, or a null value. A null value refers to nothing; it is not possible to do anything with a null value except assign it. Assignment to a variable of a reference type creates a copy of the reference rather than a copy of the referenced value. For a variable of a reference type, the default value is a null value. Value types are stored directly on the stack, either within an array or within another type; their storage can only be accessed directly. Because value types are stored directly within variables, their lifetime is determined by the lifetime of the variable that contains them. When the location containing a value type instance is destroyed, the value type instance is also destroyed. Value types are always accessed directly; it is not possible to create a reference to a value type. Prohibiting such a reference makes it impossible to refer to a value class instance that has been destroyed. Because value types are always NotInher i tab , a variable of a value type always contains a value of that type. Because of le this, the value of a value type cannot be a null value, nor can it reference an object of a more derived type. Assignment to a variable of a value type creates a copy of the value being assigned. For a variable of a value type, the default value is the result of initializing each variable member of the type to its default value. The following example shows the difference between reference types and value types:
Class Class1 Publ i c Value As Integer = 0 End Class Module Test Sub Main() Dim val1 As Integer = 0 Dim val2 As Integer = val1 val2 = 123 Dim ref1 As Class1 = New Class1() Dim ref2 As Class1 = ref1 ref2.Va lue = 123 Console .Wri teL ine ("Va lues: " & val1 & " , " & val2)
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
123
4.3
Inheritance
Console.WriteLine("Refs: " & ref1.Value & ", " & ref2.Value) End Sub End Module
The assignment to the local variable val2 does not impact the local variable val1 because both local variables are of a value type (the type Integer) and each local variable of a value type has its own storage. In contrast, the assignment ref2.Value = 123; affects the object that both ref1 and ref2 reference. One thing to note about the .NET Framework type system is that even though structures, enumerations and primitive types (except for String) are value types, they all inherit from reference types. Structures and the primitive types inherit from the reference type System.ValueType, which inherits from Object. Enumerated types inherit from the reference type System.Enum, which inherits from System.ValueType. 1.28.1 Nullable Value Types For value types, a ? modifier can be added to a type name to represent the nullable version of that type. A nullable value type can contain the same values as the non-nullable version of the type as well as the null value. Thus, for a nullable value type, assigning Nothing to a variable of the type sets the value of the variable to the null value, not the zero value of the value type. For example:
Dim x As Integer = Nothing Dim y As Integer? = Nothing ' Pr ints zero Console .Wri teL ine (x) ' Pr ints nothing (because the value of y i s the nul l value) Console .Wri teL ine (y)
A variable may also be declared to be of a nullable value type by putting a nullable type modifier on the variable name. For clarity, it is not valid to have a nullable type modifier on
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
124
4.3
Inheritance
both a variable name and a type name in the same declaration. Since nullable types are implemented using the type System.Nul lab le (Of , T) type T? is synonymous to the type the System.Nullable(Of T), and the two names can be used interchangeably. The ? modifier cannot be placed on a type that is already nullable; thus, it is not possible to declare the type Integer?? or System.Nullable(Of Integer)?. A nullable value type T? has the members of System.Nullable(Of T) as well as any operators or conversions lifted from the underlying type T into the type T?. Lifting copies operators and conversions from the underlying type, in most cases substituting nullable value types for non-nullable value types. This allows many of the same conversions and operations that apply to T to apply to T? as well. NullableTypeName ::= NonArrayTypeName ? NullableNameModifier ::= ?
125
4.3
Inheritance
... End Function Public Function CompareTo(other As Object) As Integer _ Implements IComparable.CompareTo ... End Function End Structure
A type that implements an interface also implicitly implements all of the interface's base interfaces. This is true even if the type does not explicitly list all base interfaces in the Implements clause. In this example, the TextBox structure implements both IControl and ITextBox.
Inter face IContro l Sub Paint ( ) End Inter face Inter face ITextBox Inher i t s IContro l Sub SetText( text As Str ing) End Inter face Structure TextBox Implements ITextBox ... Publ i c Sub Paint ( ) Implements ITextBox.Pa int ... End Sub Publ i c Sub SetText( text As Str ing) Implements ITextBox.SetText ... End Sub End Structure
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
126
4.3
Inheritance
Declaring that a type implements an interface in and of itself does not declare anything in the declaration space of the type. Thus, it is valid to implement two interfaces with a method by the same name. Types cannot implement a type parameter on its own, although it may involve the type parameters that are in scope.
Class C1(Of V) Implements V ' Error, can't implement type parameter directly Implements IEnumerable(Of V) ' OK, not directly implementing ... End Class
Generic interfaces can be implemented multiple times using different type arguments. However, a generic type cannot implement a generic interface using a type parameter if the supplied type parameter (regardless of type constraints) could overlap with another implementation of that interface. For example:
Interface I1(Of T) End Interface Class C1 Implements I1(Of Integer) Implements I1(Of Double) End Class Class C2(Of T) Implements I1(Of Integer) Implements I1(Of T) End Class
TypeImplementsClause ::= Implements TypeImplements StatementTerminator TypeImplements ::= NonArrayTypeName | TypeImplements Comma NonArrayTypeName
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
127
4.3
Inheritance
128
4.3
Inheritance
The Boolean value type, which represents a truth value, typically the result of a relational or logical operation. The literal is of type System.Boolean The default value of the Boolean type . is equivalent to the literal False. The Date value type, which represents a date and/or a time and maps to System.DateTime. The default value of the Date type is equivalent to the literal # 01/01/0001 12:00:00AM #. The Char value type, which represents a single Unicode character and maps to System.Char. The default value of the Char type is equivalent to the constant expression ChrW(0). The String reference type, which represents a sequence of Unicode characters and maps to System.String. The default value of the String type is a null value. PrimitiveTypeName ::= NumericTypeName | Boolean | Date | Char | String NumericTypeName ::= IntegralTypeName | FloatingPointTypeName | Decimal IntegralTypeName ::= Byte | SByte | UShort | Short | UInteger | Integer | ULong | Long FloatingPointTypeName ::= Single | Double
1.31 Enumerations
Enumerations are value types that inherit from System.Enum and symbolically represent a set of values of one of the primitive integral types. For an enumeration type E, the default value is the value produced by the expression CType(0, E). The underlying type of an enumeration must be an integral type that can represent all the enumerator values defined in the enumeration. If an underlying type is specified, it must be Byte, SByte, UShort, Short, UInteger, Integer, ULong, Long, or one of their corresponding types in the System namespace. If no underlying type is explicitly specified, the default is Integer. The following example declares an enumeration with an underlying type of Long:
Enum Color As Long Red Green Blue End Enum
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
129
4.3
Inheritance
A developer might choose to use an underlying type of Long, as in the example, to enable the use of values that are in the range of Long, but not in the range of Integer, or to preserve this option for the future. EnumDeclaration ::= [ Attributes ] [ TypeModifier+ ] Enum Identifier [ As NonArrayTypeName ] StatementTerminator EnumMemberDeclaration+ End Enum StatementTerminator 1.31.1 Enumeration Members The members of an enumeration are the enumerated values declared in the enumeration and the members inherited from class System.Enum. The scope of an enumeration member is the enumeration declaration body. This means that outside of an enumeration declaration, an enumeration member must always be qualified (unless the type is specifically imported into a namespace through a namespace import). Declaration order for enumeration member declarations is significant when constant expression values are omitted. Enumeration members implicitly have Public access only; no access modifiers are allowed on enumeration member declarations. EnumMemberDeclaration ::= [ Attributes ] Identifier [ Equals ConstantExpression ] StatementTerminator 1.31.2 Enumeration Values The enumerated values in an enumeration member list are declared as constants typed as the underlying enumeration type, and they can appear wherever constants are required. An enumeration member definition with = gives the associated member the value indicated by the constant expression. The constant expression must evaluate to an integral type that is implicitly convertible to the underlying type and must be within the range of values that can be represented by the underlying type. The following example is in error because the constant values 1.5, 2.3, and 3.3 are not implicitly convertible to the underlying integral type Long with strict semantics.
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
130
4.3
Inheritance
Option Strict On Enum Color As Long Red = 1.5 Green = 2.3 Blue = 3.3 End Enum
Multiple enumeration members may share the same associated value, as shown below:
Enum Color Red Green Blue Max = Blue End Enum
The example shows an enumeration that has two enumeration members Blue and Max that have the same associated value. If the first enumerator value definition in the enumeration has no initializer, the value of the corresponding constant is 0. An enumeration value definition without an initializer gives the enumerator the value obtained by increasing the value of the previous enumeration value by 1. This increased value must be within the range of values that can be represented by the underlying type.
Enum Color Red Green = 10 Blue End Enum Module Test Sub Main() Console .Wri teL ine (S t r i ngFromColor(Color .Red)) Console .Wri teL ine (S t r i ngFromColor(Color .Green)) Console .Wri teL ine (S t r i ngFromColor(Color .B lue) ) End Sub
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
131
4.3
Inheritance
Function StringFromColor(c As Color) As String Select Case c Case Color.Red Return String.Format("Red = " & CInt(c)) Case Color.Green Return String.Format("Green = " & CInt(c)) Case Color.Blue Return String.Format("Blue = " & CInt(c)) Case Else Return "Invalid color" End Select End Function End Module
The example above prints the enumeration values and their associated values. The output is:
Red = 0 Green = 10 Blue = 11
The reasons for the values are as follows: The enumeration value Red is automatically assigned the value 0 (since it has no initializer and is the first enumeration value member). The enumeration value Green is explicitly given the value 10. The enumeration value Blue is automatically assigned the value one greater than the enumeration value that textually precedes it. The constant expression may not directly or indirectly use the value of its own associated enumeration value (that is, circularity in the constant expression is not allowed). The following example is invalid because the declarations of A and B are circular.
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
132
4.3
Inheritance
1.32 Classes
A class is a data structure that may contain data members (constants, variables, and events), function members (methods, properties, indexers, operators, and constructors), and nested types. Classes are reference types. The following example shows a class that contains each kind of member:
Class AClass Publ i c Sub New() Console .Wri teL ine ("Construc tor " ) End Sub Publ i c Sub New(value As Integer ) MyVariable = value Console .Wri teL ine ("Construc tor " ) End Sub Publ i c Const MyConst As Integer = 12 Publ i c MyVariable As Integer = 34 Publ i c Sub MyMethod() Console .Wri teL ine ("MyClass.MyMethod") End Sub Publ i c Property MyProperty( ) As Integer Get Return MyVariable End Get Set (va lue As Integer )
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
133
4.3
Inheritance
MyVariable = value End Set End Property Default Public Property Item(index As Integer) As Integer Get Return 0 End Get Set (value As Integer) Console.WriteLine("Item(" & index & ") = " & value) End Set End Property Public Event MyEvent() Friend Class MyNestedClass End Class End Class
134
4.3
Inheritance
Console.WriteLine("a.MyVariable = " & a.MyVariable) ' Method usage. a.MyMethod() ' Property usage. a.MyProperty += 1 Console.WriteLine("a.MyProperty = " & a.MyProperty) a(1) = 1 ' Event usage. aInstance = a End Sub Sub MyHandler() Handles aInstance.MyEvent Console.WriteLine("Test.MyHandler") End Sub End Module
There are two class-specific modifiers, MustInher i tand NotInher i tab le is invalid to specify . It them both. ClassDeclaration ::= [ Attributes ] [ ClassModifier+ ] Class Identifier [ TypeParameterList ] StatementTerminator [ ClassBase ] [ TypeImplementsClause+ ] [ ClassMemberDeclaration+ ] End Class StatementTerminator ClassModifier ::= TypeModifier | MustInherit | NotInheritable | Partial 1.32.1 Class Base Specification A class declaration may include a base type specification that defines the direct base type of the class. If a class declaration has no explicit base type, the direct base type is implicitly Object. For example:
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
135
4.3
Inheritance
Class Base End Class Class Derived Inherits Base End Class
Base types cannot be a type parameter on its own, although it may involve the type parameters that are in scope.
Class C1(Of V) End Class Class C2(Of V) Inherits V End Class ' Error, type parameter used as base class
Classes may only derive from Object and classes. It is invalid for a class to derive from System.ValueType System.Enum, System.Array, System.MulticastDelegate or System.Delegate. A , generic class cannot derive from System.Attribute or from a class that derives from it. Every class has exactly one direct base class, and circularity in derivation is prohibited. It is not possible to derive from a NotInheritable class, and the accessibility domain of the base class must be the same as or a superset of the accessibility domain of the class itself. ClassBase ::= Inherits NonArrayTypeName StatementTerminator 1.32.2 Class Members The members of a class consist of the members introduced by its class member declarations and the members inherited from its direct base class. A class member declaration may have Public, Protected, Friend, Protected Friend, or Private access. When a class member declaration does not include an access modifier, the
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
136
4.3
Inheritance
declaration defaults to Publ i c access, unless it is a variable declaration; in that case it defaults to Pr ivateaccess. The scope of a class member is the class body in which the declaration occurs. If the member has Fr iendaccess, its scope extends to the class body of any derived class in the same program or any assembly that has been given Fr iendaccess, and if the member has Public, Protected, or Protected Friend access, its scope extends to the class body of any derived class in any program. ClassMemberDeclaration ::= NonModuleDeclaration | EventMemberDeclaration | VariableMemberDeclaration | ConstantMemberDeclaration | MethodMemberDeclaration | PropertyMemberDeclaration | ConstructorMemberDeclaration | OperatorDeclaration
1.33 Structures
Structures are value types that inherit from System.ValueType. Structures are similar to classes in that they represent data structures that can contain data members and function members. Unlike classes, however, structures do not require heap allocation. In the case of classes, it is possible for two variables to reference the same object, and thus possible for operations on one variable to affect the object referenced by the other variable. With structures, the variables each have their own copy of the non-Shared data, so it is not possible for operations on one to affect the other, as the following example illustrates:
Structure Point Publ i c x, y As Integer Publ i c Sub New(x As Integer , y As Integer ) Me.x = x Me.y = y
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
137
4.3
Inheritance
Given the above declaration the following code outputs the value 10:
Module Test Sub Main() Dim a As New Point (10 , 10) Dim b As Point = a a.x = 100 Console .Wri teL ine (b .x ) End Sub End Module
The assignment of a to b creates a copy of the value, and b is thus unaffected by the assignment to a.x. Had Point instead been declared as a class, the output would be 100 because a and b would reference the same object. StructureDeclaration ::= [ Attributes ] [ StructureModifier+ ] Structure Identifier [ TypeParameterList ] StatementTerminator [ TypeImplementsClause+ ] [ StructMemberDeclaration+ ] End Structure StatementTerminator StructureModifier ::= TypeModifier | Partial 1.33.1 Structure Members The members of a structure are the members introduced by its structure member declarations and the members inherited from System.ValueType. Every structure implicitly has a Public parameterless instance constructor that produces the default value of the structure. As a result, it is not possible for a structure type declaration to declare a parameterless instance constructor. A structure type is, however, permitted to declare parameterized instance constructors, as in the following example: 138
4.3
Inheritance
Structure Point Private x, y As Integer Public Sub New(x As Integer, y As Integer) Me.x = x Me.y = y End Sub End Structure
Given the above declaration, the following statements both create a Point with x and y initialized to zero.
Dim p1 As Point = New Point ( ) Dim p2 As Point = New Point (0 , 0)
Because structures directly contain their field values (rather than references to those values), structures cannot contain fields that directly or indirectly reference themselves. For example, the following code is not valid:
Structure S1 Dim f1 As S2 End Structure Structure S2 ' This would require S1 to contain itself. Dim f1 As S1 End Structure
Normally, a structure member declaration may only have Public, Friend, or Private access, but when overriding members inherited from Object, Protected and Protected Friend access may also be used. When a structure member declaration does not include an access modifier, the declaration defaults to Public access. The scope of a member declared by a structure is the structure body in which the declaration occurs. StructMemberDeclaration ::= NonModuleDeclaration | VariableMemberDeclaration | ConstantMemberDeclaration |
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
139
4.3
Inheritance
' Val id : Cal l s N1.M1.S1. ' Val id : Cal l s N1.M1.S1. ' Not val id : ambiguous.
140
4.3
Inheritance
N1.S2() ' Not valid: ambiguous. N1.M2.S2() ' Valid: Calls N1.M2.S2. End Sub End Module End Namespace
A module may only be declared in a namespace and may not be nested in another type. Standard modules may not implement interfaces, they implicitly derive from Object, and they have only Shared constructors. ModuleDeclaration ::= [ Attributes ] [ TypeModifier+ ] Module Identifier StatementTerminator [ ModuleMemberDeclaration+ ] End Module StatementTerminator 1.34.1 Standard Module Members The members of a standard module are the members introduced by its member declarations and the members inherited from Object. Standard modules may have any type of member except instance constructors. All standard module type members are implicitly Shared. Normally, a standard module member declaration may only have Public, Friend, or Private access, but when overriding members inherited from Object, the Protected and Protected Friend access modifiers may be specified. When a standard module member declaration does not include an access modifier, the declaration defaults to Public access, unless it is a variable, which defaults to Private access. As previously noted, the scope of a standard module member is the declaration containing the standard module declaration. Members inherited from Object are not included in this special scoping; those members have no scope and must always be qualified with the name of the module. If the member has Friend access, its scope extends only to namespace members declared in the same program or assemblies that have been given Friend access. ModuleMemberDeclaration ::= NonModuleDeclaration |
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
141
4.3
Inheritance
1.35 Interfaces
Interfaces are reference types that other types implement to guarantee that they support certain methods. An interface is never directly created and has no actual representation other types must be converted to an interface type. An interface defines a contract. A class or structure that implements an interface must adhere to its contract. The following example shows an interface that contains a default property I tem an event E, , a method F, and a property P:
Inter face IExample Defaul t Property I tem(index As Integer ) As Str ing Event E() Sub F( value As Integer ) Property P() As Str ing End Inter face
Interfaces may employ multiple inheritance. In the following example, the interface IComboBox inherits from both ITextBox and IListBox:
Inter face IContro l Sub Paint ( ) End Inter face Inter face ITextBox Inher i t s IContro l Sub SetText( text As Str ing)
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
142
4.3
Inheritance
End Interface Interface IListBox Inherits IControl Sub SetItems(items() As String) End Interface Interface IComboBox Inherits ITextBox, IListBox End Interface
Classes and structures can implement multiple interfaces. In the following example, the class Edi tBox derives from the class Contro l and implements both IControl and IDataBound:
Inter face IDataBound Sub Bind(b As Binder) End Inter face Publ i c Class EditBox Inher i t s Control Implements IContro l , IDataBound Publ i c Sub Paint ( ) Implements IContro l . Pa i n t ... End Sub Publ i c Sub Bind(b As Binder) Implements IDataBound.Bind ... End Sub End Class
143
4.3
Inheritance
[ InterfaceMemberDeclaration+ ] End Inter faceStatementTerminator 1.35.1 Interface Inheritance The base interfaces of an interface are the explicit base interfaces and their base interfaces. In other words, the set of base interfaces is the complete transitive closure of the explicit base interfaces, their explicit base interfaces, and so on. If an interface declaration has no explicit interface base, then there is no base interface for the type interfaces do not inherit from Object (although they do have a widening conversion to Object). In the following example, the base interfaces of IComboBox are IControl, ITextBox, and IListBox.
Inter face IContro l Sub Paint ( ) End Inter face Inter face ITextBox Inher i t s IContro l Sub SetText( text As Str ing) End Inter face Inter face IL i s tBox Inher i t s IContro l Sub SetI tems( i tems() As Str ing) End Inter face Inter face IComboBox Inher i t s ITextBox, IL i s tBox End Inter face
An interface inherits all members of its base interfaces. In other words, the IComboBox interface above inherits members SetText and SetItems as well as Paint.
144
4.3
Inheritance
A class or structure that implements an interface also implicitly implements all of the interface's base interfaces. If an interface appears more than once in the transitive closure of the base interfaces, it only contributes its members to the derived interface once. A type implementing the derived interface only has to implement the methods of the multiply defined base interface once. In the following example, Paint only needs to be implemented once, even though the class implements IComboBox and IControl.
Class ComboBox Implements IContro l , IComboBox
Sub SetText( text As Str ing) Implements IComboBox.SetText End Sub Sub SetI tems( i tems() As Str ing) Implements IComboBox.Set I tems End Sub Sub Pr int ( ) End Sub End Class Implements IComboBox.Paint
An Inherits clause has no effect on other Inherits clauses. In the following example, IDerived must qualify the name of INested with IBase.
Inter face IBase Inter face INested Sub Nested() End Inter face Sub Base() End Inter face Inter face IDer ived Inher i t s IBase, INested End Inter face ' Error : Must speci fy IBase. INested.
145
4.3
Inheritance
The accessibility domain of a base interface must be the same as or a superset of the accessibility domain of the interface itself. InterfaceBase ::= Inher i t s InterfaceBases StatementTerminator InterfaceBases ::= NonArrayTypeName | InterfaceBases Comma NonArrayTypeName 1.35.2 Interface Members The members of an interface consist of the members introduced by its member declarations and the members inherited from its base interfaces. Although interfaces do not inherit members from Object, because every class or structure that implements an interface does inherit from Object, the members of Object, including extension methods, are considered members of an interface and can be called on an interface directly without requiring a cast to Object. For example:
Inter face I1 End Inter face Class C1 Implements I1 End Class Module Test Sub Main() Dim i As I1 = New C1() Dim h As Integer = i .GetHashCode() End Sub End Module
Members of an interface with the same name as members of Object implicitly shadow Object members. Only nested types, methods, properties, and events may be members of an interface. Methods and properties may not have a body. Interface members are implicitly Public and may not specify an access modifier.
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
146
4.3
Inheritance
1.36 Arrays
An array is a reference type that contains variables accessed through indices corresponding in a one-to-one fashion with the order of the variables in the array. The variables contained in an array, also called the elements of the array, must all be of the same type, and this type is called the element type of the array. The elements of an array come into existence when an array instance is created, and cease to exist when the array instance is destroyed. Each element of an array is initialized to the default value of its type. The type System.Arrayis the base type of all array types and may not be instantiated. Every array type inherits the members declared by the System.Arraytype and is convertible to it (and Object). A one-dimensional array type with element T also implements the interface System.Collections.Generic.IList(Of T); if T is a reference type, then the array type also implements IList(Of U), where U has a widening identity or reference conversion from T. An array has a rank that determines the number of indices associated with each array element. The rank of an array determines the number of dimensions of the array. For example, an array with a rank of one is called a single-dimensional array, and an array with a rank greater than one is called a multidimensional array. The following example creates a single-dimensional array of integer values, initializes the array elements, and then prints each of them out:
Module Test Sub Main() Dim arr (5) As Integer Dim i As Integer For i = 0 To arr . Length - 1 arr ( i ) = i * i Next i
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
147
4.3
Inheritance
For i = 0 To arr.Length - 1 Console.WriteLine("arr(" & i & ") = " & arr(i)) Next i End Sub End Module
Each dimension of an array has an associated length. Dimension lengths are not part of the type of the array, but rather are established when an instance of the array type is created at run time. The length of a dimension determines the valid range of indices for that dimension: for a dimension of length N, indices can range from zero to N 1 If a dimension . is of length zero, there are no valid indices for that dimension. The total number of elements in an array is the product of the lengths of each dimension in the array. If any of the dimensions of an array has a length of zero, the array is said to be empty. The element type of an array can be any type. Array types are specified by adding a modifier to an existing type name. The modifier consists of a left parenthesis, a set of zero or more commas, and a right parenthesis. The type modified is the element type of the array, and the number of dimensions is the number of commas plus one. If more than one modifier is specified, then the element type of the array is an array. The modifiers are read left to right, with the leftmost modifier being the outermost array. In the example
Module Test Dim arr As Integer ( , ) ( , , ) ( ) End Module
148
4.3
Inheritance
the element type of arr is a two-dimensional array of three-dimensional arrays of onedimensional arrays of Integer . A variable may also be declared to be of an array type by putting an array type modifier or an array-size initialization modifier on the variable name. In that case, the array element type is the type given in the declaration, and the array dimensions are determined by the variable name modifier. For clarity, it is not valid to have an array type modifier on both a variable name and a type name in the same declaration. The following example shows a variety of local variable declarations that use array types with Integeras the element type:
Module Test Sub Main() Dim a1() As Integer Dim a2(, ) As Integer Dim a3(, , ) As Integer Dim a4 As Integer ( ) Dim a5 As Integer ( , ) Dim a6 As Integer ( , , )
' Declares 1- dimensional array of integers . ' Declares 2- dimensional array of integers . ' Declares 3- dimensional array of integers . ' Declares 1- dimensional array of integers . ' Declares 2- dimensional array of integers . ' Declares 3- dimensional array of integers .
' Declare 1- dimensional array of 2- dimensional arrays of integers Dim a7() ( , ) As Integer ' Declare 2- dimensional array of 1- dimensional arrays of integers . Dim a8(, ) ( ) As Integer Dim a9() As Integer ( ) ' Not al lowed. End Sub End Module
An array type name modifier extends to all sets of parentheses that follow it. This means that in the situations where a set of arguments enclosed in parenthesis is allowed after a type name, it is not possible to specify the arguments for an array type name. For example: 149
4.3
Inheritance
Module Test Sub Main() ' This calls the Integer constructor. Dim x As New Integer(3) ' This declares a variable of Integer(). Dim y As Integer() ' This gives an error. ' Array sizes can not be specified in a type name. Dim z As Integer()(3) End Sub End Module
In the last case, (3) is interpreted as part of the type name rather than as a set of constructor arguments. ArrayTypeName ::= NonArrayTypeName ArrayTypeModifiers ArrayTypeModifiers ::= ArrayTypeModifier+ ArrayTypeModifier ::= OpenParenthesis [ RankList ] CloseParenthesis RankList ::= Comma | RankList Comma ArrayNameModifier ::= ArrayTypeModifiers | ArraySizeInitializationModifier
1.37 Delegates
A delegate is a reference type that refers to a Shared method of a type or to an instance method of an object. The closest equivalent of a delegate in other languages is a function pointer, but whereas a function pointer can only reference Shared functions, a delegate can reference both Shared and instance methods. In the latter case, the delegate stores not
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
150
4.3
Inheritance
only a reference to the method's entry point, but also a reference to the object instance with which to invoke the method. The delegate declaration may not have a Handles clause, an Implements clause, a method body, or an End construct. The parameter list of the delegate declaration may not contain Optional or ParamArray parameters. The accessibility domain of the return type and parameter types must be the same as or a superset of the accessibility domain of the delegate itself. The members of a delegate are the members inherited from class System.Delegate. A delegate also defines the following methods: A constructor that takes two parameters, one of type Object and one of type System.IntPtr. An Invoke method that has the same signature as the delegate. A BeginInvoke method whose signature is the delegate signature, with three differences. First, the return type is changed to System.IAsyncResult. Second, two additional parameters are added to the end of the parameter list: the first of type System.AsyncCallback and the second of type Object. And finally, all ByRef parameters are changed to be ByVal. An EndInvoke method whose return type is the same as the delegate. The parameters of the method are only the delegate parameters exactly that are ByRef parameters, in the same order they occur in the delegate signature. In addition to those parameters, there is an additional parameter of type System.IAsyncResult at the end of the parameter list. There are three steps in defining and using delegates: declaration, instantiation, and invocation. Delegates are declared using delegate declaration syntax. The following example declares a delegate named SimpleDelegate that takes no arguments:
Delegate Sub SimpleDelegate( )
The next example creates a SimpleDelegate instance and then immediately calls it:
Module Test Sub F() System.Console .Wri teL ine ("Test . F " )
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
151
4.3
Inheritance
End Sub Sub Main() Dim d As SimpleDelegate = AddressOf F d() End Sub End Module
There is not much point in instantiating a delegate for a method and then immediately calling via the delegate, as it would be simpler to call the method directly. Delegates show their usefulness when their anonymity is used. The next example shows a Mult iCa l l method that repeatedly calls a SimpleDelegateinstance:
Sub Mult iCa l l ( d As SimpleDelegate, count As Integer ) Dim i As Integer For i = 0 To count - 1 d() Next i End Sub
It is unimportant to the Mult iCa l l method what the target method for the SimpleDelegateis, what accessibility this method has, or whether the method is Shared or nonshared. All that matters is that the signature of the target method is compatible with SimpleDelegate. DelegateDeclaration ::= [ Attributes ] [ TypeModifier+ ] Delegate MethodSignature StatementTerminator MethodSignature ::= SubSignature | FunctionSignature
4.3
Inheritance
declaration at compile-time to form a single type declaration. For example, the following code declares a single class Test with members Test.C1 and Test.C2. a.vb:
Publ i c Part ia l Class Test Publ i c Sub S1() End Sub End Class
b.vb:
Publ i c Class Test Publ i c Sub S2() End Sub End Class
When combining partial type declarations, at least one of the declarations must have a Partial modifier, otherwise a compile-time error results. Annotation Although it is possible to specify Partial on only one declaration among many partial declarations, it is better form to specify it on all partial declarations. In the situation where one partial declaration is visible but one or more partial declarations are hidden (such as the case of extending tool-generated code), it is acceptable to leave the Partial modifier off of the visible declaration but specify it on the hidden declarations. Only classes and structures can be declared using partial declarations. The arity of a type is considered when matching partial declarations together: two classes with the same name but different numbers of type parameters are not considered to be partial declarations of the same time. Partial declarations can specify attributes, class modifiers, Inherits statement or Implements statement. At compile time, all of the pieces of the partial declarations are combined together and used as a part of the type declaration. If there are any conflicts between attributes, modifiers, bases, interfaces, or type members, a compiletime error results. For example:
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
153
4.3
Inheritance
Public Partial Class Test1 Implements IDisposable End Class Class Test1 Inherits Object Implements IComparable End Class Public Partial Class Test2 End Class Private Partial Class Test2 End Class
The previous example declares a type Test1 that is Publ i c inherits from Object and , implements System.IDisposable and System.IComparable. The partial declarations of Test2 will cause a compile-time error because one of the declarations says that Test2 is Public and another says that Test2 is Private. Partial types with type parameters can declare constraints and variance for the type parameters, but the constraints and variance from each partial declaration must match. Thus, constraints and variance are special in that they are not automatically combined like other modifiers:
Part ia l Publ i c Class Lis t (Of T As IEnumerable) End Class ' Error : Constra in ts on T don't match Class Lis t (Of T As IComparable) End Class
The fact that a type is declared using multiple partial declarations does not affect the name lookup rules within the type. As a result, a partial type declaration can use members declared in other partial type declarations, or may implement methods on interfaces declared in other partial type declarations. For example:
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
154
4.3
Inheritance
Public Partial Class Test1 Implements IDisposable Private IsDisposed As Boolean = False End Class Class Test1 Private Sub Dispose() Implements IDisposable.Dispose If Not IsDisposed Then ... End If End Sub End Class
Initializers within a partial declaration will still be executed in declaration order; however, there is no guaranteed order of execution for initializers that occur in separate partial declarations.
155
4.3
Inheritance
arguments. A generic type that has type arguments applied to it is called a constructed type. The type arguments in a constructed type must always satisfy the constraints placed on the type parameters they match to. A type name might identify a constructed type even though it doesnt specify type parameters directly. This can occur where a type is nested within a generic class declaration, and the instance type of the containing declaration is implicitly used for name lookup:
Class Outer(Of T) Public Class Inner End Class ' Type of i is the constructed type Outer(Of T).Inner Public i As Inner End Class
A constructed type C(Of T,,Tn) is accessible when the generic type and all the type 1 arguments are accessible. For instance, if the generic type C is Public and all of the type arguments T1,,Tn are Public, then the constructed type is Public. If either the type name or one of the type arguments is Private, however, then the accessibility of the constructed type is Private. If one type argument of the constructed type is Protected and another type argument is Friend, then the constructed type is accessible only in the class and its subclasses in this assembly or any assembly that has been given Friend access. In other words, the accessibility domain for a constructed type is the intersection of the accessibility domains of its constituent parts. Annotation The fact that the accessibility domain of constructed type is the intersection of its constituted parts has the interesting side effect of defining a new accessibility level. A constructed type that contains an element that is Protected and an element that is Friend can only be accessed in contexts that can access both Friend and Protected members. However, there is no way to express this accessibility level in the language, as the accessibility Protected Friend means that an entity can be accessed in a context that can access either Friend or Protected members.
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
156
4.3
Inheritance
The base, implemented interfaces and members of constructed types are determined by substituting the supplied type arguments for each occurrence of the type parameter in the generic type. 1.39.1 Open Types and Closed Types A constructed type for who one or more type arguments are type parameters of a containing type or method is called an open type. This is because some of the type parameters of the type are still not known, so the actual shape of the type is not yet fully known. In contrast, a generic type whose type arguments are all non-type parameters is called a closed type. The shape of a closed type is always fully known. For example:
Class Base(Of T, V) End Class Class Derived(Of V) Inherits Base(Of Integer, V) End Class Class MoreDerived Inherits Derived(Of Double) End Class
The constructed type Base(Of Integer , is an open type because although the type V) parameter T has been supplied, the type parameter U has been supplied another type parameter. Thus, the full shape of the type is not yet known. The constructed type Derived(Of Double), however, is a closed type because all type parameters in the inheritance hierarchy have been supplied. Open types are defined as follows: A type parameter is an open type. An array type is an open type if its element type is an open type. A constructed type is an open type if one or more of its type arguments are an open type. A closed type is a type that is not an open type.
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
157
4.3
Inheritance
Because the program entry point cannot be in a generic type, all types used at run-time will be closed types.
158
Conversions
Conversion is the process of changing a value from one type to another. For example, a value of type Integercan be converted to a value of type Double, or a value of type Derived can be converted to a value of type Base, assuming that Base and Derived are both classes and Derived inherits from Base. Conversions may not require the value itself to change (as in the latter example), or they may require significant changes in the value representation (as in the former example). Conversions may either be widening or narrowing. A widening conversion is a conversion from a type to another type whose value domain is at least as big, if not bigger, than the original types value domain. Widening conversions should never fail. A narrowing conversion is a conversion from a type to another type whose value domain is either smaller than the original types value domain or sufficiently unrelated that extra care must be taken when doing the conversion (for example, when converting from Integer to String). Narrowing conversions, which may entail loss of information, can fail. The identity conversion (i.e. a conversion from a type to itself) and default value conversion (i.e. a conversion from Nothing) are defined for all types.
Explicit conversions, on the other hand, require cast operators. Attempting to do an explicit conversion on a value without a cast operator causes a compile-time error. The following example uses an explicit conversion to convert a Long value to an Integer value.
Module Test Sub Main() Dim longValue As Long = 134 Dim intVa lue As Integer = CInt( l ongValue)
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
159
4.3
Inheritance
Console.WriteLine(longValue & " = " & intValue) End Sub End Module
The set of implicit conversions depends on the compilation environment and the Option Str i c tstatement. If strict semantics are being used, only widening conversions may occur implicitly. If permissive semantics are being used, all widening and narrowing conversions (in other words, all conversions) may occur implicitly.
160
4.3
Inheritance
End Enum Module Test Sub Main() Dim x As Integer = 5 ' OK, even though there is no enumerated value for 5. Dim y As Values = CType(x, Values) End Sub End Module
Numeric conversions are processed at run-time as follows: For a conversion from a numeric type to a wider numeric type, the value is simply converted to the wider type. Conversions from UInteger Integer ULong, Long, or Decimal to , , Single or Double are rounded to the nearest Single or Double value. While this conversion may cause a loss of precision, it will never cause a loss of magnitude. For a conversion from an integral type to another integral type, or from Single, Double, or Decimal to an integral type, the result depends on whether integer overflow checking is on: If integer overflow is being checked: If the source is an integral type, the conversion succeeds if the source argument is within the range of the destination type. The conversion throws a System.OverflowException exception if the source argument is outside the range of the destination type. If the source is Single, Double, or Decimal, the source value is rounded up or down to the nearest integral value, and this integral value becomes the result of the conversion. If the source value is equally close to two integral values, the value is rounded to the value that has an even number in the least significant digit position. If the resulting integral value is outside the range of the destination type, a System.OverflowException exception is thrown. If integer overflow is not being checked: If the source is an integral type, the conversion always succeeds and simply consists of discarding the most significant bits of the source value.
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
161
4.3
Inheritance
If the source is Single Double, or Decimal, the conversion always succeeds and simply , consists of rounding the source value towards the nearest integral value. If the source value is equally close to two integral values, the value is always rounded to the value that has an even number in the least significant digit position. For a conversion from Double to Single, the Double value is rounded to the nearest Single value. If the Double value is too small to represent as a Single, the result becomes positive zero or negative zero. If the Double value is too large to represent as a Single, the result becomes positive infinity or negative infinity. If the Double value is NaN, the result is also NaN. For a conversion from Single or Double to Decimal, the source value is converted to Decimal representation and rounded to the nearest number after the 28th decimal place if required. If the source value is too small to represent as a Decimal, the result becomes zero. If the source value is NaN, infinity, or too large to represent as a Decimal, a System.OverflowException exception is thrown. For a conversion from Double to Single, the Double value is rounded to the nearest Single value. If the Double value is too small to represent as a Single, the result becomes positive zero or negative zero. If the Double value is too large to represent as a Single, the result becomes positive infinity or negative infinity. If the Double value is NaN, the result is also NaN.
162
4.3
Inheritance
It is not an error to convert a NotInher i tab le classes to and from interfaces that it does not implement because classes that represent COM classes may have interface implementations that are not known until run time. If a reference conversion fails at run time, a System.Inval i dCastExcept ion exception is thrown. 1.44.1 Reference Variance Conversions Generic interfaces or delegates may have variant type parameters that allow conversions between compatible variants of the type. Therefore, at runtime a conversion from a class type or an interface type to an interface type that is variant compatible with an interface type it inherits from or implements will succeed. Similarly, delegate types can be cast to and from variant compatible delegate types. For example, the delegate type
Delegate Funct ion F(Of In A, Out R)(a As A) As R
would allow a conversion from F(Of Object , Integer ) to F(Of Str ing , Integer) is, a delegate . That F which takes Object may be safely used as a delegate F which takes String. When the delegate is invoked, the target method will be expecting an object, and a string is an object. A generic delegate or interface type S(Of S1,,SN) is said to be variant compatible with a generic interface or delegate type T(Of T1,,TN) if:
S and T are both constructed from the same generic type U(Of U1,,UN).
For each type parameter UX: If the type parameter was declared without variance then SX and TX must be the same type. If the type parameter was declared In then there must be a widening identity, default, reference, array, or type parameter conversion from SX to TX. If the type parameter was declared Out then there must be a widening identity, default, reference, array, or type parameter conversion from TX to SX. When converting from a class to a generic interface with variant type parameters, if the class implements more than one variant compatible interface the conversion is ambiguous if there is not a non-variant conversion. For example:
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
163
4.3
Inheritance
Class Base End Class Class Derived1 Inherits Base End Class Class Derived2 Inherits Base End Class Class OneAndTwo Implements IEnumerable(Of Derived1) Implements IEnumerable(Of Derived2) End Class Class BaseAndOneAndTwo Implements IEnumerable(Of Base) Implements IEnumerable(Of Derived1) Implements IEnumerable(Of Derived2) End Class Module Test Sub Main() ' Error: conversion is ambiguous Dim x As IEnumerable(Of Base) = New OneAndTwo() Base) ' OK, will pick up the direct implementation of IEnumerable(Of Dim y as IEnumerable(Of Base) = New BaseAndOneAndTwo() End Sub
1.44.2 Anonymous Delegate Conversions When an expression classified as a lambda method is reclassified as a value in a context where there is no target type (for example, Dim x = Funct ion(a As Integer , b As Integer ) a + b ), the type of the resulting expression is an anonymous delegate type equivalent to the Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved. 164
4.3
Inheritance
signature of the lambda method. This anonymous delegate has a conversion to any compatible delegate type. A compatible delegate type is any delegate type that can be created using a delegate creation expression with the anonymous delegate types Invoke method as a parameter. For example:
' Anonymous type simi la r to Func(Of Object , Object , Object) Dim x = Funct ion(x , y) x + y ' OK because delegate type i s compatib le Dim y As Func(Of Integer , Integer , Integer ) = x
165
4.3
Inheritance
Because of array covariance, assignments to elements of reference type arrays include a run-time check that ensures that the value being assigned to the array element is actually of a permitted type.
Module Test Sub Fill(array() As Object, index As Integer, count As Integer, _ value As Object) Dim i As Integer For i = index To (index + count) - 1 array(i) = value Next i End Sub Sub Main() Dim strings(100) As String Fill(strings, 0, 101, "Undefined") Fill(strings, 0, 10, Nothing) Fill(strings, 91, 10, 0) End Sub End Module
In this example, the assignment to array( i ) method Fi l limplicitly includes a run-time in check that ensures that the object referenced by the variable value is either Nothing or an instance of a type that is compatible with the actual element type of array array. In method Main, the first two invocations of method Fill succeed, but the third invocation causes a System.ArrayTypeMismatchException exception to be thrown upon executing the first assignment to array(i). The exception occurs because an Integer cannot be stored in a String array. If one of the array element types is a type parameter whose type turns out to be a value type at runtime, a System.InvalidCastException exception will be thrown. For example:
Module Test Sub F(Of T As U, U)(x( ) As T) Dim y() As U = x
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
166
4.3
Inheritance
End Sub Sub Main() ' F will throw an exception because Integer() cannot be ' converted to Object() F(New Integer() { 1, 2, 3 }) End Sub End Module
Conversions also exist between an array of an enumerated type and an array of the enumerated types underlying type, provided the arrays have the same rank.
Enum Color As Byte Red Green Blue End Enum Module Test Sub Main() Dim a(10) As Color Dim b() As Integer Dim c() As Byte b = a c = a a = c End Sub End Module ' Error: Integer is not the underlying type of Color ' OK ' OK
In this example, an array of Color is converted to and from an array of Byte, Colors underlying type. The conversion to an array of Integer, however, will be an error because Integer is not the underlying type of Color.
4.3
Inheritance
value is copied from the location where it lives onto the .NET Framework heap. A reference to this location on the heap is then returned and can be stored in a reference type variable. This reference is also referred to as a boxed instance of the value type. The boxed instance has the same semantics as a reference type instead of a value type. Boxed value types can be converted back to their original value type through a process called unboxing. When a boxed value type is unboxed, the value is copied from the heap into a variable location. From that point on, it behaves as if it was a value type. When unboxing a value type, the value must be a null value or an instance of the value type. Otherwise a System.Inva l i dCastExcept ion exception is thrown. If the value is an instance of an enumerated type, that value can also be unboxed to the enumerated types underlying type or another enumerated type that has the same underlying type. A null value is treated as if it were the literal Nothing. To support nullable value types well, the value type System.Nul lab le (Of is treated specially T) when doing boxing and unboxing. Boxing a value of type Nul lab le (Of T) results in a boxed value of type T if the values HasValue property is True or a value of Nothing if the values HasValue property is False. Unboxing a value of type T to Nullable(Of T) results in an instance of Nullable(Of T) whose Value property is the boxed value and whose HasValue property is True. The value Nothing can be unboxed to Nullable(Of T) for any T and results in a value whose HasValue property is False. Because boxed value types behave like reference types, it is possible to create multiple references to the same value. For the primitive types and enumerated types, this is irrelevant because instances of those types are immutable. That is, it is not possible to modify a boxed instance of those types, so it is not possible to observe the fact that there are multiple references to the same value. Structures, on the other hand, may be mutable if its instance variables are accessible or if its methods or properties modify its instance variables. If one reference to a boxed structure is used to modify the structure, then all references to the boxed structure will see the change. Because this result may be unexpected, when a value typed as Object is copied from one location to another boxed value types will automatically be cloned on the heap instead of merely having their references copied. For example:
Class Class1 Publ i c Value As Integer = 0
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
168
4.3
Inheritance
End Class Structure Struct1 Public Value As Integer End Structure Module Test Sub Main() Dim val1 As Object = New Struct1() Dim val2 As Object = val1 val2.Value = 123 Dim ref1 As Object = New Class1() Dim ref2 As Object = ref1 ref2.Value = 123 Console.WriteLine("Values: " & val1.Value & ", " & val2.Value) Console.WriteLine("Refs: " & ref1.Value & ", " & ref2.Value) End Sub End Module
The assignment to the field of the local variable val2 does not impact the field of the local variable val1 because when the boxed Struct1 was assigned to val2, a copy of the value was made. In contrast, the assignment ref2.Value = 123 affects the object that both ref1 and ref2 references. Annotation Structure copying is not done for boxed structures typed as System.ValueType because it is not possible to late bind off of System.ValueType.
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
169
4.3
Inheritance
There is one exception to the rule that boxed value types will be copied on assignment. If a boxed value type reference is stored within another type, the inner reference will not be copied. For example:
Structure Struct1 Public Value As Object End Structure Module Test Sub Main() Dim val1 As Struct1 Dim val2 As Struct1 val1.Value = New Struct1() val1.Value.Value = 10 val2 = val1 val2.Value.Value = 123 Console.WriteLine("Values: " & val1.Value.Value & ", " & _ val2.Value.Value) End Sub End Module
This is because the inner boxed value is not copied when the value is copied. Thus, both val1 .Va lue and val2.Va lue have a reference to the same boxed value type. Annotation The fact that inner boxed value types are not copied is a limitation of the .NET type system to ensure that all inner boxed value types were copied whenever a value of type Object was copied would be prohibitively expensive.
170
4.3
Inheritance
As previously described, boxed value types can only be unboxed to their original type. Boxed primitive types, however, are treated specially when typed as Object. They can be converted to any other primitive type that they have a conversion to. For example:
Module Test Sub Main() Dim o As Object = 5 Dim b As Byte = CByte(o) ' Legal Console .Wri teL ine (b) ' Pr ints 5 End Sub End Module
Normally, the boxed Integervalue 5 could not be unboxed into a Byte variable. However, because Integer and Byte are primitive types and have a conversion, the conversion is allowed. It is important to note that converting a value type to an interface is different than a generic argument constrained to an interface. When accessing interface members on a constrained type parameter (or calling methods on Object), boxing does not occur as it does when a value type is converted to an interface and an interface member is accessed. For example, suppose an interface ICounter contains a method Increment which can be used to modify a value. If ICounter is used as a constraint, the implementation of the Increment method is called with a reference to the variable that Increment was called on, not a boxed copy:
Inter face ICounter Sub Increment() ReadOnly Property Value() As Integer End Inter face Structure Counter Implements ICounter Dim _value As Integer Property Value() As Integer Implements ICounter .Va lue Get
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
171
4.3
Inheritance
Return _value End Get End Property Sub Increment() Implements ICounter.Increment value += 1 End Sub End Structure Module Test Sub Test(Of T As ICounter)(x As T) Console.WriteLine(x.value) x.Increment() Console.WriteLine(x.value) CType(x, ICounter).Increment() Console.WriteLine(x.value) End Sub Sub Main() Dim x As Counter Test(x) End Sub End Module
The first call to Increment modifies the value in the variable x. This is not equivalent to the second call to Increment, which modifies the value in a boxed copy of x. Thus, the output of the program is:
0 1 1
1.46.1 Nullable Value Type Conversions A value type T can convert to and from the nullable version of the type, T?. The conversion from T? to T throws a System.InvalidOperationException exception if the value being
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
172
4.3
Inheritance
converted is Nothing. Also, T? has a conversion to a type S if T has an intrinsic conversion to S. And if S is a value type, then the following intrinsic conversions exist between T? and S?: A conversion of the same classification (narrowing or widening) from T? to S?. A conversion of the same classification (narrowing or widening) from T to S?. A narrowing conversion from S? to T. For example, an intrinsic widening conversion exists from Integer? to Long? because an intrinsic widening conversion exists from Integer to Long:
Dim i As Integer? = 10 Dim l As Long? = i
When converting from T? to S?, if the value of T? is Nothing, then the value of S? will be Nothing. When converting from S? to T or T? to S, if the value of T? or S? is Nothing, a System.InvalidCastException exception will be thrown. Because of the behavior of the underlying type System.Nullable(Of T), when a nullable value type T? is boxed, the result is a boxed value of type T, not a boxed value of type T?. And, conversely, when unboxing to a nullable value type T?, the value will be wrapped by System.Nullable(Of T), and Nothing will be unboxed to a null value of type T?. For example:
Dim i1? As Integer = Nothing Dim o1 As Object = i1 Console .Wri teL ine (o1 Is Nothing) o1 = 10 i1 = CType(o1, Integer?) Console .Wri teL ine ( i 1 ) ' Wil l pr int True ' Wil l
A side effect of this behavior is that a nullable value type T? appears to implement all of the interfaces of T, because converting a value type to an interface requires the type to be boxed. As a result, T? is convertible to all the interfaces that T is convertible to. It is important to note, however, that a nullable value type T? does not actually implement the interfaces of T for the purposes of generic constraint checking or reflection. For example: 173
4.3
Inheritance
Interface I1 End Interface Structure T1 Implements I1 ... End Structure Module Test Sub M1(Of T As I1)(ByVal x As T) End Sub Sub Main() Dim x? As T1 = Nothing Dim y As I1 = x M1(x) not satisfy I1 constraint End Sub End Module
174
4.3
Inheritance
175
4.3
Inheritance
From an enumerated type to its underlying numeric type, or to a numeric type that its underlying numeric type has a widening conversion to. From a constant expression of type ULong , Long, UInteger, Integer, UShort, Short, Byte, or SByte to a narrower type, provided the value of the constant expression is within the range of the destination type. Note Conversions from UInteger or Integer to Single, ULong or Long to Single or Double, or Decimal to Single or Double may cause a loss of precision, but will never cause a loss of magnitude. The other widening numeric conversions never lose any information. Reference conversions From a reference type to a base type. From a reference type to an interface type, provided that the type implements the interface or a variant compatible interface. From an interface type to Object. From an interface type to a variant compatible interface type. From a delegate type to a variant compatible delegate type. Anonymous Delegate conversions From an anonymous delegate type generated for a lambda method reclassification to any wider delegate type. Array conversions From an array type S with an element type SE to an array type T with an element type TE, provided all of the following are true:
S and T differ only in element type.
Both SE and TE are reference types or are type parameters known to be a reference type. A widening reference, array, or type parameter conversion exists from SE to TE.
176
4.3
Inheritance
From an array type S with an enumerated element type SE to an array type T with an element type TE, provided all of the following are true:
S and T differ only in element type. TE is the underlying type of SE.
Value Type conversions From a value type to a base type. From a value type to an interface type that the type implements. Nullable Value Type conversions From a type T to the type T?. From a type T? to a type S?, where there is a widening conversion from the type T to the type S. From a type T to a type S?, where there is a widening conversion from the type T to the type S. From a type T? to an interface type that the type T implements. String conversions From Char to String. From Char() to String. Type Parameter conversions From a type parameter to Object. From a type parameter to an interface type constraint or any interface variant compatible with an interface type constraint. From a type parameter to an interface implemented by a class constraint. From a type parameter to an interface variant compatible with an interface implemented by a class constraint.
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
177
4.3
Inheritance
From a type parameter to a class constraint, or a base type of the class constraint. From a type parameter T to a type parameter constraint TX, or anything TX has a widening conversion to.
178
4.3
Inheritance
From Double to Byte, SByte, UShort, Short, UInteger, Integer, ULong, Long, Decimal, or Single. From a numeric type to an enumerated type. From an enumerated type to a numeric type its underlying numeric type has a narrowing conversion to. From an enumerated type to another enumerated type. Reference conversions From a reference type to a more derived type. From a class type to an interface type, provided the class type does not implement the interface type or an interface type variant compatible with it. From an interface type to a class type. From an interface type to another interface type, provided there is no inheritance relationship between the two types and provided they are not variant compatible. Anonymous Delegate conversions From an anonymous delegate type generated for a lambda method reclassification to any narrower delegate type. Array conversions From an array type S with an element type SE, to an array type T with an element type TE, provided that all of the following are true:
S and T differ only in element type.
Both SE and TE are reference types or are type parameters not known to be value types. A narrowing reference, array, or type parameter conversion exists from SE to TE. From an array type S with an element type SE to an array type T with an enumerated element type TE, provided all of the following are true:
S and T differ only in element type.
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
179
4.3
Inheritance
Value type conversions From a reference type to a more derived value type. From an interface type to a value type, provided the value type implements the interface type. Nullable Value Type conversions From a type T? to a type T. From a type T? to a type S?, where there is a narrowing conversion from the type T to the type S. From a type T to a type S?, where there is a narrowing conversion from the type T to the type S. From a type S? to a type T, where there is a conversion from the type S to the type T. String conversions From String to Char. From String to Char(). From String to Boolean and from Boolean to String. Conversions between String and Byte, SByte, UShort, Short, UInteger, Integer, ULong, Long, Decimal, Single, or Double. From String to Date and from Date to String. Type Parameter conversions From Object to a type parameter. From a type parameter to an interface type, provided the type parameter is not constrained to that interface or constrained to a class that implements that interface. From an interface type to a type parameter.
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
180
4.3
Inheritance
From a type parameter to a derived type of a class constraint. From a type parameter T to anything a type parameter constraint TX has a narrowing conversion to.
If the conversion of T to Integer were permitted, one might easily expect that X(Of Integer).F(7) would return 7L. However, it would not, because numeric conversions are only
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
181
4.3
Inheritance
considered when the types are known to be numeric at compile time. In order to make the semantics clear, the above example must instead be written:
Class X(Of T) Public Shared Function F(t As T) As Long Return CLng(CObj(t)) ' OK, conversions permitted End Function End Class
182
4.3
Inheritance
no most encompassed type. In intuitive terms, the most encompassed type is the smallest type in the setthe one type that can be converted to each of the other types through a narrowing conversion. When collecting the candidate user-defined conversions for a type T?, the user-defined conversion operators defined by T are used instead. If the type being converted to is also a nullable value type, then any of Ts user-defined conversions operators that involve only non-nullable value types are lifted. A conversion operator from T to S is lifted to be a conversion from T? to S? and is evaluated by converting T? to T, if necessary, then evaluating the user-defined conversion operator from T to S and then converting S to S?, if necessary. If the value being converted is Nothing, however, a lifted conversion operator converts directly into a value of Nothing typed as S?. For example:
Structure S ... End Structure Structure T Publ i c Shared Widening Operator CType(ByVal v As T) As S ... End Operator End Structure Module Test Sub Main() Dim x As T? Dim y As S? y =x x = New T() y =x End Sub End Module ' Legal : y i s st i l l nul l
When resolving conversions, user-defined conversions operators are always preferred over lifted conversion operators. For example:
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
183
4.3
Inheritance
Structure S ... End Structure Structure T Public Shared Widening Operator CType(ByVal v As T) As S ... End Operator Public Shared Widening Operator CType(ByVal v As T?) As S? ... End Operator End Structure Module Test Sub Main() Dim x As T? Dim y As S? y = x lifted conversion End Sub End Module ' Calls user-defined conversion, not
At run-time, evaluating a user-defined conversion can involve up to three steps: First, the value is converted from the source type to the operand type using an intrinsic conversion, if necessary. Then, the user-defined conversion is invoked. Finally, the result of the user-defined conversion is converted to the target type using an intrinsic conversion, if necessary. It is important to note that evaluation of a user-defined conversion will never involve more than one user-defined conversion operator. 184
4.3
Inheritance
1.51.1 Most Specific Widening Conversion Determining the most specific user-defined widening conversion operator between two types is accomplished using the following steps: First, all of the candidate conversion operators are collected. The candidate conversion operators are all of the user-defined widening conversion operators in the source type and all of the user-defined widening conversion operators in the target type. Then, all non-applicable conversion operators are removed from the set. A conversion operator is applicable to a source type and target type if there is an intrinsic widening conversion operator from the source type to the operand type and there is an intrinsic widening conversion operator from the result of the operator to the target type. If there are no applicable conversion operators, then there is no most specific widening conversion. Then, the most specific source type of the applicable conversion operators is determined: If any of the conversion operators convert directly from the source type, then the source type is the most specific source type. Otherwise, the most specific source type is the most encompassed type in the combined set of source types of the conversion operators. If no most encompassed type can be found, then there is no most specific widening conversion. Then, the most specific target type of the applicable conversion operators is determined: If any of the conversion operators convert directly to the target type, then the target type is the most specific target type. Otherwise, the most specific target type is the most encompassing type in the combined set of target types of the conversion operators. If no most encompassing type can be found, then there is no most specific widening conversion. Then, if exactly one conversion operator converts from the most specific source type to the most specific target type, then this is the most specific conversion operator. If more than one such operator exists, then there is no most specific widening conversion.
185
4.3
Inheritance
1.51.2 Most Specific Narrowing Conversion Determining the most specific user-defined narrowing conversion operator between two types is accomplished using the following steps: First, all of the candidate conversion operators are collected. The candidate conversion operators are all of the user-defined conversion operators in the source type and all of the user-defined conversion operators in the target type. Then, all non-applicable conversion operators are removed from the set. A conversion operator is applicable to a source type and target type if there is an intrinsic conversion operator from the source type to the operand type and there is an intrinsic conversion operator from the result of the operator to the target type. If there are no applicable conversion operators, then there is no most specific narrowing conversion. Then, the most specific source type of the applicable conversion operators is determined: If any of the conversion operators convert directly from the source type, then the source type is the most specific source type. Otherwise, if any of the conversion operators convert from types that encompass the source type, then the most specific source type is the most encompassed type in the combined set of source types of those conversion operators. If no most encompassed type can be found, then there is no most specific narrowing conversion. Otherwise, the most specific source type is the most encompassing type in the combined set of source types of the conversion operators. If no most encompassing type can be found, then there is no most specific narrowing conversion. Then, the most specific target type of the applicable conversion operators is determined: If any of the conversion operators convert directly to the target type, then the target type is the most specific target type. Otherwise, if any of the conversion operators convert to types that are encompassed by the target type, then the most specific target type is the most encompassing type in the combined set of source types of those conversion operators. If no most encompassing type can be found, then there is no most specific narrowing conversion.
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
186
4.3
Inheritance
Otherwise, the most specific target type is the most encompassed type in the combined set of target types of the conversion operators. If no most encompassed type can be found, then there is no most specific narrowing conversion. Then, if exactly one conversion operator converts from the most specific source type to the most specific target type, then this is the most specific conversion operator. If more than one such operator exists, then there is no most specific narrowing conversion.
187
Type Members
Type members define storage locations and executable code. They can be methods, constructors, events, constants, variables, and properties.
If an event declared using a delegate type is implementing an interface event, then a compatible event is one whose underlying delegate type is the same type. Otherwise, the event uses the delegate type from the interface event it is implementing. If such an event
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
188
4.3
Inheritance
implements multiple interface events, all the interface events must have the same underlying delegate type. For example:
Interface ClickEvents Event LeftClick(x As Integer, y As Integer) Event RightClick(x As Integer, y As Integer) End Interface Class Button Implements ClickEvents ' OK. Signatures match, delegate type = ClickEvents.LeftClickHandler. Event LeftClick(x As Integer, y As Integer) _ Implements ClickEvents.LeftClick ' OK. Signatures match, delegate type = ClickEvents.RightClickHandler. Event RightClick(x As Integer, y As Integer) _ Implements ClickEvents.RightClick End Class Class Label Implements ClickEvents ' Error. Signatures match, but can't be both delegate types. Event Click(x As Integer, y As Integer) _ Implements ClickEvents.LeftClick, ClickEvents.RightClick End Class
An interface member in the implements list is specified using a type name, a period, and an identifier. The type name must be an interface in the implements list or a base interface of an interface in the implements list, and the identifier must be a member of the specified interface. A single member can implement more than one matching interface member.
Interface ILeft Sub F() End Interface
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
189
4.3
Inheritance
Interface IRight Sub F() End Interface Class Test Implements ILeft, IRight Sub F() Implements ILeft.F, IRight.F End Sub End Class
If the interface member being implemented is unavailable in all explicitly implemented interfaces because of multiple interface inheritance, the implementing member must explicitly reference a base interface on which the member is available. For example, if I1 and I2 contain a member M, and I3 inherits from I1 and I2, a type implementing I3 will implement I1.M and I2.M. If an interface shadows multiply inherited members, an implementing type will have to implement the inherited members and the member(s) shadowing them.
Inter face ILe f t Sub F() End Inter face Inter face IR ight Sub F() End Inter face Inter face ILe f tR ight Inher i t s ILe f t , IR ight Shadows Sub F() End Inter face Class Test Implements ILe f tR ight
190
4.3
Inheritance
Sub LeftF() Implements ILeft.F End Sub Sub RightF() Implements IRight.F End Sub Sub LeftRightF() Implements ILeftRight.F End Sub End Class
If the containing interface of the interface member be implemented is generic, the same type arguments as the interface being implements must be supplied. For example:
Interface I1(Of T) Function F() As T End Interface Class C1 Implements I1(Of Integer) Implements I1(Of Double) Function F1() As Integer Implements I1(Of Integer).F End Function Function F2() As Double Implements I1(Of Double).F End Function ' Error: I1(Of String) is not implemented by C1 Function F3() As String Implements I1(Of String).F End Function End Class Class C2(Of U) Implements I1(Of U) Function F() As U Implements I1(Of U).F
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
191
4.3
Inheritance
ImplementsClause ::= [ Implements ImplementsList ] ImplementsList ::= InterfaceMemberSpecifier | ImplementsList Comma InterfaceMemberSpecifier InterfaceMemberSpecifier ::= NonArrayTypeName Period IdentifierOrKeyword
1.55 Methods
Methods contain the executable statements of a program. Methods, which have an optional list of parameters and an optional return value, are either shared or nonshared. Shared methods are accessed through the class or instances of the class. Nonshared methods, also called instance methods, are accessed through instances of the class. The following example shows a class Stack that has several shared methods (Clone and Flip), and several instance methods (Push, Pop, and ToString):
Publ i c Class Stack Publ i c Shared Funct ion Clone(s As Stack) As Stack ... End Funct ion Publ i c Shared Funct ion Fl i p (s As Stack) As Stack ... End Funct ion Publ i c Funct ion Pop() As Object ... End Funct ion Publ i c Sub Push(o As Object) ... End Sub Publ i c Overr ides Funct ion ToStr ing( ) As Str ing
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
192
4.3
Inheritance
... End Function End Class Module Test Sub Main() Dim s As Stack = New Stack() Dim i As Integer While i < 10 s.Push(i) End While Dim flipped As Stack = Stack.Flip(s) Dim cloned As Stack = Stack.Clone(s) Console.WriteLine("Original stack: " & s.ToString()) Console.WriteLine("Flipped stack: " & flipped.ToString()) Console.WriteLine("Cloned stack: " & cloned.ToString()) End Sub End Module
Methods can be overloaded, which means that multiple methods may have the same name so long as they have unique signatures. The signature of a method consists of the name of the method and the number and types of its parameters. The signature of a method specifically does not include the return type or parameter modifiers. The following example shows a class with a number of F methods:
Module Test Sub F() Console .Wri teL ine ("F ( ) " ) End Sub Sub F(o As Object) Console .Wri teL ine ("F (Object ) " ) End Sub
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
193
4.3
Inheritance
Sub F(value As Integer) Console.WriteLine("F(Integer)") End Sub Sub F(a As Integer, b As Integer) Console.WriteLine("F(Integer, Integer)") End Sub Sub F(values() As Integer) Console.WriteLine("F(Integer())") End Sub Sub Main() F() F(1) F(CType(1, Object)) F(1, 2) F(New Integer() { 1, 2, 3 }) End Sub End Module
MethodMemberDeclaration ::= MethodDeclaration | ExternalMethodDeclaration InterfaceMethodMemberDeclaration ::= InterfaceMethodDeclaration 1.55.1 Regular Method Declarations There are two types of methods: subroutines, which do not return values, and functions, which do. The body and End construct of a method may only be omitted if the method is defined in an interface or has the MustOverr ide modifier. If no return type is specified on a
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
194
4.3
Inheritance
function and strict semantics are being used, a compile-time error occurs; otherwise the type is implicitly Object or the type of the method's type character. The accessibility domain of the return type and parameter types of a method must be the same as or a superset of the accessibility domain of the method itself. Subroutine and function declarations are special in that their beginning and end statements must each start at the beginning of a logical line. Additionally, the body of a non-MustOverr ide subroutine or function declaration must start at the beginning of a logical line. For example:
Module Test ' I l l ega l : Subrout ine doesnt star t the l i ne Publ i c x As Integer : Sub F() : End Sub ' I l l ega l : Fi rs t statement doesnt star t the l i ne Sub G() : Console .Wri teL ine ("G" ) End Sub ' I l l ega l : End Sub doesnt star t the l i ne Sub H() : End Sub End Module
MethodDeclaration ::= SubDeclaration | MustOverrideSubDeclaration | FunctionDeclaration | MustOverrideFunctionDeclaration InterfaceMethodDeclaration ::= InterfaceSubDeclaration | InterfaceFunctionDeclaration SubSignature ::= Sub Identifier [ TypeParameterList ] [ OpenParenthesis [ ParameterList ] CloseParenthesis ]
195
4.3
Inheritance
FunctionSignature ::= Funct ion Identifier [ TypeParameterList ] [ OpenParenthesis [ ParameterList ] CloseParenthesis ] [ As [ Attributes ] TypeName ] SubDeclaration ::= [ Attributes ] [ ProcedureModifier+ ] SubSignature [ HandlesOrImplements ] LineTerminator Block End Sub StatementTerminator MustOverrideSubDeclaration ::= [ Attributes ] MustOverrideProcedureModifier+ SubSignature [ HandlesOrImplements ] StatementTerminator InterfaceSubDeclaration ::= [ Attributes ] [ InterfaceProcedureModifier+ ] SubSignature StatementTerminator FunctionDeclaration ::= [ Attributes ] [ ProcedureModifier+ ] FunctionSignature [ HandlesOrImplements ] LineTerminator Block End Funct ion StatementTerminator MustOverrideFunctionDeclaration ::= [ Attributes ] MustOverrideProcedureModifier+ FunctionSignature [ HandlesOrImplements ] StatementTerminator InterfaceFunctionDeclaration ::= [ Attributes ] [ InterfaceProcedureModifier+ ] FunctionSignature StatementTerminator ProcedureModifier ::= AccessModifier |
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
196
4.3
Inheritance
MustOverrideProcedureModifier ::= ProcedureModifier | MustOverride InterfaceProcedureModifier ::= Shadows | Overloads HandlesOrImplements ::= HandlesClause | ImplementsClause 1.55.2 External Method Declarations An external method declaration introduces a new method whose implementation is provided external to the program. Because an external method declaration provides no actual implementation, it has no method body or End construct. External methods are implicitly shared, may not have type parameters, and may not handle events or implement interface members. If no return type is specified on a function and strict semantics are being used, a compile-time error occurs. Otherwise the type is implicitly Object or the type of the method's type character. The accessibility domain of the return type and parameter types of an external method must be the same as or a superset of the accessibility domain of the external method itself. The library clause of an external method declaration specifies the name of the external file that implements the method. The optional alias clause is a string that specifies the numeric ordinal (prefixed by a # character) or name of the method in the external file. A singlecharacter set modifier may also be specified, which governs the character set used to marshal strings during a call to the external method. The Unicode modifier marshals all strings to Unicode values, the Ansi modifier marshals all strings to ANSI values, and the Auto modifier marshals the strings according to .NET Framework rules based on the name of the method, or the alias name if specified. If no modifier is specified, the default is Ansi. If Ansi or Unicode is specified, then the method name is looked up in the external file with no modification. If Auto is specified, then method name lookup depends on the platform. If Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved. 197
4.3
Inheritance
the platform is considered to be ANSI (for example, Windows 95, Windows 98, Windows ME), then the method name is looked up with no modification. If the lookup fails, an A is appended and the lookup tried again. If the platform is considered to be Unicode (for example, Windows NT, Windows 2000, Windows XP), then a W is appended and the name is looked up. If the lookup fails, the lookup is tried again without the W. For example:
Module Test ' Al l plat fo rms bind to "ExternSub" . Declare Ansi Sub ExternSub Lib "ExternDLL" () ' Al l plat fo rms bind to "ExternSub" . Declare Unicode Sub ExternSub Lib "ExternDLL" () ' ANSI plat fo rms: bind to "ExternSub" then "ExternSubA". ' Unicode plat fo rms: bind to "ExternSubW" then "ExternSub" . Declare Auto Sub ExternSub Lib "ExternDLL" () End Module
Data types being passed to external methods are marshaled according to the .NET Framework data marshalling conventions with one exception. String variables that are passed by value (that is, ByVal x As String) are marshaled to the OLE Automation BSTR type, and changes made to the BSTR in the external method are reflected back in the string argument. This is because the type String in external methods is mutable, and this special marshalling mimics that behavior. String parameters that are passed by reference (i.e. ByRef x As String) are marshaled as a pointer to the OLE Automation BSTR type. It is possible to override these special behaviors by specifying the System.Runtime.InteropServices.MarshalAsAttribute attribute on the parameter. The example demonstrates use of external methods:
Class Path Declare Funct ion CreateDirectory Lib "kernel32" ( _ Name As Str ing , sa As Secur i tyAt t r i bu tes) As Boolean Declare Funct ion RemoveDirectory Lib "kernel32" ( _ Name As Str ing) As Boolean Declare Funct ion GetCurrentDi rectory Lib "kernel32" ( _ BufSize As Integer , Buf As Str ing) As Integer
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
198
4.3
Inheritance
Declare Function SetCurrentDirectory Lib "kernel32" ( _ Name As String) As Boolean End Class
ExternalMethodDeclaration ::= ExternalSubDeclaration | ExternalFunctionDeclaration ExternalSubDeclaration ::= [ Attributes ] [ ExternalMethodModifier+ ] Declare [ CharsetModifier ] Sub Identifier LibraryClause [ AliasClause ] [ OpenParenthesis [ ParameterList ] CloseParenthesis ] StatementTerminator ExternalFunctionDeclaration ::= [ Attributes ] [ ExternalMethodModifier+ ] Declare [ CharsetModifier ] Funct ion Identifier LibraryClause [ AliasClause ] [ OpenParenthesis [ ParameterList ] CloseParenthesis ] [ As [ Attributes ] TypeName ] StatementTerminator ExternalMethodModifier ::= AccessModifier | Shadows | Overloads CharsetModifier ::= Ansi | Unicode | Auto LibraryClause ::= Lib StringLiteral AliasClause ::= Alias StringLiteral 1.55.3 Overridable Methods The Overridable modifier indicates that a method is overridable. The Overrides modifier indicates that a method overrides a base-type overridable method that has the same signature. The NotOverridable modifier indicates that an overridable method cannot be further overridden. The MustOverride modifier indicates that a method must be overridden in derived classes.
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
199
4.3
Inheritance
There are also additional restrictions on overridable methods: A MustOverride method may not include a method body or an End construct, may not override another method, and may only appear in MustInherit classes. If a method specifies Overrides and there is no matching base method to override, a compile-time error occurs. An overriding method may not specify Shadows. A method may not override another method if the overriding method's accessibility domain is not equal to the accessibility domain of the method being overridden. The one exception is that a method overriding a Protected Friend method in another assembly that does not have Friend access must specify Protected (not Protected Friend).
Private methods may not be Overridable, NotOverridable, or MustOverride, nor may they
override other methods. Methods in NotInheritable classes may not be declared Overridable or MustOverride. The following example illustrates the differences between overridable and nonoverridable methods:
Class Base Publ i c Sub F() Console .Wri teL ine (" Base.F" ) End Sub
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
200
4.3
Inheritance
Public Overridable Sub G() Console.WriteLine("Base.G") End Sub End Class Class Derived Inherits Base Public Shadows Sub F() Console.WriteLine("Derived.F") End Sub Public Overrides Sub G() Console.WriteLine("Derived.G") End Sub End Class Module Test Sub Main() Dim d As Derived = New Derived() Dim b As Base = d b.F() d.F() b.G() d.G() End Sub End Module
In the example, class Base introduces a method F and an Overridable method G. The class Derived introduces a new method F, thus shadowing the inherited F, and also overrides the inherited method G. The example produces the following output:
Base.F Derived.F Derived.G Derived.G
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
201
4.3
Inheritance
Notice that the statement b.G() invokes Derived.G not Base.G. This is because the run-time , type of the instance (which is Derived) rather than the compile-time type of the instance (which is Base) determines the actual method implementation to invoke. 1.55.4 Shared Methods The Shared modifier indicates a method is a shared method. A shared method does not operate on a specific instance of a type and may be invoked directly from a type rather than through a particular instance of a type. It is valid, however, to use an instance to qualify a shared method. It is invalid to refer to Me, MyClass, or MyBase in a shared method. Shared methods may not be Overridable, NotOverridable, or MustOverride, and they may not override methods. Methods defined in standard modules and interfaces may not specify Shared, because they are implicitly Shared already. A method declared in a structure or class without a Shared modifier is an instance method. An instance method operates on a given instance of a type. Instance methods can only be invoked through an instance of a type and may refer to the instance through the Me expression. The following example illustrates the rules for accessing shared and instance members:
Class Test Pr ivate x As Integer Pr ivate Shared y As Integer Sub F() x = 1 ' Ok, same as Me.x = 1. y = 1 ' Ok, same as Test.y = 1. End Sub Shared Sub G() x = 1 ' Error , cannot access Me.x. y = 1 ' Ok, same as Test.y = 1. End Sub Shared Sub Main() Dim t As Test = New Test( )
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
202
4.3
Inheritance
t.x = 1 ' Ok. t.y = 1 ' Ok. Test.x = 1 ' Error, cannot access instance member through type. Test.y = 1 ' Ok. End Sub End Class
Method F shows that in an instance function member, an identifier can be used to access both instance members and shared members. Method G shows that in a shared function member, it is an error to access an instance member through an identifier. Method Main shows that in a member access expression, instance members must be accessed through instances, but shared members can be accessed through types or instances. 1.55.5 Method Parameters A parameter is a variable that can be used to pass information into and out of a method. Parameters of a method are declared by the method's parameter list, which consists of one or more parameters separated by commas. If no type is specified for a parameter and strict semantics are used, a compile-time error occurs. Otherwise the default type is Object or the type of the parameter's type character. Even under permissive semantics, if one parameter includes an As clause, all parameters must specify types. Parameters are specified as value, reference, optional, or paramarray parameters by the modifiers ByVal, ByRef, Optional, and ParamArray, respectively. A parameter that does not specify ByRef or ByVal defaults to ByVal. Parameter names are scoped to the entire body of the method and are always publicly accessible. A method invocation creates a copy, specific to that invocation, of the parameters, and the argument list of the invocation assigns values or variable references to the newly created parameters. Because external method declarations and delegate declarations have no body, duplicate parameter names are allowed in parameter lists, but discouraged. The identifier may be followed by the nullable name modifier ? to indicate that it is nullable, and also by array name modifiers to indicate that it is an array. They may be
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
203
4.3
Inheritance
combined, e.g. ByVal x?() As Integer is not allowed to use explicit array bounds; also, if . It the nullable name modifier is present then an As clause must be present. ParameterList ::= Parameter | ParameterList Comma Parameter Parameter ::= [ Attributes ] [ ParameterModifier+ ] ParameterIdentifier [ As TypeName ] [ Equals ConstantExpression ] ParameterModifier ::= ByVal | ByRef | Optional | ParamArray ParameterIdentifier ::= Identifier IdentifierModifiers 1.55.5.1 Value Parameters A value parameter is declared with an explicit ByVal modifier. If the ByVal modifier is used, the ByRef modifier may not be specified. A value parameter comes into existence with the invocation of the member the parameter belongs to, and is initialized with the value of the argument given in the invocation. A value parameter ceases to exist upon return of the member. A method is permitted to assign new values to a value parameter. Such assignments only affect the local storage location represented by the value parameter; they have no effect on the actual argument given in the method invocation. A value parameter is used when the value of an argument is passed into a method, and modifications of the parameter do not impact the original argument. A value parameter refers to its own variable, one that is distinct from the variable of the corresponding argument. This variable is initialized by copying the value of the corresponding argument. The following example shows a method F that has a value parameter named p:
Module Test Sub F(p As Integer ) Console .Wri teL ine ("p = " & p) p += 1 End Sub
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
204
4.3
Inheritance
Sub Main() Dim a As Integer = 1 Console.WriteLine("pre: a = " & a) F(a) Console.WriteLine("post: a = " & a) End Sub End Module
The example produces the following output, even though the value parameter p is modified:
pre: a = 1 p =1 post: a = 1
1.55.5.2 Reference Parameters A reference parameter is a parameter declared with a ByRef modifier. If the ByRef modifier is specified, the ByVal modifier may not be used. A reference parameter does not create a new storage location. Instead, a reference parameter represents the variable given as the argument in the method or constructor invocation. Conceptually, the value of a reference parameter is always the same as the underlying variable. A reference parameter is used when the parameter acts as an alias for a caller-provided argument. A reference parameter does not itself define a variable, but rather refers to the variable of the corresponding argument. Modifications of a reference parameter directly and immediately impact the corresponding argument. The following example shows a method Swap that has two reference parameters:
Module Test Sub Swap(ByRef a As Integer , ByRef b As Integer ) Dim t As Integer = a a =b b =t End Sub
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
205
4.3
Inheritance
Sub Main() Dim x As Integer = 1 Dim y As Integer = 2 Console.WriteLine("pre: x = " & x & ", y = " & y) Swap(x, y) Console.WriteLine("post: x = " & x & ", y = " & y) End Sub End Module
For the invocation of method Swap in class Main, a represents x, and b represents y. Thus, the invocation has the effect of swapping the values of x and y. In a method that takes reference parameters, it is possible for multiple names to represent the same storage location:
Module Test Pr ivate s As Str ing Sub F(ByRef a As Str ing , ByRef b As Str ing) s = "One" a = "Two" b = "Three" End Sub Sub G() F(s , s) End Sub End Module
In the example the invocation of method F in G passes a reference to s for both a and b. Thus, for that invocation, the names s, a, and b all refer to the same storage location, and the three assignments all modify the instance variable s.
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
206
4.3
Inheritance
If the type of the variable being passed to a reference parameter is not compatible with the reference parameter's type, or if a non-variable is passed as an argument to a reference parameter, a temporary variable may be allocated and passed to the reference parameter. The value being passed in will be copied into this temporary variable before the method is invoked and will be copied back to the original variable (if there is one) when the method returns. Thus, a reference parameter may not necessarily contain a reference to the exact storage of the variable being passed in, and any changes to the reference parameter may not be reflected in the variable until the method exits. For example:
Class Base End Class Class Derived Inherits Base End Class Module Test Sub F(ByRef b As Base) b = New Base() End Sub Property G() As Base Get End Get Set End Set End Property Sub Main() Dim d As Derived F(G) F(d) End Sub End Module ' OK. ' Throws System.InvalidCastException after F returns.
207
4.3
Inheritance
In the case of the first invocation of F, a temporary variable is created and the value of the property G is assigned to it and passed into F. Upon return from F, the value in the temporary variable is assigned back to the property of G. In the second case, another temporary variable is created and the value of d is assigned to it and passed into F. When returning from F, the value in the temporary variable is cast back to the type of the variable, Derived, and assigned to d. Since the value being passed back cannot be cast to Derived, an exception is thrown at run time. 1.55.5.3 Optional Parameters An optional parameter is declared with the Optional modifier. Parameters that follow an optional parameter in the formal parameter list must be optional as well; failure to specify the Optional modifier on the following parameters will trigger a compile-time error. An optional parameter of some type T must specify a constant expression e to be used as a default value if no argument is specified. If e evaluates to Nothing of type Object, then the default value of the T will be used as the default for the parameter. Otherwise, CType(e, T) must be a constant expression and it is taken as the default for the parameter. Optional parameters are the only situation in which an initializer on a parameter is valid. The initialization is always done as a part of the invocation expression, not within the method body itself.
Module Test Sub F(x As Integer , Optional y As Integer = 20) Console .Wri teL ine ("x = " & x & " , y = " & y) End Sub Sub Main() F(10) F(30,40) End Sub End Module
208
4.3
Inheritance
Optional parameters may not be specified in delegate or event declarations, nor in lambda expressions. 1.55.5.4 ParamArray Parameters
ParamArray parameters are declared with the ParamArray modifier. If the ParamArray modifier is present, the ByVal modifier must be specified, and no other parameter may use the ParamArray modifier. The ParamArray parameter's type must be a one-dimensional
array, and it must be the last parameter in the parameter list. A ParamArray parameter represents an indeterminate number of parameters of the type of the ParamArray. Within the method itself, a ParamArray parameter is treated as its declared type and has no special semantics. A ParamArray parameter is implicitly optional, with a default value of an empty one-dimensional array of the type of the ParamArray. A ParamArray permits arguments to be specified in one of two ways in a method invocation: The argument given for a ParamArray can be a single expression of a type that widens to the ParamArray type. In this case, the ParamArray acts precisely like a value parameter. Alternatively, the invocation can specify zero or more arguments for the ParamArray, where each argument is an expression of a type that is implicitly convertible to the element type of the ParamArray. In this case, the invocation creates an instance of the ParamArray type with a length corresponding to the number of arguments, initializes the elements of the array instance with the given argument values, and uses the newly created array instance as the actual argument. Except for allowing a variable number of arguments in an invocation, a ParamArray is precisely equivalent to a value parameter of the same type, as the following example illustrates.
Module Test Sub F(ParamArray args( ) As Integer ) Dim i As Integer Console .Wri te ( "Ar ray contains " & args.Length & " elements:" ) For Each i In args Console .Wri te ( " " & i )
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
209
4.3
Inheritance
Next i Console.WriteLine() End Sub Sub Main() Dim a As Integer() = { 1, 2, 3 } F(a) F(10, 20, 30, 40) F() End Sub End Module
The first invocation of F simply passes the array a as a value parameter. The second invocation of F automatically creates a four-element array with the given element values and passes that array instance as a value parameter. Likewise, the third invocation of F creates a zero-element array and passes that instance as a value parameter. The second and third invocations are precisely equivalent to writing:
F(New Integer ( ) {10, 20, 30, 40}) F(New Integer ( ) {}) ParamArray parameters may not be specified in delegate or event declarations.
1.55.6 Event Handling Methods can declaratively handle events raised by objects in instance or shared variables. To handle events, a method declaration specifies the Handles keyword and lists one or more events. An event in the Handles list is specified by two identifiers separated by a period: The first identifier must be an instance or shared variable in the containing type that specifies the WithEvents modifier or the MyBase or Me keyword; otherwise, a compile-time Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved. 210
4.3
Inheritance
error occurs. This variable contains the object that will raise the events handled by this method. The second identifier must specify a member of the type of the first identifier. The member must be an event, and may be shared. If a shared variable is specified for the first identifier, then the event must be shared, or an error results. A handler method M is considered a valid event handler for an event E if the statement AddHandler E, AddressOf M would also be valid. Unlike an AddHandler statement, however, explicit event handlers allow handling an event with a method with no arguments regardless of whether strict semantics are being used or not:
Option Str i c t On Class C1 Event E(x As Integer ) End Class Class C2 withEvents C1 As New C1() ' Val id Sub M1() Handles C1.E End Sub Sub M2() ' Inva l i d AddHandler C1.E, AddressOf M1 End Sub End Class
A single member can handle multiple matching events, and multiple methods may handle a single event. A method's accessibility has no effect on its ability to handle events. The following example shows how a method can handle events:
Class Raiser Event E1()
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
211
4.3
Inheritance
Sub Raise() RaiseEvent E1 End Sub End Class Module Test WithEvents x As Raiser Sub E1Handler() Handles x.E1 Console.WriteLine("Raised") End Sub Sub Main() x = New Raiser() x.Raise() x.Raise() End Sub End Module
A type inherits all event handlers provided by its base type. A derived type cannot in any way alter the event mappings it inherits from its base types, but may add additional handlers to the event. HandlesClause ::= [ Handles EventHandlesList ] EventHandlesList ::= EventMemberSpecifier | EventHandlesList Comma EventMemberSpecifier EventMemberSpecifier ::= Identifier Period IdentifierOrKeyword | MyBase Period IdentifierOrKeyword |
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
212
4.3
Inheritance
1.55.7 Extension Methods Methods can be added to types from outside of the type declaration using extension methods. Extension methods are methods with the System.Runtime.CompilerServ ices .Extens ionAtt r i bu te attribute applied to them. They can only be declared in standard modules and must have at least one parameter, which specifies the type the method extends. For example, the following extension method extends the type Str ing :
Imports System.Runtime.CompilerServ ices Module Str ingExtens ions <Extension> _ Sub Pr int ( s As Str ing) Console .Wri teL ine (s ) End Sub End Module
Annotation Although Visual Basic requires extension methods to be declared in a standard module, other languages such as C# may allow them to be declared in other kinds of types. As long as the methods follow the other conventions outlined here and the containing type is not an open generic type and cannot be instantiated, Visual Basic will recognize the extension methods. When an extension method is invoked, the instance it is being invoked on is passed to the first parameter. The first parameter cannot be declared Optional or ParamArray. Any type, including a type parameter, can appear as the first parameter of an extension method. For example, the following methods extend the types Integer(), any type that implements System.Collections.Generic.IEnumerable(Of T), and any type at all:
Imports System.Runtime.CompilerServ ices Module Extensions
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
213
4.3
Inheritance
<Extension> _ Sub PrintArray(a() As Integer) ... End Sub <Extension> _ Sub PrintList(Of T)(a As IEnumerable(Of T)) ... End Sub <Extension> _ Sub Print(Of T)(a As T) ... End Sub End Module
As the previous example shows, interfaces can be extended. Interface extension methods supply the implementation of the method, so types that implement an interface that has extension methods defined on it still only implement the members originally declared by the interface. For example:
Imports System.Runtime.CompilerServices Interface IAction Sub DoAction() End Interface Module IActionExtensions <Extension> _ Public Sub DoAnotherAction(i As IAction) i.DoAction() End Sub End Module Class C Implements IAction
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
214
4.3
Inheritance
Sub DoAction() Implements IAction.DoAction ... End Sub ' ERROR: Cannot implement extension method IAction.DoAnotherAction Sub DoAnotherAction() Implements IAction.DoAnotherAction ... End Sub End Class
Extension methods can also have type constraints on their type parameters and, just as with non-extension generic methods, type argument can be inferred:
Imports System.Runtime.CompilerServices Module IEnumerableComparableExtensions <Extension> _ Public Function Sort(Of T As IComparable(Of T))(i As IEnumerable(Of T)) _ As IEnumerable(Of T) ... End Function
Extension methods can also be accessed through implicit instance expressions within the type being extended:
Imports System.Runtime.CompilerServices Class C1 Sub M1() Me.M2() M2() End Sub End Class Module C1Extensions <Extension> _ Sub M2(c As C1)
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
215
4.3
Inheritance
For the purposes of accessibility, extension methods are also treated as members of the standard module they are declared in they have no extra access to the members of the type they are extending beyond the access they have by virtue of their declaration context. Extensions methods are only available when the standard module method is in scope. Otherwise, the original type will not appear to have been extended. For example:
Imports System.Runtime.CompilerServices Class C1 End Class Namespace N1 Module C1Extensions <Extension> _ Sub M1(c As C1) ... End Sub End Module End Namespace Module Test Sub Main() Dim c As New C1() ' Error: c has no member named "M1" c.M1() End Sub End Module
Referring to a type when only an extension method on the type is available will still produce a compile-time error.
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
216
4.3
Inheritance
It is important to note that extension methods are considered to be members of the type in all contexts where members are bound, such as the strongly-typed For Each pattern. For example:
Imports System.Runtime.CompilerServ ices Class C1 End Class Class C1Enumerator ReadOnly Property Current( ) As C1 Get ... End Get End Property Funct ion MoveNext() As Boolean ... End Funct ion End Class Module C1Extensions <Extension> _ Funct ion GetEnumerator(c As C1) As C1Enumerator ... End Funct ion End Module Module Test Sub Main() Dim c As New C1() ' Val id For Each o As Object In c ... Next o
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
217
4.3
Inheritance
Delegates can also be created that refer to extension methods. Thus, the code:
Delegate Sub D1() Module Test Sub Main() Dim s As String = "Hello, World!" Dim d As D1 d = AddressOf s.Print d() End Sub End Module
Annotation Visual Basic normally inserts a check on an instance method call that causes a System.Nul lReferenceExcept ion occur if the instance the method is being invoked on is to Nothing. In the case of extension methods, there is no efficient way to insert this check, so extension methods will need to explicitly check for Nothing.
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
218
4.3
Inheritance
Additionally, a value type will be boxed when being passed as a ByVal argument to a parameter typed as an interface. This implies that side effects of the extension method will operate on a copy of the structure instead of the original. While the language puts no restrictions on the first argument of an extension method, it is recommended that extension methods are not used to extend value types or that when extending value types, the first parameter is passed ByRef to ensure that side effects operate on the original value. 1.55.8 Partial Methods A partial method is a method that specifies a signature but not the body of the method. The body of the method can be supplied by another method declaration with the same name and signature, most likely in another partial declaration of the type. For example: a.vb:
' Designer generated code Publ i c Part ia l Class MyForm Pr ivate Part ia l Sub Val idateContro l s ( ) End Sub Publ i c Sub New() ' In i t i a l i z e contro l s ... Val idateContro l s ( ) End Sub End Class
b.vb:
Publ i c Part ia l Class MyForm Publ i c Sub Val idateContro l s ( ) ' Val idat i on log ic goes here ...
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
219
4.3
Inheritance
In this example, a partial declaration of the class MyForm declares a partial method Val idateContro lwith no implementation. The constructor in the partial declaration calls the s partial method, even though there is no body supplied in the file. The other partial declaration of MyForm then supplies the implementation of the method. Partial methods can be called regardless of whether a body has been supplied; if no method body is supplied, the call is ignored. For example:
Publ i c Class C1 Pr ivate Part ia l End Sub Sub M1()
made.
Publ i c Sub New() ' Since no implementat ion i s suppl ied , th is cal l M1() End Sub
wil l
not be
Any expressions that are passed in as arguments to a partial method call that is ignored are ignored also and not evaluated. Annotation This means that partial methods are a very efficient way of providing behavior that is defined across two partial types, since the partial methods have no cost if they are not used. The partial method declaration must be declared as Private and must always be a subroutine with no statements in its body. Partial methods cannot themselves implement interface methods, although the method that supplies their body can. Only one method can supply a body to a partial method. A method supplying a body to a partial method must have the same signature as the partial method, the same constraints on any type parameters, the same declaration modifiers, and the same parameter and type parameter names. Attributes on the partial method and the method that supplies its
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
220
4.3
Inheritance
body are merged, as are any attributes on the methods parameters. Similarly, the list of events that the methods handle is merged. For example:
Class C1 Event E1() Event E2() Private Partial Sub S() Handles Me.E1 End Sub ' Handles both E1 and E2 Private Sub S() Handles Me.E2 ... End Sub End Class
1.56 Constructors
Constructors are special methods that allow control over initialization. They are run after the program begins or when an instance of a type is created. Unlike other members, constructors are not inherited and do not introduce a name into a type's declaration space. Constructors may only be invoked by object-creation expressions or by the .NET Framework; they may never be directly invoked. Note Constructors have the same restriction on line placement that subroutines have. The beginning statement, end statement and block must all appear at the beginning of a logical line. ConstructorMemberDeclaration ::= [ Attributes ] [ ConstructorModifier+ ] Sub New [ OpenParenthesis [ ParameterList ] CloseParenthesis ] LineTerminator [ Block ] End Sub StatementTerminator ConstructorModifier ::= AccessModifier | Shared
221
4.3
Inheritance
1.56.1 Instance Constructors Instance constructors initialize instances of a type and are run by the .NET Framework when an instance is created. The parameter list of a constructor is subject to the same rules as the parameter list of a method. Instance constructors may be overloaded. All constructors in reference types must invoke another constructor. If the invocation is explicit, it must be the first statement in the constructor method body. The statement can either invoke another of the type's instance constructors for example, Me.New(. . . )or MyClass.New(. . . or if it is not a structure it can invoke an instance constructor of the ) type's base type for example, MyBase.New(...). It is invalid for a constructor to invoke itself. If a constructor omits a call to another constructor, MyBase.New() is implicit. If there is no parameterless base type constructor, a compile-time error occurs. Because Me is not considered to be constructed until after the call to a base class constructor, the parameters to a constructor invocation statement cannot reference Me, MyClass, or MyBase implicitly or explicitly. When a constructor's first statement is of the form MyBase.New(...), the constructor implicitly performs the initializations specified by the variable initializers of the instance variables declared in the type. This corresponds to a sequence of assignments that are executed immediately after invoking the direct base type constructor. Such ordering ensures that all base instance variables are initialized by their variable initializers before any statements that have access to the instance are executed. For example:
Class A Protected x As Integer = 1 End Class Class B Inher i t s A Pr ivate y As Integer = x Publ i c Sub New() Console .Wri teL ine ("x = " & x & " , y = " & y)
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
222
4.3
Inheritance
When New B() is used to create an instance of B, the following output is produced:
x = 1, y = 1
The value of y is 1 because the variable initializer is executed after the base class constructor is invoked. Variable initializers are executed in the textual order they appear in the type declaration. When a type declares only Pr ivateconstructors, it is not possible in general for other types to derive from the type or create instances of the type; the only exception is types nested within the type. Pr ivateconstructors are commonly used in types that contain only Shared members. If a type contains no instance constructor declarations, a default constructor is automatically provided. The default constructor simply invokes the parameterless constructor of the direct base type. If the direct base type does not have an accessible parameterless constructor, a compile-time error occurs. The declared access type for the default constructor is Public unless the type is MustInherit, in which case the default constructor is Protected. Annotation The default access for a MustInherit types default constructor is Protected because MustInherit classes cannot be created directly. So there is no point in making the default constructor Public. In the following example a default constructor is provided because the class contains no constructor declarations:
Class Message Dim sender As Object Dim text As Str ing End Class
223
4.3
Inheritance
Class Message Dim sender As Object Dim text As String Sub New() End Sub End Class
Default constructors that are emitted into a designer generated class marked with the attribute Microsof t .V i sua lBas ic .Compi le rServ i ces .Des ignerGeneratedAtt r i butethe method will call Sub In i t i a l i z eComponent()it exists, after the call to the base constructor. , if Annotation This allows designer generated files, such as those created by the WinForms designer, to omit the constructor in the designer file. This enables the programmer to specify it themselves, if they so choose. 1.56.2 Shared Constructors Shared constructors initialize a type's shared variables; they are run after the program begins executing, but before any references to a member of the type. A shared constructor specifies the Shared modifier, unless it is in a standard module in which case the Shared modifier is implied. Unlike instance constructors, shared constructors have implicit public access, have no parameters, and may not call other constructors. Before the first statement in a shared constructor, the shared constructor implicitly performs the initializations specified by the variable initializers of the shared variables declared in the type. This corresponds to a sequence of assignments that are executed immediately upon entry to the constructor. The variable initializers are executed in the textual order they appear in the type declaration. The following example shows an Employee class with a shared constructor that initializes a shared variable:
Imports System.Data
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
224
4.3
Inheritance
Class Employee Private Shared ds As DataSet Shared Sub New() ds = New DataSet() End Sub Public Name As String Public Salary As Decimal End Class
A separate shared constructor exists for each closed generic type. Because the shared constructor is executed exactly once for each closed type, it is a convenient place to enforce run-time checks on the type parameter that cannot be checked at compile-time via constraints. For example, the following type uses a shared constructor to enforce that the type parameter is Integeror Double:
Class EnumHolder(Of T) Shared Sub New() I f Not GetType(T) . I sEnum() Then Throw New ArgumentException("T must be an enumerated type." ) End I f End Sub End Class
Exactly when shared constructors are run is mostly implementation dependent, though several guarantees are provided if a shared constructor is explicitly defined: Shared constructors are run before the first access to any static field of the type. Shared constructors are run before the first invocation of any static method of the type. Shared constructors are run before the first invocation of any constructor for the type. The above guarantees do not apply in the situation where a shared constructor is implicitly created for shared initializers. The output from the following example is uncertain, because the exact ordering of loading and therefore of shared constructor execution is not defined:
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
225
4.3
Inheritance
Module Test Sub Main() A.F() B.F() End Sub End Module Class A Shared Sub New() Console.WriteLine("Init A") End Sub Public Shared Sub F() Console.WriteLine("A.F") End Sub End Class Class B Shared Sub New() Console.WriteLine("Init B") End Sub Public Shared Sub F() Console.WriteLine("B.F") End Sub End Class
or
Init B Init A
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
226
4.3
Inheritance
A.F B.F
By contrast, the following example produces predictable output. Note that the Shared constructor for the class A never executes, even though class B derives from it:
Module Test Sub Main() B.G() End Sub End Module Class A Shared Sub New() Console .Wri teL ine (" In i t End Sub End Class Class B Inher i t s A Shared Sub New() Console .Wri teL ine (" In i t End Sub Publ i c Shared Sub G() Console .Wri teL ine ("B .G" ) End Sub End Class B")
A")
It is also possible to construct circular dependencies that allow Shared variables with variable initializers to be observed in their default value state, as in the following example:
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
227
4.3
Inheritance
Class A Public Shared X As Integer = B.Y + 1 End Class Class B Public Shared Y As Integer = A.X + 1 Shared Sub Main() Console.WriteLine("X = " & A.X & ", Y = " & B.Y) End Sub End Class
To execute the Main method, the system first loads class B. The Shared constructor of class B proceeds to compute the initial value of Y, which recursively causes class A to be loaded because the value of A.X is referenced. The Shared constructor of class A in turn proceeds to compute the initial value of X, and in doing so fetches the default value of Y, which is zero. A.X is thus initialized to 1. The process of loading A then completes, returning to the calculation of the initial value of Y, the result of which becomes 2. Had the Main method instead been located in class A, the example would have produced the following output:
X = 2, Y = 1
Avoid circular references in Shared variable initializers since it is generally impossible to determine the order in which classes containing such references are loaded.
1.57 Events
Events are used to notify code of a particular occurrence. An event declaration consists of an identifier, either a delegate type or a parameter list, and an optional Implements clause. If a delegate type is specified, the delegate type may not have a return type. If a parameter list is specified, it may not contain Optional or ParamArray parameters. The accessibility domain of the parameter types and/or delegate type must be the same as, or
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
228
4.3
Inheritance
a superset of, the accessibility domain of the event itself. Events may be shared by specifying the Shared modifier. In addition to the member name added to the type's declaration space, an event declaration implicitly declares several other members. Given an event named X, the following members are added to the declaration space: If the form of the declaration is a method declaration, a nested delegate class named XEventHandler is introduced. The nested delegate class matches the method declaration and has the same accessibility as the event. The attributes in the parameter list apply to the parameters of the delegate class. A Pr ivateinstance variable typed as the delegate, named XEvent. A method named add_X, which takes the delegate type and has the same access type as the event. A method named remove_X , which takes the delegate type and has the same access type as the event. If a type attempts to declare a name that matches one of the above names, a compile-time error will result, and the implicit add_X and remove_X declarations are ignored for the purposes of name binding. It is not possible to override or overload any of the introduced members, although it is possible to shadow them in derived types. For example, the class declaration
Class Raiser Publ i c Event Constructed( i As Integer ) End Class
229
4.3
Inheritance
CType( _ [Delegate].Combine(ConstructedEvent, d), _ Raiser.ConstructedEventHandler) End Sub Public Sub remove_Constructed(d As ConstructedEventHandler) ConstructedEvent = _ CType( _ [Delegate].Remove(ConstructedEvent, d), _ Raiser.ConstructedEventHandler) End Sub End Class
Declaring an event without specifying a delegate type is the simplest and most compact syntax, but has the disadvantage of declaring a new delegate type for each event. For example, in the following example, three hidden delegate types are created, even though all three events have the same parameter list:
Public Class Button Public Event Click(sender As Object, e As EventArgs) Public Event DoubleClick(sender As Object, e As EventArgs) Public Event RightClick(sender As Object, e As EventArgs) End Class
In the following example, the events simply use the same delegate, EventHandler:
Publ i c Delegate Sub EventHandler (sender As Object , e As EventArgs) Publ i c Class Button Publ i c Event Cl ick As EventHandler Publ i c Event DoubleCl i ck As EventHandler Publ i c Event RightCl i ck As EventHandler End Class
Events can be handled in one of two ways: statically or dynamically. Statically handling events is simpler and only requires a WithEvents variable and a Handles clause. In the following example, class Form1 statically handles the event Click of object Button:
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
230
4.3
Inheritance
Public Class Form1 Public WithEvents Button1 As New Button() Public Sub Button1_Click(sender As Object, e As EventArgs) _ Handles Button1.Click Console.WriteLine("Button1 was clicked!") End Sub End Class
Dynamically handling events is more complex because the event must be explicitly connected and disconnected to in code. The statement AddHandler adds a handler for an event, and the statement RemoveHandler removes a handler for an event. The next example shows a class Form1 that adds Button1_Click as an event handler for Button1's Click event:
Publ i c Class Form1 Publ i c Sub New() ' Add Button1_Cl i ck as an event handler for Button1's Cl ick event. AddHandler Button1.C l i ck , AddressOf Button1_Cl i ck End Sub Pr ivate Button1 As Button = New Button() Sub Button1_Cl i ck (sender As Object , e As EventArgs) Console .Wri teL ine ("But ton1 was cl i cked!" ) End Sub Publ i c Sub Disconnect( ) RemoveHandler Button1.C l i ck , End Sub End Class AddressOf Button1_Cl i ck
In method Disconnect, the event handler is removed. EventMemberDeclaration ::= RegularEventMemberDeclaration | CustomEventMemberDeclaration
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
231
4.3
Inheritance
RegularEventMemberDeclaration ::= [ Attributes ] [ EventModifiers+ ] Event Identifier ParametersOrType [ ImplementsClause ] StatementTerminator InterfaceEventMemberDeclaration ::= [ Attributes ] [ InterfaceEventModifiers+ ] Event Identifier ParametersOrType StatementTerminator ParametersOrType ::= [ OpenParenthesis [ ParameterList ] CloseParenthesis ] | As NonArrayTypeName EventModifiers ::= AccessModifier | Shadows | Shared InterfaceEventModifiers ::= Shadows 1.57.1 Custom Events As discussed in the previous section, event declarations implicitly define a field, an add_ method, and a remove_ method that are used to keep track of event handlers. In some situations, however, it may be desirable to provide custom code for tracking event handlers. For example, if a class defines forty events of which only a few will ever be handled, using a hash table instead of forty fields to track the handlers for each event may be more efficient. Custom events allow the add_X and remove_X methods to be defined explicitly, which enables custom storage for event handlers. Custom events are declared in the same way that events that specify a delegate type are declared, with the exception that the keyword Custom must precede the Event keyword. A custom event declaration contains three declarations: an AddHandler declaration, a RemoveHandler declaration and a RaiseEvent declaration. None of the declarations can have any modifiers, although they can have attributes. For example:
Class Test Pr ivate Handlers As EventHandler Publ i c Custom Event TestEvent As EventHandler AddHandler (va lue As EventHandler )
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
232
4.3
Inheritance
Handlers = CType([Delegate].Combine(Handlers, value), _ EventHandler) End AddHandler RemoveHandler(value as EventHandler) Handlers = CType([Delegate].Remove(Handlers, value), _ EventHandler) End RemoveHandler RaiseEvent(sender As Object, e As EventArgs) Dim TempHandlers As EventHandler = Handlers If TempHandlers IsNot Nothing Then TempHandlers(sender, e) End If End RaiseEvent End Event End Class
The AddHandler and RemoveHandler declaration take one ByVal parameter, which must be of the delegate type of the event. When an AddHandler or RemoveHandler statement is executed (or a Handles clause automatically handles an event), the corresponding declaration will be called. The RaiseEvent declaration takes the same parameters as the event delegate and will be called when a RaiseEvent statement is executed. All of the declarations must be provided and are considered to be subroutines. Note AddHandler, RemoveHandler and RaiseEvent declarations have the same restriction on line placement that subroutines have. The beginning statement, end statement and block must all appear at the beginning of a logical line. In addition to the member name added to the type's declaration space, a custom event declaration implicitly declares several other members. Given an event named X, the following members are added to the declaration space: A method named add_X, corresponding to the AddHandler declaration. A method named remove_X, corresponding to the RemoveHandler declaration.
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
233
4.3
Inheritance
A method named f i r e _X corresponding to the RaiseEvent declaration. , If a type attempts to declare a name that matches one of the above names, a compile-time error will result, and the implicit declarations are all ignored for the purposes of name binding. It is not possible to override or overload any of the introduced members, although it is possible to shadow them in derived types. Note Custom is not a reserved word. CustomEventMemberDeclaration ::= [ Attributes ] [ EventModifiers+ ] Custom Event Identifier As TypeName [ ImplementsClause ] StatementTerminator EventAccessorDeclaration+ End Event StatementTerminator EventAccessorDeclaration ::= AddHandlerDeclaration | RemoveHandlerDeclaration | RaiseEventDeclaration AddHandlerDeclaration ::= [ Attributes ] AddHandler OpenParenthesis ParameterList CloseParenthesis LineTerminator [ Block ] End AddHandler StatementTerminator RemoveHandlerDeclaration ::= [ Attributes ] RemoveHandler OpenParenthesis ParameterList CloseParenthesis LineTerminator [ Block ] End RemoveHandler StatementTerminator RaiseEventDeclaration ::= [ Attributes ] RaiseEvent OpenParenthesis ParameterList CloseParenthesis LineTerminator
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
234
4.3
Inheritance
[ Block ]
End RaiseEvent StatementTerminator
1.58 Constants
A constant is a constant value that is a member of a type. Constants are implicitly shared. If the declaration contains an As clause, the clause specifies the type of the member introduced by the declaration. If the type is omitted and strict semantics are being used, a compile-time error occurs; otherwise the type of the constant is implicitly Object. The type of a constant may only be a primitive type or Object. If a constant is typed as Object and there is no type character, the real type of the constant will be the type of the constant expression. Otherwise, the type of the constant is the type of the constant's type character. The following example shows a class named Constants that has two public constants:
Class Constants Publ i c Const A As Integer = 1 Publ i c Const B As Integer = A + 1 End Class
Constants can be accessed through the class, as in the following example, which prints out the values of Constants.A and Constants.B.
Module Test Sub Main() Console .Wri teL ine (Constants .A & " , " & Constants .B) End Sub End Module
A constant declaration that declares multiple constants is equivalent to multiple declarations of single constants. The following example declares three constants in one declaration statement.
Class A Protected Const x As Integer = 1, y As Long = 2, z As Short = 3 End Class
235
4.3
Inheritance
Class A Protected Const x As Integer = 1 Protected Const y As Long = 2 Protected Const z As Short = 3 End Class
The accessibility domain of the type of the constant must be the same as or a superset of the accessibility domain of the constant itself. The constant expression must yield a value of the constant's type or of a type that is implicitly convertible to the constant's type. The constant expression may not be circular; that is, a constant may not be defined in terms of itself. The compiler automatically evaluates the constant declarations in the appropriate order. In the following example, the compiler first evaluates Y, then Z, and finally X, producing the values 10, 11, and 12, respectively.
Class A Publ i c Const X As Integer = B.Z + 1 Publ i c Const Y As Integer = 10 End Class Class B Publ i c Const Z As Integer = A.Y + 1 End Class
When a symbolic name for a constant value is desired, but the type of the value is not permitted in a constant declaration or when the value cannot be computed at compile time by a constant expression, a read-only variable may be used instead. ConstantMemberDeclaration ::= [ Attributes ] [ ConstantModifier+ ] Const ConstantDeclarators StatementTerminator ConstantModifier ::= AccessModifier | Shadows ConstantDeclarators ::= ConstantDeclarator | ConstantDeclarators Comma ConstantDeclarator
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
236
4.3
Inheritance
A variable declared with the Shared modifier is a shared variable. A shared variable identifies exactly one storage location regardless of the number of instances of the type that are created. A shared variable comes into existence when a program begins executing, and ceases to exist when the program terminates. A shared variable is shared only among instances of a particular closed generic type. For example, the program:
Class C(Of V) Shared InstanceCount As Integer = 0 Publ i c Sub New() InstanceCount += 1 End Sub Publ i c Shared ReadOnly Property Count() As Integer
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
237
4.3
Inheritance
Get Return InstanceCount End Get End Property End Class Class Application Shared Sub Main() Dim x1 As New C(Of Integer)() Console.WriteLine(C(Of Integer).Count) Dim x2 As New C(Of Double)() Console.WriteLine(C(Of Integer).Count) Dim x3 As New C(Of Integer)() Console.WriteLine(C(Of Integer).Count) End Sub End Class
Prints out:
1 1 2
A variable declared without the Shared modifier is called an instance variable. Every instance of a class contains a separate copy of all instance variables of the class. An instance variable of a reference type comes into existence when a new instance of that type is created, and ceases to exist when there are no references to that instance and the Fina l i ze method has executed. An instance variable of a value type has exactly the same lifetime as the variable to which it belongs. In other words, when a variable of a value type comes into existence or ceases to exist, so does the instance variable of the value type. If the declarator contains an As clause, the clause specifies the type of the members introduced by the declaration. If the type is omitted and strict semantics are being used, a compile-time error occurs. Otherwise the type of the members is implicitly Object or the type of the members' type character.
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
238
4.3
Inheritance
Note There is no ambiguity in the syntax: if a declarator omits a type, it will always use the type of a following declarator. The accessibility domain of an instance or shared variable's type or array element type must be the same as or a superset of the accessibility domain of the instance or shared variable itself. The following example shows a Color class that has internal instance variables named redPart greenPart, and bluePart: ,
Class Color Fr iend redPart As Short Fr iend bluePart As Short Fr iend greenPart As Short Publ i c Sub New(red As Short , blue As Short , green As Short) redPart = red bluePart = blue greenPart = green End Sub End Class
VariableMemberDeclaration ::= [ Attributes ] VariableModifier+ VariableDeclarators StatementTerminator VariableModifier ::= AccessModifier | Shadows | Shared | ReadOnly | WithEvents |
Dim
239
4.3
Inheritance
VariableDeclarator ::= VariableIdentifiers As ObjectCreationExpression | VariableIdentifiers [ As TypeName ] [ Equals Expression ] VariableIdentifiers ::= VariableIdentifier | VariableIdentifiers Comma VariableIdentifier VariableIdentifier ::= Identifier IdentifierModifiers 1.59.1 Read-Only Variables When an instance or shared variable declaration includes a ReadOnly modifier, assignments to the variables introduced by the declaration may only occur as part of the declaration or in a constructor in the same class. Specifically, assignments to a read-only instance or shared variable are permitted only in the following situations: In the variable declaration that introduces the instance or shared variable (by including a variable initializer in the declaration). For an instance variable, in the instance constructors of the class that contains the variable declaration. The instance variable can only be accessed in an unqualified manner or through Me or MyClass. For a shared variable, in the shared constructor of the class that contains the shared variable declaration. A shared read-only variable is useful when a symbolic name for a constant value is desired, but when the type of the value is not permitted in a constant declaration, or when the value cannot be computed at compile time by a constant expression. An example of the first such application follows, in which color shared variables are declared ReadOnly to prevent them from being changed by other programs:
Class Color Fr iend redPart As Short Fr iend bluePart As Short Fr iend greenPart As Short
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
240
4.3
Inheritance
Public Sub New(red As Short, blue As Short, green As Short) redPart = red bluePart = blue greenPart = green End Sub Public Public Public Public End Class Shared Shared Shared Shared ReadOnly ReadOnly ReadOnly ReadOnly Red As Color = New Color(&HFF, 0, 0) Blue As Color = New Color(0, &HFF, 0) Green As Color = New Color(0, 0, &HFF) White As Color = New Color(&HFF, &HFF, &HFF)
Constants and read-only shared variables have different semantics. When an expression references a constant, the value of the constant is obtained at compile time, but when an expression references a read-only shared variable, the value of the shared variable is not obtained until run time. Consider the following application, which consists of two separate programs. file1.vb:
Namespace Program1 Public Class Utils Public Shared ReadOnly X As Integer = 1 End Class End Namespace
file2.vb:
Namespace Program2 Module Test Sub Main() Console.WriteLine(Program1.Utils.X) End Sub End Module End Namespace
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
241
4.3
Inheritance
The namespaces Program1 and Program2 denote two programs that are compiled separately. Because variable Program1.Utils.X is declared as Shared ReadOnly, the value output by the Console.WriteLine statement is not known at compile time, but rather is obtained at run time. Thus, if the value of X is changed and Program1 is recompiled, the Console.WriteLine statement will output the new value even if Program2 is not recompiled. However, if X had been a constant, the value of X would have been obtained at the time Program2 was compiled, and would have remained unaffected by changes in Program1 until Program2 was recompiled. 1.59.2 WithEvents Variables A type can declare that it handles some set of events raised by one of its instance or shared variables by declaring the instance or shared variable that raises the events with the WithEvents modifier. For example:
Class Raiser Publ i c Event E1() Publ i c Sub Raise( ) RaiseEvent E1 End Sub End Class Module Test Pr ivate WithEvents x As Raiser Pr ivate Sub E1Handler ( ) Handles x.E1 Console .Wri teL ine ("Ra ised" ) End Sub Publ i c Sub Main() x = New Raiser ( ) End Sub End Module
In this example, the method E1Handler handles the event E1 that is raised by the instance of the type Raiser stored in the instance variable x. Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved. 242
4.3
Inheritance
The WithEvents modifier causes the variable to be renamed with a leading underscore and replaced with a property of the same name that does the event hookup. For example, if the variable's name is F, it is renamed to _F and a property F is implicitly declared. If there is a collision between the variable's new name and another declaration, a compile-time error will be reported. Any attributes applied to the variable are carried over to the renamed variable. The implicit property created by a WithEvents declaration takes care of hooking and unhooking the relevant event handlers. When a value is assigned to the variable, the property first calls the remove method for the event on the instance currently in the variable (unhooking the existing event handler, if any). Next the assignment is made, and the property calls the add method for the event on the new instance in the variable (hooking up the new event handler). The following code is equivalent to the code above for the standard module Test:
Module Test Pr ivate _x As Raiser Publ i c Property x() As Raiser Get Return _x End Get Set (Value As Raiser ) ' Unhook any exis t i ng handlers . I f _x IsNot Nothing Then RemoveHandler _x.E1, AddressOf E1Handler End I f ' Change value. _x = Value ' Hook- up new handlers . I f _x IsNot Nothing Then AddHandler _x.E1, AddressOf E1Handler End I f
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
243
4.3
Inheritance
End Set End Property Sub E1Handler() Console.WriteLine("Raised") End Sub Sub Main() x = New Raiser() End Sub End Module
It is not valid to declare an instance or shared variable as WithEvents if the variable is typed as a structure. In addition, WithEvents may not be specified in a structure, and WithEvents and ReadOnly cannot be combined. 1.59.3 Variable Initializers Instance and shared variable declarations in classes and instance variable declarations (but not shared variable declarations) in structures may include variable initializers. For Shared variables, variable initializers correspond to assignment statements that are executed after the program begins, but before the Shared variable is first referenced. For instance variables, variable initializers correspond to assignment statements that are executed when an instance of the class is created. Structures cannot have instance variable initializers because their parameterless constructors cannot be modified. Consider the following example:
Class Test Publ i c Shared x As Double = Math.Sqrt (2 .0 ) Publ i c i As Integer = 100 Publ i c s As Str ing = "Hel lo" End Class Module TestModule Sub Main() Dim a As New Test( )
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
244
4.3
Inheritance
Console.WriteLine("x = " & Test.x & ", i = " & a.i & ", s = " & a.s) End Sub End Module
An assignment to x occurs when the class is loaded, and assignments to i and s occur when a new instance of the class is created. It is useful to think of variable initializers as assignment statements that are automatically inserted in the block of the type's constructor. The following example contains several instance variable initializers.
Class A Pr ivate x As Integer = 1 Pr ivate y As Integer = - 1 Pr ivate count As Integer Publ i c Sub New() count = 0 End Sub Publ i c Sub New(n As Integer ) count = n End Sub End Class Class B Inher i t s A Pr ivate sqrt2 As Double = Math.Sqrt (2 .0 ) Pr ivate i tems As ArrayL is t = New ArrayL is t (100) Pr ivate max As Integer Publ i c Sub New()
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
245
4.3
Inheritance
Me.New(100) items.Add("default") End Sub Public Sub New(n As Integer) MyBase.New(n - 1) max = n End Sub End Class
The example corresponds to the code shown below, where each comment indicates an automatically inserted statement.
Class A Private x, y, count As Integer Public Sub New() MyBase.New ' Invoke object() constructor. x = 1 ' This is a variable initializer. y = -1 ' This is a variable initializer. count = 0 End Sub Public Sub New(n As Integer) MyBase.New ' Invoke object() constructor. x = 1 ' This is a variable initializer. y = - 1 ' This is a variable initializer. count = n End Sub End Class Class B Inherits A Private sqrt2 As Double Private items As ArrayList Private max As Integer
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
246
4.3
Inheritance
Public Sub New() Me.New(100) items.Add("default") End Sub Public Sub New(n As Integer) MyBase.New(n - 1) sqrt2 = Math.Sqrt(2.0) ' This is a variable initializer. items = New ArrayList(100) ' This is a variable initializer. max = n End Sub End Class
All variables are initialized to the default value of their type before any variable initializers are executed. For example:
Class Test Public Shared b As Boolean Public i As Integer End Class Module TestModule Sub Main() Dim t As New Test() Console.WriteLine("b = " & Test.b & ", i = " & t.i) End Sub End Module
Because b is automatically initialized to its default value when the class is loaded and i is automatically initialized to its default value when an instance of the class is created, the preceding code produces the following output:
b = False , i = 0
Each variable initializer must yield a value of the variable's type or of a type that is implicitly convertible to the variable's type. A variable initializer may be circular or refer to
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
247
4.3
Inheritance
a variable that will be initialized after it, in which case the value of the referenced variable is its default value for the purposes of the initializer. Such an initializer is of dubious value. There are three forms of variable initializers: regular initializers, array-size initializers, and object initializers. The first two forms appear after an equal sign that follows the type name, the latter two are part of the declaration itself. Only one form of initializer may be used on any particular declaration. 1.59.3.1 Regular Initializers A regular initializer is an expression that is implicitly convertible to the type of the variable. It appears after an equal sign that follows the type name and must be classified as a value. For example:
Module Test Dim x As Integer = 10 Dim y As Integer = 20 Sub Main() Console.WriteLine("x = " & x & ", y = " & y) End Sub End Module
If a variable declaration has a regular initializer, then only a single variable can be declared at a time. For example:
Module Test Sub Main() ' OK, only one variable declared at a time. Dim x As Integer = 10, y As Integer = 20 ' Error: Cant initialize multiple variables at once. Dim a, b As Integer = 10 End Sub End Module
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
248
4.3
Inheritance
1.59.3.2 Object Initializers An object initializer is specified using an object creation expression in the place of the type name. An object initializer is equivalent to a regular initializer assigning the result of the object creation expression to the variable. So
Module TestModule Sub Main() Dim x As New Test(10) End Sub End Module
is equivalent to
Module TestModule Sub Main() Dim x As Test = New Test(10) End Sub End Module
The parenthesis in an object initializer is always interpreted as the argument list for the constructor and never as array type modifiers. A variable name with an object initializer cannot have an array type modifier or a nullable type modifier. 1.59.3.3 Array-Size Initializers An array-size initializer is a modifier on the name of the variable that gives a set of dimension upper bounds denoted by expressions. The upper bound expressions must be classified as values and must be implicitly convertible to Integer The set of upper bounds is . equivalent to a variable initializer of an array-creation expression with the given upper bounds. The number of dimensions of the array type is inferred from the array size initializer. So
Module Test Sub Main() Dim x(5, 10) As Integer End Sub End Module
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
249
4.3
Inheritance
is equivalent to
Module Test Sub Main() Dim x As Integer(,) = New Integer(5, 10) {} End Sub End Module
All upper bounds must be equal to or greater than -1, and all dimensions must have an upper bound specified. If the element type of the array being initialized is itself an array type, the array-type modifiers go to the right of the array-size initializer. For example
Module Test Sub Main() Dim x(5,10)(,,) As Integer End Sub End Module
declares a local variable x whose type is a two-dimensional array of three-dimensional arrays of Integer initialized to an array with bounds of 0..5 in the first dimension and 0..10 in , the second dimension. It is not possible to use an array size initializer to initialize the elements of a variable whose type is an array of arrays. A variable declaration with an array-size initializer cannot have an array type modifier on its type or a regular initializer. ArraySizeInitializationModifier ::= OpenParenthesis BoundList CloseParenthesis [ ArrayTypeModifiers ] BoundList::= Bound | BoundList Comma Bound Bound ::= Expression | 0 To Expression
250
4.3
Inheritance
1.59.4 System.MarshalByRefObject Classes Classes that derive from the class System.MarshalByRefObject are marshaled across context boundaries using proxies (that is, by reference) rather than through copying (that is, by value). This means that an instance of such a class may not be a true instance but instead may just be a stub that marshals variable accesses and method calls across a context boundary. As a result, it is not possible to create a reference to the storage location of variables defined on such classes. This means that variables typed as classes derived from System.MarshalByRefObject cannot be passed to reference parameters, and methods and variables of variables typed as value types may not be accessed. Instead, Visual Basic treats variables defined on such classes as if they were properties (since the restrictions are the same on properties). There is one exception to this rule: a member implicitly or explicitly qualified with Me is exempt from the above restrictions, because Me is always guaranteed to be an actual object, not a proxy.
1.60 Properties
Properties are a natural extension of variables; both are named members with associated types, and the syntax for accessing variables and properties is the same. Unlike variables, however, properties do not denote storage locations. Instead, properties have accessors, which specify the statements to execute in order to read or write their values. Properties are defined with property declarations. The first part of a property declaration resembles a field declaration. The second part includes a Get accessor and/or a Set accessor. In the example below, the Button class defines a Caption property.
Publ i c Class Button Pr ivate capt ionValue As Str ing Publ i c Property Caption( ) As Str ing Get Return capt ionValue End Get
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
251
4.3
Inheritance
Set (Value As String) captionValue = value Repaint() End Set End Property ... End Class
Based on the Button class above, the following is an example of use of the Caption property:
Dim okButton As Button = New Button() okButton.Capt ion = "OK" ' Invokes Set accessor . Dim s As Str ing = okButton.Capt ion ' Invokes Get accessor .
Here, the Set accessor is invoked by assigning a value to the property, and the Get accessor is invoked by referencing the property in an expression. If no type is specified for a property and strict semantics are being used, a compile-time error occurs; otherwise the type of the property is implicitly Object or the type of the property's type character. A property declaration may contain either a Get accessor, which retrieves the value of the property, a Set accessor, which stores the value of the property, or both. Because a property implicitly declares methods, a property may be declared with the same modifiers as a method. If the property is defined in an interface or defined with the MustOverride modifier, the property body and the End construct must be omitted; otherwise, a compile-time error occurs. The index parameter list makes up the signature of the property, so properties may be overloaded on index parameters but not on the type of the property. The index parameter list is the same as for a regular method. However, none of the parameters may be modified with the ByRef modifier and none of them may be named Value (which is reserved for the implicit value parameter in the Set accessor). A property may be declared as follows:
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
252
4.3
Inheritance
If the property specifies no property type modifier, the property must have both a Get accessor and a Set accessor. The property is said to be a read-write property. If the property specifies the ReadOnly modifier, the property must have a Get accessor and may not have a Set accessor. The property is said to be read-only property. It is a compiletime error for a read-only property to be the target of an assignment. If the property specifies the WriteOnly modifier, the property must have a Set accessor and may not have a Get accessor. The property is said to be write-only property. It is a compiletime error to reference a write-only property in an expression except as the target of an assignment or as an argument to a method. The Get and Set accessors of a property are not distinct members, and it is not possible to declare the accessors of a property separately. The following example does not declare a single read-write property. Rather, it declares two properties with the same name, one read-only and one write-only:
Class A Pr ivate nameValue As Str ing ' Error , contains a dupl i ca te member name. Publ i c ReadOnly Property Name() As Str ing Get Return nameValue End Get End Property ' Error , contains a dupl i ca te member name. Publ i c WriteOnly Property Name() As Str ing Set (Value As Str ing) nameValue = value End Set End Property End Class
Since two members declared in the same class cannot have the same name, the example causes a compile-time error.
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
253
4.3
Inheritance
By default, the accessibility of a propertys Get and Set accessors is the same as the accessibility of the property itself. However, the Get and Set accessors can also specify accessibility separately from the property. In that case, the accessibility of an accessor must be more restrictive than the accessibility of the property and only one accessor can have a different accessibility level from the property. Access types are considered more or less restrictive as follows:
Private is more restrictive than Public, Protected Friend, Protected, or Friend. Friend is more restrictive than Protected Friend or Public. Protected is more restrictive than Protected Friend or Public. Protected Friend is more restrictive than Public.
When one of a propertys accessors is accessible but the other one is not, the property is treated as if it was read-only or write-only. For example:
Class A Publ i c Property P() As Integer Get ... End Get Pr ivate Set (Value As Integer ) ... End Set End Property End Class Module Test Sub Main() Dim a As A = New A() ' Error : A.P i s read- only in th is context . a.P = 10 End Sub End Module
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
254
4.3
Inheritance
When a derived type shadows a property, the derived property hides the shadowed property with respect to both reading and writing. In the following example, the P property in B hides the P property in A with respect to both reading and writing:
Class A Publ i c WriteOnly Property P() As Integer Set (Value As Integer ) End Set End Property End Class Class B Inher i t s A Publ i c Shadows ReadOnly Property P() As Integer Get End Get End Property End Class Module Test Sub Main() Dim x As B = New B B.P = 10 End Sub End Module ' Error , B.P i s read- only .
The accessibility domain of the return type or parameter types must be the same as or a superset of the accessibility domain of the property itself. A property may only have one Set accessor and one Get accessor. Except for differences in declaration and invocation syntax, Overridable, NotOverridable, Overrides, MustOverride, and MustInherit properties behave exactly like Overridable, NotOverridable, Overrides, MustOverride, and MustInherit methods. When a property is overridden, the overriding property must be of the same type (read-write, read-only, writeonly). An Overridable property cannot contain a Private accessor. Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved. 255
4.3
Inheritance
In the following example X is an Overr idableread-only property, Y is an Overridable readwrite property, and Z is a MustOverride read-write property.
MustInher i t Class A Pr ivate _y As Integer Publ i c Overr idable ReadOnly Property X() As Integer Get Return 0 End Get End Property Publ i c Overr idable Property Y() As Integer Get Return _y End Get Set (Value As Integer ) _y = value End Set End Property Publ i c MustOverr ide Property Z() As Integer End Class
Because Z is MustOverride, the containing class A must be declared MustInherit. By contrast, a class that derives from class A is shown below:
Class B Inher i t s A Pr ivate _z As Integer Publ i c Overr ides ReadOnly Property X() As Integer Get Return MyBase.X + 1 End Get End Property
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
256
4.3
Inheritance
Public Overrides Property Y() As Integer Get Return MyBase.Y End Get Set (Value As Integer) If value < 0 Then MyBase.Y = 0 Else MyBase.Y = Value End If End Set End Property Public Overrides Property Z() As Integer Get Return _z End Get Set (Value As Integer) _z = Value End Set End Property End Class
Here, the declarations of properties X, Y, and Z override the base properties. Each property declaration exactly matches the accessibility modifiers, type, and name of the corresponding inherited property. The Get accessor of property X and the Set accessor of property Y use the MyBase keyword to access the inherited properties. The declaration of property Z overrides the MustOverride property thus, there are no outstanding MustOverride members in class B, and B is permitted to be a regular class. Properties can be used to delay initialization of a resource until the moment it is first referenced. For example:
Imports System.IO Publ i c Class ConsoleStreams
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
257
4.3
Inheritance
Private Shared reader As TextReader Private Shared writer As TextWriter Private Shared errors As TextWriter Public Shared ReadOnly Property [In]() As TextReader Get If reader Is Nothing Then reader = Console.In End If Return reader End Get End Property Public Shared ReadOnly Property Out() As TextWriter Get If writer Is Nothing Then writer = Console.Out End If Return writer End Get End Property Public Shared ReadOnly Property [Error]() As TextWriter Get If errors Is Nothing Then errors = Console.Error End If Return errors End Get End Property End Class
The ConsoleStreamsclass contains three properties, In, Out, and Error, that represent the standard input, output, and error devices, respectively. By exposing these members as properties, the ConsoleStreams class can delay their initialization until they are actually used. For example, upon first referencing the Out property, as in
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
258
4.3
Inheritance
world" underlying TextWri terfor the output device is , the ) initialized. But if the application makes no reference to the In and Error properties, then no objects are created for those devices. ConsoleStreams.Out.Wri teL ine ("he l l o ,
PropertyMemberDeclaration ::= RegularPropertyMemberDeclaration | MustOverridePropertyMemberDeclaration | AutoPropertyMemberDeclaration PropertySignature ::= Property Identifier [ OpenParenthesis [ ParameterList ] CloseParenthesis ] [ As [ Attributes ] TypeName ] RegularPropertyMemberDeclaration ::= [ Attributes ] [ PropertyModifier+ ] PropertySignature [ ImplementsClause ] LineTerminator PropertyAccessorDeclaration+ End Property StatementTerminator MustOverridePropertyMemberDeclaration ::= [ Attributes ] MustOverridePropertyModifier+ PropertySignature [ ImplementsClause ] StatementTerminator AutoPropertyMemberDeclaration ::= [ Attributes ] [ AutoPropertyModifier+ ] Property Identifier [ OpenParenthesis [ ParameterList ] CloseParenthesis ] [ As [ Attributes ] TypeName ] [ Equals Expression ] [ ImplementsClause ] LineTerminator | [ Attributes ] [ AutoPropertyModifier+ ] Property Identifier [ OpenParenthesis [ ParameterList ] CloseParenthesis ] As [ Attributes ] New [ NonArrayTypeName [ OpenParenthesis [ ArgumentList ] CloseParenthesis ] ] [ ObjectCreationExpressionInitializer ] [ ImplementsClause ] LineTerminator
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
259
4.3
Inheritance
InterfacePropertyMemberDeclaration ::= [ Attributes ] [ InterfacePropertyModifier+ ] PropertySignature StatementTerminator AutoPropertyModifier ::= AccessModifier | Shadows | Shared | Overridable | NotOverridable | Overrides |
Overloads
MustOverridePropertyModifier ::= PropertyModifier | MustOverride InterfacePropertyModifier ::= Shadows | Overloads | Default | ReadOnly |
WriteOnly
PropertyAccessorDeclaration ::= PropertyGetDeclaration | PropertySetDeclaration 1.60.1 Get Accessor Declarations A Get accessor (getter) is declared by using a property Get declaration. A property Get declaration consists of the keyword Get followed by a statement block. Given a property named P, a Get accessor declaration implicitly declares a method with the name get_P with the same modifiers, type, and parameter list as the property. If the type contains a
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
260
4.3
Inheritance
declaration with that name, a compile-time error results, but the implicit declaration is ignored for the purposes of name binding. A special local variable, which is implicitly declared in the Get accessor body's declaration space with the same name as the property, represents the return value of the property. The local variable has special name resolution semantics when used in expressions. If the local variable is used in a context that expects an expression that is classified as a method group, such as an invocation expression, then the name resolves to the function rather than to the local variable. For example:
ReadOnly Property F( i As Integer ) As Integer Get I f i = 0 Then F =1 ' Sets the return value. Else F = F( i - 1) ' Recurs ive cal l . End I f End Get End Property
The use of parentheses can cause ambiguous situations (such as F(1) where F is a property whose type is a one-dimensional array). In all ambiguous situations, the name resolves to the property rather than the local variable. For example:
ReadOnly Property F( i As Integer ) As Integer ( ) Get I f i = 0 Then F = new Integer (2) { 1, 2, 3 } Else F = F( i 1) ' Recurs ive cal l , not index. End I f End Get End Property
When control flow leaves the Get accessor body, the value of the local variable is passed back to the invocation expression. Because invoking a Get accessor is conceptually
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
261
4.3
Inheritance
equivalent to reading the value of a variable, it is considered bad programming style for Get accessors to have observable side effects, as illustrated in the following example:
Class Counter Pr ivate Value As Integer Publ i c ReadOnly Property NextValue() As Integer Get Value += 1 Return Value End Get End Property End Class
The value of the NextValue property depends on the number of times the property has previously been accessed. Thus, accessing the property produces an observable side effect, and the property should instead be implemented as a method. The "no side effects" convention for Get accessors does not mean that Get accessors should always be written to simply return values stored in variables. Indeed, Get accessors often compute the value of a property by accessing multiple variables or invoking methods. However, a properly designed Get accessor performs no actions that cause observable changes in the state of the object. Note Get accessors have the same restriction on line placement that subroutines have. The beginning statement, end statement and block must all appear at the beginning of a logical line. PropertyGetDeclaration ::= [ Attributes ] [ AccessModifier ] Get LineTerminator [ Block ] End Get StatementTerminator 1.60.2 Set Accessor Declarations A Set accessor (setter) is declared by using a property set declaration. A property set declaration consists of the keyword Set, an optional parameter list, and a statement block. Given a property named P, a setter declaration implicitly declares a method with the name Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved. 262
4.3
Inheritance
set_P with the same modifiers and parameter list as the property. If the type contains a
declaration with that name, a compile-time error results, but the implicit declaration is ignored for the purposes of name binding. If a parameter list is specified, it must have one member, that member must have no modifiers except ByVal, and its type must be the same as the type of the property. The parameter represents the property value being set. If the parameter is omitted, a parameter named Value is implicitly declared. Note Set accessors have the same restriction on line placement that subroutines have. The beginning statement, end statement and block must all appear at the beginning of a logical line. PropertySetDeclaration ::= [ Attributes ] [ AccessModifier ] Set [ OpenParenthesis [ ParameterList ] CloseParenthesis ] LineTerminator [ Block ] End Set StatementTerminator 1.60.3 Default Properties A property that specifies the modifier Default is called a default property. Any type that allows properties may have a default property, including interfaces. The default property may be referenced without having to qualify the instance with the name of the property. Thus, given a class
Class Test Publ i c Defaul t ReadOnly Property I tem(i As Integer ) As Integer Get Return i End Get End Property End Class
the code
Module TestModule Sub Main()
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
263
4.3
Inheritance
Dim x As Test = New Test() Dim y As Integer y = x(10) End Sub End Module
is equivalent to
Module TestModule Sub Main() Dim x As Test = New Test() Dim y As Integer y = x.Item(10) End Sub End Module
Once a property is declared Defaul t all of the properties overloaded on that name in the , inheritance hierarchy become the default property, whether they have been declared Defaul t or not. Declaring a property Default in a derived class when the base class declared a default property by another name does not require any other modifiers such as Shadows or Overrides. This is because the default property has no identity or signature and so cannot be shadowed or overloaded. For example:
Class Base Publ i c ReadOnly Defaul t Property I tem(i As Integer ) As Integer Get Console .Wri teL ine ("Base = " & i ) End Get End Property End Class Class Derived Inher i t s Base ' This hides I tem, but does not change the defaul t property .
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
264
4.3
Inheritance
Public Shadows ReadOnly Property Item(i As Integer) As Integer Get Console.WriteLine("Derived = " & i) End Get End Property End Class Class MoreDerived Inherits Derived ' This declares a new default property, but not Item. ' This does not need to be declared Shadows Public ReadOnly Default Property Value(i As Integer) As Integer Get Console.WriteLine("MoreDerived = " & i) End Get End Property End Class Module Test Sub Main() Dim x As MoreDerived = New MoreDerived() Dim y As Integer Dim z As Derived = x y = x(10) y = x.Item(10) y = z(10) End Sub End Module MoreDerived = 10 Derived = 10 Base = 10
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
265
4.3
Inheritance
All default properties declared within a type must have the same name and, for clarity, must specify the Defaul t modifier. Because a default property with no index parameters would cause an ambiguous situation when assigning instances of the containing class, default properties must have index parameters. Furthermore, if one property overloaded on a particular name includes the Defaul t modifier, all properties overloaded on that name must specify it. Default properties may not be Shared, and at least one accessor of the property must not be Private. 1.60.4 Automatically Implemented Properties If a property omits declaration of any accessors, an implementation of the property will be automatically supplied unless the property is declared in an interface or is declared MustOverride. Only read/write properties with no arguments can be automatically implemented; otherwise, a compile-time error occurs. An automatically implemented property x, even one overriding another property, introduces a private local variable _x with the same type as the property. If there is a collision between the local variable's name and another declaration, a compile-time error will be reported. The automatically implemented propertys Get accessor returns the value of the local and the propertys Set accessor that sets the value of the local. For example, the declaration:
Publ i c Property x() As Integer
As with variable declarations, an automatically implemented property can include an initializer. For example:
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
266
4.3
Inheritance
Public Property x() As Integer = 10 Public Shared Property y() As New Customer() With { .Name = "Bob" }
Annotation When an automatically implemented property is initialized, it is initialized through the property, not the underlying field. This is so overriding properties can intercept the initialization if they need to. Array initializers are allowed on automatically implemented properties, except that there is no way to specify the array bounds explicitly. For example:
' Valid Property x As Integer() = {1, 2, 3} Property y As Integer(,) = {{1, 2, 3}, {12, 13, 14}, {11, 10, 9}} ' Invalid Property x4(5) As Short
1.61 Operators
Operators are methods that define the meaning of an existing Visual Basic operator for the containing class. When the operator is applied to the class in an expression, the operator is compiled into a call to the operator method defined in the class. Defining an operator for a class is also known as overloading the operator. It is not possible to overload an operator that already exists; in practice, this primarily applies to conversion operators. For example, it is not possible to overload the conversion from a derived class to a base class:
Class Base End Class Class Derived ' Cannot redefine conversion from Derived to Base, ' conversion will be ignored. Public Shared Widening Operator CType(s As Derived) As Base ... End Operator End Class
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
267
4.3
Inheritance
Operator declarations do not explicitly add names to the containing types declaration space; however they do implicitly declare a corresponding method starting with the characters op_. The following sections list the corresponding method names with each operator. There are three classes of operators that can be defined: unary operators, binary operators and conversion operators. All operator declarations share certain restrictions: Operator declarations must always be Publ i c and Shared. The Public modifier can be omitted in contexts where the modifier will be assumed. The parameters of an operator cannot be declared ByRef, Optional or ParamArray. The type of at least one of the operands or the return value must be the type that contains the operator. There is no function return variable defined for operators. Therefore, the Return statement must be used to return values from an operator body. The only exception to these restrictions applies to nullable value types. Since nullable value types do not have an actual type definition, a value type can declare user-defined operators for the nullable version of the type. When determining whether a type can declare a particular user-defined operator, the ? modifiers are first dropped off of all of the types involved in the declaration for the purposes of validity checking. This relaxation does not apply to the return type of the IsTrue and IsFalse operators; they must still return Boolean, not Boolean?.
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
268
4.3
Inheritance
The precedence and associativity of an operator cannot be modified by an operator declaration. Note Operators have the same restriction on line placement that subroutines have. The beginning statement, end statement and block must all appear at the beginning of a logical line. OperatorDeclaration ::= [ Attributes ] [ OperatorModifier+ ] Operator OverloadableOperator OpenParenthesis ParameterList CloseParenthesis [ As [ Attributes ] TypeName ] LineTerminator [ Block ] End Operator StatementTerminator OperatorModifier ::= Public | Shared | Overloads | Shadows | Widening | Narrowing OverloadableOperator ::= + | - | * | / | \ | & | Like | Mod | And | Or | Xor | ^ | < < | > > | = | < > | > | < | > = | < = | Not | IsTrue | IsFalse | CType 1.61.1 Unary Operators The following unary operators can be overloaded: The unary plus operator + (corresponding method: op_UnaryPlus) The unary minus operator - (corresponding method: op_UnaryNegation) The logical Not operator (corresponding method: op_OnesComplement) The IsTrue and IsFalse operators (corresponding methods: op_True, op_False) All overloaded unary operators must take a single parameter of the containing type and may return any type, except for IsTrue and IsFalse, which must return Boolean. If the containing type is a generic type, the type parameters must match the containing types type parameters. For example,
Structure Complex ...
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
269
4.3
Inheritance
Public Shared Operator +(v As Complex) As Complex Return v End Operator End Structure
If a type overloads one of IsTrueor IsFa l se then it must overload the other as well. If only , one is overloaded, a compile-time error results. Note IsTrueand IsFa l seare not reserved words. 1.61.2 Binary Operators The following binary operators can be overloaded: The addition +, subtraction -, multiplication *, division /, integral division \, modulo Mod and exponentiation ^ operators (corresponding method: op_Addition, op_Subtraction, op_Multiply, op_Division, op_IntegerDivision, op_Modulus, op_Exponent) The relational operators =, <>, <, >, <=, >= (corresponding methods: op_Equality, op_Inequality, op_LessThan, op_GreaterThan, op_LessThanOrEqual, op_GreaterThanOrEqual) Note While the equality operator can be overloaded, the assignment operator (used only in assignment statements) cannot be overloaded. The Like operator (corresponding method: op_Like) The concatenation operator & (corresponding method: op_Concatenate) The logical And, Or and Xor operators (corresponding methods: op_BitwiseAnd, op_BitwiseOr, op_ExclusiveOr) The shift operators << and >> (corresponding methods: op_LeftShift, op_RightShift) All overloaded binary operators must take the containing type as one of the parameters. If the containing type is a generic type, the type parameters must match the containing types type parameters. The shift operators further restrict this rule to require the first parameter to be of the containing type; the second parameter must always be of type Integer. The following binary operators must be declared in pairs:
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
270
4.3
Inheritance
Operator = and operator <> Operator > and operator < Operator >= and operator <= If one of the pair is declared, then the other must also be declared with matching parameter and return types, or a compile-time error will result. Annotation The purpose of requiring paired declarations of relational operators is to try and ensure at least a minimum level of logical consistency in overloaded operators. In contrast to the relational operators, overloading both the division and integral division operators is strongly discouraged, although not an error. Annotation In general, the two types of division should be entirely distinct: a type that supports division is either integral (in which case it should support \) or not (in which case it should support /). We considered making it an error to define both operators, but because their languages do not generally distinguish between two types of division the way Visual Basic does, we felt it was safest to allow the practice but strongly discourage it. Compound assignment operators cannot be overloaded directly. Instead, when the corresponding binary operator is overloaded, the compound assignment operator will use the overloaded operator. For example:
Structure Complex ... Publ i c Shared Operator +(x As Complex, y As Complex) _ As Complex ... End Operator End Structure Module Test
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
271
4.3
Inheritance
Sub Main() Dim c1, c2 As Complex ' Calls the overloaded + operator c1 += c2 End Sub End Module
1.61.3 Conversion Operators Conversion operators define new conversions between types. These new conversions are called user-defined conversions. A conversion operator converts from a source type, indicated by the parameter type of the conversion operator, to a target type, indicated by the return type of the conversion operator. Conversions must be classified as either widening or narrowing. A conversion operator declaration that includes the Widening keyword introduces a user-defined widening conversion (corresponding method: op_Impl i c i). A conversion operator declaration that includes the Narrowing keyword t introduces a user-defined narrowing conversion (corresponding method: op_Explicit). In general, user-defined widening conversions should be designed to never throw exceptions and never lose information. If a user-defined conversion can cause exceptions (for example, because the source argument is out of range) or loss of information (such as discarding high-order bits), then that conversion should be defined as a narrowing conversion. In the example:
Structure Digi t Dim value As Byte Publ i c Sub New(value As Byte) i f value < 0 OrElse value > 9 Then Throw New ArgumentException( ) Me.value = value End Sub Publ i c Shared Widening Operator CType(d As Digi t ) Return d.value End Operator As Byte
272
4.3
Inheritance
the conversion from Digi t to Byte is a widening conversion because it never throws exceptions or loses information, but the conversion from Byte to Digit is a narrowing conversion since Digit can only represent a subset of the possible values of a Byte. Unlike all other type members that can be overloaded, the signature of a conversion operator includes the target type of the conversion. This is the only type member for which the return type participates in the signature. The widening or narrowing classification of a conversion operator, however, is not part of the operators signature. Thus, a class or structure cannot declare both a widening conversion operator and a narrowing conversion operator with the same source and target types. A user-defined conversion operator must convert either to or from the containing type for example, it is possible for a class C to define a conversion from C to Integer and from Integer to C, but not from Integer to Boolean. If the containing type is a generic type, the type parameters must match the containing types type parameters. Also, it is not possible to redefine an intrinsic (i.e. non-user-defined) conversion. As a result, a type cannot declare a conversion where: The source type and the destination type are the same. Both the source type and the destination type are not the type that defines the conversion operator. The source type or the destination type is an interface type. The source type and destination types are related by inheritance (including Object). The only exception to these rules applies to nullable value types. Since nullable value types do not have an actual type definition, a value type can declare user-defined conversions for the nullable version of the type. When determining whether a type can declare a particular user-defined conversion, the ? modifiers are first dropped off of all of the types involved in the declaration for the purposes of validity checking. Thus, the following declaration is valid because S can define a conversion from S to T:
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
273
4.3
Inheritance
Structure T ... End Structure Structure S Public Shared Widening Operator CType(ByVal v As S?) As T ... End Operator End Structure
The following declaration is not valid, however, because structure S cannot define a conversion from S to S:
Structure S Public Shared Widening Operator CType(ByVal v As S) As S? ... End Operator End Structure
1.61.4 Operator Mapping Because the set of operators that Visual Basic supports may not exactly match the set of operators that other languages on the .NET Framework, some operators are mapped specially onto other operators when being defined or used. Specifically: Defining an integral division operator will automatically define a regular division operator (usable only from other languages) that will call the integral division operator. Overloading the Not, And, and Or operators will overload only the bitwise operator from the perspective of other languages that distinguish between logical and bitwise operators. A class that overloads only the logical operators in a language that distinguishes between logical and bitwise operators (i.e. a languages that uses op_LogicalNot, op_LogicalAnd, and op_LogicalOr for Not, And, and Or, respectively) will have their logical operators mapped onto the Visual Basic logical operators. If both the logical and bitwise operators are overloaded, only the bitwise operators will be used.
274
4.3
Inheritance
Overloading the << and >> operators will overload only the signed operators from the perspective of other languages that distinguish between signed and unsigned shift operators. A class that overloads only an unsigned shift operator will have the unsigned shift operator mapped onto the corresponding Visual Basic shift operator. If both an unsigned and signed shift operator is overloaded, only the signed shift operator will be used.
275
Statements
Statements represent executable code. A method is executed by first initializing all of its parameters to their correct values and initializing all of its local variables to the default value of their types. After parameter and local variable initialization, the method body block is executed. After the method block has been executed, execution returns to the caller of the method. Statement ::= LabelDeclarationStatement | LocalDeclarationStatement | WithStatement | SyncLockStatement | EventStatement | AssignmentStatement | InvocationStatement | ConditionalStatement | LoopStatement | ErrorHandlingStatement | BranchStatement | ArrayHandlingStatement | UsingStatement Annotation The Microsoft Visual Basic Compiler only allows statements which start with a keyword or an identifier. Thus, for instance, the invocation statement Cal l (Console) .Wr i teL ine is allowed, but the invocation statement (Console) .Wr i teL ine not. is
4.3
Inheritance
Label declaration statements must appear at the beginning of a logical line and labels may be either an identifier or an integer literal. Because both label declaration statements and invocation statements can consist of a single identifier, a single identifier at the beginning of a local line is always considered a label declaration statement. Label declaration statements must always be followed by a colon, even if no statements follow on the same logical line. Labels have their own declaration space and do not interfere with other identifiers. The following example is valid and uses the name variable x both as a parameter and as a label.
Funct ion F(x As Integer ) As Integer I f x >= 0 Then GoTo x End I f x = -x x: Return x End Funct ion
The scope of a label is the body of the method containing it. For the sake of readability, statement productions that involve multiple substatements are treated as a single production in this specification, even though the substatements may each be by themselves on a labeled line. Block ::= [ Statements+ ] LabelDeclarationStatement ::= LabelName : LabelName ::= Identifier | IntLiteral Statements ::= [ Statement ] | Statements : [ Statement ]
277
4.3
Inheritance
1.62.1 Local Variables and Parameters A method invocation creates a copy, specific to that invocation, of the local variables and parameters of the method. A local variable or parameter comes into existence when control enters the method body that contains the local variable declaration or parameter declaration and ceases to exist when control leaves the method. All locals are initialized to their type's default value. Local variables and parameters are always publicly accessible. It is an error to refer to a local variable in a textual position that precedes its declaration, as the following example illustrates:
Class A Private i As Integer = 0 Sub F() i = 1 Dim i As Integer i = 2 End Sub Sub G() Dim a As Integer = 1 Dim b As Integer = a End Sub End Class
In the F method above, the first assignment to i specifically does not refer to the field declared in the outer scope. Rather, it refers to the local variable, and it is in error because it textually precedes the declaration of the variable. In the G method, a subsequent variable declaration refers to a local variable declared in an earlier variable declaration within the same local variable declaration. Each block in a method creates a declaration space for local variables. Names are introduced into this declaration space through local variable declarations in the method body and through the parameter list of the method, which introduces names into the outermost block's declaration space. Blocks do not allow shadowing of names through 278
4.3
Inheritance
nesting: once a name has been declared in a block, the name may not be redeclared in any nested blocks. Thus, in the following example, the F and G methods are in error because the name i is declared in the outer block and cannot be redeclared in the inner block. However, the H and I methods are valid because the two i's are declared in separate non-nested blocks.
Class A Sub F() Dim i As Integer = 0 I f True Then Dim i As Integer = 1 End I f End Sub Sub G() I f True Then Dim i As Integer = 0 End I f Dim i As Integer = 1 End Sub Sub H() I f True Dim End I f I f True Dim End I f End Sub Then i As Integer = 0 Then i As Integer = 1
279
4.3
Inheritance
When the method is a function, a special local variable is implicitly declared in the method body's declaration space with the same name as the method representing the return value of the function. The local variable has special name resolution semantics when used in expressions. If the local variable is used in a context that expects an expression classified as a method group, such as an invocation expression, then the name resolves to the function rather than to the local variable. For example:
Function F(i As Integer) As Integer If i = 0 Then F = 1 ' Sets the return value. Else F = F(i - 1) ' Recursive call. End If End Function
The use of parentheses can cause ambiguous situations (such as F(1), where F is a function whose return type is a one-dimensional array); in all ambiguous situations, the name resolves to the function rather than the local variable. For example:
Funct ion F( i As Integer ) As Integer ( ) I f i = 0 Then F = new Integer (2) { 1, 2, 3 } Else F = F( i - 1) ' Recurs ive cal l , not an index. End I f End Funct ion
When control flow leaves the method body, the value of the local variable is passed back to the invocation expression. If the method is a subroutine, there is no such implicit local variable, and control simply returns to the invocation expression.
280
4.3
Inheritance
281
4.3
Inheritance
shape (i.e. array type modifiers) as the local declaration statement. Note that it is possible that the inferred element type may still be an array type. For example:
Option Infer On Module Test Sub Main() ' Error: initializer is not an array type Dim x() = 1 ' Type is Integer() Dim y() = New Integer() {} ' Type is Integer()() Dim z() = New Integer()() {} ' Type is Integer()()() Dim a()() = New Integer()()() {} ' Error: initializer does not have same array shape Dim b()() = New Integer(,)() {} End Sub End Module
If no type is specified on a local declaration statement that has a nullable type modifier, then the type of the local declaration is the nullable version of the inferred type or the inferred type itself if it a nullable value type already. If the inferred type is not a value type that can be made nullable, a compile-time error occurs. If both a nullable type modifier and an array size or array type modifier are placed on a local declaration statement with no type, then the nullable type modifier is considered to apply to the element type of the array and the previous steps are used to determine the element type. Variable initializers on local declaration statements are equivalent to assignment statements placed at the textual location of the declaration. Thus, if execution branches over the local declaration statement, the variable initializer is not executed. If the local declaration statement is executed more than once, the variable initializer is executed an
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
282
4.3
Inheritance
equal number of times. Static variables only execute their initializer the first time. If an exception occurs while initializing a static variable, the static variable is considered initialized with the default value of the static variable's type. The following example shows the use of initializers:
Module Test Sub F() Static x As Integer = 5 Console.WriteLine("Static variable x = " & x) x += 1 End Sub Sub Main() Dim i As Integer For i = 1 to 3 F() Next i label: i = 3 Dim y As Integer = 8 If i > 0 Then Console.WriteLine("Local variable y = " & y) y -= 1 i -= 1 Goto label End If End Sub End Module
283
4.3
Inheritance
Initializers on static locals are thread-safe and protected against exceptions during initialization. If an exception occurs during a static local initializer, the static local will have its default value and not be initialized. A static local initializer
Module Test Sub F() Static x As Integer = 5 End Sub End Module
is equivalent to
Imports System.Threading Imports Microsoft.VisualBasic.CompilerServices Module Test Class InitFlag Public State As Short End Class Private xInitFlag As InitFlag = New InitFlag() Sub F() Dim x As Integer If xInitFlag.State <> 1 Then Monitor.Enter(xInitFlag) Try If xInitFlag.State = 0 Then xInitFlag.State = 2 x = 5 Else If xInitFlag.State = 2 Then Throw New IncompleteInitialization()
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
284
4.3
Inheritance
End If Finally xInitFlag.State = 1 Monitor.Exit(xInitFlag) End Try End If End Sub End Module
Local variables, local constants, and static variables are scoped to the statement block in which they are declared. Static variables are special in that their names may only be used once throughout the entire method. For example, it is not valid to specify two static variable declarations with the same name even if they are in different blocks. LocalDeclarationStatement ::= LocalModifier VariableDeclarators StatementTerminator LocalModifier ::= Stat i c | Dim | Const 1.63.1 Implicit Local Declarations In addition to local declaration statements, local variables can also be declared implicitly through use. A simple name expression that uses a name that does not resolve to something else declares a local variable by that name. For example:
Option Expl i c i t Off
Module Test Sub Main() x = 10 y = 20 Console .Wri teL ine (x + y) End Sub End Module
Implicit local declaration only occurs in expression contexts that can accept an expression classified as a variable. The exception to this rule is that a local variable may not be implicitly declared when it is the target of a function invocation expression, indexing expression, or a member access expression.
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
285
4.3
Inheritance
Implicit locals are treated as if they are declared at the beginning of the containing method. Thus, they are always scoped to the entire method body, even if declared inside of a lambda expression. For example, the following code:
Option Explicit Off Module Test Sub Main() Dim x = Sub() a = 10 End Sub Dim y = Sub() Console.WriteLine(a) End Sub x() y() End Sub End Module
Will print the value 10. Implicit locals are typed as Object if no type character was attached to the variable name; otherwise the type of the variable is the type of the type character. Local variable type inference is not used for implicit locals. If explicit local declaration is specified by the compilation environment or by Option Expl i c i,t all local variables must be explicitly declared and implicit variable declaration is disallowed.
286
4.3
Inheritance
Function F() As Integer Return 10 End Sub End Structure Module TestModule Sub Main() Dim y As Test With y .x = 10 Console.WriteLine(.x) .x = .F() End With End Sub End Module
It is invalid to branch into a With statement block from outside of the block. WithStatement ::= With Expression StatementTerminator [ Block ] End With StatementTerminator
287
4.3
Inheritance
Public Function Add() As Integer SyncLock Me count += 1 Add = count End SyncLock End Function Public Function Subtract() As Integer SyncLock Me count -= 1 Subtract = count End SyncLock End Function End Class
The example above synchronizes on the specific instance of the class Test to ensure that no more than one thread of execution can add or subtract from the count variable at a time for a particular instance. The SyncLock block is implicitly contained by a Try statement whose Finally block calls the Shared method System.Threading.Monitor.Exit on the expression. This ensures the lock is freed even when an exception is thrown. As a result, it is invalid to branch into a SyncLock block from outside of the block, and a SyncLock block is treated as a single statement for the purposes of Resume and Resume Next. The above example is equivalent to the following code:
Class Test Pr ivate count As Integer = 0 Publ i c Funct ion Add() As Integer Try System.Threading.Moni tor . Enter (Me) count += 1 Add = count Fina l l y System.Threading.Moni tor . Ex i t (Me)
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
288
4.3
Inheritance
End Try End Function Public Function Subtract() As Integer Try System.Threading.Monitor.Enter(Me) count -= 1 Subtract = count Finally System.Threading.Monitor.Exit(Me) End Try End Function End Class
289
4.3
Inheritance
The RaiseEvent statement is processed as a call to the Invoke method of the event's delegate, using the supplied parameters, if any. If the delegates value is Nothing, no exception is thrown. If there are no arguments, the parentheses may be omitted. For example:
Class Raiser Publ i c Event E1(Count As Integer ) Publ i c Sub Raise( ) Stat i c RaiseCount As Integer = 0 RaiseCount += 1 RaiseEvent E1(RaiseCount) End Sub End Class Module Test Pr ivate WithEvents x As Raiser Pr ivate Sub E1Handler (Count As Integer ) Handles x.E1 Console .Wri teL ine ("Ra ise #" & Count) End Sub Publ i c Sub Main() x = New Raiser x.Raise( ) x.Raise( ) x.Raise( ) End Sub End Module
' Pr ints "Raise #1". ' Pr ints "Raise #2". ' Pr ints "Raise #3".
290
4.3
Inheritance
Static RaiseCount As Integer = 0 Dim TemporaryDelegate As E1EventHandler RaiseCount += 1 ' Use a temporary to avoid a race condition. TemporaryDelegate = E1Event If Not TemporaryDelegate Is Nothing Then TemporaryDelegate.Invoke(RaiseCount) End If End Sub End Class
RaiseEventStatement ::= RaiseEvent IdentifierOrKeyword [ OpenParenthesis [ ArgumentList ] CloseParenthesis ] StatementTerminator 1.66.2 AddHandler and RemoveHandler Statements Although most event handlers are automatically hooked up through WithEvents variables, it may be necessary to dynamically add and remove event handlers at run time. AddHandler and RemoveHandler statements do this. Each statement takes two arguments: the first argument must be an expression that is classified as an event access and the second argument must be an expression that is classified as a value. The second argument's type must be the delegate type associated with the event access. For example:
Publ i c Class Form1 Publ i c Sub New() ' Add Button1_Cl i ck as an event handler for Button1's Cl ick event. AddHandler Button1.C l i ck , AddressOf Button1_Cl i ck End Sub Pr ivate Button1 As Button = New Button() Sub Button1_Cl i ck (sender As Object , e As EventArgs) Console .Wri teL ine ("But ton1 was cl i cked!" ) End Sub
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
291
4.3
Inheritance
Public Sub Disconnect() RemoveHandler Button1.Click, AddressOf Button1_Click End Sub End Class
Given an event E, the statement calls the relevant add_E or remove_E method on the instance to add or remove the delegate as a handler for the event. Thus, the above code is equivalent to:
Publ i c Class Form1 Publ i c Sub New() Button1.add_Cl i ck (AddressOf Button1_Cl i ck ) End Sub Pr ivate Button1 As Button = New Button() Sub Button1_Cl i ck (sender As Object , e As EventArgs) Console .Wri teL ine ("But ton1 was cl i cked!" ) End Sub Publ i c Sub Disconnect( ) Button1.remove_Cl i ck (AddressOf Button1_Cl i ck ) End Sub End Class
AddHandlerStatement ::= AddHandler Expression Comma Expression StatementTerminator RemoveHandlerStatement ::= RemoveHandler Expression Comma Expression StatementTerminator
292
4.3
Inheritance
AssignmentStatement ::= RegularAssignmentStatement | CompoundAssignmentStatement | MidAssignmentStatement 1.67.1 Regular Assignment Statements A simple assignment statement stores the result of an expression in a variable. The expression on the left side of the assignment operator must be classified as a variable or a property access, while the expression on the right side of the assignment operator must be classified as a value. The type of the expression must be implicitly convertible to the type of the variable or property access. If the variable being assigned into is an array element of a reference type, a run-time check will be performed to ensure that the expression is compatible with the array-element type. In the following example, the last assignment causes a System.ArrayTypeMismatchExcept ion be thrown, because an instance of ArrayL is t to cannot be stored in an element of a String array.
Dim sa(10) As Str ing Dim oa As Object( ) = sa oa(0) = Nothing ' This i s al lowed. oa(1) = "Hel lo" ' This i s al lowed. oa(2) = New ArrayL is t ( ) ' System.ArrayTypeMismatchExcept ion i s thrown.
If the expression on the left side of the assignment operator is classified as a variable, then the assignment statement stores the value in the variable. If the expression is classified as a property access, then the assignment statement turns the property access into an invocation of the Set accessor of the property with the value substituted for the value parameter. For example:
Module Test Pr ivate PValue As Integer Publ i c Property P As Integer Get Return PValue
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
293
4.3
Inheritance
End Get Set (Value As Integer) PValue = Value End Set End Property Sub Main() ' The following two lines are equivalent. P = 10 set_P(10) End Sub End Module
If the target of the variable or property access is typed as a value type but not classified as a variable, a compile-time error occurs. For example:
Structure S Public F As Integer End Structure Class C Private PValue As S Public Property P As S Get Return PValue End Get Set (Value As S) PValue = Value End Set End Property End Class Module Test Sub Main()
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
294
4.3
Inheritance
Dim ct As C = New C() Dim rt As Object = new C() ' Compile-time error: ct.P not classified as variable. ct.P.F = 10 ' Run-time exception. rt.P.F = 10 End Sub End Module
Note The semantics of the assignment depend on the type of the variable or property to which it is being assigned. If the variable to which it is being assigned is a value type, the assignment copies the value of the expression into the variable. If the variable to which it is being assigned is a reference type, the assignment copies the reference, not the value itself, into the variable. If the type of the variable is Object, the assignment semantics are determined by whether the value's type is a value type or a reference type at run time. Annotation For intrinsic types such as Integerand Date, reference and value assignment semantics are the same because the types are immutable. As a result, the language is free to use reference assignment on boxed intrinsic types as an optimization. From a value perspective, the result is the same. Because the equals character (=) is used both for assignment and for equality, there is an ambiguity between a simple assignment and an invocation statement in situations such as x = y.ToStr i ng(In all such cases, the assignment statement takes precedence over the . ) equality operator. This means that the example expression is interpreted as x = (y.ToString()) rather than (x = y).ToString(). RegularAssignmentStatement ::= Expression Equals Expression StatementTerminator 1.67.2 Compound Assignment Statements A compound assignment statement takes the form V OP= E (where OP is a valid binary operator). The expression on the left side of the assignment operator must be classified as Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved. 295
4.3
Inheritance
a variable or property access, while the expression on the right side of the assignment operator must be classified as a value. The compound assignment statement is equivalent to the statement V = V OP E with the difference that the variable on the left side of the compound assignment operator is only evaluated once. The following example demonstrates this difference:
Module Test Function GetIndex() As Integer Console.WriteLine("Getting index") Return 1 End Function Sub Main() Dim a(2) As Integer Console.WriteLine("Simple assignment") a(GetIndex()) = a(GetIndex()) + 1 Console.WriteLine("Compound assignment") a(GetIndex()) += 1 End Sub End Module
The expression a(GetIndex() ) evaluated twice for simple assignment but only once for is compound assignment, so the code prints:
Simple assignment Gett ing index Gett ing index Compound assignment Gett ing index
CompoundAssignmentStatement ::= Expression CompoundBinaryOperator [ LineTerminator ] Expression StatementTerminator CompoundBinaryOperator ::= ^ = | * = | / = | \ = | + = | - = | & = | < < = |
> > =
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
296
4.3
Inheritance
1.67.3 Mid Assignment Statement A Mid assignment statement assigns a string into another string. The left side of the assignment has the same syntax as a call to the function Microsof t .V i sua lBas i c .S t r i ngs .Mid . The first argument is the target of the assignment and must be classified as a variable or a property access whose type is implicitly convertible to and from String. The second parameter is the 1-based start position that corresponds to where the assignment should begin in the target string and must be classified as a value whose type must be implicitly convertible to Integer. The optional third parameter is the number of characters from the right-side value to assign into the target string and must be classified as a value whose type is implicitly convertible to Integer. The right side is the source string and must be classified as a value whose type is implicitly convertible to String. The right side is truncated to the length parameter, if specified, and replaces the characters in the left-side string, starting at the start position. If the right side string contained fewer characters than the third parameter, only the characters from the right side string will be copied. The following example displays ab123fg:
Module Test Sub Main() Dim s1 As Str ing = "abcdefg" Dim s2 As Str ing = "1234567" Mid$(s1, 3, 3) = s2 Console .Wri teL ine (s1) End Sub End Module
Note Mid is not a reserved word. MidAssignmentStatement ::= Mid [ $ ] OpenParenthesis Expression Comma Expression [ Comma Expression ] CloseParenthesis Equals Expression StatementTerminator
297
4.3
Inheritance
298
4.3
Inheritance
a = b Else b = a End If ' Line If statement If a < b Then a = b Else b = a End Sub End Module
IfStatement ::= BlockIfStatement | LineIfThenStatement BlockIfStatement ::= I f BooleanExpression [ Then ] StatementTerminator [ Block ] [ ElseIfStatement+ ] [ ElseStatement ] End If StatementTerminator ElseIfStatement ::= ElseIf BooleanExpression [ Then ] StatementTerminator [ Block ] ElseStatement ::= Else StatementTerminator [ Block ] LineIfThenStatement ::= If BooleanExpression Then Statements [ Else Statements ] StatementTerminator The line version of the If statement binds less tightly than :, and its Else binds to the lexically nearest preceding If that is allowed by the syntax. For example, the following two versions are equivalent:
I f True Then _ I f True Then Console .Wri teL ine ("a" ) Else Console .Wri teL ine ("b" ) _ Else Console .Wri teL ine ("c " ) : Console .Wri teL ine ("d" )
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
299
4.3
Inheritance
If True Then If True Then Console.WriteLine("a") Else Console.WriteLine("b") End If Console.WriteLine("c") : Console.WriteLine("d") End If
All statements other than label declaration statements are allowed inside a line I f statement, including block statements. However, they may not use LineTerminators as StatementTerminators except inside multi-line lambda expressions. For example:
' Al lowed, since i t uses : instead of LineTerminator to separate statements I f b Then With New Str ing("a" (0 ) , 5 ) : Console .Wri teL ine ( . Length) : End With ' Disal l owed, since i t uses a LineTerminator I f b then With New Str ing("a" (0 ) , 5) Console .Wri teL ine ( . Length) End With ' Al lowed, since i t only uses LineTerminator ins ide a mult i - l i ne lambda I f b Then Cal l Sub() Console .Wri teL ine ("a" ) End Sub. Invoke()
1.69.2 Select Case Statements A Select Case statement executes statements based on the value of an expression. The expression must be classified as a value. When a Select Case statement is executed, the Select expression is evaluated first, and the Case statements are then evaluated in order of textual declaration. The first Case statement that evaluates to True has its block executed. If no Case statement evaluates to True and there is a Case Else statement, that block is executed. Once a block has finished executing, execution passes to the end of the Select statement.
300
4.3
Inheritance
Execution of a Case block is not permitted to "fall through" to the next switch section. This prevents a common class of bugs that occur in other languages when a Case terminating statement is accidentally omitted. The following example illustrates this behavior:
Module Test Sub Main() Dim x As Integer = 10 Select Case x Case 5 Console .Wri teL ine ("x Case 10 Console .Wri teL ine ("x Case 20 - 10 Console .Wri teL ine ("x Case 30 Console .Wri teL ine ("x End Select End Sub End Module
Although Case 10 and Case 20 - 10 select for the same value, Case 10 is executed because it precedes Case 20 - 10 textually. When the next Case is reached, execution continues after the Select statement. A Case clause may take two forms. One form is an optional Is keyword, a comparison operator, and an expression. The expression is converted to the type of the Select expression; if the expression is not implicitly convertible to the type of the Select expression, a compile-time error occurs. If the Select expression is E, the comparison operator is Op, and the Case expression is E1, the case is evaluated as E OP E1. The operator must be valid for the types of the two expressions; otherwise a compile-time error occurs.
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
301
4.3
Inheritance
The other form is an expression optionally followed by the keyword To and a second expression. Both expressions are converted to the type of the Select expression; if either expression is not implicitly convertible to the type of the Select expression, a compile-time error occurs. If the Select expression is E, the first Case expression is E1, and the second Case expression is E2, the Case is evaluated either as E = E1 (if no E2 is specified) or (E >= E1) And (E <= E2). The operators must be valid for the types of the two expressions; otherwise a compile-time error occurs. SelectStatement ::= Select [ Case ] Expression StatementTerminator [ CaseStatement+ ] [ CaseElseStatement ] End Select StatementTerminator CaseStatement ::= Case CaseClauses StatementTerminator [ Block ] CaseClauses ::= CaseClause | CaseClauses Comma CaseClause CaseClause ::= [ Is [ LineTerminator ] ] ComparisonOperator [ LineTerminator ] Expression | Expression [ To Expression ] ComparisonOperator ::= = | < > | < | > | > = | < = CaseElseStatement ::= Case Else StatementTerminator [ Block ]
302
4.3
Inheritance
DoLoopStatement | ForStatement | ForEachStatement 1.70.1 While...End While and Do...Loop Statements A While or Do loop statement loops based on a Boolean expression. A While loop statement loops as long as the Boolean expression evaluates to true; a Do loop statement may contain a more complex condition. An expression may be placed after the Do keyword or after the Loop keyword, but not after both. The Boolean expression is evaluated as per Section 1.93. (Note: this does not require the expression to have Boolean type). It is also valid to specify no expression at all; in that case, the loop will never exit. If the expression is placed after Do, it will be evaluated before the loop block is executed on each iteration. If the expression is placed after Loop, it will be evaluated after the loop block has executed on each iteration. Placing the expression after Loop will therefore generate one more loop than placement after Do. The following example demonstrates this behavior:
Module Test Sub Main() Dim x As Integer x =3 Do While x = 1 Console .Wri teL ine ("F i r s t Loop Do
loop")
Console .Wri teL ine ("Second loop") Loop While x = 1 End Sub End Module
303
4.3
Inheritance
In the case of the first loop, the condition is evaluated before the loop executes. In the case of the second loop, the condition is executed after the loop executes. The conditional expression must be prefixed with either a While keyword or an Unti l keyword. The former breaks the loop if the condition evaluates to false, the latter when the condition evaluates to true. Note Unti l is not a reserved word. WhileStatement ::= While BooleanExpression StatementTerminator [ Block ] End While StatementTerminator DoLoopStatement ::= DoTopLoopStatement | DoBottomLoopStatement DoTopLoopStatement ::= Do [ WhileOrUntil BooleanExpression ] StatementTerminator [ Block ] Loop StatementTerminator DoBottomLoopStatement ::= Do StatementTerminator [ Block ] Loop WhileOrUntil BooleanExpression StatementTerminator WhileOrUntil ::= While | Until 1.70.2 For...Next Statements A For...Next statement loops based on a set of bounds. A For statement specifies a loop control variable, a lower bound expression, an upper bound expression, and an optional step value expression. The loop control variable is specified either through an identifier followed by an optional As clause or an expression, which is interpreted as follows: If the loop control variable is an identifier with an As clause, the identifier defines a new local variable of the type specified in the As clause, scoped to the entire For loop.
304
4.3
Inheritance
If the loop control variable is an identifier without an As clause, then the identifier is first resolved using the simple name resolution rules (see Section 1.78.4). If by these rules the identifier is undefined or is classified as a type, and local type inference is being used, then the identifier defines a new local variable whose scope is the entire For loop and whose type is inferred from the bound and step expressions. If the identifier is classified as a variable, then the loop control variable is that variable. Otherwise, it is a compile-error. If the loop control variable is an expression, the expression must be classified as a variable. A loop control variable cannot be used by another enclosing For . . .Next statement. The type of the loop control variable of a For statement determines the type of the iteration and must be one of:
Byte, SByte, UShort, Short, UInteger, Integer, ULong, Long, Decimal, Single, Double
An enumerated type
Object
A type T that has the following operators, where B is a type that can be used in a Boolean expression:
Publ i c Publ i c Publ i c Publ i c Shared Shared Shared Shared Operator Operator Operator Operator >= (op1 As T, op2 As T) As B <= (op1 As T, op2 As T) As B - (op1 As T, op2 As T) As T + (op1 As T, op2 As T) As T
The bound and step expressions must be implicitly convertible to the type of the loop control variable and must be classified as values. At compile time, the type of the loop control variable is inferred by choosing the widest type among the lower bound, upper bound, and step expression types. If there is no widening conversion between two of the types, then a compile-time error occurs. At run time, if the type of the loop control variable is Object, then the type of the iteration is inferred the same as at compile time, with two exceptions. First, if the bound and step expressions are all of integral types but have no widest type, then the widest type that encompasses all three types will be inferred. And second, if the type of the loop control variable is inferred to be String, Double will be inferred instead. If, at run time, no loop
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
305
4.3
Inheritance
control type can be determined or if any of the expressions cannot be converted to the loop control type, a System.Inval i dCastExcept ion occur. Once a loop control type has will been chosen at the beginning of the loop, the same type will be used throughout the iteration, regardless of changes made to the value in the loop control variable. A For statement must be closed by a matching Next statement. A Next statement without a variable matches the innermost open For statement, while a Next statement with one or more loop control variables will, from left to right, match the For loops that match each variable. If a variable matches a For loop that is not the most nested loop at that point, a compile-time error results. At the beginning of the loop, the three expressions are evaluated in textual order and the lower bound expression is assigned to the loop control variable. If the step value is omitted, it is implicitly the literal 1, converted to the type of the loop control variable. The three expressions are only ever evaluated at the beginning of the loop. At the beginning of each loop, the control variable is compared to see if it is greater than the end point if the step expression is positive, or less than the end point if the step expression is negative. If it is, the For loop terminates; otherwise the loop block executes. If the loop control variable is not a primitive type, the comparison operator is determined by whether the expression step >= step step is true or false. At the Next statement, the step value is added to the control variable and execution returns to the top of the loop. It is not valid to branch into a For loop from outside the loop. ForStatement ::= For LoopControlVariable Equals Expression To Expression [ Step Expression ] StatementTerminator [ Block ] [ Next [ NextExpressionList ] StatementTerminator ] LoopControlVariable ::= Identifier [ IdentifierModifiers As TypeName ] | Expression
306
4.3
Inheritance
NextExpressionList ::= Expression | NextExpressionList Comma Expression 1.70.3 For Each...Next Statements A For Each. . .Next statement loops based on the elements in an expression. A For Each statement specifies a loop control variable and an enumerator expression. The loop control variable is specified either through an identifier followed by an optional As clause or an expression, which is interpreted as follows: If the loop control variable is an identifier with an As clause, the identifier defines a new local variable of the type specified in the As clause, scoped to the entire For Each loop. If the loop control variable is an identifier without an As clause, then the identifier is first resolved using the simple name resolution rules (see Section 1.78.4). If by these rules the identifier is undefined or is classified as a type, and local variable type inference is being used, then the identifier defines a new local variable whose scope is the entire For loop and whose type is the element type of the collection (Object if the enumerator expression is typed as Object). If the identifier is classified as a variable, then the loop control variable is that variable. Otherwise it is an compile-error. If the loop control variable is an expression, the expression must be classified as a variable. The enumerator expression must be classified as a value and its type must be a collection type or Object. If the type of the enumerator expression is Object, then all processing is deferred until run-time. Otherwise, a conversion must exist from the element type of the collection to the type of the loop control variable. The loop control variable cannot be used by another enclosing For Each statement. A For Each statement must be closed by a matching Next statement. A Next statement without a loop control variable matches the innermost open For Each. A Next statement with one or more loop control variables will, from left to right, match the For Each loops that have the same loop control variable. If a variable matches a For Each loop that is not the most nested loop at that point, a compile-time error occurs. A type C is said to be a collection type if:
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
307
4.3
Inheritance
this property is the element type of the collection type. It implements the interface System.Collections.Generic.IEnumerable(Of T), in which case the element type of the collection is considered to be T. It implements the interface System.Collections.IEnumerable, in which case the element type of the collection is considered to be Object. Following is an example of a class that can be enumerated:
Publ i c Class IntegerCol lec t i on Pr ivate integers(10) As Integer Publ i c Class IntegerCol lec t i onEnumerator Pr ivate col l ec t i on As IntegerCol lec t i on Pr ivate index As Integer = - 1 Fr iend Sub New(c As IntegerCol lec t i on ) col l ec t i on = c End Sub Publ i c Funct ion MoveNext() As Boolean index += 1 Return index <= 10 End Funct ion Publ i c ReadOnly Property Current As Integer Get
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
308
4.3
Inheritance
If index < 0 OrElse index > 10 Then Throw New System.InvalidOperationException() End If Return collection.integers(index) End Get End Property End Class Public Sub New() Dim i As Integer For i = 0 To 10 integers(i) = I Next i End Sub Public Function GetEnumerator() As IntegerCollectionEnumerator Return New IntegerCollectionEnumerator(Me) End Function End Class
Before the loop begins, the enumerator expression is evaluated. If the type of the expression does not satisfy the design pattern, then the expression is cast to System.Col lec t i ons . I Enumerable System.Col lec t i ons .Gener ic . I Enumerable(Of the or . If T) expression type implements the generic interface, the generic interface is preferred at compile-time but the non-generic interface is preferred at run-time. If the expression type implements the generic interface multiple times, the statement is considered ambiguous and a compile-time error occurs. Annotation The non-generic interface is preferred in the late bound case, because picking the generic interface would mean that all the calls to the interface methods would involve type parameters. Since it is not possible to know the matching type arguments at run-time, all such calls would have to be made using late-bound calls. This would be slower than calling
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
309
4.3
Inheritance
the non-generic interface because the non-generic interface could be called using compiletime calls.
GetEnumerator is called on the resulting value and the return value of the function is stored in a temporary. Then at the beginning of each loop, MoveNext is called on the temporary. If it returns False, the loop terminates. If it returns True, the Current property is retrieved,
coerced to the type of the iterator variable (regardless of whether the conversion is implicit or explicit), and assigned to the iterator variable; then the loop block executes. Annotation The current element of the iteration is converted to the type of the loop control variable even if the conversion is explicit because there is no convenient place to introduce a conversion operator in the statement. This became particularly troublesome when working with the most common collection type, System.Collections.ArrayList, because its element type is Object. This would have required casts in a great many loops, something we felt was not ideal. Ironically, generics enabled the creation of a strongly-typed collection, System.Collections.Generic.List(Of T), which might have made us rethink this design point, but for compatibilitys sake, this cannot be changed now. When the Next statement is reached, execution returns to the top of the loop. If a variable is specified after the Next keyword, it must be the same as the first variable after the For Each. For example, consider the following code:
Module Test Sub Main() Dim i As Integer Dim c As IntegerCol lec t i on = New IntegerCol lec t i on ( ) For Each i In c Console .Wri teL ine ( i ) Next i End Sub End Module
310
4.3
Inheritance
Module Test Sub Main() Dim i As Integer Dim c As IntegerCollection = New IntegerCollection() Dim e As IntegerCollection.IntegerCollectionEnumerator e = c.GetEnumerator() While e.MoveNext() i = e.Current Console.WriteLine(i) End While End Sub End Module
If the type E of the enumerator implements System.IDisposablethen the enumerator is , disposed upon exiting the loop by calling the Dispose method. This ensures that resources held by the enumerator are released. If the method containing the For Each statement does not use unstructured error handling, then the For Each statement is wrapped in a Try statement with the Dispose method called in the Finally to ensure cleanup. Note The System.Array type is a collection type, and since all array types derive from System.Array, any array type expression is permitted in a For Each statement. For singledimensional arrays, the For Each statement enumerates the array elements in increasing index order, starting with index 0 and ending with index Length - 1. For multidimensional arrays, the indices of the rightmost dimension are increased first. For example, the following code prints 1 2 3 4:
Module Test Sub Main() Dim x( , ) As Integer = { { 1, 2 }, { 3, 4 } } Dim i As Integer For Each i In x Console .Wri te ( i & " ")
311
4.3
Inheritance
It is not valid to branch into a For Each statement block from outside the block. ForEachStatement ::= For Each LoopControlVariable In [ LineTerminator ] Expression StatementTerminator [ Block ] [ Next [ NextExpressionList ] StatementTerminator ]
312
4.3
Inheritance
Catch e As Exception Console.WriteLine("Caught exception!") Finally Console.WriteLine("Exiting try.") End Try End Sub End Module
A Try statement is made up of three kinds of blocks: try blocks, catch blocks, and finally blocks. A try block is a statement block that contains the statements to be executed. A catch block is a statement block that handles an exception. A finally block is a statement block that contains statements to be run when the Try statement is exited, regardless of whether an exception has occurred and been handled. A Try statement, which can only contain one try block and one finally block, must contain at least one catch block or finally block. It is invalid to explicitly transfer execution into a try block except from within a catch block in the same statement. StructuredErrorStatement ::= ThrowStatement | TryStatement TryStatement ::= Try StatementTerminator [ Block ] [ CatchStatement+ ] [ FinallyStatement ] End Try StatementTerminator 1.71.1.1 Finally Blocks A Finally block is always executed when execution leaves any part of the Try statement. No explicit action is required to execute the Finally block; when execution leaves the Try statement, the system will automatically execute the Finally block and then transfer execution to its intended destination. The Finally block is executed regardless of how execution leaves the Try statement: through the end of the Try block, through the end of a
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
313
4.3
Inheritance
Catch block, through an Exi t Try statement, through a GoTo statement, or by not handling a
thrown exception. It is invalid to explicitly transfer execution into a Finally block; it is also invalid to transfer execution out of a Finally block except through an exception. FinallyStatement ::= Finally StatementTerminator [ Block ] 1.71.1.2 Catch Blocks If an exception occurs while processing the Try block, each Catch statement is examined in textual order to determine if it handles the exception. The identifier specified in a Catch clause represents the exception that has been thrown. If the identifier contains an As clause, then the identifier is considered to be declared within the Catch block's local declaration space. Otherwise, the identifier must be a local variable (not a static variable) that was defined in a containing block. A Catch clause with no identifier will catch all exceptions derived from System.Exception. A Catch clause with an identifier will only catch exceptions whose types are the same as or derived from the type of the identifier. The type must be System.Exception, or a type derived from System.Exception. When an exception is caught that derives from System.Exception, a reference to the exception object is stored in the object returned by the function Microsoft.VisualBasic.Information.Err. A Catch clause with a When clause will only catch exceptions when the expression evaluates to True; the type of the expression must be a Boolean expression as per Section 1.93. A When clause is only applied after checking the type of the exception, and the expression may refer to the identifier representing the exception, as this example demonstrates:
Module Test Sub Main() Dim i As Integer = 5 Try
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
314
4.3
Inheritance
Throw New ArgumentException() Catch e As OverflowException When i = 5 Console.WriteLine("First handler") Catch e As ArgumentException When i = 4 Console.WriteLine("Second handler") Catch When i = 5 Console.WriteLine("Third handler") End Try End Sub End Module
If a Catch clause handles the exception, execution transfers to the Catch block. At the end of the Catch block, execution transfers to the first statement following the Try statement. The Try statement will not handle any exceptions thrown in a Catch block. If no Catch clause handles the exception, execution transfers to a location determined by the system. It is invalid to explicitly transfer execution into a Catch block. CatchStatement ::= Catch [ Identifier [ As NonArrayTypeName ] ] [ When BooleanExpression ] StatementTerminator [ Block ] 1.71.1.3 Throw Statement The Throw statement raises an exception, which is represented by an instance of a type derived from System.Exception. If the expression is not classified as a value or is not a type derived from System.Exception, then a compile-time error occurs. If the expression evaluates to a null value at run time, then a System.NullReferenceException exception is raised instead.
315
4.3
Inheritance
A Throw statement may omit the expression within a catch block of a Try statement, as long as there is no intervening finally block. In that case, the statement rethrows the exception currently being handled within the catch block. For example:
Sub Test(x As Integer ) Try Throw New Except ion( ) Catch I f x = 0 Then Throw ' OK, rethrows exception from above. Else Try I f x = 1 Then Throw ' OK, rethrows exception from above. End I f Fina l l y Throw ' Inva l i d , ins ide of a Fina l l y . End Try End I f End Try End Sub
ThrowStatement ::= Throw [ Expression ] StatementTerminator 1.71.2 Unstructured Exception-Handling Statements Unstructured exception handling is a method of handling errors by indicating statements to branch to when an exception occurs. Unstructured exception handling is implemented using three statements: the Error statement, the On Error statement, and the Resume statement. For example:
Module Test Sub ThrowException( ) Error 5 End Sub Sub Main() On Error Goto GotExcept ion
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
316
4.3
Inheritance
ThrowException() Exit Sub GotException: Console.WriteLine("Caught exception!") Resume Next End Sub End Module
When a method uses unstructured exception handling, a single structured exception handler is established for the entire method that catches all exceptions. (Note that in constructors this handler does not extend over the call to the call to New at the beginning of the constructor.) The method then keeps track of the most recent exception-handler location and the most recent exception that has been thrown. At entry to the method, the exception-handler location and the exception are both set to Nothing. When an exception is thrown in a method that uses unstructured exception handling, a reference to the exception object is stored in the object returned by the function Microsoft.VisualBasic.Information.Err. UnstructuredErrorStatement ::= ErrorStatement | OnErrorStatement | ResumeStatement 1.71.2.1 Error Statement An Error statement throws a System.Exception exception containing a Visual Basic 6 exception number. The expression must be classified as a value and its type must be implicitly convertible to Integer. ErrorStatement ::= Error Expression StatementTerminator 1.71.2.2 On Error Statement An On Error statement modifies the most recent exception-handling state. It may be used in one of four ways:
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
317
4.3
Inheritance
On Error GoTo -1 resets the most recent exception to Nothing. On Error GoTo 0 resets the most recent exception-handler location to Nothing. On Error GoTo LabelName establishes the label as the most recent exception-handler
location. This statement cannot be used in a method that contains a lambda or query expression.
On Error Resume Next, establishes the Resume Next behavior as the most recent exception-
GotoStatement |
Resume Next
1.71.2.3 Resume Statement A Resume statement returns execution to the statement that caused the most recent exception. If the Next modifier is specified, execution returns to the statement that would have been executed after the statement that caused the most recent exception. If a label name is specified, execution returns to the label. Because the SyncLock statement contains an implicit structured error-handling block, Resume and Resume Next have special behaviors for exceptions that occur in SyncLock statements. Resume returns execution to the beginning of the SyncLock statement, while Resume Next returns execution to the next statement following the SyncLock statement. For example, consider the following code:
Class LockClass End Class Module Test Sub Main() Dim Fi rs tT ime As Boolean = True
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
318
4.3
Inheritance
Dim Lock As LockClass = New LockClass() On Error Goto Handler SyncLock Lock Console.WriteLine("Before exception") Throw New Exception() Console.WriteLine("After exception") End SyncLock Console.WriteLine("After SyncLock") Exit Sub Handler: If FirstTime Then FirstTime = False Resume Else Resume Next End If End Sub End Module
The first time through the SyncLock statement, Resume returns execution to the beginning of the SyncLock statement. The second time through the SyncLock statement, Resume Next returns execution to the end of the SyncLock statement. Resume and Resume Next are not allowed within a SyncLock statement. In all cases, when a Resume statement is executed, the most recent exception is set to Nothing. If a Resume statement is executed with no most recent exception, the statement
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
319
4.3
Inheritance
raises a System.Except ionexception containing the Visual Basic error number 20 (Resume without error). ResumeStatement ::= Resume [ ResumeClause ] StatementTerminator ResumeClause ::= Next | LabelName
320
4.3
Inheritance
ExitStatement | ContinueStatement | StopStatement | EndStatement | ReturnStatement GotoStatement ::= GoTo LabelName StatementTerminator ExitStatement ::= Exi t ExitKind StatementTerminator ExitKind ::= Do | For | While | Select | Sub | Function | Property | Try ContinueStatement ::= Continue ContinueKind StatementTerminator ContinueKind ::= Do | For | While StopStatement ::= Stop StatementTerminator EndStatement ::= End StatementTerminator ReturnStatement ::= Return [ Expression ] StatementTerminator
321
4.3
Inheritance
the given number of dimensions is incompatible with the variable or property at run time a compile-time error occurs. For example:
Module Test Sub Main() Dim o As Object Dim b() As Byte Dim i(,) As Integer ' The next two statements are equivalent. ReDim o(10,30) o = New Object(10, 30) {} ' The next two statements are equivalent. ReDim b(10) b = New Byte(10) {} ' Error: Incorrect number of dimensions. ReDim i(10, 30, 40) End Sub End Module
If the Preservekeyword is specified, then the expressions must also be classifiable as a value, and the new size for each dimension except for the rightmost one must be the same as the size of the existing array. The values in the existing array are copied into the new array: if the new array is smaller, the existing values are discarded; if the new array is bigger, the extra elements will be initialized to the default value of the element type of the array. For example, consider the following code:
Module Test Sub Main() Dim x(5, 5) As Integer x(3, 3) = 3 ReDim Preserve x(5, 6) Console .Wri teL ine (x(3 , 3) & " , " & x(3, 6))
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
322
4.3
Inheritance
If the existing array reference is a null value at run time, no error is given. Other than the rightmost dimension, if the size of a dimension changes, a System.ArrayTypeMismatchExcept ion will be thrown. Note Preserveis not a reserved word. RedimStatement ::= ReDim [ Preserve ] RedimClauses StatementTerminator RedimClauses ::= RedimClause | RedimClauses Comma RedimClause RedimClause ::= Expression ArraySizeInitializationModifier 1.73.2 Erase Statement An Erase statement sets each of the array variables or properties specified in the statement to Nothing. Each expression in the statement must be classified as a variable or property access whose type is an array type or Object. For example:
Module Test Sub Main() Dim x() As Integer = New Integer (5 ) {} ' The fo l l ow ing two statements are equiva lent . Erase x x = Nothing End Sub End Module
323
4.3
Inheritance
The Using statement automates the process of acquiring a resource, executing a set of statements, and then disposing of the resource. The statement can take two forms: in one, the resource is a local variable declared as a part of the statement and treated as a regular local variable declaration statement; in the other, the resource is the result of an expression. If the resource is a local variable declaration statement then the type of the local variable declaration must be a type that can be implicitly converted to System.IDisposableThe . declared local variables are read-only, scoped to the Using statement block and must include an initializer. If the resource is the result of an expression then the expression must
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
324
4.3
Inheritance
be classified as a value and must be of a type that can be implicitly converted to System.IDisposableThe expression is only evaluated once, at the beginning of the . statement. The Using block is implicitly contained by a Try statement whose finally block calls the method IDisposable.Dispose on the resource. This ensures the resource is disposed even when an exception is thrown. As a result, it is invalid to branch into a Using block from outside of the block, and a Using block is treated as a single statement for the purposes of Resume and Resume Next. If the resource is Nothing, then no call to Dispose is made. Thus, the example:
Using f As C = New C() ... End Using
is equivalent to:
Dim f As C = New C() Try ... Fina l l y I f f IsNot Nothing Then f .D i spose( ) End I f End Try
A Using statement that has a local variable declaration statement may acquire multiple resources at a time, which is equivalent to nested Using statements. For example, a Using statement of the form:
Using r1 As R = New R() , r2 As R = New R() r1.F ( ) r2.F( ) End Using
is equivalent to:
Using r1 As R = New R() Using r2 As R = New R()
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
325
4.3
Inheritance
UsingStatement ::= Using UsingResources StatementTerminator [ Block ] End Using StatementTerminator UsingResources ::= VariableDeclarators | Expression
326
Expressions
An expression is a sequence of operators and operands that specifies a computation of a value, or that designates a variable or constant. This chapter defines the syntax, order of evaluation of operands and operators, and meaning of expressions. Expression ::= SimpleExpression | TypeExpression | MemberAccessExpression | DictionaryAccessExpression | InvocationExpression | IndexExpression | NewExpression | CastExpression | OperatorExpression | ConditionalExpression | LambdaExpression | QueryExpression | XMLLiteralExpression | XMLMemberAccessExpression
4.3
Inheritance
A lambda method, which is an anonymous method. A property group, which is a set of properties overloaded on the same name. A property group may have an associated target expression. A property access. Every property access has an associated type, namely the type of the property. A property access may have an associated target expression. A late-bound access, which represents a method or property access deferred until runtime. A late-bound access may have an associated target expression and an associated type argument list. The type of a late-bound access is always Object. An event access. Every event access has an associated type, namely the type of the event. An event access may have an associated target expression. An event access may appear as the first argument of the RaiseEvent AddHandler , and RemoveHandler statements. In any , other context, an expression classified as an event access causes a compile-time error. An array literal, which represents the initial values of an array whose type has not yet been determined. Void. This occurs when the expression is an invocation of a subroutine. An expression classified as void is only valid in the context of an invocation statement. A default value. Only the literal Nothing produces this classification. The final result of an expression is usually a value or a variable, with the other categories of expressions functioning as intermediate values that are only permitted in certain contexts. Note that expressions whose type is a type parameter can be used in statements and expressions that require the type of an expression to have certain characteristics (such as being a reference type, value type, deriving from some type, etc.) if the constraints imposed on the type parameter satisfy those characteristics. 1.75.1 Expression Reclassification Normally, when an expression is used in a context that requires a classification different from that of the expression, a compile-time error occurs for example, attempting to assign a value to a literal. However, in many cases it is possible to change an expression's Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved. 328
4.3
Inheritance
classification through the process of reclassification. The following types of expressions can be reclassified: 1. A variable can be reclassified as a value. The value stored in the variable is fetched. 2. A method group can be reclassified as a value. The method group expression is interpreted as an invocation expression with the associated target expression and type parameter list, and empty parentheses (that is, f is interpreted as f ( ) and f(Of Integer) is interpreted as f(Of Integer)()). This reclassification may result in the expression being further reclassified as void. 3. A method pointer can be reclassified as a value. This reclassification can only occur in the context of a conversion where the target type is known. The method pointer expression is interpreted as the argument to a delegate instantiation expression of the appropriate type with the associated type argument list. For example:
Delegate Sub D(i As Integer ) Module Test Sub F( i As Integer ) End Sub Sub Main() Dim del As D ' The next two l i nes are equiva lent . del = AddressOf F del = New D(AddressOf F) End Sub End Module
4. A lambda method can be reclassified as a value. If the reclassification occurs in the context of a conversion where the target type is known, then one of two reclassifications can occur: 4.1. If the target type is a delegate type, the lambda method is interpreted as the argument to a delegate-construction expression of the appropriate type.
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
329
4.3
Inheritance
4.2. If the target type is System.Linq.Express ions .Express ion(Of T) T is a delegate , and type, then the lambda method is interpreted as if it was being used in delegateconstruction expression for T and then converted to an expression tree. Annotation The exact translation between lambda methods and expression trees may not be fixed between versions of the compiler and is beyond the scope of this specification. For Microsoft Visual Basic 10.0, all lambda expressions may be converted to expression trees subject to the following restrictions: 1. Only single-line lambda expressions without ByRef parameters may be converted to expression trees. Of the single-line Sub lambdas, only invocation statements may be converted to expression trees. 2. Anonymous type expressions cannot be converted to expression trees if an earlier field initializer is used to initialize a subsequent field initializer, e.g. New With {.a=1, .b=.a} 3. Object initializer expressions cannot be converted to expression trees if a member of the current object being initialized is used in one of the field initializers, e.g. New C1 With
{.a=1, .b=.Method1()}
4. Multi-dimensional array creation expressions can only be converted to expression trees if they declare their element type explicitly. 5. Late-binding expressions cannot be converted to expression trees. 6. When a variable or field is passed ByRef to an invocation expression but does not have exactly the same type as the ByRef parameter, or when a property is passed ByRef, normal VB semantics are that a copy of the argument is passed ByRef and its final value is then copied back into the variable or field or property. In expression trees, the copy-back does not happen. All these restrictions apply to nested lambda expressions as well. If the target type is not known, then the lambda method is interpreted as the argument to a delegate instantiation expression of an anonymous delegate type with the same signature of the lambda method. If strict semantics are being used and the type of any
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
330
4.3
Inheritance
of the parameters are omitted, a compile-time error occurs; otherwise, Object is substituted for any missing parameter type. For example:
Module Test Sub Main() ' Type of x wil l be equiva lent to Func(Of Object , Object , Object) Dim x = Funct ion(a , b) a + b ' Type of y wil l be equiva lent to Action(Of Object , Object) Dim y = Sub(a, b) Console .Wri teL ine (a + b) End Sub End Module
5. A property group can be reclassified as a property access. The property group expression is interpreted as an index expression with empty parentheses (that is, f is interpreted as f ( ) ). 6. A property access can be reclassified as a value. The property access expression is interpreted as an invocation expression of the Get accessor of the property. If the property has no getter, then a compile-time error occurs. 7. A late-bound access can be reclassified as a late-bound method or late-bound property access. In a situation where a late-bound access can be reclassified both as a method access and as a property access, reclassification to a property access is preferred. 8. A late-bound access can be reclassified as a value. 9. An array literal can be reclassified as a value. The type of the value is determined as follows: 9.1. If the reclassification occurs in the context of a conversion where the target type is known and the target type is an array type, then that target type is used. If the target type is System.Col lec t i ons .Gener ic . I L i s t , f T) (O System.Col lec t i ons .Gener ic . I Co l l e c t i on (OfSystem.Collections.Generic.IEnumerable(Of , or T) T), and the array literal has one level of nesting, then the type T() is used as the target type.
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
331
4.3
Inheritance
9.2. Otherwise, an array of rank equal to the level of nesting is used. The element type of the array is determined by the dominant type of the elements in the initializer; if no dominant type can be determined, Object is used. For example:
' x Is GetType(Double(,,)) Dim x = { { { 1, 2.0 }, { 3, 4 } }, { { 5, 6 }, { 7, 8 } } }.GetType() ' y Is GetType(Integer()) Dim y = { 1, 2, 3 }.GetType() ' z Is GetType(Object()) Dim z = { 1, "2" }.GetType() ' Error: Inconsistent nesting Dim a = { { 10 }, { 20, 30 } }.GetType()
10.Otherwise, a compile-time error occurs. Annotation This causes a slight change in behavior between version 9.0 and version 10.0 of the language. In the past, array element initializers did not affect local variable type inference and now they do. So Dim a() = { 1, 2, would have inferred Object( )as the type of a in 3 } version 9.0 of the language and Integer() in version 10.0. The reclassification then reinterprets the array literal as an array-creation expression. So the examples:
Dim x As Double = { 1, 2, 3, 4 } Dim y = { "a" , "b" }
The default value Nothing can be reclassified as a value. In a context where the target type is known, the result is the default value of the target type. In a context where the target type is not known, the result is a null value of type Object.
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
332
4.3
Inheritance
A namespace expression, type expression, event access expression, or void expression cannot be reclassified. Multiple reclassifications can be done at the same time. For example:
Module Test Sub F(i As Integer) End Sub ReadOnly Property P() As Integer Get End Get End Sub Sub Main() F(P) End Property End Module
In this case, the property group expression P is first reclassified from a property group to a property access and then reclassified from a property access to a value. The fewest number of reclassifications are performed to reach a valid classification in the context.
4.3
Inheritance
Str ingconversions are always done in the current culture of the execution environment at
run time. Note that constant coercion expressions can only ever use intrinsic conversions. The +, and Not unary operators, provided the operand and result is of a type listed above. The +, , *, ^, Mod, /, \, <<, >>, &, And, Or, Xor, AndAlso, OrElse, =, <, >, <>, <=, and => binary operators, provided each operand and result is of a type listed above. The conditional operator If, provided each operand and result is of a type listed above. The following run-time functions:
Microsoft.VisualBasic.Strings.ChrW Microsoft.VisualBasic.Strings.Chr, if the constant value is between 0 and 128 Microsoft.VisualBasic.Strings.AscW, if the constant string is not empty Microsoft.VisualBasic.Strings.Asc, if the constant string is not empty
The following constructs are not permitted in constant expressions: Implicit binding through a With context. Constant expressions of an integral type (ULong, Long, UInteger, Integer, UShort, Short, SByte, or Byte) can be implicitly converted to a narrower integral type, and constant expressions of type Double can be implicitly converted to Single, provided the value of the constant expression is within the range of the destination type. These narrowing conversions are allowed regardless of whether permissive or strict semantics are being used. ConstantExpression ::= Expression
334
4.3
Inheritance
late-binding, including for the purposes of overload resolution. Note that, unlike the earlybound case, invoking or accessing a Shared member late-bound will cause the invocation target to be evaluated at run time. If the expression is an invocation expression for a member defined on System.Object late binding will not take place. , In general, late-bound accesses are resolved at run time by looking up the identifier on the actual run-time type of the expression. If late-bound member lookup fails at run time, a System.Miss ingMemberExcept ionexception is thrown. Because late-bound member lookup is done solely off the run-time type of the associated target expression, an object's run-time type is never an interface. Therefore, it is impossible to access interface members in a latebound member access expression.
The arguments to a la te - bound member access are evaluated in the order they appear in the member access express ion: not the order in which parameters are declared in the la te - bound member. The fo l l ow ing example i l l u s t r a tes th is Publ i c Sub f (ByVal x As Integer , ByVal y As Integer ) End Sub End Class Module Module1 Sub Main() Console .Wri te ( "Ear l y - bound: ") Dim c As C = New C c. f ( y :=t ( "y" ) , x:=t("x" ) )
Console .Wri te (vbCrL f & "Late- bound: ") Dim o As Object = New C o. f (y :=t ( " y" ) , End Sub
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
x:=t("x" ) )
335
4.3
Inheritance
Function t(ByVal s As String) As Integer Console.Write(s) Return 0 End Function End Module
Because late-bound overload resolution is done on the run-time type of the arguments, it is possible that an expression might produce different results based on whether it is evaluated at compile time or run time. The following example illustrates this difference:
Class Base End Class Class Derived Inherits Base End Class Module Test Sub F(b As Base) Console.WriteLine("F(Base)") End Sub Sub F(d As Derived) Console.WriteLine("F(Derived)") End Sub Sub Main() Dim b As Base = New Derived() Dim o As Object = b
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
336
4.3
Inheritance
337
4.3
Inheritance
1.78.3 Instance Expressions An instance expression is the keyword Me , MyClass, or MyBase. An instance expression, which may only be used within the body of a non-shared method, constructor, or property accessor, is classified as a value. Instance expressions cannot be used in the arguments to constructor invocations. The three different instance expression types are: The keyword Me represents the instance of the type containing the method or property accessor being executed. The keyword MyClass is equivalent to Me, but all method invocations on it are treated as if the method being invoked is non-overridable. Thus, the method called will not be affected by the run-time type of the value on which the method is being called. MyClass can only be used as the target expression of a member access or in a Handles clause. The keyword MyBase represents the instance of the type containing the method or property accessor being executed cast to its direct base type. All method invocations on it are treated as if the method being invoked is non-overridable. MyBase can only be used as the target expression of a member access. A member access using MyBase is called a base access. The following example demonstrates how instance expressions can be used:
Class Base Publ i c Overr idable Sub F() Console .Wri teL ine ("Base.F" ) End Sub End Class Class Derived Inher i t s Base Publ i c Overr ides Sub F() Console .Wri teL ine ("Der ived.F" ) End Sub Publ i c Sub G() MyClass.F ( )
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
338
4.3
Inheritance
End Sub End Class Class MoreDerived Inherits Derived Public Overrides Sub F() Console.WriteLine("MoreDerived.F") End Sub Public Sub H() MyBase.F() End Sub End Class Module Test Sub Main() Dim x As MoreDerived = new MoreDerived() x.F() x.G() x.H() End Sub End Module
339
4.3
Inheritance
1.78.4 Simple Name Expressions A simple name expression consists of a single identifier followed by an optional type argument list. The name is resolved and classified by the following simple name resolution rules: 1. Starting with the immediately enclosing block and continuing with each enclosing outer block (if any), if the identifier matches the name of a local variable, static variable, constant local, method type parameter, or parameter, then the identifier refers to the matching entity. If the identifier matches a local variable, static variable, or constant local and a type argument list was provided, a compile-time error occurs. If the identifier matches a method type parameter and a type argument list was provided, no match occurs and resolution continues. If the identifier matches a local variable, the local variable matched is the implicit function or Get accessor return local variable, and the expression is part of an invocation expression, invocation statement, or an AddressOf expression, then no match occurs and resolution continues. The expression is classified as a variable if it is a local variable, static variable, or parameter. The expression is classified as a type if it is a method type parameter. The expression is classified as a value if it is a constant local. 2. For each nested type containing the expression, starting from the innermost and going to the outermost, if a lookup of the identifier in the type produces a match with an accessible member: 2.1. If the matching type member is a type parameter, then the result is classified as a type and is the matching type parameter. If a type argument list was provided, no match occurs and resolution continues. 2.2. Otherwise, if the type is the immediately enclosing type and the lookup identifies a non-shared type member, then the result is the same as a member access of the form Me.E(Of A) where E is the identifier and A is the type argument , list, if any.
340
4.3
Inheritance
2.3.
Otherwise, the result is exactly the same as a member access of the form
T.E(Of A)where T is the type containing the matching member, E is the identifier, , and A is the type argument list, if any. In this case, it is an error for the identifier to
refer to a non-shared member. 3. For each nested namespace, starting from the innermost and going to the outermost namespace, do the following: 3.1. If the namespace contains an accessible type with the given name and has the same number of type parameters as was supplied in the type argument list, if any, then the identifier refers to that type and is classified as a type. 3.2. Otherwise, if no type argument list was supplied and the namespace contains a namespace member with the given name, then the identifier refers to that namespace and is classified as a namespace. 3.3. Otherwise, if the namespace contains one or more accessible standard modules, and a member name lookup of the identifier produces an accessible match in exactly one standard module, then the result is exactly the same as a member access of the form M.E(Of A), where M is the standard module containing the matching member, E is the identifier, and A is the type argument list, if any. If the identifier matches accessible type members in more than one standard module, a compile-time error occurs. 4. If the source file has one or more import aliases, and the identifier matches the name of one of them, then the identifier refers to that namespace or type. If a type argument list is supplied, a compile-time error occurs. 5. If the source file containing the name reference has one or more imports: 5.1. If the identifier matches in exactly one import the name of an accessible type with the same number of type parameters as was supplied in the type argument list, if any, or a type member, then the identifier refers to that type or type member. If the identifier matches in more than one import the name of an accessible type with the same number of type parameters as was supplied in the type argument list, if any, or an accessible type member, a compile-time error occurs.
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
341
4.3
Inheritance
5.2. Otherwise, if no type argument list was supplied and the identifier matches in exactly one import the name of a namespace with accessible types, then the identifier refers to that namespace. If no type argument list was supplied and the identifier matches in more than one import the name of a namespace with accessible types, a compile-time error occurs. 5.3. Otherwise, if the imports contain one or more accessible standard modules, and a member name lookup of the identifier produces an accessible match in exactly one standard module, then the result is exactly the same as a member access of the form M.E(Of A) where M is the standard module containing the , matching member, E is the identifier, and A is the type argument list, if any. If the identifier matches accessible type members in more than one standard module, a compile-time error occurs. 6. If the compilation environment defines one or more import aliases, and the identifier matches the name of one of them, then the identifier refers to that namespace or type. If a type argument list is supplied, a compile-time error occurs. 7. If the compilation environment defines one or more imports: 7.1. If the identifier matches in exactly one import the name of an accessible type with the same number of type parameters as was supplied in the type argument list, if any, or a type member, then the identifier refers to that type or type member. If the identifier matches in more than one import the name of an accessible type with the same number of type parameters as was supplied in the type argument list, if any, or a type member, a compile-time error occurs. 7.2. Otherwise, if no type argument list was supplied and the identifier matches in exactly one import the name of a namespace with accessible types, then the identifier refers to that namespace. If no type argument list was supplied and the identifier matches in more than one import the name of a namespace with accessible types, a compile-time error occurs. 7.3. Otherwise, if the imports contain one or more accessible standard modules, and a member name lookup of the identifier produces an accessible match in exactly one standard module, then the result is exactly the same as a member
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
342
4.3
Inheritance
access of the form M.E(Of A) where M is the standard module containing the , matching member, E is the identifier, and A is the type argument list, if any. If the identifier matches accessible type members in more than one standard module, a compile-time error occurs. 8. Otherwise, the name given by the identifier is undefined. A simple name expression that is undefined is a compile-time error. Normally, a name can only occur once in a particular namespace. However, because namespaces can be declared across multiple .NET assemblies, it is possible to have a situation where two assemblies define a type with the same fully qualified name. In that case, a type declared in the current set of source files is preferred over a type declared in an external .NET assembly. Otherwise, the name is ambiguous and there is no way to disambiguate the name. SimpleNameExpression ::= Identifier [ OpenParenthesis Of TypeArgumentList CloseParenthesis ] 1.78.5 AddressOf Expressions An AddressOf expression is used to produce a method pointer. The expression consists of the AddressOf keyword and an expression that must be classified as a method group or a late-bound access. The method group cannot refer to constructors. The result is classified as a method pointer, with the same associated target expression and type argument list (if any) as the method group. AddressOfExpression ::= AddressOf Expression
343
4.3
Inheritance
IsExpression | GetXmlNamespaceExpression 1.79.1 GetType Expressions A GetType expression consists of the keyword GetType and the name of a type. A type expression is classified as a value, and the value of the expression is the reflection (System.Type) class that represents that type. If the expression is a type parameter, the expression will return the System.Type object that corresponds to the type argument supplied for the type parameter at run-time. The type name in a GetType expression is special in two ways: The type name is allowed to be System.Void, the only place in the language where this type name may be referenced. The type name may be a constructed generic type with the type arguments omitted. This allows the GetType expression to return the System.Type object that corresponds to the generic type itself. The following example demonstrates the GetType expression:
Module Test Sub Main() Dim t As Type() = { GetType( Integer ) , GetType(System.Int32) , _ GetType(Str i ng ) , GetType(Double( ) ) } Dim i As Integer For i = 0 To t .Length 1 Console .Wri teL ine ( t ( i ) . Name) Next i End Sub End Module
344
4.3
Inheritance
String Double[]
GetTypeExpression ::= GetType OpenParenthesis GetTypeTypeName CloseParenthesis GetTypeTypeName ::= TypeName | OpenTypeName OpenTypeName ::= QualifiedOpenTypeName | NullableOpenTypeName QualifiedOpenTypeName ::= Identifier [ TypeArityList ] | Global Period IdentifierOrKeyword [ TypeArityList ] | QualifiedOpenTypeName Period IdentifierOrKeyword [ TypeArityList ] TypeArityList ::= ( Of [ CommaList ] ) CommaList ::= Comma | CommaList Comma NullableOpenTypeName ::= QualifiedOpenTypeName ? 1.79.2 TypeOf...Is Expressions A TypeOf...Is expression is used to check whether the run-time type of a value is compatible with a given type. The first operand must be classified as a value, cannot be a reclassified lambda method, and must be of a reference type or an unconstrained type parameter type. The second operand must be a type name. The result of the expression is classified as a value and is a Boolean value. The expression evaluates to True if the run-time type of the operand has an identity, default, reference, array, value type, or type parameter conversion to the type, False otherwise. A compile-time error occurs if no conversion exists between the type of the expression and the specific type. TypeOfIsExpression ::= TypeOf Expression Is [ LineTerminator ] TypeName
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
345
4.3
Inheritance
1.79.3 Is Expressions An Is or IsNot expression is used to do a reference equality comparison. Each expression must be classified as a value and the type of each expression must be a reference type, an unconstrained type parameter type, or a nullable value type. If the type of one expression is an unconstrained type parameter type or nullable value type, however, the other expression must be the literal Nothing. The result is classified as a value and is typed as Boolean. An Is operation evaluates to True if both values refer to the same instance or both values are Nothing, or False otherwise. An IsNot operation evaluates to False if both values refer to the same instance or both values are Nothing, or True otherwise. IsExpression ::= Expression Is [ LineTerminator ] Expression | Expression IsNot [ LineTerminator ] Expression 1.79.4 GetXmlNamespace Expressions A GetXmlNamespace expression consists of the keyword GetXmlNamespace and the name of an XML namespace declared by the source file or compilation environment. An XML namespace expression is classified as a value, and the value of the expression is an instance of System.Xml.Linq.XNamespace that represents that XML namespace. If that type is not available, then a compile-time error will occur. For example:
Imports <xmlns:db="http: / / example.org /database"> Module Test Sub Main() Dim db = GetXmlNamespace(db) ' The fo l l ow ing are equiva lent Dim customer1 = _ New System.Xml.L inq .XElement(db + "customer" , "Bob") Dim customer2 = <db:customer>Bob</>
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
346
4.3
Inheritance
Everything between the parentheses is considered part of the namespace name, so XML rules around things such as whitespace apply. For example:
Imports <xmlns:db-ns="http://example.org/database"> Module Test Sub Main() ' Error, XML name expected Dim db1 = GetXmlNamespace( db-ns ) ' Error, ')' expected Dim db2 = GetXmlNamespace(db _ ) ' OK. Dim db3 = GetXmlNamespace(db-ns) End Sub End Module
The XML namespace expression can also be omitted, in which case the expression returns the object that represents the default XML namespace. GetXmlNamespaceExpression ::= GetXmlNamespace XMLNamespaceName ] CloseParenthesis OpenParenthesis [
4.3
Inheritance
1. If E is omitted, then the expression from the immediately containing With statement is substituted for E and the member access is performed. If there is no containing With statement, a compile-time error occurs. 2. If E is classified as a namespace or E is the keyword Global, then the member lookup is done in the context of the specified namespace. If I is the name of an accessible member of that namespace with the same number of type parameters as was supplied in the type argument list, if any, then the result is that member. The result is classified as a namespace or a type depending on the member. Otherwise, a compile-time error occurs. 3. If E is a type or an expression classified as a type, then the member lookup is done in the context of the specified type. If I is the name of an accessible member of E, then E.I is evaluated and classified as follows: 3.1. If I is the keyword New and E is not an enumeration then a compile-time error occurs. 3.2. If I identifies a type with the same number of type parameters as was supplied in the type argument list, if any, then the result is that type. 3.3. If I identifies one or more methods, then the result is a method group with the associated type argument list and no associated target expression. 3.4. If I identifies one or more properties and no type argument list was supplied, then the result is a property group with no associated target expression. 3.5. If I identifies a shared variable and no type argument list was supplied, then the result is either a variable or a value. If the variable is read-only, and the reference occurs outside the shared constructor of the type in which the variable is declared, then the result is the value of the shared variable I in E. Otherwise, the result is the shared variable I in E. 3.6. If I identifies a shared event and no type argument list was supplied, the result is an event access with no associated target expression. 3.7. If I identifies a constant and no type argument list was supplied, then the result is the value of that constant.
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
348
4.3
Inheritance
3.8. If I identifies an enumeration member and no type argument list was supplied, then the result is the value of that enumeration member. 3.9. Otherwise, E. I is an invalid member reference, and a compile-time error occurs. 4. If E is classified as a variable or value, the type of which is T, then the member lookup is done in the context of T. If I is the name of an accessible member of T, then E.I is evaluated and classified as follows: 4.1. If I is the keyword New, E is an instance expression (i.e. Me, MyBase, or MyClass), and no type arguments were supplied, then the result is a method group representing the instance constructors of the type of E with an associated target expression of E and no type argument list. Otherwise, a compile-time error occurs. If I identifies one or more methods, including extension methods if T is not Object, then the result is a method group with the associated type argument list and an associated target expression of E.
4.2.
4.3. If I identifies one or more properties and no type arguments were supplied, then the result is a property group with an associated target expression of E. 4.4. If I identifies a shared variable or an instance variable and no type arguments were supplied, then the result is either a variable or a value. If the variable is readonly, and the reference occurs outside a constructor of the class in which the variable is declared appropriate for the kind of variable (shared or instance), then the result is the value of the variable I in the object referenced by E. If T is a reference type, then the result is the variable I in the object referenced by E. Otherwise, if T is a value type and the expression E is classified as a variable, the result is a variable; otherwise the result is a value. 4.5. If I identifies an event and no type arguments were supplied, the result is an event access with an associated target expression of E. 4.6. If I identifies a constant and no type arguments were supplied, then the result is the value of that constant.
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
349
4.3
Inheritance
4.7. If I identifies an enumeration member and no type arguments were supplied, then the result is the value of that enumeration member. 4.8. If T is Object, then the result is a late-bound member lookup classified as a late-bound access with the associated type argument list and an associated target expression of E. 5. Otherwise, E.I is an invalid member reference, and a compile-time error occurs. When a member access expression begins with the keyword Global, the keyword represents the outermost unnamed namespace, which is useful in situations where a declaration shadows an enclosing namespace. The Global keyword allows escaping out to the outermost namespace in that situation. For example:
Class System End Class Module Test Sub Main() ' Error : Class System does not contain Console System.Console .Wri teL ine ("Hel l o , world!" ) ' Legal , binds to System in outermost namespace Global .System.Console .Wr i teL ine ("Hel l o , world!" ) End Sub End Module
In the above example, the first method call is invalid because the identifier System binds to the class System, not the namespace System. The only way to access the System namespace is to use Global to escape out to the outermost namespace. If the member being accessed is shared, any expression on the left side of the period is superfluous and is not evaluated unless the member access is done late-bound. For example, consider the following code:
Class C Publ i c Shared F As Integer = 10
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
350
4.3
Inheritance
End Class Module Test Public Function ReturnC() As C Console.WriteLine("Returning a new instance of C.") Return New C() End Function Public Sub Main() Console.WriteLine("The value of F is: " & ReturnC().F) End Sub End Module
It prints The value of F i s : because the function ReturnC does not need to be called to 10 provide an instance of C to access the shared member F. MemberAccessExpression ::= [ MemberAccessBase ] Period IdentifierOrKeyword [ OpenParenthesis Of TypeArgumentList CloseParenthesis ] MemberAccessBase ::= Expression | NonArrayTypeName | Global | MyClass |
MyBase
1.80.1 Identical Type and Member Names It is not uncommon to name members using the same name as their type. In that situation, however, inconvenient name hiding can occur:
Enum Color Red Green Yel low End Enum
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
351
4.3
Inheritance
Class Test ReadOnly Property Color() As Color Get Return Color.Red End Get End Property Shared Function DefaultColor() As Color Return Color.Green ' Binds to the instance property! End Function End Class
In the previous example, the simple name Color in Defaul tColor binds to the instance property instead of the type. Because an instance member cannot be referenced in a shared member, this would normally be an error. However, a special rule allows access to the type in this case. If the base expression of a member access expression is a simple name and binds to a constant, field, property, local variable or parameter whose type has the same name, then the base expression can refer either to the member or the type. This can never result in ambiguity because the members that can be accessed off of either one are the same. 1.80.2 Default Instances In some situations, classes derived from a common base class usually or always have only a single instance. For example, most windows shown in a user interface only ever have one instance showing on the screen at any time. To simplify working with these types of classes, Visual Basic can automatically generate default instances of the classes that provide a single, easily referenced instance for each class. Default instances are always created for a family of types rather than for one particular type. So instead of creating a default instance for a class Form1 that derives from Form, default instances are created for all classes derived from Form. This means that each individual class that derives from the base class does not have to be specially marked to have a default instance.
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
352
4.3
Inheritance
The default instance of a class is represented by a compiler-generated property that returns the default instance of that class. The property generated as a member of a class called the group class that manages allocating and destroying default instances for all classes derived from the particular base class. For example, all of the default instance properties of classes derived from Form may be collected in the MyForms class. If an instance of the group class is returned by the expression My.Forms, then the following code accesses the default instances of derived classes Form1 and Form2:
Class Form1 Inher i t s Form Publ i c x As Integer End Class Class Form2 Inher i t s Form Publ i c y As Integer End Class Module Main Sub Main() My.Forms.Form1.x = 10 Console .Wri teL ine (My.Forms.Form2.y) End Sub End Module
Default instances will not be created until the first reference to them; fetching the property representing the default instance causes the default instance to be created if it has not already been created or has been set to Nothing. To allow testing for the existence of a default instance, when a default instance is the target of an Is or IsNot operator, the default instance will not be created. Thus, it is possible to test whether a default instance is Nothing or some other reference without causing the default instance to be created. Default instances are intended to make it easy to refer to the default instance from outside of the class that has the default instance. Using a default instance from within a class that defines it might cause confusion as to which instance is being referred to, i.e. the default instance or the current instance. For example, the following code modifies only the value x
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
353
4.3
Inheritance
in the default instance, even though it is being called from another instance. Thus the code would print the value 5 instead of 10:
Class Form1 Inher i t s Form Publ i c x As Integer = 5 Publ i c Sub ChangeX() Form1.x = 10 End Sub End Class Module Main Sub Main() Dim f As Form1 = New Form1() f .ChangeX() Console .Wri teL ine ( f . x ) End Sub End Module
To prevent this kind of confusion, it is not valid to refer to a default instance from within an instance method of the default instances type. 1.80.2.1 Default Instances and Type Names A default instance may also be accessible directly through its types name. In this case, in any expression context where the type name is not allowed the expression E, where E represents the fully qualified name of the class with a default instance, is changed to E, where E represents an expression that fetches the default instance property. For example, if default instances for classes derived from Form allow accessing the default instance through the type name, then the following code is equivalent to the code in the previous example:
Module Main Sub Main() Form1.x = 10
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
354
4.3
Inheritance
This also means that a default instance that is accessible through its types name is also assignable through the type name. For example, the following code sets the default instance of Form1 to Nothing:
Module Main Sub Main() Form1 = Nothing End Sub End Module
Note that the meaning of E. I were E represents a class and I represents a shared member does not change. Such an expression still accesses the shared member directly off of the class instance and does not reference the default instance. 1.80.2.2 Group Classes The Microsoft.VisualBasic.MyGroupCollectionAttribute attribute indicates the group class for a family of default instances. The attribute has four parameters: The parameter TypeToCollect specifies the base class for the group. All instantiable classes without open type parameters that derive from a type with this name (regardless of type parameters) will automatically have a default instance. The parameter CreateInstanceMethodName specifies the method to call in the group class to create a new instance in a default instance property. The parameter DisposeInstanceMethodName specifies the method to call in the group class to dispose of a default instance property if the default instance property is assigned the value Nothing. The parameter DefaultInstanceAlias specifics the expression E to substitute for the class name if the default instances are accessible directly through their type name. If this parameter is Nothing or an empty string, default instances on this group type are not accessible directly through their types name.
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
355
4.3
Inheritance
Annotation In all current implementations of the Visual Basic language, the Defaul t I ns tanceAl ias parameter is ignored, except in compiler-provided code. Multiple types can be collected into the same group by separating the names of the types and methods in the first three parameters using commas. There must be the same number of items in each parameter, and the list elements are matched in order. For example, the following attribute declaration collects types that derive from C1, C2 or C3 into a single group:
<Microsoft .V i sua lBas ic .MyGroupCol lec t i on( "C1, C2, C3", _ "CreateC1, CreateC2, CreateC3", _ "DisposeC1, DisposeC2, DisposeC3", My.Cs" )> Publ i c NotInher i tab le Class MyCs End Class
The signature of the create method must be of the form Shared Function <Name>(Of T As {New, <Type>})(Instance Of T) As T. The dispose method must be of the form Shared Sub <Name>(Of T As <Type>)(ByRef Instance Of T). Thus, the group class for the example in the preceding section could be declared as follows:
<Microsoft .V i sua lBas ic .MyGroupCol lec t i on( " Form", "Create" , _ "Dispose" , "My.Forms")> _ Publ i c NotInher i tab le Class MyForms Pr ivate Shared Funct ion Create(Of T As {New, Form}) _ ( Ins tance As T) As T I f Instance Is Nothing Then Return New T() Else Return Instance End I f End Funct ion Pr ivate Shared Sub Dispose(Of T As Form)(ByRef Instance As T) Instance.C lose( )
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
356
4.3
Inheritance
If a source file declared a derived class Form1, the generated group class would be equivalent to:
<Microsoft .V i sua lBas ic .MyGroupCol lec t i on( " Form", "Create" , _ "Dispose" , "My.Forms")> _ Publ i c NotInher i tab le Class MyForms Pr ivate Shared Funct ion Create(Of T As {New, Form}) _ ( Ins tance As T) As T I f Instance Is Nothing Then Return New T() Else Return Instance End I f End Funct ion Pr ivate Shared Sub Dispose(Of T As Form)(ByRef Instance As T) Instance.C lose( ) Instance = Nothing End Sub Pr ivate m_Form1 As Form1 Publ i c Property Form1() As Form1 Get Return Create(m_Form1) End Get Set (Value As Form1) I f Value IsNot Nothing AndAlso Value IsNot m_Form1 Then Throw New ArgumentException( _ "Property can only be set to Nothing. " ) End I f Dispose(m_Form1) End Set
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
357
4.3
Inheritance
1.80.3 Extension Method Collection Extension methods for the member access expression E. I are collected by gathering all of the extension methods with the name I that are available in the current context. In other words, the search for extension methods named I is conducted similar to that of a simple name expression: First, each nested type containing the expression is checked, starting from the innermost and going to the outermost. Then, each nested namespace is checked, starting from the innermost and going to the outermost namespace. Then, the imports in the source file are checked. Then, the imports defined by the compilation environment are checked. An extension method is collected only if there is a widening native conversion from the target expression type to the type of the first parameter of the extension method. And unlike regular simple name expression binding, the search collects all extension methods; the collection does not stop when an extension method is found. For example:
Imports System.Runtime.CompilerServ ices Class C1 End Class Namespace N1 Module N1C1Extensions <Extension> _ Sub M1(c As C1, x As Integer ) End Sub End Module End Namespace
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
358
4.3
Inheritance
Namespace N1.N2 Module N2C1Extensions <Extension> _ Sub M1(c As C1, y As Double) End Sub End Module End Namespace Namespace N1.N2.N3 Module Test Sub Main() Dim x As New C1() ' Calls N1C1Extensions.M1 x.M1(10) End Sub End Module End Namespace
In this example, even though N2C1Extensions.M1 is found before N1C1Extensions.M1 they , both are considered as extension methods. Once all of the extension methods have been collected, they are then curried. Currying takes the target of the extension method call and applies it to the extension method call, resulting in a new method signature with the first parameter removed (because it has been specified). For example:
Imports System.Runtime.CompilerServ ices Module Ext1 <Extension> _ Sub M(x As Integer , y As Integer ) End Sub End Module Module Ext2 <Extension> _ Sub M(x As Integer , y As Double) End Sub
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
359
4.3
Inheritance
End Module Module Main Sub Test() Dim v As Integer = 10 ' The curried method signatures considered are: ' Ext1.M(y As Integer) ' Ext2.M(y As Double) v.M(10) End Sub End Module
In the above example, the curried result of applying v to Ext1.M is the method signature Sub M(y As Integer). In addition to removing the first parameter of the extension method, currying also removes any method type parameters that are a part of the type of the first parameter. When currying an extension method with method type parameter, type inference is applied to the first parameter and the result is fixed for any type parameters that are inferred. If type inference fails, the method is ignored. For example:
Imports System.Runtime.CompilerServ ices Module Ext1 <Extension> _ Sub M(Of T, U)(x As T, y As U) End Sub End Module Module Ext2 <Extension> _ Sub M(Of T)(x As T, y As T) End Sub End Module Module Main
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
360
4.3
Inheritance
Sub Test() Dim v As Integer = 10 ' The curried method signatures considered are: ' Ext1.M(Of U)(y As U) ' Ext2.M(y As Integer) v.M(10) End Sub End Module
In the above example, the curried result of applying v to Ext1.M is the method signature Sub M(Of U)(y As U), because the type parameter T is inferred as a result of the currying and is now fixed. Because the type parameter U was not inferred as a part of the currying, it remains an open parameter. Similarly, because the type parameter T is inferred as a result of applying v to Ext2.M, the type of parameter y becomes fixed as Integer. It will not be inferred to be any other type. When currying the signature, all constraints except for New constraints are also applied. If the constraints are not satisfied, or depend on a type that was not inferred as a part of currying, the extension method is ignored. For example:
Imports System.Runtime.CompilerServ ices Module Ext1 <Extension> _ Sub M1(Of T As Structure) (x As T, y As Integer ) End Sub <Extension> _ Sub M2(Of T As U, U)(x As T, y As U) End Sub End Module Module Main Sub Test( ) Dim s As Str ing = "abc" ' Error : Str ing does not sat i s f y the Structure constra in t
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
361
4.3
Inheritance
s.M1(10) ' Error: T depends on U, which cannot be inferred s.M2(10) End Sub End Module
Annotation One of the main reasons for doing currying of extension methods is that it allows query expressions to infer the type of the iteration before evaluating the arguments to a query pattern method. Since most query pattern methods take lambda expressions, which require type inference themselves, this greatly simplifies the process of evaluating a query expression. Unlike normal interface inheritance, extension methods that extend two interfaces that do not relate to one another are available, as long as they do not have the same curried signature:
Imports System.Runtime.CompilerServices Interface I1 End Interface Interface I2 End Interface Class C1 Implements I1, I2 End Class Module I1Ext <Extension> _ Sub M1(i As I1, x As Integer) End Sub <Extension> _
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
362
4.3
Inheritance
Sub M2(i As I1, x As Integer) End Sub End Module Module I2Ext <Extension> _ Sub M1(i As I2, x As Integer) End Sub <Extension> _ Sub M2(I As I2, x As Double) End Sub End Module Module Main Sub Test() Dim c As New C1() ' Error: M is ambiguous between I1Ext.M1 and I2Ext.M1. c.M1(10) ' Calls I1Ext.M2 c.M2(10) End Sub End Module
Finally, it is important to remember that extension methods are not considered when doing late binding:
Module Test Sub Main() Dim o As Object = ' Ignores extension methods o.M1() End Sub End Module
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
363
4.3
Inheritance
If an exclamation point is specified with no expression, the expression from the immediately containing With statement is assumed. If there is no containing With statement, a compile-time error occurs. DictionaryAccessExpression ::= [ Expression ] ! IdentifierOrKeyword
4.3
Inheritance
the delegate type and the target expression becomes the associated target expression of the method group. An argument list has two sections: positional arguments and named arguments. Positional arguments are expressions and must precede any named arguments. Named arguments start with an identifier that can match keywords, followed by := and an expression. If the method group only contains one accessible method, including both instance and extension methods, and that method takes no arguments and is a function, then the method group is interpreted as an invocation expression with an empty argument list and the result is used as the target of an invocation expression with the provided argument list(s). For example:
Class C1 Funct ion M1() As Integer ( ) Return New Integer ( ) { 1, 2, 3 } End Sub End Class Module Test Sub Main() Dim c As New C1() ' Pr ints 3 Console .Wri teL ine (c .M1(2) ) End Sub End Module
Otherwise, overload resolution is applied to the methods to pick the most applicable method for the given argument list(s). If the most applicable method is a function, then the result of the invocation expression is classified as a value typed as the return type of the function. If the most applicable method is a subroutine, then the result is classified as void. If the most applicable method is a partial method that has no body, then the invocation expression is ignored and the result is classified as void. For an early-bound invocation expression, the arguments are evaluated in the order in which the corresponding parameters are declared in the target method. For a late-bound
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
365
4.3
Inheritance
member access expression, they are evaluated in the order in which they appear in the member access expression: see Section 1.77, Late-Bound Expressions. InvocationExpression ::= Expression [ OpenParenthesis [ ArgumentList ] CloseParenthesis ] ArgumentList ::= PositionalArgumentList | PositionalArgumentList Comma NamedArgumentList | PositionalArgumentList ::= [ Expression ] | PositionalArgumentList Comma [ Expression ] NamedArgumentList ::= IdentifierOrKeyword ColonEquals Expression | NamedArgumentList Comma IdentifierOrKeyword ColonEquals Expression 1.82.1 Overloaded Method Resolution Given a method group, the most applicable method in the group for an argument list is determined using the following steps. If, after applying a particular step, no members remain in the set, then a compile-time error occurs. If only one member remains in the set, then that member is the most applicable member. The steps are: 1. First, if no type arguments have been supplied, apply type inference to any methods which have type parameters. If type inference succeeds for a method, then the inferred type arguments are used for that particular method. If type inference fails for a method, then that method is eliminated from the set. 2. Next, eliminate all members from the set that are inaccessible or not applicable to the argument list. 3. Next, eliminate all members from the set that require narrowing conversions to be applicable to the argument list, except for the case where the argument expression type is Object.
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
NamedArgumentList
366
4.3
Inheritance
4. Next, eliminate all remaining members from the set that require narrowing coercions to be applicable to the argument list. If the set is empty, the type containing the method group is not an interface, and strict semantics are not being used, the invocation target expression is reclassified as a late-bound method access. Otherwise, the normal rules apply. Annotation The justification for this rule is that if a program is loosely-typed (that is, most or all variables are declared as Object), overload resolution can be difficult because all conversions from Object are narrowing. Rather than have the overload resolution fail in many situations (requiring strong typing of the arguments to the method call), resolution the appropriate overloaded method to call is deferred until run time. This allows the loosely-typed call to succeed without additional casts. An unfortunate side-effect of this, however, is that performing the late-bound call requires casting the call target to Object. In the case of a structure value, this means that the value must be boxed to a temporary. If the method eventually called tries to change a field of the structure, this change will be lost once the method returns. Interfaces are excluded from this special rule because late binding always resolves against the members of the runtime class or structure type, which may have different names than the members of the interfaces they implement. 5. Next, if any instance methods remain in the set, eliminate all extension methods from the set. For example:
Imports System.Runtime.CompilerServ ices Class C3 Sub M1(d As Integer ) End Sub End Class Module C3Extensions <Extension> _ Sub M1(c3 As C3, c As Long)
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
367
4.3
Inheritance
End Sub <Extension> _ Sub M1(c3 As C3, c As Short) End Sub End Module Module Test Sub Main() Dim c As New C3() Dim sVal As Short = 10 Dim lVal As Long = 20 ' Calls C3.M1, since C3.M1 is applicable. c.M1(sVal) ' Calls C3Extensions.M1 since C3.M1 requires a narrowing conversion c.M1(lVal) End Sub
Annotation Extension methods are ignored if there are applicable instance methods to guarantee that adding an import (that might bring new extension methods into scope) will not cause a call on an existing instance method to rebind to an extension method. Given the broad scope of some extension methods (i.e. those defined on interfaces and/or type parameters), this is a safer approach to binding to extension methods. 6. Next, if, given any two members of the set, M and N, M is more applicable than N to the argument list, eliminate N from the set. If more than one member remains in the set and the remaining members are not equally applicable to the argument list, a compiletime error results. 7. Otherwise, given any two members of the set, M and N, apply the following tie-breaking rules, in order:
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
368
4.3
Inheritance
If M has fewer parameters from an expanded paramarray than N, eliminate N from the set. For example:
Module Test Sub F(a As Object , ParamArray b As Object( ) ) Console .Wri teL ine ("F (Object , Object( ) ) " ) End Sub Sub F(a As Object , b As Object , ParamArray c As Object( ) ) Console .Wri teL ine ("F (Object , Object , Object( ) ) " ) End Sub Sub Main() F(1) F(1, 2) F(1, 2, 3) End Sub End Module
Annotation When a class declares a method with a paramarray parameter, it is not uncommon to also include some of the expanded forms as regular methods. By doing so it is possible to avoid the allocation of an array instance that occurs when an expanded form of a method with a paramarray parameter is invoked. 7.1. If M is defined in a more derived type than N, eliminate N from the set. For example:
Class Base Sub F(Of T, U)(x As T, y As U) End Sub End Class
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
369
4.3
Inheritance
Class Derived Inherits Base Overloads Sub F(Of T, U)(x As U, y As T) End Sub End Class Module Test Sub Main() Dim d As New Derived() ' Calls Derived.F d.F(10, 10) End Sub End Module
This rule also applies to the types that extension methods are defined on. For example:
Imports System.Runtime.CompilerServices Class Base End Class Class Derived Inherits Base End Class Module BaseExt <Extension> _ Sub M(b As Base, x As Integer) End Sub End Module Module DerivedExt <Extension> _
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
370
4.3
Inheritance
Sub M(d As Derived, x As Integer) End Sub End Module Module Test Sub Main() Dim b As New Base() Dim d As New Derived() ' Calls BaseExt.M b.M(10) ' Calls DerivedExt.M d.M(10) End Sub End Module
7.2. If M and N are extension methods and the target type of M is a class or structure and the target type of N is an interface, eliminate N from the set. For example:
Imports System.Runtime.CompilerServ ices Inter face I1 End Inter face Class C1 Implements I1 End Class Module Ext1 <Extension> _ Sub M(i As I1 , x As Integer ) End Sub End Module Module Ext2
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
371
4.3
Inheritance
<Extension> _ Sub M(c As C1, y As Integer) End Sub End Module Module Test Sub Main() Dim c As New C1() ' Calls Ext2.M, because Ext1.M is hidden since it extends ' an interface. c.M(10) ' Calls Ext1.M CType(c, I1).M(10) End Sub End Module
7.3. If M and N are extension methods and the target type of M has fewer type parameters than the target type of N, eliminate N from the set. For example:
Imports System.Runtime.CompilerServ ices Class C1 End Class Module Ext1 <Extension> _ Sub M1(c As C1, x As Integer ) End Sub End Module Module Ext2 <Extension> _ Sub M1(Of T)(x As T, y As Integer ) End Sub End Module
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
372
4.3
Inheritance
Module Test Sub Main() Dim c As New C1() ' Calls Ext1.M1 since Ext2.M1 target type is more generic. c.M1(10) End Sub End Module
7.4. 7.5.
If M is less generic than N, eliminate N from the set. If M is not an extension method and N is, eliminate N from the set.
7.6. If M and N are extension methods and M was found before N, eliminate N from the set. For example:
Imports System.Runtime.CompilerServ ices Class C1 End Class Namespace N1 Module N1C1Extensions <Extension> _ Sub M1(c As C1, x As Integer ) End Sub End Module End Namespace Namespace N1.N2 Module N2C1Extensions <Extension> _ Sub M1(c As C1, y As Integer ) End Sub End Module End Namespace
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
373
4.3
Inheritance
Namespace N1.N2.N3 Module Test Sub Main() Dim x As New C1() ' Calls N2C1Extensions.M1 x.M1(10) End Sub End Module End Namespace
If the extension methods were found in the same step, then those extension methods are ambiguous. The call may always be disambiguated using the name of the standard module containing the extension method and calling the extension method as if it was a regular member. For example:
Imports System.Runtime.CompilerServices Class C1 End Class Module C1ExtA <Extension> _ Sub M(c As C1) End Sub End Module Module C1ExtB <Extension> _ Sub M(c As C1) End Sub End Module Module Main Sub Test() Dim c As New C1()
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
374
4.3
Inheritance
' Ambiguous between C1ExtA.M and ' Calls C1ExtA.M ' Calls C1ExtB.M
7.7. If M and N both required type inference to produce type arguments, and M did not require determining the dominant type for any of its type arguments (i.e. each the type arguments inferred to a single type), but N did, eliminate N from the set. Annotation This rule ensures that overload resolution that succeeded in previous versions (where inferring multiple types for a type argument would cause an error), continue to produce the same results. 7.8. If one or more arguments are AddressOf or lambda expressions, and all of the corresponding delegate types in M match exactly, but not all do in N, eliminate N from the set. Annotation As with the rule on type inference, this prevents resolution that succeeded in previous versions from changing. 7.9. If one or more arguments are AddressOf or lambda expressions, and all of the corresponding delegate types in M are widening conversions, but not all are in N, eliminate N from the set. 7.10. If the overload resolution is being done to resolve the target of a delegatecreation expression from an AddressOf expression and M is a function, while N is a subroutine, eliminate N from the set. 8. Otherwise, the call is ambiguous and a compile-time error occurs.
375
4.3
Inheritance
1.82.1.1 Applicability A member M is considered equally applicable as N if their signatures are the same or if each parameter type in M is the same as the corresponding parameter type in N. Annotation Two members can end up in a method group with the same signature due to extension methods. Two members can also be equally applicable but not have the same signature due to type parameters or paramarray expansion. A member M is considered more applicable than N if their signatures are different and at least one parameter type in M is more applicable than a parameter type in N, and no parameter type in N is more applicable than a parameter type in M. Given a pair of parameters Mj and Nj that matches an argument Aj, the type of Mj is considered more applicable than the type of Nj if one of the following conditions is true: 1. Mj and Nj have identical types, or 2. There exists a widening conversion from the type of Mj to the type Nj, or Annotation Note that because parameters types are being compared without regard to the actual argument in this case, the widening conversion from constant expressions to a numeric type the value fits into is not considered in this case. 3. Aj is the literal 0, Mj is a numeric type and Nj is an enumerated type, or Annotation This rule is necessary because the literal 0 widens to any enumerated type. Since an enumerated type widens to its underlying type, this means that overload resolution on 0 will, by default, prefer enumerated types over numeric types. We received a lot of feedback that this behavior was counterintuitive. 4. Mj is Byte and Nj is SByte, or 5. Mj is Short and Nj is UShort, or
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
376
4.3
Inheritance
6. Mj is Integerand Nj is UInteger, or 7. Mj is Long and Nj is ULong. Annotation The rules about the numeric types are necessary because the signed and unsigned numeric types of a particular size only have narrowing conversions between them. Thus, the above rules break the tie between the two types in favor of the more natural numeric type. This is particularly important when doing overload resolution on a type that widens to both the signed and unsigned numeric types of a particular size (for example, a numeric literal that fits into both). 8. Mj and Nj are delegate function types and the return type of Mj is more specific than the return type of Nj. If Aj is classified as a lambda method, and Mj or Nj is System.Linq.Expressions.Expression(Of T), then the type argument of the type (assuming it is a delegate type) is substituted for the type being compared. Annotation It is interesting to note that the previous rule differs slightly from C#, in that C# requires that the delegate function types have identical parameter lists before comparing return types, while Visual Basic does not. 1.82.1.2 Genericity A member M is determined to be less generic than a member N as follows: 1. If, for each pair of matching parameters Mj and Nj, Mj is less or equally generic than Nj with respect to type parameters on the method, and at least one Mj is less generic with respect to type parameters on the method. 2. Otherwise, if for each pair of matching parameters Mj and Nj, Mj is less or equally generic than Nj with respect to type parameters on the type, and at least one Mj is less generic with respect to type parameters on the type, then M is less generic than N. A parameter M is considered to be equally generic to a parameter N if their types Mt and Nt both refer to type parameters or both dont refer to type parameters. M is considered to be less generic than N if Mt does not refer to a type parameter and Nt does. Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved. 377
4.3
Inheritance
For example:
Class C1(Of T) Sub S1(Of U)(x As U, y As T) End Sub Sub S1(Of U)(x As U, y As U) End Sub Sub S2(x As Integer, y As T) End Sub Sub S2(x As T, y As T) End Sub End Class Module Test Sub Main() Dim x As C1(Of Integer) = New C1(Of Integer) x.S1(10, 10) x.S2(10, 10) End Sub End Module ' Calls S1(U, T) ' Calls S2(Integer, T)
Extension method type parameters that were fixed during currying are considered type parameters on the type, not type parameters on the method. For example:
Imports System.Runtime.CompilerServices Module Ext1 <Extension> _ Sub M1(Of T, U)(x As T, y As U, z As U) End Sub End Module Module Ext2
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
378
4.3
Inheritance
<Extension> _ Sub M1(Of T, U)(x As T, y As U, z As T) End Sub End Module Module Test Sub Main() Dim i As Integer = 10 i.M1(10, 10) End Sub End Module
In practice, the rules for determining whether one method is more applicable than another method are intended to find the overload that is closest to the actual arguments supplied. If there is a method whose parameter types match the argument types, then that method is obviously the closest. Barring that, one method is closer than another if all of its parameter types are wider than (or the same as) the parameter types of the other method. If neither methods parameters are wider than the other, then there is no way for to determine which method is closer to the arguments. Note The context of the method usage is not used in overload resolution (except for determining accessibility). Annotation The context of the method call is not generally considered when doing overload resolution to avoid adding even more complexity to an already complex algorithm. The ability for mere mortals to grasp the results of overload resolution is already tenuous one, but adding a myriad of contextual clues would make it impossible. This does mean that in some situations overload resolution will choose the wrong overload, such as when it picks an instance method when doing overload resolution off of a type name. Also note that because of the named parameter syntax, the ordering of the actual and formal parameters may not be the same. 379
4.3
Inheritance
1.82.2 Applicable Methods A method is applicable to a set of type arguments, positional arguments, and named arguments if the method can be invoked using the two argument lists. The argument lists are matched against the parameter lists as follows: 1. First, match each positional argument in order to the list of method parameters. If there are more positional arguments than parameters and the last parameter is not a paramarray, the method is not applicable. Otherwise, the paramarray parameter is expanded with parameters of the paramarray element type to match the number of positional arguments. If a positional argument is omitted, the method is not applicable. 2. Next, match each named argument to a parameter with the given name. If one of the named arguments fails to match, matches a paramarray parameter, or matches an argument already matched with another positional or named argument, the method is not applicable. 3. Next, if parameters that have not been matched are not optional, the method is not applicable. If optional parameters remain, the default value specified in the optional parameter declaration is matched to the parameter. If an Object parameter does not specify a default value: 3.1. If the parameter has a
System.Runtime.CompilerServ ices . ID i spatchConstantAtt r attribute, then the i bute expression New System.Runtime. In teropServices .Di spatchWrapper(Nothing) is used.
3.2.
System.Runtime.CompilerServ ices . IUnknownConstantAtt r i bu te attribute, then the expression New System.Runtime. In teropServices .UnknownWrapper(Nothing)used. is
3.3.
4. Finally, if type arguments have been specified, they are matched against the type parameter list. If the two lists do not have the same number of elements, the method is
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
380
4.3
Inheritance
not applicable, unless the type argument list is empty. If the type argument list is empty, type inferencing is used to try and infer the type argument list. If type inferencing fails, the method is not applicable. Otherwise, the type arguments are filled in the place of the type parameters in the signature. The type of each argument expression must be implicitly convertible to the type of the parameter it matches. If the parameter is a reference parameter, the argument expression must be also implicitly convertible from the type of the parameter. The type arguments, if any, must satisfy the constraints, if any, on the matching type parameters. If a single argument expression matches a paramarray parameter and the type of the argument expression is convertible to both the type of the paramarray parameter and the paramarray element type, the method is applicable in both its expanded and unexpanded forms, with two exceptions. If the conversion from the type of the argument expression to the paramarray type is narrowing, then the method is only applicable in its expanded form. If the argument expression is the literal Nothing, then the method is only applicable in its unexpanded form. For example:
Module Test Sub F(ParamArray a As Object( ) ) Dim o As Object For Each o In a Console .Wri te (o .GetType() . Fu l lName) Console .Wri te ( " ") Next o Console .Wri teL ine ( ) End Sub Sub Main() Dim a As Object( ) = { 1, "Hel lo" , Dim o As Object = a F(a) F(CType(a, Object) ) F(o)
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
123.456 }
381
4.3
Inheritance
In the first and last invocations of F, the normal form of F is applicable because a widening conversion exists from the argument type to the parameter type (both are of type Object()), and the argument is passed as a regular value parameter. In the second and third invocations, the normal form of F is not applicable because no widening conversion exists from the argument type to the parameter type (conversions from Object to Object() are narrowing). However, the expanded form of F is applicable, and a one-element Object() is created by the invocation. The single element of the array is initialized with the given argument value (which itself is a reference to an Object()). 1.82.3 Passing Parameters If a parameter is a value parameter, the matching argument expression must be classified as a value. The value is converted to the type of the parameter and passed in as the parameter at run time. If the parameter is a reference parameter and the matching argument expression is classified as a variable whose type is the same as the parameter, then a reference to the variable is passed in as the parameter at run time. Otherwise, if the matching argument expression is classified as a variable, value, or property access, then a temporary variable of the type of the parameter is allocated. Before the method invocation at run time, the argument expression is reclassified as a value, converted to the type of the parameter, and assigned to the temporary variable. Then a reference to the temporary variable is passed in as the parameter. After the method invocation is evaluated, if the argument expression is classified as a variable or property access, the temporary variable is assigned to the variable expression or the property access expression. If the property access expression has no Set accessor, then the assignment is not performed. Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved. 382
4.3
Inheritance
1.82.4 Conditional Methods If the target method to which an invocation expression refers is a subroutine that is not a member of an interface and if the method has one or more System.Diagnost i cs .Condi t i ona lAt t r i attributes, evaluation of the expression depends on bu te the conditional compilation constants defined at that point in the source file. Each instance of the attribute specifies a string, which names a conditional compilation constant. Each conditional compilation constant is evaluated as if it were part of a conditional compilation statement. If the constant evaluates to True, the expression is evaluated normally at run time. If the constant evaluates to False, the expression is not evaluated at all. When looking for the attribute, the most derived declaration of an overridable method is checked. Note The attribute is not valid on functions or interface methods and is ignored if specified on either kind of method. Thus, conditional methods will only appear in invocation statements. 1.82.5 Type Argument Inference When a method with type parameters is called without specifying type arguments, type argument inference is used to try and infer type arguments for the call. This allows a more natural syntax to be used for calling a method with type parameters when the type arguments can be trivially inferred. For example, given the following method declaration:
Module Uti l Funct ion Choose(Of T)(b As Boolean, f i r s t I f b Then Return f i r s t Else Return second End I f End Funct ion End Class As T, second As T) As T
it is possible to invoke the Choose method without explicitly specifying a type argument: 383
4.3
Inheritance
' calls Choose(Of Integer) Dim i As Integer = Util.Choose(True, 5, 213) ' calls Choose(Of String) Dim s As String = Util.Choose(False, "a", "b")
Through type argument inference, the type arguments Integerand Str ingare determined from the arguments to the method. Type argument inference occurs before expression reclassification is performed on lambda methods or method pointers in the argument list, since reclassification of those two kinds of expressions may require the type of the parameter to be known. Given a set of arguments A1, A2, , AN, a set of matching parameters P1, P2, , PN and a set of method type parameters T1, T2, , TN, the dependencies between the arguments and method type parameters are first collected as follows: If AN is the Nothing literal, no dependencies are generated. If AN is a lambda method and the type of PN is a constructed delegate type or System.Linq.Expressions.Expression(Of T), where T is a constructed delegate type, If the type of a lambda method parameter will be inferred from the type of the corresponding parameter PN, and the type of the parameter depends on a method type parameter TN, then AN has a dependency on TN. If the type of a lambda method parameter is specified and the type of the corresponding parameter PN depends on a method type parameter TN, then TN has a dependency on AN. If the return type of PN depends on a method type parameter TN, then TN has a dependency on AN. If AN is a method pointer and the type of PN is a constructed delegate type, If the return type of PN depends on a method type parameter TN, then TN has a dependency on AN. If PN is a constructed type and the type of PN depends on a method type parameter TN, then TN has a dependency on AN. Otherwise, no dependency is generated.
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
384
4.3
Inheritance
After collecting dependencies, any arguments that have no dependencies are eliminated. If any method type parameters have no outgoing dependencies (i.e. the method type parameter does not depend on an argument), then type inference fails. Otherwise, the remaining arguments and method type parameters are grouped into strongly connected components. A strongly connected component is a set of arguments and method type parameters, where any element in the component is reachable via dependencies on other elements. The strongly connected components are then topologically sorted and processed in topological order: If the strongly typed component contains only one element, If the element has already been marked complete, skip it. If the element is an argument, then add type hints from the argument to the method type parameters that depend on it and mark the element as complete. If the argument is a lambda method with parameters that still need inferred types, then infer Object for the types of those parameters. If the element is a method type parameter, then infer the method type parameter to be the dominant type among the argument type hints and mark the element as complete. If a type hint has an array element restriction on it, then only conversions that are valid between arrays of the given type are considered (i.e. covariant and intrinsic array conversions). If a type hint has a generic argument restriction on it, then only identity conversions are considered. If no dominant type can be chosen, inference fails. If any lambda method argument types depend on this method type parameter, the type is propagated to the lambda method. If the strongly typed component contains more than one element, then the component contains a cycle. For each method type parameter that is an element in the component, if the method type parameter depends on an argument that is not marked complete, convert that dependency into an assertion that will be checked at the end of the inference process. 385
4.3
Inheritance
Restart the inference process at the point at which the strongly typed components were determined. If type inference succeeds for all of the method type parameters, then any dependencies that were changed into assertions are checked. An assertion succeeds if the type of the argument is implicitly convertible to the inferred type of the method type parameter. If an assertion fails, then type argument inference fails. Given an argument type TA for an argument A and a parameter type TP for a parameter P, type hints are generated as follows: If TP does not involve any method type parameters then no hints are generated. If TP and TA are array types of the same rank, then replace TA and TP with the element types of TA and TP and restart this process with an array element restriction. If TP is a method type parameter, then TA is added as a type hint with the current restriction, if any. If A is a lambda method and TP is a constructed delegate type or System.Linq.Expressions.Expression(Of T), where T is a constructed delegate type, for each lambda method parameter type TL and corresponding delegate parameter type TD, replace TA with TL and TP with TD and restart the process with no restriction. Then replace TA with the return type of the lambda method and TP with the return type of the delegate type and restart the process with no restriction. If A is a method pointer and TP is a constructed delegate type, use the parameter types of TP to determine which method pointed is most applicable to TP. If there is a method that is most applicable, replace TA with the return type of the method and TP with the return type of the delegate type and restart the process with no restriction. Otherwise, TP must be a constructed type. Given TG, the generic type of TP, If TA is TG, inherits from TG, or implements the type TG exactly once, then for each matching type argument TAX from TA and TPX from TP, replace TA with TAX and TP with TPX and restart the process with a generic argument restriction. Otherwise, type inference fails for the generic method.
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
386
4.3
Inheritance
The success of type inference does not, in and of itself, guarantee that the method is applicable.
387
4.3
Inheritance
the indexing is late-bound. The expression results in a late-bound property access typed as Object. The associated target expression is either the target expression, if it is a value, or the associated target expression of the property group. At run time the expression is processed as follows: If the expression is classified as a late-bound property group, the expression may result in a method group, a property group, or a value (if the member is an instance or shared variable). If the result is a method group or property group, overload resolution is applied to the group to determine the correct method for the argument list. If overload resolution fails, a System.Ref lec t i on .AmbiguousMatchExcept ion exception is thrown. Then the result is processed either as a property access or as an invocation and the result is returned. If the invocation is of a subroutine, the result is Nothing. If the run-time type of the target expression is an array type or System.Array the result of , the index expression is the value of the variable at the specified index. Otherwise, the run-time type of the expression must have a default property and the index is performed on the property group that represents all of the default properties on the type. If the type has no default property, then a System.Miss ingMemberExceptionexception is thrown. IndexExpression ::= Expression OpenParenthesis [ ArgumentList ] CloseParenthesis
388
4.3
Inheritance
A New expression is classified as a value and the result is the new instance of the type. NewExpression ::= ObjectCreationExpression | ArrayExpression | AnonymousObjectCreationExpression 1.84.1 Object-Creation Expressions An object-creation expression is used to create a new instance of a class type or a structure type. The type of an object creation expression must be a class type, a structure type, or a type parameter with a New constraint and cannot be a MustInher i tclass. Given an object creation expression of the form New T(A), where T is a class type or structure type and A is an optional argument list, overload resolution determines the correct constructor of T to call. A type parameter with a New constraint is considered to have a single, parameterless constructor. If no constructor is callable, a compile-time error occurs; otherwise the expression results in the creation of a new instance of T using the chosen constructor. If there are no arguments, the parentheses may be omitted. Where an instance is allocated depends on whether the instance is a class type or a value type. New instances of class types are created on the system heap, while new instances of value types are created directly on the stack. An object-creation expression can optionally specify a list of member initializers after the constructor arguments. These member initializers are prefixed with the keyword With, and the initializer list is interpreted as if it was in the context of a With statement. For example, given the class:
Class Customer Dim Name As Str ing Dim Address As Str ing End Class
The code:
Module Test Sub Main() Dim x As New Customer() With { .Name = "Bob Smith" , _
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
389
4.3
Inheritance
Each initializer must specify a name to assign, and the name must be a non-ReadOnly instance variable or property of the type being constructed; the member access will not be late bound if the type being constructed is Object. Each member in a type can only be initialized once. The initializer expressions, however, may refer to each other. For example:
Module Test Sub Main() Dim x As New Customer() With { .Name = "Bob Smith" , _ .Address = .Name & " St. " } End Sub End Module
The initializers are assigned left-to-right, so if an initializer refers to a member that has not been initialized yet, it will see whatever value the instance variable after the constructor ran:
Module Test Sub Main()
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
390
4.3
Inheritance
' The value of Address will be " St." since Name has not been ' assigned yet. Dim x As New Customer() With { .Address = .Name & " St." } End Sub End Module
If the type being created is a collection type and has an instance method named Add (including extension methods), then the object-creation expression can specify a collection initializer prefixed by the keyword From. An object-creation expression cannot specify both
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
391
4.3
Inheritance
a member initializer and a collection initializer. Each element in the collection initializer is passed as an argument to an invocation of the Add function. Any shared Add methods are ignored. For example:
Dim l i s t = New Lis t (Of Integer ) ( ) From { 1, 2, 3, 4 }
is equivalent to:
Dim l i s t = New Lis t (Of Integer ) ( ) l i s t . Add(1) l i s t . Add(2) l i s t . Add(3)
If an element is a collection initializer itself, each element of the sub-collection initializer will be passed as an individual argument to the Add function. For example, the following:
Dim dic t = Dict ionary(Of Integer , Str ing) From { { 1, "One" },{ 2, "Two" } }
is equivalent to:
Dim dic t = New Dict ionary(Of Integer , Str ing) dic t .Add(1 , "One") dic t .Add(2, "Two")
This expansion is always done and is only ever done one level deep; after that, subinitializers are considered array literals. For example:
' Error : Lis t (Of T) does not have an Add method that takes two parameters. Dim l i s t = New Lis t (Of Integer ( ) ) ( ) From { { 1, 2 }, { 3, 4 } } ' OK, th is in i t i a l i z e s the dic t i onary with ( In teger , Integer ( ) ) Dim dic t = New Dict ionary(Of Integer , Integer ( ) ) ( ) From _ { { 1, { 2, 3 } }, { 3, { 4, 5 } } } pairs .
4.3
Inheritance
ObjectCreationExpressionInitializer ::= ObjectMemberInitializer | ObjectCollectionInitializer ObjectMemberInitializer ::= With OpenCurlyBrace FieldInitializerList CloseCurlyBrace FieldInitializerList ::= FieldInitializer | FieldInitializerList Comma FieldInitializer FieldInitializer ::= [ [ Key ] . IdentifierOrKeyword Equals ] Expression ObjectCollectionInitializer ::= From CollectionInitializer CollectionInitializer ::= OpenCurlyBrace [ CollectionElementList ] CloseCurlyBrace CollectionElementList ::= CollectionElement | CollectionElementList Comma CollectionElement CollectionElement ::= Expression | CollectionInitializer 1.84.2 Array Expressions An array expression is used to create a new instance of an array type. There are two types of array expressions: array creation expressions, and array literals. 1.84.2.1 Array creation expressions If an array size initialization modifier is supplied, the resulting array type is derived by deleting each of the individual arguments from the array size initialization argument list. The value of each argument determines the upper bound of the corresponding dimension in the newly allocated array instance. If the expression has a non-empty collection initializer, each argument in the argument list must be a constant, and the rank and dimension lengths specified by the expression list must match those of the collection initializer.
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
393
4.3
Inheritance
Dim a() As Integer = New Integer(2) {} Dim b() As Integer = New Integer(2) { 1, 2, 3 } Dim c(,) As Integer = New Integer(1, 2) { { 1, 2, 3 } , { 4, 5, 6 } } ' Error, length/initializer mismatch. Dim d() As Integer = New Integer(2) { 0, 1, 2, 3 }
If an array size initialization modifier is not supplied, then the type name must be an array type and the collection initializer must be empty or have the same number of levels of nesting as the rank of the specified array type. All of the elements in the innermost nesting level must be implicitly convertible to the element type of the array and must be classified as a value. The number of elements in each nested collection initializer must always be consistent with the size of the other collections at the same level. The individual dimension lengths are inferred from the number of elements in each of the corresponding nesting levels of the collection initializer. If the collection initializer is empty, the length of each dimension is zero.
Dim e() As Integer = New Integer() { 1, 2, 3 } Dim f(,) As Integer = New Integer(,) { { 1, 2, 3 } , { 4, 5, 6 } } ' Error: Inconsistent numbers of elements! Dim g(,) As Integer = New Integer(,) { { 1, 2 }, { 4, 5, 6 } } ' Error: Inconsistent levels of nesting! Dim h(,) As Integer = New Integer(,) { 1, 2, { 3, 4 } }
The outermost nesting level of a collection initializer corresponds to the leftmost dimension of an array, and the innermost nesting level corresponds to the rightmost dimension. The example:
Dim array As Integer(,) = _ { { 0, 1 }, { 2, 3 }, { 4, 5 }, { 6, 7 }, { 8, 9 } }
394
4.3
Inheritance
0) 0) 0) 0)
= = = =
2: 4: 6: 8:
1) 1) 1) 1)
= = = =
3 5 7 9
If the collection initializer is empty (that is, one that contains curly braces but no initializer list) and the bounds of the dimensions of the array being initialized are known, the empty collection initializer represents an array instance of the specified size where all the elements have been initialized to the element type's default value. If the bounds of the dimensions of the array being initialized are not known, the empty collection initializer represents an array instance in which all dimensions are size zero. An array instance's rank and length of each dimension are constant for the entire lifetime of the instance. In other words, it is not possible to change the rank of an existing array instance, nor is it possible to resize its dimensions. 1.84.2.2 Array Literals An array literal denotes an array whose element type, rank, and bounds are inferred from a combination of the expression context and a collection initializer. This is explained in Section 1.75.1, Expression Reclassification. For example:
' array of integers Dim a = {1, 2, 3} ' array of shorts Dim b = {1S, 2S, 3S} ' array of shorts whose type is taken from the context Dim c As Short() = {1, 2, 3} ' array of type Integer(,) Dim d = {{1, 0}, {0, 1}} ' jagged array of rank ()() Dim e = {({1, 0}), ({0, 1})} ' error: inconsistent rank Dim f = {{1}, {2, 3}}
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
395
4.3
Inheritance
The format and requirements for the collection initializer in an array literal is exactly the same as that for the collection initializer in an array creation expression. Annotation An array literal does not create the array in and of itself; instead, it is the reclassification of the expression into a value that causes the array to be created. For instance, the conversion CType(new Integer( ) {1,2,3}, Short( ) ) possible because there is no is not conversion from Integer ( ) Short(); but the expression CType({1,2,3},Short()) is possible to because it first reclassifies the array literal into the array creation expression New Short() {1,2,3}. ArrayExpression ::= ArrayCreationExpression | ArrayLiteralExpression ArrayCreationExpression ::= New NonArrayTypeName ArrayNameModifier CollectionInitializer ArrayLiteralExpression ::= CollectionInitializer 1.84.3 Delegate-Creation Expressions A delegate-creation expression is used to create a new instance of a delegate type. The argument of a delegate-creation expression must be an expression classified as a method pointer or a lambda method. If the argument is a method pointer, one of the methods referenced by the method pointer must be applicable to the signature of the delegate type. A method M is applicable to a delegate type D if:
M is not Partial or has a body.
396
4.3
Inheritance
The parameter types of M each have a conversion from the type of the corresponding parameter type of D, and their modifiers (i.e. ByRef, ByVal) match. The return type of M, if any, has a conversion to the return type of D. If the method pointer references a late-bound access, then the late-bound access is assumed to be to a function that has the same number of parameters as the delegate type. If strict semantics are not being used and there is only one method referenced by the method pointer, but it is not applicable due to the fact that it has no parameters and the delegate type does, then the method is considered applicable and the parameters or return value are simply ignored. For example:
Delegate Sub F(x As Integer ) Module Test Sub M() End Sub Sub Main() ' Val id Dim x As F = AddressOf M End Sub End Module
Annotation This relaxation is only allowed when strict semantics are not being used because of extension methods. Because extension methods are only considered if a regular method was not applicable, it is possible for an instance method with no parameters to hide an extension method with parameters for the purpose of delegate construction. If more than one method referenced by the method pointer is applicable to the delegate type, then overload resolution is used to pick between the candidate methods. The types of the parameters to the delegate are used as the types of the arguments for the purposes of overload resolution. If no one method candidate is most applicable, a compile-time error occurs. In the following example, the local variable is initialized with a delegate that refers Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved. 397
4.3
Inheritance
to the second Square method because that method is more applicable to the signature and return type of DoubleFunc.
Delegate Funct ion DoubleFunc(x As Double) As Double Module Test Funct ion Square(x As Single) As Single Return x * x End Funct ion Funct ion Square(x As Double) As Double Return x * x End Funct ion Sub Main() Dim a As New DoubleFunc(AddressOf Square) End Sub End Module
Had the second Square method not been present, the first Square method would have been chosen. If strict semantics are specified by the compilation environment or by Option Strict, then a compile-time error occurs if the most specific method referenced by the method pointer is narrower than the delegate signature. A method M is considered narrower than a delegate type D if: A parameter type of M has a widening conversion to the corresponding parameter type of D. Or, the return type, if any, of M has a narrowing conversion to the return type of D. If type arguments are associated with the method pointer, only methods with the same number of type arguments are considered. If no type arguments are associated with the method pointer, type inference is used when matching signatures against a generic method. Unlike other normal type inference, the return type of the delegate is used when inferring type arguments, but return types are still not considered when determining the least generic overload. The following example shows both ways of supplying a type argument to a delegate-creation expression:
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
398
4.3
Inheritance
Delegate Function D(s As String, i As Integer) As Integer Delegate Function E() As Integer Module Test Public Function F(Of T)(s As String, t1 As T) As T End Function Public Function G(Of T)() As T End Function Sub Main() Dim d1 As D = AddressOf f(Of Integer) Dim d2 As D = AddressOf f Dim e1 As E = AddressOf g(Of Integer) Dim e2 As E = AddressOf g End Sub End Module ' OK, type arg explicit ' OK, type arg inferred ' OK, type arg explicit ' OK, infer from return
In the above example, a non-generic delegate type was instantiated using a generic method. It is also possible to create an instance of a constructed delegate type using a generic method. For example:
Delegate Function Predicate(Of U)(u1 As U, u2 As U) As Boolean Module Test Function Compare(Of T)(t1 As List(of T), t2 As List(of T)) As Boolean ... End Function Sub Main() Dim p As Predicate(Of List(Of Integer)) p = AddressOf Compare(Of Integer) End Sub End Module
399
4.3
Inheritance
If the argument to the delegate-creation expression is a lambda method, the lambda method must be applicable to the signature of the delegate type. A lambda method L is applicable to a delegate type D if: If L has parameters, D has the same number of parameters. (If L has no parameters, the parameters of D are ignored.) The parameter types of L each have a conversion to the type of the corresponding parameter type of D, and their modifiers (i.e. ByRef, ByVal) match. If D is a function, the return type of L has a conversion to the return type of D. (If D is a subroutine, the return value of L is ignored.) If the parameter type of a parameter of L is omitted, then the type of the corresponding parameter in D is inferred; if the parameter of L has array or nullable name modifiers, a compile-time error results. Once all of the parameter types of L are available, then the type of the expression in the lambda method is inferred. For example:
Delegate Funct ion F(x As Integer , y As Long) As Long Module Test Sub Main() ' b in fe r red to Integer , c and return type in fe r red to Long Dim a As F = Funct ion(b , c) b + c ' e and return type in fe r red to Integer , f in fe r red to Long Dim d As F = Funct ion(e , f ) e + CInt( f ) End Sub End Module
In some situations where delegate signature does not exactly match the lambda method or method signature, the .NET Framework may not support the delegate creation natively. In that situation, a lambda method expression is used to match the two methods. For example:
Delegate Funct ion IntFunc(x As Integer ) As Integer Module Test
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
400
4.3
Inheritance
Function SquareString(x As String) As String Return CInt(x) * CInt(x) End Function Sub Main() ' The following two lines are equivalent Dim a As New IntFunc(AddressOf SquareString) Dim b As New IntFunc( _ Function(x As Integer) CInt(SquareString(CStr(x)))) End Sub End Module
The result of a delegate-creation expression is a delegate instance that refers to the matching method with the associated target expression (if any) from the method pointer expression. If the target expression is typed as a value type, then the value type is copied onto the system heap because a delegate can only point to a method of an object on the heap. The method and object to which a delegate refers remain constant for the entire lifetime of the delegate. In other words, it is not possible to change the target or object of a delegate after it has been created. 1.84.4 Anonymous Object-Creation Expressions An object-creation expression with member initializers can also omit the type name entirely. In that case, an anonymous type is constructed based on the types and names of the members initialized as a part of the expression. For example:
Module Test Sub Main() Dim Customer = New With { .Name = "John Smith", .Age = 34 } Console.WriteLine(Customer.Name) End Sub End Module
The type created by an anonymous object-creation expression is a class that has no name, inherits directly from Object, and has a set of properties with the same name as the members assigned to in the member initializer list. The type of each property is inferred
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
401
4.3
Inheritance
using the same rules as local variable type inference. Generated anonymous types also override ToStr ing returning a string representation of all members and their values. (The , exact format of this string is beyond the scope of this specification). By default, the properties generated by the anonymous type are read-write. It is possible to mark an anonymous type property as read-only by using the Key modifier. The Key modifier specifies that the field can be used to uniquely identify the value the anonymous type represents. In addition to making the property read-only, it also causes the anonymous type to override Equals and GetHashCode and to implement the interface System.IEquatable(Of T) (filling in the anonymous type for T). The members are defined as follows:
Function Equals(obj As Object) As Boolean and Function Equals(val As T) As Boolean are
implemented by validating that the two instances are of the same type and then comparing each Key member using Object.Equals. If all Key members are equal, then Equals returns True, otherwise Equals returns False.
Function GetHashCode() As Integer is implemented such that that if Equals is true for two instances of the anonymous type, then GetHashCode will return the same value. The hash starts with a seed value and then, for each Key member, in order multiplies the hash by 31 and adds the Key members hash value (provided by GetHashCode) if the member is not a reference type or nullable value type with the value of Nothing.
402
4.3
Inheritance
End Sub Public ReadOnly Property ZipCode As Integer Get Return _zipCode End Get Set (value As Integer) _zipCode = value End Set End Property Public Property State As String Get Return _state End Get Set (value As Integer) _state = value End Set End Property Public Overrides Function Equals(obj As Object) As Boolean Dim val As $Anonymous1 = TryCast(obj, $Anonymous1) Return Equals(val) End Function Public Overloads Function Equals(val As $Anonymous1) As Boolean _ Implements IEquatable(Of $Anonymous1).Equals If val Is Nothing Then Return False End If If Not Object.Equals(_zipCode, val._zipCode) Then Return False End If
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
403
4.3
Inheritance
Return True End Function Public Overrides Function GetHashCode() As Integer Dim hash As Integer = 0 hash = hash Xor _zipCode.GetHashCode() Return hash End Function Public Overrides Function ToString() As String Return "{ Key .ZipCode = " & _zipCode & ", .State = " & _state & " }" End Function
To simplify the situation where an anonymous type is created from the fields of another type, field names can be inferred directly from expressions in the following cases: A simple name expression x infers the name x. A member access expression x.y infers the name y. A dictionary lookup expression x!y infers the name y. An invocation or index expression with no arguments x() infers the name x. An XML member access expression x.<y>, x. . .<y> x.@y infers the name y. , An XML member access expression that is the target of a member access expression x.<y>.z infers the name z. An XML member access expression that is the target of an invocation or index expression with no arguments x.<y>.z() infers the name z. An XML member access expression that is the target of an invocation or index expression x.<y>(0) infers the name y. The initializer is interpreted as an assignment of the expression to the inferred name. For example, the following initializers are equivalent: Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved. 404
4.3
Inheritance
Class Address Public Street As String Public City As String Public State As String Public ZIP As String End Class Class C1 Sub Test(a As Address) Dim cityState1 = New With { .City = a.City, .State = a.State } Dim cityState2 = New With { a.City, a.State } End Sub End Class
If a member name is inferred that conflicts with an existing member of the type, such as GetHashCode , then a compile time error occurs. Unlike regular member initializers, anonymous object-creation expressions do not allow member initializers to have circular references, or to refer to a member before it has been initialized. For example:
Module Test Sub Main() ' Error : Circu lar references Dim x = New With { .a = .b , .b = .a } ' Error : Referr i ng to .b before i t has been assigned to Dim y = New With { .a = .b , .b = 10 } ' Error : Referr i ng to .a before i t has been assigned to Dim z = New With { .a = .a } End Sub End Module
If two anonymous class creation expressions occur within the same method and yield the same resulting shapeif the property order, property names, and property types all match they will both refer to the same anonymous class. The method scope of an instance or shared member variable with an initializer is the constructor in which the variable is initialized. Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved. 405
4.3
Inheritance
Annotation It is possible that a compiler may choose to unify anonymous types further, such as at the assembly level, but this cannot be relied upon at this time. AnonymousObjectCreationExpression ::= New ObjectMemberInitializer
Annotation The conversion set that DirectCast and TryCast support are restricted because they implement native CLR conversions. The purpose of DirectCast is to provide the functionality of the unbox instruction, while the purpose of TryCast is to provide the functionality of the isinst instruction. Since they map onto CLR instructions, supporting conversions not directly supported by the CLR would defeat the intended purpose.
DirectCast converts expressions that are typed as Object differently than CType. When converting an expression of type Object whose run-time type is a primitive value type, DirectCast throws a System.InvalidCastException exception if the specified type is not the same as the run-time type of the expression or a System.NullReferenceException if the expression evaluates to Nothing.
Annotation As noted above, DirectCast maps directly onto the CLR instruction unbox when the type of the expression is Object. In contrast, CType turns into a call to a runtime helper to do the conversion so that conversions between primitive types can be supported. In the case
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
406
4.3
Inheritance
when an Object expression is being converted to a primitive value type and the type of the actual instance match the target type, DirectCastwill be significantly faster than CType.
TryCast converts expressions but does not throw an exception if the expression cannot be converted to the target type. Instead, TryCast will result in Nothing if the expression cannot
Annotation As noted above, TryCast maps directly onto the CLR instruction isinst. By combining the type check and the conversion into a single operation, TryCast can be cheaper than doing a TypeOfIs and then a CType. If no conversion exists from the type of the expression to the specified type, a compiletime error occurs. Otherwise, the expression is classified as a value and the result is the value produced by the conversion. CastExpression ::= DirectCast OpenParenthesis Expression Comma TypeName CloseParenthesis | TryCast OpenParenthesis Expression Comma TypeName CloseParenthesis | CType OpenParenthesis Expression Comma TypeName CloseParenthesis | CastTarget OpenParenthesis Expression CloseParenthesis
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
407
4.3
Inheritance
CastTarget ::= CBool | CByte | CChar | CDate | CDec | CDbl | CInt | CLng | CObj | CSByte | CShort | CSng | CStr | CUInt | CULng | CUShort
408
4.3
Inheritance
Unary negation Multiplicative Integer division Modulus Additive Concatenation Shift Relational Logical NOT Logical AND Logical OR Logical XOR
+, *, / \ Mod +, & <<, >> =, <> , <, >, <=, >=, Like, Is, IsNot Not And, AndAlso Or, OrElse Xor
When an expression contains two operators with the same precedence, the associativity of the operators controls the order in which the operations are performed. All binary operators are left-associative, meaning that operations are performed from left to right. Precedence and associativity can be controlled using parenthetical expressions. 1.86.2 Object Operands In addition to the regular types supported by each operator, all operators support operands of type Object. Operators applied to Object operands are handled similarly to late-bound method calls made on Object values: the run-time type of the operands, rather than the compile-time type, determines the validity and type of the operation. If strict semantics are specified by the compilation environment or by Option Strict, any operators with operands of type Object cause a compile-time error, except for the TypeOf...Is, Is and IsNot operators. When given one or more operands of type Object, the outcome of the operation is the result of applying the operator to the operand types if the run-time types of the operands are
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
409
4.3
Inheritance
types that are supported by the operator. The value Nothing is treated as the default value of the type of the other operand in a binary operator expression. In a unary operator expression, or if both operands are Nothing in a binary operator expression, the type of the operation is Integer or the only result type of the operator, if the operator does not result in Integer. The result of the operation is always then cast back to Object. If the operand types have no valid operator, a System.InvalidCastException exception is thrown. Conversions at run time are done without regard to whether they are implicit or explicit. If the result of a numeric binary operation would produce an overflow exception (regardless of whether integer overflow checking is on or off), then the result type is promoted to the next wider numeric type, if possible. For example, consider the following code:
Module Test Sub Main() Dim o As Object = CObj(CByte(2) ) * CObj(CByte(255)) Console .Wri teL ine (o .GetType() . ToStr i ng ( ) & " = " & o) End Sub End Module
If no wider numeric type is available to hold the number, a System.OverflowException exception is thrown. 1.86.3 Operator Resolution Given an operator type and a set of operands, operator resolution determines which operator to use for the operands. When resolving operators, user-defined operators will be considered first, using the following steps: First, all of the candidate operators are collected. The candidate operators are all of the user-defined operators of the particular operator type in the source type and all of the user-defined operators of the particular type in the target type. If the source type and destination type are related, common operators are only considered once.
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
410
4.3
Inheritance
Then, overload resolution is applied to the operators and operands to select the most specific operator. When collecting the candidate operators for a type T?, the operators of type T are used instead. Any of Ts user-defined operators that involve only non-nullable value types are also lifted. A lifted operator uses the nullable version of any value types, with the exception the return types of IsTrue and IsFalse (which must be Boolean). Lifted operators are evaluated by converting the operands to their non-nullable version, then evaluating the user-defined operator and then converting the result type to its nullable version. If ether operand is Nothing, the result of the expression is a value of Nothing typed as the nullable version of the result type. For example:
Structure T ... End Structure Structure S Publ i c Shared Operator +(ByVal op1 As S, ByVal op2 As T) As T ... End Operator End Structure Module Test Sub Main() Dim x As S? Dim y, z As T? ' Val id , as S + T = T i s l i f t ed to S? + T? = T? z =x +y End Sub End Module
If the operator is a binary operator and one of the operands is reference type, the operator is also lifted, but any binding to the operator produces an error. For example:
Structure S1 Public F1 As Integer
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
411
4.3
Inheritance
Public Shared Operator +(left As S1, right As String) As S1 ... End Operator End Structure Module Test Sub Main() Dim a? As S1 Dim s As String ' Error: '+' is not defined for S1? and String a=a+s End Sub End Module
Annotation This rule exists because there has been consideration whether we wish to add nullpropagating reference types in a future version, in which case the behavior in the case of binary operators between the two types would change. As with conversions, user-defined operators are always preferred over lifted operators. When resolving overloaded operators, there may be differences between classes defined in Visual Basic and those defined in other languages: In other languages, Not, And , and Or may be overloaded both as logical operators and bitwise operators. Upon import from an external assembly, either form is accepted as a valid overload for these operators. However, for a type which defines both logical and bitwise operators, only the bitwise implementation will be considered. In other languages, >> and << may be overloaded both as signed operators and unsigned operators. Upon import from an external assembly, either form is accepted as a valid overload. However, for a type which defines both signed and unsigned operators, only the signed implementation will be considered.
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
412
4.3
Inheritance
If no user-defined operator is most specific to the operands, then intrinsic operators will be considered. If no intrinsic operator is defined for the operands, then a compile-time error results. A type T that has an intrinsic operator also defines that same operator for T?. The result of the operator on T? will be the same as for T, except that if either operand is Nothing, the result of the operator will be Nothing (i.e. the null value is propagated). For the purposes of resolving the type of an operation, the ? is removed from any operands that have them, the type of the operation is determined, and a ? is added to the type of the operation if any of the operands were nullable value types. For example:
Dim v1? As Integer = 10 Dim v2 As Long = 20 ' Type of operat ion wil l be Long? Console .Wri teL ine (v1 + v2)
Each operator lists the intrinsic types it is defined for and the type of the operation performed given the operand types. The result of type of a intrinsic operation follows these general rules: If all operands are of the same type, and the operator is defined for the type, then no conversion occurs and the operator for that type is used. Any operand whose type is not defined for the operator is converted using the following steps and the operator is resolved against the new types: The operand is converted to the next widest type that is defined for both the operator and the operand and to which it is implicitly convertible. If there is no such type, then the operand is converted to the next narrowest type that is defined for both the operator and the operand and to which it is implicitly convertible. If there is no such type or the conversion cannot occur, a compile-time error occurs. Otherwise, the operands are converted to the wider of the operand types and the operator for that type is used. If the narrower operand type cannot be implicitly converted to the wider operator type, a compile-time error occurs. 413
4.3
Inheritance
Despite these general rules, however, there are a number of special cases called out in the operator results tables. Note For formatting reasons, the operator type tables abbreviate the predefined names to their first two characters. So By is Byte, UI is UInteger St is String, etc. Err , means that there is no operation defined for the given operand types.
4.3
Inheritance
1.87.1 Unary Plus Operator The unary plus operator is defined for the Byte, SByte, UShort, Short, UInteger Integer, ULong, Long, Single, Double, and Decimal types. Operation Type: Bo Sh SB SB By By Sh Sh US US In In UI UI Lo Lo UL UL De De Si Si Do Do Da Err Ch Err St Do Ob Ob
UnaryPlusExpression ::= + Expression 1.87.2 Unary Minus Operator The unary minus operator is defined for the following types:
SByte, Short, Integer, and Long. The result is computed by subtracting the operand from
zero. If integer overflow checking is on and the value of the operand is the maximum negative SByte, Short, Integer, or Long, a System.OverflowException exception is thrown. Otherwise, if the value of the operand is the maximum negative SByte, Short, Integer, or Long, the result is that same value, and the overflow is not reported.
Single and Double. The result is the value of the operand with its sign inverted, including the
values 0 and Infinity. If the operand is NaN, the result is also NaN.
Decimal. The result is computed by subtracting the operand from zero.
UnaryMinusExpression ::= - Expression 1.87.3 Addition Operator The addition operator computes the sum of the two operands. The addition operator is defined for the following types:
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
415
4.3
Inheritance
Byte, SByte, UShort, Short, UInteger, Integer, ULong, and Long. If integer overflow checking is on and the sum is outside the range of the result type, a System.OverflowException
exception is thrown. Otherwise, overflows are not reported, and any significant high-order bits of the result are discarded.
Single and Double. The sum is computed according to the rules of IEEE 754 arithmetic. Decimal. If the resulting value is too large to represent in the decimal format, a System.OverflowException exception is thrown. If the result value is too small to represent in
Note
The System.DateTime type defines overloaded addition operators. Because System.DateTime is equivalent to the intrinsic Date type, these operators is also available on the Date type. Operation Type: Bo Bo Sh SB By Sh U S In UI Lo UL SB By SB SB Sh Sh By Sh Sh Sh Sh Sh U S In In US In US In In In In In In In UI Lo Lo UI Lo UI Lo UI Lo Lo Lo Lo Lo Lo Lo Lo Lo UL De De UL De UL De UL De UL D e De De De De De De De De De Si Si Si Si Si Si Si Si Si Si D o Do Do Do Do Do Do Do Do Do D a Err Err Err Err Err Err Err Err Err Ch Err Err Err Err Err Err Err Err Err St Do Do Do Do Do Do Do Do Do O b Ob Ob Ob Ob Ob Ob Ob Ob Ob 416
4.3
Inheritance
D e Si D o D a Ch St O b
De
Si Si
Do Do Do
Do Do Do St St St
Ob Ob Ob Ob Ob Ob Ob
AdditionOperatorExpression ::= Expression + [ LineTerminator ] Expression 1.87.4 Subtraction Operator The subtraction operator subtracts the second operand from the first operand. The subtraction operator is defined for the following types:
Byte, SByte, UShort, Short, UInteger, Integer, ULong, and Long. If integer overflow checking is on and the difference is outside the range of the result type, a System.OverflowException
exception is thrown. Otherwise, overflows are not reported, and any significant high-order bits of the result are discarded.
Single and Double. The difference is computed according to the rules of IEEE 754 arithmetic. Decimal. If the resulting value is too large to represent in the decimal format, a System.OverflowException exception is thrown. If the result value is too small to represent in
the decimal format, the result is 0. Note The System.DateTime type defines overloaded subtraction operators. Because System.DateTime is equivalent to the intrinsic Date type, these operators is also available on the Date type.
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
417
4.3
Inheritance
Operation Type: Bo Bo Sh SB By Sh U S In UI Lo UL D e Si D o D a Ch St SB By SB SB Sh Sh By Sh Sh Sh Sh Sh U S In In US In US In In In In In In In UI Lo Lo UI Lo UI Lo UI Lo Lo Lo Lo Lo Lo Lo Lo Lo UL De De UL De UL De UL De UL D e De De De De De De De De De De Si Si Si Si Si Si Si Si Si Si Si Si D o Do Do Do Do Do Do Do Do Do Do Do Do D a Err Err Err Err Err Err Err Err Err Err Err Err Err Ch Err Err Err Err Err Err Err Err Err Err Err Err Err Err St Do Do Do Do Do Do Do Do Do Do Do Do Err Err Do O b Ob Ob Ob Ob Ob Ob Ob Ob Ob Ob Ob Ob Err Err Ob
418
4.3
Inheritance
O b
Ob
SubtractionOperatorExpression ::= Expression - [ LineTerminator ] Expression 1.87.5 Multiplication Operator The multiplication operator computes the product of two operands. The multiplication operator is defined for the following types:
Byte, SByte, UShort, Short, UInteger, Integer, ULong, and Long. If integer overflow checking is on and the product is outside the range of the result type, a System.OverflowException
exception is thrown. Otherwise, overflows are not reported, and any significant high-order bits of the result are discarded.
Single and Double. The product is computed according to the rules of IEEE 754 arithmetic. Decimal. If the resulting value is too large to represent in the decimal format, a System.OverflowException exception is thrown. If the result value is too small to represent in
the decimal format, the result is 0. Operation Type: Bo Bo Sh SB By Sh U S In SB By SB SB Sh Sh By Sh Sh Sh Sh Sh U S In In US In US In In In In In In In UI Lo Lo UI Lo UI Lo Lo Lo Lo Lo Lo Lo Lo UL De De UL De UL De D e De De De De De De Si Si Si Si Si Si Si D o Do Do Do Do Do Do D a Err Err Err Err Err Err Ch Err Err Err Err Err Err St Do Do Do Do Do Do O b Ob Ob Ob Ob Ob Ob 419
4.3
Inheritance
UI Lo UL D e Si D o D a Ch St O b
UI
Lo Lo
UL De UL
De De De De
Si Si Si Si Si
Do Do Do Do Do Do
Do Do Do Do Do Do Err Err Do
Ob Ob Ob Ob Ob Ob Err Err Ob Ob
MultiplicationOperatorExpression ::= Expression * [ LineTerminator ] Expression 1.87.6 Division Operators Division operators compute the quotient of two operands. There are two division operators: the regular (floating-point) division operator and the integer division operator. The regular division operator is defined for the following types:
Singleand Double. The quotient is computed according to the rules of IEEE 754 arithmetic. Decimal. If the value of the right operand is zero, a System.Div ideByZeroExcept ion exception
is thrown. If the resulting value is too large to represent in the decimal format, a System.OverflowException exception is thrown. If the result value is too small to represent in the decimal format, the result is zero. The scale of the result, before any rounding, is the
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
420
4.3
Inheritance
closest scale to the preferred scale which will preserve a result equal to the exact result. The preferred scale is the scale of the first operand less the scale of the second operand. According to normal operator resolution rules, regular division purely between operands of types such as Byte, Short, Integer, and Long would cause both operands to be converted to type Decimal. However, when doing operator resolution on the division operator when neither type is Decimal, Double is considered narrower than Decimal. This convention is followed because Double division is more efficient than Decimal division. Operation Type: Bo Bo Do SB By Sh U S In UI Lo UL D e Si SB By Do Do Do Do Do Sh Do Do Do Do U S Do Do Do Do Do In Do Do Do Do Do Do UI Do Do Do Do Do Do Do Lo Do Do Do Do Do Do Do Do UL Do Do Do Do Do Do Do Do Do D e De De De De De De De De De De Si Si Si Si Si Si Si Si Si Si Si Si D o Do Do Do Do Do Do Do Do Do Do Do D a Err Err Err Err Err Err Err Err Err Err Err Ch Err Err Err Err Err Err Err Err Err Err Err St Do Do Do Do Do Do Do Do Do Do Do O b Ob Ob Ob Ob Ob Ob Ob Ob Ob Ob Ob
421
4.3
Inheritance
D o D a Ch St O b
Do
Err Err
Do Err Err Do
Ob Err Err Ob Ob
The integer division operator is defined for Byte, SByte, UShort, Short, UInteger, Integer, ULong, and Long. If the value of the right operand is zero, a System.DivideByZeroException exception is thrown. The division rounds the result towards zero, and the absolute value of the result is the largest possible integer that is less than the absolute value of the quotient of the two operands. The result is zero or positive when the two operands have the same sign, and zero or negative when the two operands have opposite signs. If the left operand is the maximum negative SByte, Short, Integer, or Long, and the right operand is 1, an overflow occurs; if integer overflow checking is on, a System.OverflowException exception is thrown. Otherwise, the overflow is not reported and the result is instead the value of the left operand. Annotation As the two operands for unsigned types will always be zero or positive, the result is always zero or positive. As the result of the expression will always be less than or equal to the largest of the two operands, it is not possible for an overflow to occur. As such integer overflow checking is not performed for integer divide with two unsigned integers. The result is the type as that of the left operand. Operation Type: Bo SB By Sh U S In UI Lo UL D e Si D o D a Ch St O b 422
4.3
Inheritance
Bo Sh SB By Sh U S In UI Lo UL D e Si D o D a Ch St O b
SB SB
Sh Sh By
Sh Sh Sh Sh
In In US In US
In In In In In In
Lo Lo UI Lo UI Lo UI
Lo Lo Lo Lo Lo Lo Lo Lo
Lo Lo UL Lo UL Lo UL Lo UL
Lo Lo Lo Lo Lo Lo Lo Lo Lo Lo
Lo Lo Lo Lo Lo Lo Lo Lo Lo Lo Lo
Lo Lo Lo Lo Lo Lo Lo Lo Lo Lo Lo Lo
Err Err Err Err Err Err Err Err Err Err Err Err Err
Err Err Err Err Err Err Err Err Err Err Err Err Err Err
Lo Lo Lo Lo Lo Lo Lo Lo Lo Lo Lo Lo Err Err Lo
Ob Ob Ob Ob Ob Ob Ob Ob Ob Ob Ob Ob Err Err Ob Ob
423
4.3
Inheritance
FPDivisionOperatorExpression ::= Expression / [ LineTerminator ] Expression IntegerDivisionOperatorExpression ::= Expression \ [ LineTerminator ] Expression 1.87.7 Mod Operator The Mod (modulo) operator computes the remainder of the division between two operands. The Mod operator is defined for the following types:
Byte, SByte, UShort, Short, UInteger, Integer, ULong and Long. The result of x Mod y is the value produced by x (x \ y) * y. If y is zero, a System.DivideByZeroException exception is thrown.
arithmetic.
Decimal. If the value of the right operand is zero, a System.DivideByZeroException exception
is thrown. If the resulting value is too large to represent in the decimal format, a System.OverflowException exception is thrown. If the result value is too small to represent in the decimal format, the result is zero. Operation Type: Bo Bo Sh SB By Sh U S In UI SB By SB SB Sh Sh By Sh Sh Sh Sh Sh U S In In US In US In In In In In In In UI Lo Lo UI Lo UI Lo UI Lo Lo Lo Lo Lo Lo Lo Lo UL De De UL De UL De UL D e De De De De De De De Si Si Si Si Si Si Si Si D o Do Do Do Do Do Do Do D a Err Err Err Err Err Err Err Ch Err Err Err Err Err Err Err St Do Do Do Do Do Do Do O b Ob Ob Ob Ob Ob Ob Ob 424
4.3
Inheritance
Lo UL D e Si D o D a Ch St O b
Lo
De UL
De De De
Si Si Si Si
Do Do Do Do Do
Do Do Do Do Do Err Err Do
Ob Ob Ob Ob Ob Err Err Ob Ob
ModuloOperatorExpression ::= Expression Mod [ LineTerminator ] Expression 1.87.8 Exponentiation Operator The exponentiation operator computes the first operand raised to the power of the second operand. The exponentiation operator is defined for type Double. The value is computed according to the rules of IEEE 754 arithmetic. Operation Type: Bo Bo Do SB SB By Do Do Do Do Sh Do Do U S Do Do In Do Do UI Do Do Lo Do Do UL Do Do D e Do Do Si Do Do D o Do Do D a Err Err Ch Err Err St Do Do O b Ob Ob 425
4.3
Inheritance
By Sh U S In UI Lo UL D e Si D o D a Ch St O b
Do
Do Do
Do Do Do
Do Do Do Do
Do Do Do Do Do
Do Do Do Do Do Do
Do Do Do Do Do Do Do
Do Do Do Do Do Do Do Do
Do Do Do Do Do Do Do Do Do
Do Do Do Do Do Do Do Do Do Do
Err Err Err Err Err Err Err Err Err Err Err
Err Err Err Err Err Err Err Err Err Err Err Err
Do Do Do Do Do Do Do Do Do Do Err Err Do
Ob Ob Ob Ob Ob Ob Ob Ob Ob Ob Err Err Ob Ob
426
4.3
Inheritance
The = operator tests whether the two operands are equal. The <> operator tests whether the two operands are not equal. The < operator tests whether the first operand is less than the second operand. The > operator tests whether the first operand is greater than the second operand. The <= operator tests whether the first operand is less than or equal to the second operand. The >= operator tests whether the first operand is greater than or equal to the second operand. The relational operators are defined for the following types:
Boolean. The operators compare the truth values of the two operands. True is considered to be less than False, which matches with their numeric values. Byte, SByte, UShort, Short, UInteger, Integer, ULong, and Long. The operators compare the
754 standard.
Decimal. The operators compare the numeric values of the two decimal operands. Date. The operators return the result of comparing the two date/time values. Char. The operators return the result of comparing the two Unicode values. String. The operators return the result of comparing the two values using either a binary
comparison or a text comparison. The comparison used is determined by the compilation environment and the Option Compare statement. A binary comparison determines whether the numeric Unicode value of each character in each string is the same. A text comparison does a Unicode text comparison based on the current culture in use on the .NET Framework. When doing a string comparison, a null value is equivalent to the string literal "". Operation Type:
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
427
4.3
Inheritance
Bo Bo Bo SB By Sh U S In UI Lo UL D e Si D o D a Ch St O b
SB By SB SB Sh Sh By
Sh Sh Sh Sh Sh
U S In In US In US
In In In In In In In
UI Lo Lo UI Lo UI Lo UI
Lo Lo Lo Lo Lo Lo Lo Lo Lo
UL De De UL De UL De UL De UL
D e De De De De De De De De De De
Si Si Si Si Si Si Si Si Si Si Si Si
D o Do Do Do Do Do Do Do Do Do Do Do Do
D a Err Err Err Err Err Err Err Err Err Err Err Err Da
Ch Err Err Err Err Err Err Err Err Err Err Err Err Err Ch
St Bo Do Do Do Do Do Do Do Do Do Do Do Da St St
O b Ob Ob Ob Ob Ob Ob Ob Ob Ob Ob Ob Ob Ob Ob Ob Ob
428
4.3
Inheritance
RelationalOperatorExpression ::= Expression = [ LineTerminator ] Expression | Expression < > [ LineTerminator ] Expression | Expression < [ LineTerminator ] Expression | Expression > [ LineTerminator ] Expression | Expression < = [ LineTerminator ] Expression | Expression > = [ LineTerminator ] Expression
429
4.3
Inheritance
comparisons and ordering are based on the numeric Unicode values. If text comparisons are being used, character comparisons and ordering are based on the current locale being used on the .NET Framework. In some languages, special characters in the alphabet represent two separate characters and vice versa. For example, several languages use the character to represent the characters a and e when they appear together, while the characters and O can be used to represent the character . When using text comparisons, the Like operator recognizes such cultural equivalences. In that case, an occurrence of the single special character in either pattern or string matches the equivalent two-character sequence in the other string. Similarly, a single special character in pattern enclosed in brackets (by itself, in a list, or in a range) matches the equivalent two-character sequence in the string and vice versa. In a Like expression where both operands are Nothing or one operand has an intrinsic conversion to String and the other operand is Nothing, Nothing is treated as if it were the empty string literal "". Operation Type: Bo Bo St SB By Sh U S In UI Lo SB By St St St St St Sh St St St St U S St St St St St In St St St St St St UI St St St St St St St Lo St St St St St St St St UL St St St St St St St St D e St St St St St St St St Si St St St St St St St St D o St St St St St St St St D a St St St St St St St St Ch St St St St St St St St St St St St St St St St St O b Ob Ob Ob Ob Ob Ob Ob Ob 430
4.3
Inheritance
UL D e Si D o D a Ch St O b
St
St St
St St St
St St St St
St St St St St
St St St St St St
St St St St St St St
Ob Ob Ob Ob Ob Ob Ob Ob
431
4.3
Inheritance
Bo Bo St SB By Sh U S In UI Lo UL D e Si D o D a Ch St O b
SB By St St St St St
Sh St St St St
U S St St St St St
In St St St St St St
UI St St St St St St St
Lo St St St St St St St St
UL St St St St St St St St St
D e St St St St St St St St St St
Si St St St St St St St St St St St
D o St St St St St St St St St St St St
D a St St St St St St St St St St St St St
Ch St St St St St St St St St St St St St St
St St St St St St St St St St St St St St St St
O b Ob Ob Ob Ob Ob Ob Ob Ob Ob Ob Ob Ob Ob Ob Ob Ob
432
4.3
Inheritance
When the logical operators And and Or are lifted for the type Boolean?, they are extended to encompass three-valued Boolean logic as such:
And evaluates to true if both operands are true; false if one of the operands is false; Nothing
otherwise.
Or evaluates to true if either operand is true; false is both operands are false; Nothing
433
4.3
Inheritance
x = Nothing y = True If x Or y Then ' Will execute End If End Sub End Module
Annotation Ideally, the logical operators And and Or would be lifted using three-valued logic for any type that can be used in a Boolean expression (i.e. a type that implements IsTrue and IsFalse), in the same way that AndAlso and OrElse short circuit across any type that can be used in a Boolean expression. Unfortunately, three-valued lifting is only applied to Boolean?, so user-defined types that desire three-valued logic must do so manually by defining And and Or operators for their nullable version. No overflows are possible from these operations. The enumerated type operators do the bitwise operation on the underlying type of the enumerated type, but the return value is the enumerated type. Not Operation Type: Bo Bo SB SB By By Sh Sh US US In In UI UI Lo Lo UL UL De Lo Si Lo Do Lo Da Err Ch Err St Lo Ob Ob
And, Or, Xor Operation Type: Bo Bo Bo SB SB By SB SB Sh Sh Sh Sh Sh U S In In In In In UI Lo Lo Lo Lo Lo UL Lo Lo D e Lo Lo Si Lo Lo D o Lo Lo D a Err Err Ch Err Err St Bo Lo O b Ob Ob 434
4.3
Inheritance
By Sh U S In UI Lo UL D e Si D o D a Ch St O b
By
Sh Sh
US In US
In In In In
UI Lo UI Lo UI
Lo Lo Lo Lo Lo Lo
UL Lo UL Lo UL Lo UL
Lo Lo Lo Lo Lo Lo Lo Lo
Lo Lo Lo Lo Lo Lo Lo Lo Lo
Lo Lo Lo Lo Lo Lo Lo Lo Lo Lo
Err Err Err Err Err Err Err Err Err Err Err
Err Err Err Err Err Err Err Err Err Err Err Err
Lo Lo Lo Lo Lo Lo Lo Lo Lo Lo Err Err Lo
Ob Ob Ob Ob Ob Ob Ob Ob Ob Ob Err Err Ob Ob
LogicalOperatorExpression ::= Not Expression | Expression And [ LineTerminator ] Expression | Expression Or [ LineTerminator ] Expression | Expression Xor [ LineTerminator ] Expression
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
435
4.3
Inheritance
1.91.1 Short-circuiting Logical Operators The AndAlso and OrElse operators are the short-circuiting versions of the And and Or logical operators. Because of their short circuiting behavior, the second operand is not evaluated at run time if the operator result is known after evaluating the first operand. The short-circuiting logical operators are evaluated as follows: If the first operand in an AndAlso operation evaluates to False, the expression returns False. Otherwise, the second operand is evaluated and a logical And operation is performed on the two results. If the first operand in an OrElse operation evaluates to True, the expression returns True. Otherwise, the second operand is evaluated and a logical Or operation is performed on its two results. The AndAlso and OrElse operators are defined for the type Boolean, or for any type T that overloads the following operators:
Publ i c Shared Operator IsTrue(op As T) As Boolean Publ i c Shared Operator IsFa l se (op As T) As Boolean
When evaluating the AndAlso or OrElse operators, the first operand is evaluated only once, and the second operand is either not evaluated or evaluated exactly once. For example, consider the following code:
Module Test Funct ion TrueValue() As Boolean Console .Wri te ( " True") Return True End Funct ion Funct ion FalseValue() As Boolean Console .Wri te ( " False" ) Return False
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
436
4.3
Inheritance
End Function Sub Main() Console.Write("And:") If FalseValue() And TrueValue() Then End If Console.WriteLine() Console.Write("Or:") If TrueValue() Or FalseValue() Then End If Console.WriteLine() Console.Write("AndAlso:") If FalseValue() AndAlso TrueValue() Then End If Console.WriteLine() Console.Write("OrElse:") If TrueValue() OrElse FalseValue() Then End If Console.WriteLine() End Sub End Module
In the lifted form of the AndAlso and OrElse operators, if the first operand was a null Boolean?, then the second operand is evaluated but the result is always a null Boolean?. Operation Type:
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
437
4.3
Inheritance
Bo Bo Bo SB By Sh U S In UI Lo UL D e Si D o D a Ch St O b
SB By Bo Bo Bo Bo Bo
Sh Bo Bo Bo Bo
U S Bo Bo Bo Bo Bo
In Bo Bo Bo Bo Bo Bo
UI Bo Bo Bo Bo Bo Bo Bo
Lo Bo Bo Bo Bo Bo Bo Bo Bo
UL Bo Bo Bo Bo Bo Bo Bo Bo Bo
D e Bo Bo Bo Bo Bo Bo Bo Bo Bo Bo
Si Bo Bo Bo Bo Bo Bo Bo Bo Bo Bo Bo
D o Bo Bo Bo Bo Bo Bo Bo Bo Bo Bo Bo Bo
D a Err Err Err Err Err Err Err Err Err Err Err Err Err
Ch Err Err Err Err Err Err Err Err Err Err Err Err Err Err
St Bo Bo Bo Bo Bo Bo Bo Bo Bo Bo Bo Bo Err Err Bo
O b Ob Ob Ob Ob Ob Ob Ob Ob Ob Ob Ob Ob Err Err Ob Ob
438
4.3
Inheritance
ShortCircuitLogicalOperatorExpression ::= Expression AndAlso [ LineTerminator ] Expression | Expression OrElse [ LineTerminator ] Expression
4.3
Inheritance
If the shift amount is zero, the result of the operation is identical to the value of the first operand. No overflows are possible from these operations. Operation Type: Bo Sh SB SB By By Sh Sh US US In In UI UI Lo Lo UL UL De Lo Si Lo Do Lo Da Err Ch Err St Lo Ob Ob
ShiftOperatorExpression ::= Expression < < [ LineTerminator ] Expression | Expression > > [ LineTerminator ] Expression
Annotation It is interesting to note that if Option Strict is off, an expression that has a narrowing conversion to Boolean will be accepted without a compile-time error but the language will still prefer an IsTrue operator if it exists. This is because Option Strict only changes what is and isnt accepted by the language, and never changes the actual meaning of an expression. Thus, IsTrue has to always be preferred over a narrowing conversion, regardless of Option Strict.
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
440
4.3
Inheritance
For example, the following class does not define a widening conversion to Boolean. As a result, its use in the I f statement causes a call to the IsTrue operator.
Class MyBool Publ i c Shared Widening Operator CType(b As Boolean) As MyBool ... End Operator Publ i c Shared Narrowing Operator CType(b As MyBool) As Boolean ... End Operator Publ i c Shared Operator IsTrue(b As MyBool) As Boolean ... End Operator Publ i c Shared Operator IsFa l se (b As MyBool) As Boolean ... End Operator End Class Module Test Sub Main() Dim b As New MyBool I f b Then Console .Wri teL ine ("T rue" ) End Sub End Module
If a Boolean expression is typed as or converted to Boolean or Boolean?, then it is true if the value is True and false otherwise. Otherwise, a Boolean expression calls the IsTrue operator and returns True if the operator returned True; otherwise it is false (but never calls the IsFalse operator). In the following example, Integer has a narrowing conversion to Boolean, so a null Integer? has a narrowing conversion to both Boolean? (yielding a null Boolean) and to Boolean (which
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
441
4.3
Inheritance
throws an exception). The narrowing conversion to Boolean? is preferred, and so the value of i as a Boolean expression is therefore False.
Dim i As Integer? = Nothing I f i Then Console .Wri teL ine ( )
442
4.3
Inheritance
A lambda expression begins with the keyword Funct ionor Sub and a parameter list. Parameters in a lambda expression cannot be declared Optional or ParamArray and cannot have attributes. Unlike regular methods, omitting a parameter type for a lambda method does not automatically infer Object. Instead, when a lambda method is reclassified, the omitted parameter types are inferred from the target type. In the previous example, the lambda expression could have been written as Function(x) x * 2, and it would have inferred the type of x to be Integer when the lambda method was used to create an instance of the IntFunc delegate type. Unlike local variable inference, if a lambda method parameter omits a type but has an array or nullable name modifier, a compile-time error occurs. Lambda expressions can either be single-line or multi-line. Single-line Function lambda expressions contain a single expression that represents the value returned from the lambda method. Single-line Sub lambda expressions contain a single statement. For example:
Module Test Sub Do(a() As Integer , act ion As Action(Of Integer ) ) For index As Integer = 0 To a.Length - 1 act ion(a( i ndex)) Next index End Sub Sub Main() Dim a() As Integer = { 1, 2, 3, 4 } Do(a, Sub(x As Integer ) Console .Wri teL ine (x) ) End Sub End Module
Single-line lambda constructs bind less tightly than all other expressions and statements. Thus, for example, Function() x + 5 is equivalent to Function() (x+5) rather than (Function() x) + 5. To avoid ambiguity, a single-line Sub lambda expression may not contain a Dim statement or a label declaration statement. Also, unless it is enclosed in parentheses, a single-line Sub lambda expression may not be immediately followed by a colon :, a member access operator ., a dictionary member access operator ! or an open parenthesis (. It may not contain any block statement (With, SyncLock, IfEndIf, While, For, Do, Using) nor OnError nor Resume. Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved. 443
4.3
Inheritance
Annotation In the lambda expression Funct ion( i ) x=i , the body is interpreted as an expression (which tests whether x and i are equal). But in the lambda expression Sub(i) x=i, the body is interpreted as a statement (which assigns i to x). A multi-line lambda expression contains a statement block and must end with an appropriate End statement (i.e. End Function or End Sub). As with regular methods, a multiline lambda methods Function or Sub statement and End statements must be on their own lines. For example:
' Error : Funct ion statement must be on i t s own l i ne ! Dim x = Sub(x As Integer ) : Console .Wri teL ine (x) : End Sub ' OK Dim y = Sub(x As Integer ) Console .Wri teL ine (x) End Sub
Multi-line Function lambda expressions can declare a return type but cannot put attributes on it. If a multi-line Function lambda expression does not declare a return type but the return type can be inferred from the context in which the lambda expression is used (see Section 1.75.1 on Expression Reclassification), then that return type is used. Otherwise the return type of the function is calculated from the dominant type of the expressions in all of the Return statements in the statement block. If there are no Return statements, or if there is no dominant type among them, and strict semantics are being used, a compile-time error occurs; otherwise the return type is implicitly Object. Note that the return type is calculated from all Return statements, even if they are not reachable. For example:
' Return type i s Double Dim x = Funct ion( ) Return 10 Return 10.50 End Funct ion
There is no implicit return variable, as there is no name for the variable. The statement blocks inside multi-line lambda expressions have the following restrictions: Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved. 444
4.3
Inheritance
On Error and Resume statements are not allowed, although Try statements are allowed.
Static locals cannot be declared in multi-line lambda expressions. It is not possible to branch into or out of the statement block of a multi-line lambda expression, although the normal branching rules apply within it. For example:
Label1: Dim x = Sub()
' Error : Cannot branch out Goto Label1 ' OK: Wholly with in the lamba. Goto Label2:
A lambda expression is roughly equivalent to an anonymous method declared on the containing type. The initial example is roughly equivalent to:
Module Test Delegate Funct ion IntFunc(x As Integer ) As Integer Sub Apply(a( ) As Integer , func As IntFunc) For index As Integer = 0 To a.Length - 1 a( index) = func(a( index) ) Next index End Sub Funct ion $Lambda1(x As Integer ) As Integer Return x * 2 End Funct ion Sub Main() Dim a() As Integer = { 1, 2, 3, 4 }
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
445
4.3
Inheritance
Apply(a, AddressOf $Lambda1) For Each value In a Console.Write(value & " ") Next value End Sub End Module
LambdaExpression ::= SingleLineLambda | MultiLineLambda SingleLineLambda ::= Funct ion [ OpenParenthesis [ ParametertList ] CloseParenthesis ] Expression | Sub [ OpenParenthesis [ ParametertList ] CloseParenthesis ] Statement MultiLineLambda ::= MultiLineFunctionLambda | MultiLineSubLambda MultiLineFunctionLambda ::= Funct ion [ OpenParenthesis [ ParametertList ] CloseParenthesis ] [ As TypeName ] LineTerminator Block
End Function
1.94.1 Closures Lambda expressions have access to all of the variables in scope, including local variables or parameters defined in the containing method and lambda expressions. When a lambda expression refers to a local variable or parameter, the lambda expression captures the variable being referred to into a closure. A closure is an object that lives on the heap Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved. 446
4.3
Inheritance
instead of on the stack, and when a variable is captured, all references to the variable are redirected to the closure. This enables lambda expressions to continue to refer to local variables and parameters even after the containing method is complete. For example:
Module Test Delegate Function D() As Integer Function M() As D Dim x As Integer = 10 Return Function() x End Function Sub Main() Dim y As D = M() ' Prints 10 Console.WriteLine(y()) End Sub End Module
447
4.3
Inheritance
End Function Sub Main() Dim y As D = M() ' Prints 10 Console.WriteLine(y()) End Sub End Module
A closure captures a new copy of a local variable each time it enters the block in which the local variable is declared, but the new copy is initialized with the value of the previous copy, if any. For example:
Module Test Delegate Function D() As Integer Function M() As D() Dim a(9) As D For i As Integer = 0 To 9 Dim x a(i) = Function() x x += 1 Next i Return a End Function Sub Main() Dim y() As D = M() For i As Integer = 0 To 9 Console.Write(y(i)() & " ") Next i End Sub End Module
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
448
4.3
Inheritance
prints
1 2 3 4 5 6 7 8 9 10
instead of
9 9 9 9 9 9 9 9 9 9
Because closures have to be initialized when entering a block, it is not allowed to GoTo into a block with a closure from outside of that block, although it is allowed to Resume into a block with a closure. For example:
Module Test Sub Main() Dim a = 10 I f a = 10 Then L1: Dim x = Funct ion( ) a ' Val id , source i s with in block GoTo L2 End I f ' ERROR: target i s ins ide block with closure GoTo L1 End Sub End Module
L2:
Because they cannot be captured into a closure, the following cannot appear inside of a lambda expression: Reference parameters. Instance expressions (Me , MyClass, MyBase), if the type of Me is not a class. The members of an anonymous type-creation expression, if the lambda expression is part of the expression. For example:
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
449
4.3
Inheritance
' Error: Lambda cannot refer to anonymous type field Dim x = New With { .a = 12, .b = Function() .a } ReadOnly instance variables in instance constructors or ReadOnly shared variables in shared
constructors where the variables are used in a non-value context. For example:
Class C1 ReadOnly F1 As Integer Sub New() ' Val id , doesnt modify F1 Dim x = Funct ion( ) F1 ' Error , tr i es to modify F1 Dim f = Funct ion( ) ModifyValue(F1) End Sub Sub ModifyValue(ByRef x As Integer ) End Sub End Class
A query expression must start with a From or an Aggregate operator and can end with any query operator. The result of a query expression is classified as a value; the result type of the expression depends on the result type of the last query operator in the expression.
450
4.3
Inheritance
QueryExpression ::= FromOrAggregateQueryOperator | QueryExpression QueryOperator FromOrAggregateQueryOperator ::= FromQueryOperator | AggregateQueryOperator JoinOrGroupJoinQueryOperator ::= JoinQueryOperator | GroupJoinQueryOperator QueryOperator ::= FromQueryOperator | AggregateQueryOperator | SelectQueryOperator | DistinctQueryOperator | WhereQueryOperator | OrderByQueryOperator | PartitionQueryOperator | LetQueryOperator | GroupByQueryOperator | JoinOrGroupJoinQueryOperator 1.95.1 Range Variables Some query operators introduce a special kind of variable called a range variable. Range variables are not real variables; instead, they represent the individual values during the evaluation of the query over the input collections. Range variables are scoped from the introducing query operator to the end of a query expression, or to a query operator such as Select that hides them. For example, in the following query
Dim waCusts = _ From cust As Customer In Customers _ Where cust .S ta te = "WA"
the From query operator introduces a range variable cust typed as Customer that represents each customer in the Customers collection. The following Where query operator then refers to the range variable cust in the filter expression to determine whether to filter an individual customer out of the resulting collection.
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
451
4.3
Inheritance
There are two types of range variables: collection range variables and expression range variables. Collection range variables take their values from the elements of the collections being queried. The collection expression in a collection range variable declaration must be classified as a value whose type is queryable. If the type of a collection range variable is omitted, it is inferred to be the element type of the collection expression, or Object if the collection expression does not have an element type (i.e. only defines a Cast method). If the collection expression is not queryable (i.e. the element type of the collection cannot be inferred), a compile-time error results. An expression range variable is a range variable whose value is calculated by an expression rather than a collection. In the following example, the Select query operator introduces an expression range variable named ci tySta te calculated from two fields:
Dim ci tyStates = _ From cust As Customer In Customers _ Select ci tyState = cust .C i t y & " , " & cust .S ta te _ Where ci tyState . Length( ) < 10
An expression range variable is not required to reference another range variable, although such a variable may be of dubious value. The expression assigned to an expression range variable must be classified as a value and must be implicitly convertible to the type of the range variable, if given. Only in a Let operator may an expression range variable have its type specified. In other operators, or if its type is not specified, then local variable type inference is used to determine the type of the range variable. A range variable must follow the rules for declaring local variables in respect to shadowing. Thus, a range variable cannot hide the name of a local variable or parameter in the enclosing method or another range variable (unless the query operator specifically hides all current range variables in scope). CollectionRangeVariableDeclarationList ::= CollectionRangeVariableDeclaration | CollectionRangeVariableDeclarationList Comma CollectionRangeVariableDeclaration
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
452
4.3
Inheritance
CollectionRangeVariableDeclaration ::= Identifier [ As TypeName ] In [ LineTerminator ] Expression ExpressionRangeVariableDeclarationList ::= ExpressionRangeVariableDeclaration | ExpressionRangeVariableDeclarationList Comma ExpressionRangeVariableDeclaration ExpressionRangeVariableDeclaration ::= [ Identifier [ As TypeName ] Equals ] Expression 1.95.2 Queryable Types Query expressions are implemented by translating the expression into calls to well-known methods on a collection type. These well-defined methods define the element type of the queryable collection as well as the result types of query operators executed on the collection. Each query operator specifies the method or methods that the query operator is generally translated into, although the specific translation is implementation dependent. The methods are given in the specification using a general format that looks like:
Funct ion Select (se lec tor As Func(Of T, R)) As CR
The following applies to the methods: The method must be an instance or extension member of the collection type and must be accessible. The method may be generic, provided that is possible to infer all type arguments. The method may be overloaded, in which case overload resolution is used to determine the exactly method to use. Another delegate type may be used in place of the delegate Func type, provided that it has the same signature, including return type, as the matching Func type. The type System.Linq.Express ions .Express ion(Of D) be used in place of the delegate Func may type, provided that D is a delegate type that has the same signature, including return type, as the matching Func type.
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
453
4.3
Inheritance
The type T represents the element type of the input collection. All of the methods defined by a collection type must have the same input element type for the collection type to be queryable. The type S represents the element type of the second input collection in the case of query operators that perform joins. The type K represents a key type in the case of query operators that have a set of range variables that act as keys. The type N represents a type that is used as a numeric type (although it could still be a user-defined type and not an intrinsic numeric type). The type B represents a type that can be used in a Boolean expression. The type R represents the element type of the result collection, if the query operator produces a result collection. R depends on the number of range variables in scope at the conclusion of the query operator. If a single range variable is in scope, then R is the type of that range variable. In the example
Dim custNames = _ From c In Customers _ Select c.Name
the result of the query will be a collection type with an element type of String. If multiple range variables are in scope, then R is an anonymous type that contains all of the range variables in scope as Key fields. In the example:
Dim custAndOrderNames = _ From c In Customers, o In c.Orders _ Select Name = c.Name, ProductName = o.ProductName
the result of the query will be a collection type with an element type of an anonymous type with a read-only property named Name of type String and a read-only property named ProductName of type String. Within a query expression, anonymous types generated to contain range variables are transparent, which means that range variables are always available without qualification. For example, in the previous example the range variables c and o could be Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved. 454
4.3
Inheritance
accessed without qualification in the Select query operator, even though the input collections element type was an anonymous type. The type CX represents a collection type, not necessarily the input collection type, whose element type is some type X. A queryable collection type must satisfy one of the following conditions, in order of preference: It must define a conforming Select method. It must have have one of the following methods
Funct ion AsEnumerable( ) As CT Funct ion AsQueryable( ) As CT
which can be called to obtain a queryable collection. If both methods are provided, AsQueryable is preferred over AsEnumerable . It must have a method
Funct ion Cast(Of T)( ) As CT
which can be called with the type of the range variable to produce a queryable collection. Because determining the element type of a collection occurs independently of an actual method invocation, the applicability of specific methods cannot be determined. Thus, when determining the element type of a collection if there are instance methods that match wellknown methods, then any extension methods that match well-known methods are ignored. Query operator translation occurs in the order in which the query operators occur in the expression. It is not necessary for a collection object to implement all of the methods needed by all the query operators, although every collection object must at least support the Select query operator. If a needed method is not present, a compile-time error occurs. When binding well-known method names, non-methods are ignored for the purpose of multiple inheritance in interfaces and extension method binding, although shadowing semantics still apply. For example:
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
455
4.3
Inheritance
Class Q1 Public Function [Select](selector As Func(Of Integer, Integer)) As Q1 End Function End Class Class Q2 Inherits Q1 Public [Select] As Integer End Class Module Test Sub Main() Dim qs As New Q2() ' Error: Q2.Select still hides Q1.Select Dim zs = From q In qs Select q End Sub End Module
1.95.3 Default Query Indexer Every queryable collection type whose element type is T and does not already have a default property is considered to have a default property of the following general form:
Publ i c ReadOnly Defaul t Property I tem(index As Integer ) As T Get Return Me.ElementAtOrDefaul t ( i ndex) End Get End Property
The default property can only be referred to using the default property access syntax; the default property cannot be referred to by name. For example:
Dim customers As IEnumerable(Of Customer) = . . . Dim customerThree = customers(2)
456
4.3
Inheritance
If the collection type does not have an ElementAtOrDefaul tmember, a compile-time error will occur. 1.95.4 From Query Operator The From query operator introduces a collection range variable that represents the individual members of a collection to be queried. For example, the query expression:
From c As Customer In Customers . . .
When a From query operator declares multiple collection range variables or is not the first From query operator in the query expression, each new collection range variable is cross joined to the existing set of range variables. The result is that the query is evaluated over the cross-product of all the elements in the joined collections. For example, the expression:
From c In Customers _ From e In Employees _ ...
457
4.3
Inheritance
The range variables introduced in previous query operators can be used in a later From query operator. For example, in the following query expression the second From query operator refers to the value of the first range variable:
From c As Customer In Customers _ From o As Order In c.Orders _ Select c.Name, o
Multiple range variables in a From query operator or multiple From query operators are only supported if the collection type contains one or both of the following methods:
Funct ion SelectMany(se lector As Func(Of T, CR)) As CR Funct ion SelectMany(se lector As Func(Of T, CS) , _ resul t sSe lec tor As Func(Of T, S, R)) As
The code
Dim xs() As Integer = . . . Dim ys() As Integer = . . . Dim zs = From x In xs, y In ys . . .
is generally translated to
Dim xs() As Integer = . . . Dim ys() As Integer = . . . Dim zs = _ xs.Se lectMany( _ Funct ion(x As Integer ) ys, _ Funct ion(x As Integer , y As Integer ) New With {x, y}). . .
Note From is not a reserved word. FromQueryOperator ::= [ LineTerminator ] From [ LineTerminator ] CollectionRangeVariableDeclarationList
458
4.3
Inheritance
1.95.5 Join Query Operator The Jo inquery operator joins existing range variables with a new collection range variable, producing a single collection whose elements have been joined together based on an equality expression. For example:
Dim customersAndOrders = _ From cust In Customers _ Jo in ord In Orders On cust . ID Equals ord.CustomerID
The equality expression is more restricted than a regular equality expression: Both expressions must be classified as a value. Both expressions must reference at least one range variable. The range variable declared in the join query operator must be referenced by one of the expressions, and that expression must not reference any other range variables. If the types of the two expressions are not the exact same type, then If the equality operator is defined for the two types, both expressions are implicitly convertible to it, and it is not Object, then convert both expressions to that type. Otherwise, if there is a dominant type that both expressions can be implicitly converted to, then convert both expressions to that type. Otherwise, a compile-time error occurs. The expressions are compared using hash values (i.e. by calling GetHashCode()) rather than by using equality operators for efficiency. A Jo inquery operator may do multiple joins or equality conditions in the same operator. A Join query operator is only supported if the collection type contains a method:
Funct ion Jo in ( i nner As CS, _ outerSelector As Func(Of T, K), _ innerSelector As Func(Of S, K), _ resul tSe lec tor As Func(Of T, S, R)) As CR
The code
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
459
4.3
Inheritance
Dim xs() As Integer = ... Dim ys() As Integer = ... Dim zs = From x In xs _ Join y In ys On x Equals y _ ...
is generally translated to
Dim xs() As Integer = ... Dim ys() As Integer = ... Dim zs = _ xs.Join( _ ys, _ Function(x As Integer) x, _ Function(y As Integer) y, _ Function(x As Integer, y As Integer) New With {x, y})...
Note Jo in On and Equals are not reserved words. , JoinQueryOperator ::= [ LineTerminator ] Join [ LineTerminator ] CollectionRangeVariableDeclaration [ JoinOrGroupJoinQueryOperator ] [ LineTerminator ] On [ LineTerminator ] JoinConditionList JoinConditionList ::= JoinCondition | JoinConditionList And [ LineTerminator ] JoinCondition JoinCondition ::= Expression Equals [ LineTerminator ] Expression 1.95.6 Let Query Operator The Let query operator introduces an expression range variable. This allows calculating an intermediate value once that will be used multiple times in later query operators. For example:
Dim taxedPr ices = _ From o In Orders _ Let tax = o.Pr i ce * 0.088 _
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
460
4.3
Inheritance
Where tax > 3.50 _ Select o.Price, tax, total = o.Price + tax
A Let query operator is only supported if the collection type contains a method:
Funct ion Select (se lec tor As Func(Of T, R)) As CR
The code
Dim xs() As Integer = . . . Dim zs = From x In xs _ Let y = x * 10 _ ...
is generally translated to
Dim xs() As Integer = . . . Dim zs = _ xs.Se lect (Funct ion(x As Integer ) New With {x, .y = x * 10}). . .
LetQueryOperator ::= [ LineTerminator ] Let [ LineTerminator ] ExpressionRangeVariableDeclarationList 1.95.7 Select Query Operator The Select query operator is like the Let query operator in that it introduces expression range variables; however, a Select query operator hides the currently available range variables instead of adding to them. Also, the type of an expression range variable introduced by a Select query operator is always inferred using local variable type inference rules; an explicit type cannot be specified, and if no type can be inferred, a compile-time error occurs. For example, in the query:
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
461
4.3
Inheritance
Dim smiths = _ From cust In Customers _ Select name = cust.name _ Where name.EndsWith("Smith")
the Where query operator only has access to the name range variable introduced by the Select operator; if the Where operator had tried to reference cust, a compile-time error would have occurred. Instead of explicitly specifying the names of the range variables, a Select query operator can infer the names of the range variables, using the same rules as anonymous type object creation expressions. For example:
Dim custAndOrderNames = _ From cust In Customers, ord In cust .Orders _ Select cust .name, ord.ProductName _ Where name.EndsWith("Smith" )
If the name of the range variable is not supplied and a name cannot be inferred, a compiletime error occurs. If the Select query operator contains only a single expression, no error occurs if a name for that range variable cannot be inferred but the range variable is nameless. For example:
Dim custAndOrderNames = _ From cust In Customers, ord In cust .Orders _ Select cust .Name & " bought " & ord.ProductName _ Take 10
If there is an ambiguity in a Select query operator between assigning a name to a range variable and an equality expression, the name assignment is preferred. For example:
Dim badCustNames = _ From c In Customers _ Let name = " John Smith" _ Select name = c.Name ' Creates a range var iab le named "name" Dim goodCustNames = _ From c In Customers _
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
462
4.3
Inheritance
Each expression in the Select query operator must be classified as a value. A Select query operator is supported only if the collection type contains a method:
Funct ion Select (se lec tor As Func(Of T, R)) As CR
The code
Dim xs() As Integer = . . . Dim zs = From x In xs _ Select x, y = x * 10 _ ...
is generally translated to
Dim xs() As Integer = . . . Dim zs = _ xs.Se lect (Funct ion(x As Integer ) New With {x, .y = x * 10}). . .
SelectQueryOperator ::= [ LineTerminator ] Select [ LineTerminator ] ExpressionRangeVariableDeclarationList 1.95.8 Distinct Query Operator The Dist inc t query operator restricts the values in a collection only to those with distinct values, as determined by comparing the element type for equality. For example, the query:
Dim dist i nc tCustomerPr i ce = _ From cust In Customers, ord In cust .Orders _ Select cust .Name, ord.Pr i ce _ Dist i nc t
will only return one row for each distinct pairing of customer name and order price, even if the customer has multiple orders with the same price. A Dist i nc t query operator is supported only if the collection type contains a method:
Funct ion Dist i nc t ( ) As CT
463
4.3
Inheritance
The code
Dim xs() As Integer = ... Dim zs = From x In xs _ Distinct _ ...
is generally translated to
Dim xs() As Integer = ... Dim zs = xs.Distinct()...
Note Dist inc t not a reserved word. is DistinctQueryOperator ::= [ LineTerminator ] Dist i nc t [ LineTerminator ] 1.95.9 Where Query Operator The Where query operator restricts the values in a collection to those that satisfy a given condition. A Where query operator takes a Boolean expression that is evaluated for each set of range variable values; if the value of the expression is true, then the values appear in the output collection, otherwise the values are skipped. For example, the query expression:
From cust In Customers, ord In Orders _ Where cust . ID = ord.CustomerID _ ...
A Where query operator is supported only if the collection type contains a method:
Funct ion Where(predicate As Func(Of T, B)) As CT
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
464
4.3
Inheritance
The code
Dim xs() As Integer = ... Dim zs = From x In xs _ Where x < 10 _ ...
is generally translated to
Dim xs() As Integer = ... Dim zs = _ xs.Where(Function(x As Integer) x < 10)...
Note Where is not a reserved word. WhereQueryOperator ::= [ LineTerminator ] Where [ LineTerminator ] BooleanExpression 1.95.10 Partition Query Operators The Take query operator results in the first n elements of a collection. When used with the While modifier, the Take operator results in the first n elements of a collection that satisfy a Boolean expression. The Skip operator skips the first n elements of a collection and then returns the remainder of the collection. When used in conjunction with the While modifier, the Skip operator skips the first n elements of a collection that satisfy a Boolean expression and then returns the rest of the collection. The expressions in a Take or Skip query operator must be classified as a value. A Take query operator is supported only if the collection type contains a method:
Funct ion Take(count As N) As CT
A Skip query operator is supported only if the collection type contains a method:
Funct ion Skip(count As N) As CT
A Take While query operator is supported only if the collection type contains a method:
Funct ion TakeWhile(pred ica te As Func(Of T, B)) As CT
A Skip While query operator is supported only if the collection type contains a method:
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
465
4.3
Inheritance
The code
Dim xs() As Integer = ... Dim zs = From x In xs _ Skip 10 _ Take 5 _ Skip While x < 10 _ Take While x > 5 _ ...
is generally translated to
Dim xs() As Integer = ... Dim zs = _ xs.Skip(10). _ Take(5). _ SkipWhile(Function(x) x < 10). _ TakeWhile(Function(x) x > 5)...
Note Take and Skip are not reserved words. PartitionQueryOperator ::= [ LineTerminator ] Take [ LineTerminator ] Take [ LineTerminator ] Skip [ LineTerminator ] Skip [ LineTerminator ] Expression | While [ LineTerminator ] BooleanExpression | [ LineTerminator ] Expression | While [ LineTerminator ] BooleanExpression
1.95.11 Order By Query Operator The Order By query operator orders the values that appear in the range variables. An Order By query operator takes expressions that specify the key values that should be used to order the iteration variables. For example, the following query returns products sorted by price:
Dim productsByPr ice = _ From p In Products _
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
466
4.3
Inheritance
An ordering can be marked as Ascending, in which case smaller values come before larger values, or Descending, in which case larger values come before smaller values. The default for an ordering if none is specified is Ascending. For example, the following query returns products sorted by price with the most expensive product first:
Dim productsByPr iceDesc = _ From p In Products _ Order By p.Pr i ce Descending _ Select p.Name
The Order By query operator may specify multiple expressions for ordering, in which case the collection is ordered in a nested manner. For example, the following query orders customers by state, then by city within each state and then by ZIP code within each city:
Dim customersByLocat ion = _ From c In Customers _ Order By c.State , c.C i ty , c.ZIP _ Select c.Name, c.State , c.C i ty , c.ZIP
The expressions in an Order By query operator must be classified as a value. An Order By query operator is supported only if the collection type contains one or both of the following methods:
Funct ion OrderBy(keySelector As Func(Of T, K)) As CT Funct ion OrderByDescending(keySelector As Func(Of T, K)) As CT
The return type CT must be an ordered collection. An ordered collection is a collection type that contains one or both of the methods:
Funct ion ThenBy(keySelector As Func(Of T, K)) As CT Funct ion ThenByDescending(keySelector As Func(Of T, K)) As CT
The code
Dim xs() As Integer = . . . Dim zs = From x In xs _
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
467
4.3
Inheritance
is generally translated to
Dim xs() As Integer = ... Dim zs = _ xs.OrderBy(Function(x) x).ThenByDescending(Function(x) x Mod 2)...
Annotation Because query operators simply map syntax to methods that implement a particular query operation, order preservation is not dictated by the language and is determined by the implementation of the operator itself. This is very similar to user-defined operators in that the implementation to overload the addition operator for a user-defined numeric type may not perform anything resembling an addition. Of course, to preserve predictability, implementing something that does not match user expectations is not recommended. Note Order and By are not reserved words. OrderByQueryOperator ::= [ LineTerminator ] Order By [ LineTerminator ] OrderExpressionList OrderExpressionList ::= OrderExpression | OrderExpressionList Comma OrderExpression OrderExpression ::= Expression [ Ordering ] Ordering ::= Ascending | Descending 1.95.12 Group By Query Operator The Group By query operator groups the range variables in scope based on one or more expressions, and then produces new range variables based on those groupings. For example, the following query groups all customers by State, and then computes the count and average age of each group:
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
468
4.3
Inheritance
Dim averageAges = _ From cust In Customers _ Group By cust.State _ Into Count(), Average(cust.Age)
The Group By query operator has three clauses: the optional Group clause, the By clause, and the Into clause. The Group clause has the same syntax and effect as a Select query operator, except that it only affects the range variables available in the Into clause and not the By clause. For example:
Dim averageAges = _ From cust In Customers _ Group cust .Age By cust .S ta te _ Into Count() , Average(Age)
The By clause declares expression range variables that are used as key values in the grouping operation. The Into clause allows the declaration of expression range variables that calculate aggregations over each of the groups formed by the By clause. Within the Into clause, the expression range variable can only be assigned an expression which is a method invocation of an aggregate function. An aggregate function is a function on the collection type of the group (which may not necessarily be the same collection type of the original collection) which looks like either of the following methods:
Funct ion <name>() As <type> Funct ion <name>(selector As Func(Of T, R)) As R
If an aggregate function takes a delegate argument, then the invocation expression can have an argument expression that must be classified as a value. The argument expression can use the range variables that are in scope; within the call to an aggregate function, those range variables represent the values in the group being formed, not all of the values in the collection. For example, in the original example in this section the Average function calculates the average of the customers ages per state rather than for all of the customers together. All collection types are considered to have the aggregate function Group defined on it, which takes no parameters and simply returns the group. Other standard aggregate functions that a collection type may provide are:
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
469
4.3
Inheritance
Count and LongCount , which return the count of the elements in the group or the count of the elements in the group that satisfy a Boolean expression. Count and LongCount are
Sum, which returns the sum of an expression across all the elements in the group. Sum is
expression and is supported only if the collection type contains one of the methods:
Funct ion Any() As B Funct ion Any(predicate As Func(Of T, B)) As B
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
470
4.3
Inheritance
Al l, which determines whether a Boolean expression is true for all elements in the group. Al l
returns a value that can be used in a Boolean expression and is supported only if the collection type contains a method:
Funct ion Al l (p red icate As Func(Of T, B)) As B
After a Group By query operator, the range variables previously in scope are hidden, and the range variables introduced by the By and Into clauses are available. A Group By query operator is supported only if the collection type contains the method:
Funct ion GroupBy(keySelector As Func(Of T, K), _ resul tSe lec tor As Func(Of K, CT, R)) As CR
Range variable declarations in the Group clause are supported only if the collection type contains the method:
Funct ion GroupBy(keySelector As Func(Of T, K), _ elementSelector As Func(Of T, S), _ resul tSe lec tor As Func(Of K, CS, R)) As CR
The code
Dim xs() As Integer = . . . Dim zs = From x In xs _ Group y = x * 10, z = x / 10 By evenOdd = x Mod 2 _ Into Sum(y), Average(z) _ ...
is generally translated to
Dim xs() As Integer = . . . Dim zs = _ xs.GroupBy( _ Funct ion(x As Integer ) x Mod 2, _ Funct ion(x As Integer ) New With {.y = x * 10, .z = x / 10}, _ Funct ion(evenOdd, group) New With { _ evenOdd, _ .Sum = group.Sum(Funct ion(e ) e.y) , _ .Average = group.Average(Funct ion(e) e.z)}) . . .
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
471
4.3
Inheritance
Note Group, By, and Into are not reserved words. GroupByQueryOperator ::= [ LineTerminator ] Group [ [ LineTerminator ] ExpressionRangeVariableDeclarationList ] [ LineTerminator ] By [ LineTerminator ] ExpressionRangeVariableDeclarationList [ LineTerminator ] Into [ LineTerminator ] ExpressionRangeVariableDeclarationList 1.95.13 Aggregate Query Operator The Aggregate query operator performs a similar function as the Group By operator, except it allows aggregating over groups that have already been formed. Because the group has already been formed, the Into clause of an Aggregate query operator does not hide the range variables in scope (in this way, Aggregate is more like a Let, and Group By is more like a Select). For example, the following query aggregates the total of all the orders placed by customers in Washington:
Dim orderTota ls = _ From cust In Customers _ Where cust .S ta te = "WA" _ Aggregate order In cust .Orders _ Into Sum(order .Tota l )
The result of this query is a collection whose element type is an anonymous type with a property named cust typed as Customer and a property named Sum typed as Integer. Unlike Group By, additional query operators can be placed between the Aggregate and Into clauses. Between an Aggregate clause and the end of the Into clause, all range variables in scope, including those declared by the Aggregate clause can be used. For example, the following query aggregates the sum total of all the orders placed by customers in Washington before 2006:
Dim orderTota ls = _ From cust In Customers _ Where cust .S ta te = "WA" _
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
472
4.3
Inheritance
Aggregate order In cust.Orders _ Where order.Date <= #01/01/2006# _ Into Sum = Sum(order.Total)
The Aggregate operator can also be used to start a query expression. In this case, the result of the query expression will be the single value computed by the Into clause. For example, the following query calculates the sum of all the order totals before January 1st, 2006:
Dim ordersTota l = _ Aggregate order In Orders _ Where order .Date <= #01/01/2006# _ Into Sum(order .Tota l )
The result of the query is a single Integervalue. An Aggregate query operator is always available (although the aggregate function must be also be available for the expression to be valid). The code
Dim xs() As Integer = . . . Dim zs = _ Aggregate x In xs _ Where x < 5 _ Into Sum()
is generally translated to
Dim xs() As Integer = . . . Dim zs = _ xs.Where(Funct ion(x) x < 5).Sum()
Note Aggregate and Into are not reserved words. AggregateQueryOperator ::= [ LineTerminator ] Aggregate [ LineTerminator ] CollectionRangeVariableDeclaration [ QueryOperator+ ] [ LineTerminator ] Into [ LineTerminator ] ExpressionRangeVariableDeclarationList 473
4.3
Inheritance
1.95.14 Group Join Query Operator The Group Jo inquery operator combines the functions of the Join and Group By query operators into a single operator. Group Join joins two collections based on matching keys extracted from the elements, grouping together all of the elements on the right side of the join that match a particular element on the left side of the join. Thus, the operator produces a set of hierarchical results. For example, the following query produces elements that contain a single customers name, a group of all of their orders, and the total amount of all of those orders:
Dim custsWithOrders = _ From cust In Customers _ Group Jo in order In Orders On cust . ID Equals order .CustomerID _ Into Orders = Group, OrdersTota l = Sum(order .Tota l ) _ Select cust .Name, Orders, OrdersTota l
The result of the query is a collection whose element type is an anonymous type with three properties: Name, typed as String, Orders typed as a collection whose element type is Order, and OrdersTotal, typed as Integer. A Group Join query operator is supported only if the collection type contains the method:
Funct ion GroupJo in( i nner As CS, _ outerSelector As Func(Of T, K), _ innerSelector As Func(Of S, K), _ resul tSe lec tor As Func(Of T, CS, R)) As
The code
Dim xs() As Integer = . . . Dim ys() As Integer = . . . Dim zs = From x In xs _ Group Jo in y in ys On x Equals y _ Into g = Group _ ...
is generally translated to
Dim xs() As Integer = . . . Dim ys() As Integer = . . .
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
474
4.3
Inheritance
Dim zs = _ xs.GroupJoin( _ ys, _ Function(x As Integer) x, _ Function(y As Integer) y, _ Function(x, group) New With {x, .g = group})...
Note Group, Jo in and Into are not reserved words. , GroupJoinQueryOperator ::= [ LineTerminator ] Group Join [ LineTerminator ] CollectionRangeVariableDeclaration [ JoinOrGroupJoinQueryOperator ] [ LineTerminator ] On [ LineTerminator ] JoinConditionList [ LineTerminator ] Into [ LineTerminator ] ExpressionRangeVariableDeclarationList
4.3
Inheritance
type of the first operand when determining the dominant type for the result type of the expression. For example:
Module Test Sub Main() Dim x?, y As Integer Dim a?, b As Long a = If(x, a) y = If(x, 0) End Sub End Module ' Result type: Long? ' Result type: Integer
In both forms of the expression, if an operand is Nothing, its type is not used to determine the dominant type. In the case of the expression I f (<express ion>, Nothing, Nothing) , the dominant type is considered to be Object. ConditionalExpression ::= If OpenParenthesis BooleanExpression Comma Expression Comma Expression CloseParenthesis | If OpenParenthesis Expression Comma Expression CloseParenthesis
476
4.3
Inheritance
An XML literal expression can take the form of an XML document, an XML element, an XML processing instruction, an XML comment, or a CDATA section. Annotation This specification contains only enough of a description of XML to describe the behavior of the Visual Basic language. More information on XML can be found at http://www.w3.org/TR/REC-xml/. XMLLiteralExpression ::= XMLDocument | XMLElement | XMLProcessingInstruction | XMLComment | XMLCDATASection 1.97.1 Lexical rules XML literal expressions are interpreted using the lexical rules of XML instead of the lexical rules of regular Visual Basic code. The two sets of rules generally differ in the following ways: White space is significant in XML. As a result, the grammar for XML literal expressions explicitly states where white space is allowed. Whitespace is not preserved, except when it occurs in the context of character data within an element. For example:
' The following element preserves no whitespace Dim e1 = _ <customer> <name>Bob</> </> ' The following element preserves all of the whitespace Dim e2 = _ <customer>
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
477
4.3
Inheritance
Bob </>
XML end-of-line whitespace is normalized according to the XML specification. XML is case-sensitive. Keywords must match casing exactly, or else a compile-time error will occur. Line terminators are considered white space in XML. As a result, no line-continuation characters are needed in XML literal expressions. XML does not accept full-width characters. If full-width characters are used, a compile-time error will occur. XMLCharacter ::= < Unicode tab character (0x0009) > | < Unicode linefeed character (0x000A) > | < Unicode carriage return character (0x000D) > | < Unicode characters 0x0020 0xD7FF > | < Unicode characters 0xE000 0xFFFD > | < Unicode characters 0x10000 0x10FFFF > XMLString ::= XMLCharacter+ XMLWhitespace ::= XMLWhitespaceCharacter+ XMLWhitespaceCharacter ::= < Unicode carriage return character (0x000D) > | < Unicode linefeed character (0x000A) > | < Unicode space character (0x0020) > | < Unicode tab character (0x0009) > XMLNameCharacter ::= XMLLetter | XMLDigit | . | - | _ | : | XMLCombiningCharacter | XMLExtender XMLNameStartCharacter ::= XMLLetter | _ | : XMLName ::= XMLNameStartCharacter [ XMLNameCharacter+ ]
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
478
4.3
Inheritance
XMLLetter ::= < Unicode character as defined in the Letter production of the XML 1.0 specification > XMLDigit ::= < Unicode character as defined in the Digit production of the XML 1.0 specification > XMLCombiningCharacter ::= < Unicode character as defined in the CombiningChar production of the XML 1.0 specification > XMLExtender ::= < Unicode character as defined in the Extender production of the XML 1.0 specification > 1.97.2 Embedded expressions XML literal expressions can contain embedded expressions. An embedded expression is a Visual Basic expression that is evaluated and used to fill in one or more values at the location of embedded expression. For example, the following code places the string John Smith as the value of the XML element:
Dim name as Str ing = " John Smith" Dim element As System.Xml.L inq .XElement = <customer><%=name %></customer>
Expressions can be embedded in a number of contexts. For example, the following code produces an element named customer:
Dim name As Str ing = "customer" Dim element As System.Xml.L inq .XElement = <<%=name %>>John Smith</>
Each context where an embedded expression can be used specifies the types that will be accepted. When within the context of the expression part of an embedded expression, the normal lexical rules for Visual Basic code still apply so, for example, line continuations must be used:
' Visual Basic express ion uses l i ne cont inuat ion , XML does not Dim element As System.Xml.L inq .XElement = _
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
479
4.3
Inheritance
Smith</>
XMLEmbeddedExpression ::= < % = [ LineTerminator ] Expression [ LineTerminator ] % > 1.97.3 XML Documents An XML document results in a value typed as System.Xml.Linq.XDocument. Unlike the XML 1.0 specification, XML documents in XML literal expressions are required to specify the XML document prologue; XML literal expressions without the XML document prologue are interpreted as their individual entity. For example:
Dim doc As System.Xml.L inq .XDocument = _ <?xml vers ion="1.0"?> <?instruc t i on?> <customer>Bob</> Dim pi As System.Xml.L inq .XProcess ing Ins t ruc t i on = _ <?instruc t i on?>
An XML document can contain an embedded expression whose type can be any type; at runtime, however, the object must satisfy the requirements of the XDocument constructor or a run-time error will occur. Unlike regular XML, XML document expressions do not support DTDs (Document Type Declarations). Also, the encoding attribute, if supplied, will be ignored since the encoding of the Xml literal expression is always the same as the encoding of the source file itself. Annotation Although the encoding attribute is ignored, it is still valid attribute in order to maintain the ability to include any valid Xml 1.0 documents in source code. XMLDocument ::= XMLDocumentPrologue [ XMLMisc+ ] XMLDocumentBody [ XMLMisc+ ] 480
4.3
Inheritance
XMLVersion ::= XMLWhitespace version [ XMLWhitespace ] = [ XMLWhitespace ] XMLVersionNumberValue XMLVersionNumberValue ::= SingleQuoteCharacter 1 . 0 SingleQuoteCharacter | DoubleQuoteCharacter 1 . 0 DoubleQuoteCharacter XMLEncoding ::= XMLWhitespace encoding [ XMLWhitespace ] = [ XMLWhitespace ] XMLEncodingNameValue XMLEncodingNameValue ::= SingleQuoteCharacter XMLEncodingName SingleQuoteCharacter | DoubleQuoteCharacter XMLEncodingName DoubleQuoteCharacter XMLEncodingName ::= XMLLatinAlphaCharacter [ XMLEncodingNameCharacter+ ] XMLEncodingNameCharacter ::= XMLUnderscoreCharacter | XMLLatinAlphaCharacter | XMLNumericCharacter | XMLPeriodCharacter | XMLDashCharacter XMLLatinAlphaCharacter ::= < Unicode Latin alphabetic character (0x0041-0x005a, 0x0061-0x007a) > XMLNumericCharacter ::= < Unicode digit character (0x0030-0x0039) > XMLHexNumericCharacter ::= XMLNumericCharacter | < Unicode Latin hex alphabetic character (0x0041-0x0046, 0x0061-0x0066) >
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
481
4.3
Inheritance
XMLPeriodCharacter ::= < Unicode period character (0x002e) > XMLUnderscoreCharacter ::= < Unicode underscore character (0x005f) > XMLDashCharacter ::= < Unicode dash character (0x002d) > XMLStandalone ::= XMLWhitespace standalone [ XMLWhitespace ] = [ XMLWhitespace ] XMLYesNoValue XMLYesNoValue ::= SingleQuoteCharacter XMLYesNo SingleQuoteCharacter | DoubleQuoteCharacter XMLYesNo DoubleQuoteCharacter XMLYesNo ::= yes | no XMLMisc ::= XMLComment | XMLProcessingInstruction | XMLWhitespace XMLDocumentBody ::= XMLElement | XMLEmbeddedExpression 1.97.4 XML Elements An XML element results in a value typed as System.Xml.L inq .XElementUnlike regular XML, . XML elements can omit the name in the closing tag and the current most-nested element will be closed. For example:
Dim name = <name>Bob</>
Attribute declarations in an XML element result in values typed as System.Xml.L inq .XAtt r i bu te . Attribute values are normalized according to the XML specification. When the value of an attribute is Nothing the attribute will not be created, so the attribute value expression will not have to be checked for Nothing. For example:
Dim expr = Nothing ' Throws nul l argument exception Dim direct = New System.Xml.L inq .XElement( _
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
482
4.3
Inheritance
"Name", _ New System.Xml.Linq.XAttribute("Length", expr)) ' Doesn't throw exception, the result is <Name/> Dim literal = <Name Length=<%= expr %>/>
XML elements and attributes can contain nested expressions in the following places: The name of the element, in which case the embedded expression must be a value of a type implicitly convertible to System.Xml.L inq .XName For example: .
Dim name = <<%="name" %>>Bob</>
The name of an attribute of the element, in which case the embedded expression must be a value of a type implicitly convertible to System.Xml.L inq .XName For example: .
Dim name = <name <%= " length" %>="3">Bob</>
The value of an attribute of the element, in which case the embedded expression can be a value of any type. For example:
Dim name = <name length=<%= 3 %>>Bob</>
An attribute of the element, in which case the embedded expression can be a value of any type. For example:
Dim name = <name <%= new XAttr ibute (" l ength" , 3) %>>Bob</>
The content of the element, in which case the embedded expression can be a value of any type. For example:
Dim name = <name>< ="Bob" %></> %
If the type of the embedded expression is Object( ) the array will be passed as a , paramarray to the XElement constructor. XMLElement ::= XMLEmptyElement | XMLElementStart XMLContent XMLElementEnd
483
4.3
Inheritance
XMLEmptyElement ::= < XMLQualifiedNameOrExpression [ XMLAttribute+ ] [ XMLWhitepace ] / > XMLElementStart ::= < XMLQualifiedNameOrExpression [ XMLAttribute+ ] [ XMLWhitespace ] > XMLElementEnd ::= < / > | < / XMLQualifiedName [ XMLWhitespace ] > XMLContent ::= [ XMLCharacterData ] [ XMLNestedContent [ XMLCharacterData ] ]+ XMLCharacterData ::= < Any XMLCharacterDataString that does not contain the string "]]>" > XMLCharacterDataString ::= < Any Unicode character except < or & >+ XMLNestedContent ::= XMLElement | XMLReference | XMLCDATASection | XMLProcessingInstruction | XMLComment | XMLEmbeddedExpression XMLAttribute ::= XMLWhitespace XMLAttributeName [ XMLWhitespace ] = [ XMLWhitespace ] XMLAttributeValue | XMLWhitespace XMLEmbeddedExpression XMLAttributeName ::= XMLQualifiedNameOrExpression | XMLNamespaceAttributeName XMLAttributeValue ::= DoubleQuoteCharacter [ XMLAttributeDoubleQuoteValueCharacter+ ]
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
484
4.3
Inheritance
DoubleQuoteCharacter | SingleQuoteCharacter [ XMLAttributeSingleQuoteValueCharacter+ ] SingleQuoteCharacter | XMLEmbeddedExpression XMLAttributeDoubleQuoteValueCharacter ::= < Any XMLCharacter except <, &, or DoubleQuoteCharacter > | XMLReference XMLAttributeSingleQuoteValueCharacter ::= < Any XMLCharacter except <, &, or SingleQuoteCharacter > | XMLReference XMLReference ::= XMLEntityReference | XMLCharacterReference XMLEntityReference ::= & XMLEntityName ; XMLEntityName ::= l t | gt | amp | apos | quot XMLCharacterReference ::= & # XMLNumericCharacter+ ; | & # x XMLHexNumericCharacter+ ; 1.97.5 XML Namespaces XML elements can contain XML namespace declarations, as defined by the XML namespaces 1.0 specification. The restrictions on defining the namespaces xml and xmlns are enforced and will produce compile-time errors. XML namespace declarations cannot have an embedded expression for their value; the value supplied must be a non-empty string literal. For example:
' Declares a val id namespace Dim customer = <db:customer xmlns:db="http: / / example.org /database">Bob</> ' Error : xmlns cannot be re- def ined Dim bad1 = <elem xmlns:xmlns="http: / / example.org /namespace"/>
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
485
4.3
Inheritance
' Error: cannot have an embedded expression Dim bad2 = <elem xmlns:db=<%= "http://example.org/database" %>>Bob</>
Annotation This specification contains only enough of a description of XML namespace to describe the behavior of the Visual Basic language. More information on XML namespaces can be found at http://www.w3.org/TR/REC-xml-names/. XML element and attribute names can be qualified using namespace names. Namespaces are bound as in regular XML, with the exception that any namespace imports declared at the file level are considered to be declared in a context enclosing the declaration, which is itself enclosed by any namespace imports declared by the compilation environment. If a namespace name cannot be found, a compile-time error occurs. For example:
Imports System.Xml.Linq Imports <xmlns:db="http://example.org/database"> Module Test Sub Main() ' Binds to the imported namespace above. Dim c1 = <db:customer>Bob</> ' Binds to the namespace declaration in the element Dim c2 = _ <db:customer xmlns:db="http://example.org/databaseother">Mary</> ' Binds to the inner namespace declaration Dim c3 = _ <database xmlns:db="http://example.org/database-one"> <db:customer xmlns:db="http://example.org/databasetwo">Joe</> </> ' Error: namespace db2 cannot be found Dim c4 = _
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
486
4.3
Inheritance
XML namespaces declared in an element do not apply to XML literals inside embedded expressions. For example:
' Error: Namespace prefix 'db' is not declared Dim customer = _ <db:customer xmlns:db="http://example.org/database"> <%= <db:customer>Bob</> %> </>
Annotation This is because the embedded expression can be anything, including a function call. If the function call contained an XML literal expression, it is not clear whether programmers would expect the XML namespace to be applied or ignored. XMLNamespaceAttributeName ::= XMLPrefixedNamespaceAttributeName | XMLDefaultNamespaceAttributeName XMLPrefixedNamespaceAttributeName ::= xmlns : XMLNamespaceName XMLDefaultNamespaceAttributeName ::=
xmlns
XMLNamespaceName ::= XMLNamespaceNameStartCharacter [ XMLNamespaceNameCharacter+ ] XMLNamespaceNameStartCharacter ::= < Any XMLNameCharacter except : > XMLNamespaceNameCharacter ::= XMLLetter | _ XMLQualifiedNameOrExpression ::= XMLQualifiedName | XMLEmbeddedExpression 487
4.3
Inheritance
XMLQualifiedName ::= XMLPrefixedName | XMLUnprefixedName XMLPrefixedName ::= XMLNamespaceName : XMLNamespaceName XMLUnprefixedName ::= XMLNamespaceName 1.97.6 XML Processing Instructions An XML processing instruction results in a value typed as System.Xml.L inq .XProcess ing Ins t ruc.t XML processing instructions cannot contain i on embedded expressions, as they are valid syntax within the processing instruction. XMLProcessingInstruction ::= < ? XMLProcessingTarget [ XMLWhitespace [ XMLProcessingValue ] ] ? > XMLProcessingTarget ::= < Any XMLName except a casing permutation of the string "xml" > XMLProcessingValue ::= < Any XMLString that does not contain the string "?>" > 1.97.7 XML Comments An XML comment results in a value typed as System.Xml.Linq.XComment. XML comments cannot contain embedded expressions, as they are valid syntax within the comment. XMLComment ::= < ! - - [ XMLCommentCharacter+ ] - - > XMLCommentCharacter ::= < Any XMLCharacter except dash (0x002D) > | - < Any XMLCharacter except dash (0x002D) > 1.97.8 CDATA sections A CDATA section results in a value typed as System.Xml.Linq.XCData. CDATA sections cannot contain embedded expressions, as they are valid syntax within the CDATA section.
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
488
4.3
Inheritance
XMLCDATASection ::=
< ! [ CDATA [ [ XMLCDATASectionString ] ] ] >
XMLCDATASectionString ::= < Any XMLString that does not contain the string "]]>" >
Attribute access, in which a Visual Basic identifier follows a dot and an at sign, or an XML name follows a dot and an at sign. For example:
Dim customer = <customer age="30"/> Dim customerAge = customer.@age
489
4.3
Inheritance
Annotation The Attr i buteValue extension method (as well as the related extension property Value) is not currently defined in any assembly. If the extension members are needed, they are automatically defined in the assembly being produced. Descendents access, in which an XML names follows three dots. For example:
Dim company = _ <company> <customers> <customer>Bob</> <customer>Mary</> <customer>Joe</> </> </> Dim customers = company.. .<customer>
The base expression of an XML member access expression must be a value and must be of the type: If an element or descendents access, System.Xml.L inq .XContainer a derived type, or or System.Col lec t i ons .Gener ic . I Enumerable(Of a derived type, where T is or T) System.Xml.Linq.XContainer or a derived type. If an attribute access, System.Xml.Linq.XElement or a derived type, or System.Collections.Generic.IEnumerable(Of T) or a derived type, where T is System.Xml.Linq.XElement or a derived type.
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
490
4.3
Inheritance
Names in XML member access expressions cannot be empty. They can be namespace qualified, using any namespaces defined by imports. For example:
Imports <xmlns:db="http://example.org/database"> Module Test Sub Main() Dim customer = _ <db:customer> <db:name>Bob</> </> Dim name = customer.<db:name> End Sub End Module
Whitespace is not allowed after the dot(s) in an XML member access expression, or between the angle brackets and the name. For example:
Dim customer = _ <customer age="30"> <name>Bob</> </> ' All the following are error cases Dim age = customer.@ age Dim name = customer.< name > Dim names = customer...< name >
If the types in the System.Xml.L inqnamespace are not available, then an XML member access expression will cause a compile-time error. XMLMemberAccessExpression ::= Expression . [ LineTerminator ] < XMLQualifiedName > | Expression . @ [ LineTerminator ] < XMLQualifiedName > | Expression . @ [ LineTerminator ] IdentifierOrKeyword | Expression . . . [ LineTerminator ] < XMLQualifiedName >
491
Documentation Comments
Documentation comments are specially formatted comments in the source that can be analyzed to produce documentation about the code they are attached to. The basic format for documentation comments is XML. When the compiling code with documentation comments, the compiler may optionally emit an XML file that represents the sum total of the documentation comments in the source. This XML file can then be used by other tools to produce printed or online documentation. This chapter describes document comments and recommended XML tags to use with document comments.
Documentation comments must be well formed XML according to http://www.w3.org/TR/REC-xml. If the XML is not well formed, a warning is generated and the documentation file will contain a comment saying that an error was encountered. Although developers are free to create their own set of tags, a recommended set is defined in the next section. Some of the recommended tags have special meanings:
492
4.3
Inheritance
The <param> tag is used to describe parameters. The parameter specified by a <param> tag must exist and all parameters of the type member must be described in the documentation comment. If either condition is not true, the compiler issues a warning. The cref attribute can be attached to any tag to provide a reference to a code element. The code element must exist; at compile-time the compiler replaces the name with the ID string representing the member. If the code element does not exist, the compiler issues a warning. When looking for a name described in a cref attribute, the compiler respects Imports statements that appear within the containing source file. The <summary> tag is intended to be used by a documentation viewer to display additional information about a type or member. Note that the documentation file does not provide full information about a type and members, only what is contained in the document comments. To get more information about a type or member, the documentation file must be used in conjunction with reflection on the actual type or member.
493
4.3
Inheritance
<paramref> Identifies that a word is a parameter name <permission> Documents the security accessibility of a member <remarks> Describes a type <returns> Describes the return value of a method <see> Specifies a link <seealso> Generates a See Also entry <sum mary>
1.100.1 <c> This tag specifies that a fragment of text within a description should use a font like that used for a block of code. (For lines of actual code, use <code> .) Syntax:
<c>text to be set l i ke code</c>
Example:
' ' ' <remarks> ' ' ' Class <c>Point</c> models a point in a two- dimensional plane. ' ' ' </remarks> Publ i c Class Point End Class
1.100.2 <code> This tag specifies that one or more lines of source code or program output should use a fixed-width font. (For small code fragments, use <c> .) Syntax:
<code>source code or program output</code>
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
494
4.3
Inheritance
Example:
''' <summary> ''' This method changes the point's location by the given x- and ''' y-offsets. ''' <example> ''' For example: ''' <code> ''' Dim p As Point = New Point(3,5) ''' p.Translate(-1,3) ''' </code> ''' results in <c>p</c>'s having the value (2,8). ''' </example> ''' </summary> Public Sub Translate(x As Integer, y As Integer) Me.x += x Me.y += y End Sub
1.100.3 <example> This tag allows example code within a comment to show how an element can be used. Ordinarily, this will involve use of the tag <code> as well. Syntax:
<example>descr ipt i on</example>
Example: See <code> for an example. 1.100.4 <exception> This tag provides a way to document the exceptions a method can throw. Syntax:
<exception cref="member">descr ipt i on</except ion>
Example:
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
495
4.3
Inheritance
Public Module DataBaseOperations ''' <exception cref="MasterFileFormatCorruptException"></exception> ''' <exception cref="MasterFileLockedOpenException"></exception> Public Sub ReadRecord(flag As Integer) If Flag = 1 Then Throw New MasterFileFormatCorruptException() ElseIf Flag = 2 Then Throw New MasterFileLockedOpenException() End If ' ... End Sub End Module
1.100.5 <include> This tag is used to include information from an external well-formed XML document. An XPath expression is applied to the XML document to specify what XML should be included from the document. The <include> tag is then replaced with the selected XML from the external document. Syntax:
<include f i l e=" f i l e name" path="xpath">
496
4.3
Inheritance
1.100.6 <list> This tag is used to create a list or table of items. It may contain a <lis theader> block to define the heading row of either a table or definition list. (When defining a table, only an entry for term in the heading need be supplied.) Each item in the list is specified with an <item> block. When creating a definition list, both term and description must be specified. However, for a table, bulleted list, or numbered list, only description need be specified. Syntax:
<lis t type="bul le t " | "number" | "table"> <lis theader> <term>term</term> <descr ipt i on>descr ip t i on</descr ip t i on> </l i s theader> <item> <term>term</term> <descr ipt i on>descr ip t i on</descr ip t i on> </i tem> <item> <term>term</term> <descr ipt i on>descr ip t i on</descr ip t i on> </i tem> </l i s t>
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
497
4.3
Inheritance
Example:
Public Class TestClass ''' <remarks> ''' Here is an example of a bulleted list: ''' <list type="bullet"> ''' <item> ''' <description>Item 1.</description> ''' </item> ''' <item> ''' <description>Item 2.</description> ''' </item> ''' </list> ''' </remarks> Public Shared Sub Main() End Sub End Class
1.100.7 <para> This tag is for use inside other tags, such as <remarks> or <returns>, and permits structure to be added to text. Syntax:
<para>content</para>
Example:
' ' ' <sum mary> ' ' ' T h i s i s the entry point of the Point class test ing program. ' ' ' <para>This program tests each method and operator , and ' ' ' i s intended to be run after any non- trv ia l maintenance has ' ' ' been performed on the Point class .</para> ' ' ' </summary> Publ i c Shared Sub Main() End Sub
498
4.3
Inheritance
1.100.8 <param> This tag describes a parameter for a method, constructor, or indexed property. Syntax:
<param name="name">description</param>
Example:
''' <summary> ''' This method changes the point's location to the given ''' coordinates. ''' </summary> ''' <param name="x"><c>x</c> is the new x-coordinate.</param> ''' <param name="y"><c>y</c> is the new y-coordinate.</param> Public Sub Move(x As Integer, y As Integer) Me.x = x Me.y = y End Sub
1.100.9 <paramref> This tag indicates that a word is a parameter. The documentation file can be processed to format this parameter in some distinct way. Syntax:
<paramref name="name"/>
Example:
''' <summary> ''' This constructor initializes the new Point to ''' (<paramref name="x"/>,<paramref name="y"/>). ''' </summary> ''' <param name="x"><c>x</c> is the new Point's x-coordinate.</param> ''' <param name="y"><c>y</c> is the new Point's y-coordinate.</param> Public Sub New(x As Integer, y As Integer) Me.x = x
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
499
4.3
Inheritance
1.100.10 <permission> This tag documents the security accessibility of a member Syntax:
<permission cref="member">description</permission>
Example:
''' <permission cref="System.Security.PermissionSet">Everyone can ''' access this method.</permission> Public Shared Sub Test() End Sub
1.100.11 <remarks> This tag specifies overview information about a type. (Use <summary> to describe the members of a type.) Syntax:
<remarks>descr ipt i on</remarks>
Example:
' ' ' <remarks> ' ' ' Class <c>Point</c> models a point in a two- dimensional plane. ' ' ' </remarks> Publ i c Class Point End Class
1.100.12 <returns> This tag describes the return value of a method. Syntax:
<returns>descr ipt i on</returns>
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
500
4.3
Inheritance
Example:
''' <summary> ''' Report a point's location as a string. ''' </summary> ''' <returns> ''' A string representing a point's location, in the form (x,y), without ''' any leading, training, or embedded whitespace. ''' </returns> Public Overrides Function ToString() As String Return "(" & x & "," & y & ")" End Sub
1.100.13 <see> This tag allows a link to be specified within text. (Use <seealso> to indicate text that is to appear in a See Also section.) Syntax:
<see cref="member"/>
Example:
' ' ' <sum mary> ' ' ' This method changes the point ' s locat i on to the given ' ' ' coordinates . ' ' ' </summary> ' ' ' <see cref="Transla te" /> Publ i c Sub Move(x As Integer , y As Integer ) Me.x = x Me.y = y End Sub ''' ''' ''' ''' ''' <sum mary> This method changes the point ' s locat i on by the given x- and y- offsets . </summary> <see cref="Move"/>
501
4.3
Inheritance
1.100.14 <seealso> This tag generates an entry for the See Also section. (Use <see> to specify a link from within text.) Syntax:
<seealso cref="member"/>
Example:
' ' ' <sum mary> ' ' ' This method determines whether two Points have the same locat i on . ' ' ' </summary> ' ' ' <seealso cref="operator=="/> ' ' ' <seealso cref="operator !=" /> Publ i c Overr ides Funct ion Equals(o As Object) As Boolean ' ... End Funct ion
1.100.15 <summary> This tag describes a type member. (Use <remarks> to describe a type itself.) Syntax:
<sum mary>descr ipt i on</summary>
Example:
' ' ' <sum mary> ' ' ' This constructor in i t i a l i z e s the new Point to (0,0 ) . ' ' ' </summary> Publ i c Sub New() Me.New(0,0) End Sub
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
502
4.3
Inheritance
Example:
''' <typeparam name="T"> ''' The base item type. Must implement IComparable. ''' </typeparam> Public Class ItemManager(Of T As IComparable) End Class
Example:
''' <value> ''' Property <c>X</c> represents the point's x-coordinate. ''' </value> Public Property X() As Integer Get Return _x End Get Set (Value As Integer) _x = Value End Set End Property
503
4.3
Inheritance
1.101 ID Strings
When generating the documentation file, the compiler generates an ID string for each element in the source code that is tagged with a documentation comment that uniquely identifies it. This ID string can be used by external tools to identify which element in a compiled assembly corresponds to the document comment. ID strings are generated as follows: No white space is placed in the string. The first part of the string identifies the kind of member being documented, via a single character followed by a colon. The following kinds of members are defined, with the corresponding character in parenthesis after it: events (E), fields (F), methods including constructors and operators (M), namespaces (N), properties (P) and types (T). An exclamation point (!) indicates an error occurred while generating the ID string, and the rest of the string provides information about the error. The second part of the string is the fully qualified name of the element, starting at the global namespace. The name of the element, its enclosing type(s), and namespace are separated by periods. If the name of the item itself has periods, they are replaced by the pound sign (#). (It is assumed that no element has this character in its name.) The name of a type with type parameters ends with a backquote (`) followed by a number that represents the number of type parameters on the type. It is important to remember that because nested types have access to the type parameters of the types containing them, nested types implicitly contain the type parameters of their containing types, and those types are counted in their type parameter totals in this case. For methods and properties with arguments, the argument list follows, enclosed in parentheses. For those without arguments, the parentheses are omitted. The arguments are separated by commas. The encoding of each argument is the same as a CLI signature, as follows: Arguments are represented by their fully qualified name. For example, Integer becomes System.Int32 String becomes System.String, Object becomes System.Object, and so , on. Arguments having the ByRef modifier have a '@' following their type name. Arguments having the ByVal, Optional or ParamArray modifier have no special notation. Arguments that are arrays are represented as [lowerbound:size, , lowerbound:size] where the number of
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
504
4.3
Inheritance
commas is the rank 1, and the lower bounds and size of each dimension, if known, are represented in decimal. If a lower bound or size is not specified, it is omitted. If the lower bound and size for a particular dimension are omitted, the ':' is omitted as well. Arrays of arrays are represented by one "[]" per level. 1.101.1 ID string examples The following examples each show a fragment of VB code, along with the ID string produced from each source element capable of having a documentation comment: Types are represented using their fully qualified name.
Enum Color Red Blue Green End Enum Namespace Acme Interface IProcess End Interface Structure ValueType ... End Structure Class Widget Public Class NestedClass End Class Public Interface IMenuItem End Interface Public Delegate Sub Del(i As Integer) Public Enum Direction North
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
505
4.3
Inheritance
South East West End Enum End Class End Namespace "T:Color" "T:Acme.IProcess" "T:Acme.ValueType" "T:Acme.Widget" "T:Acme.Widget.NestedClass" "T:Acme.Widget.IMenuItem" "T:Acme.Widget.Del" "T:Acme.Widget.Direction"
506
4.3
Inheritance
Constructors.
Namespace Acme Class Widget Shared Sub New() End Sub Public Sub New() End Sub Public Sub New(s As String) End Sub End Class End Namespace "M:Acme.Widget.#cctor" "M:Acme.Widget.#ctor" "M:Acme.Widget.#ctor(System.String)"
Methods.
Namespace Acme Structure ValueType Public Sub M(i As Integer) End Sub End Structure Class Widget Public Class NestedClass
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
507
4.3
Inheritance
Public Sub M(i As Integer) End Sub End Class Public Shared Sub M0() End Sub Public Sub M1(c As Char, ByRef f As Float, _ ByRef v As ValueType) End Sub Public Sub M2(x1() As Short, x2(,) As Integer, _ x3()() As Long) End Sub Public Sub M3(x3()() As Long, x4()(,,) As Widget) End Sub Public Sub M4(Optional i As Integer = 1) Public Sub M5(ParamArray args() As Object) End Sub End Class End Namespace "M:Acme.ValueType.M(System.Int32)" "M:Acme.Widget.NestedClass.M(System.Int32)" "M:Acme.Widget.M0" "M:Acme.Widget.M1(System.Char,System.Single@,Acme.ValueType@)" "M:Acme.Widget.M2(System.Int16[],System.Int32[0:,0:],System.Int64[][])" "M:Acme.Widget.M3(System.Int64[][],Acme.Widget[0:,0:,0:][])" "M:Acme.Widget.M4(System.Int32) M:Acme.Widget.M5(System.Object[])"
Properties.
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
508
4.3
Inheritance
Namespace Acme Class Widget Public Property Width() As Integer Get End Get Set (Value As Integer) End Set End Property Public Default Property Item(i As Integer) As Integer Get End Get Set (Value As Integer) End Set End Property Public Default Property Item(s As String, _ i As Integer) As Integer Get End Get Set (Value As Integer) End Set End Property End Class End Namespace "P:Acme.Widget.Width" "P:Acme.Widget.Item(System.Int32)" "P:Acme.Widget.Item(System.String,System.Int32)"
Events
Namespace Acme Class Widget Public Event AnEvent As EventHandler Public Event AnotherEvent() End Class
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
509
4.3
Inheritance
Operators.
Namespace Acme Class Widget Public Shared Operator +(x As Widget) As Widget End Operator Public Shared Operator +(x1 As Widget, x2 As Widget) As Widget End Operator End Class End Namespace "M:Acme.Widget.op_UnaryPlus(Acme.Widget)" "M:Acme.Widget.op_Addition(Acme.Widget,Acme.Widget)"
510
4.3
Inheritance
Namespace Graphics ''' <remarks> ''' Class <c>Point</c> models a point in a two-dimensional ''' plane. ''' </remarks> Public Class Point ''' <summary> ''' Instance variable <c>x</c> represents the point's x-coordinate. ''' </summary> Private _x As Integer ''' <summary> ''' Instance variable <c>y</c> represents the point's y-coordinate. ''' </summary> Private _y As Integer ''' <value> ''' Property <c>X</c> represents the point's x-coordinate. ''' </value> Public Property X() As Integer Get Return _x End Get Set(Value As Integer) _x = Value End Set End Property ''' <value> ''' Property <c>Y</c> represents the point's y-coordinate. ''' </value> Public Property Y() As Integer Get Return _y End Get Set(Value As Integer)
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
511
4.3
Inheritance
_y = Value End Set End Property ''' <summary> ''' This constructor initializes the new Point to (0,0). ''' </summary> Public Sub New() Me.New(0, 0) End Sub ''' <summary> ''' This constructor initializes the new Point to ''' (<paramref name="x"/>,<paramref name="y"/>). ''' </summary> ''' <param name="x"><c>x</c> is the new Point's ''' x-coordinate.</param> ''' <param name="y"><c>y</c> is the new Point's ''' y-coordinate.</param> Public Sub New(x As Integer, y As Integer) Me.X = x Me.Y = y End Sub ''' <summary> ''' This method changes the point's location to the given ''' coordinates. ''' </summary> ''' <param name="x"><c>x</c> is the new x-coordinate.</param> ''' <param name="y"><c>y</c> is the new y-coordinate.</param> ''' <see cref="Translate"/> Public Sub Move(x As Integer, y As Integer) Me.X = x Me.Y = y End Sub
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
512
4.3
Inheritance
''' <summary> ''' This method changes the point's location by the given x- and ''' y-offsets. ''' <example> ''' For example: ''' <code> ''' Dim p As Point = New Point(3, 5) ''' p.Translate(-1, 3) ''' </code> ''' results in <c>p</c>'s having the value (2,8). ''' </example> ''' </summary> ''' <param name="x"><c>x</c> is the relative x-offset.</param> ''' <param name="y"><c>y</c> is the relative y-offset.</param> ''' <see cref="Move"/> Public Sub Translate(x As Integer, y As Integer) Me.X += x Me.Y += y End Sub ''' <summary> ''' This method determines whether two Points ''' location. ''' </summary> ''' <param name="o"><c>o</c> is the object to ''' current object.</param> ''' <returns> ''' True if the Points have the same location ''' exact same type; otherwise, false. ''' </returns> ''' <seealso cref="Operator op_Equality"/> ''' <seealso cref="Operator op_Inequality"/> Public Overrides Function Equals(o As Object) If o Is Nothing Then Return False End If
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
As Boolean
513
4.3
Inheritance
If o Is Me Then Return True End If If Me.GetType() Is o.GetType() Then Dim p As Point = CType(o, Point) Return (X = p.X) AndAlso (Y = p.Y) End If Return False End Function ''' <summary> ''' Report a point's location as a string. ''' </summary> ''' <returns> ''' A string representing a point's location, in the form ''' (x,y), without any leading, training, or embedded whitespace. ''' </returns> Public Overrides Function ToString() As String Return "(" & X & "," & Y & ")" End Function ''' <summary> ''' This operator determines whether two Points have the ''' same location. ''' </summary> ''' <param name="p1"><c>p1</c> is the first Point to be compared. ''' </param> ''' <param name="p2"><c>p2</c> is the second Point to be compared. ''' </param> ''' <returns> ''' True if the Points have the same location and they ''' have the exact same type; otherwise, false. ''' </returns> ''' <seealso cref="Equals"/> ''' <seealso cref="op_Inequality"/> Public Shared Operator =(p1 As Point, p2 As Point) As Boolean
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
514
4.3
Inheritance
If p1 Is Nothing OrElse p2 Is Nothing Then Return False End If If p1.GetType() Is p2.GetType() Then Return (p1.X = p2.X) AndAlso (p1.Y = p2.Y) End If Return False End Operator ''' <summary> ''' This operator determines whether two Points have the ''' same location. ''' </summary> ''' <param name="p1"><c>p1</c> is the first Point to be comapred. ''' </param> ''' <param name="p2"><c>p2</c> is the second Point to be compared. ''' </param> ''' <returns> ''' True if the Points do not have the same location and ''' the exact same type; otherwise, false. ''' </returns> ''' <seealso cref="Equals"/> ''' <seealso cref="op_Equality"/> Public Shared Operator <>(p1 As Point, p2 As Point) As Boolean Return Not p1 = p2 End Operator ''' <summary> ''' This is the entry point of the Point class testing program. ''' <para>This program tests each method and operator, and ''' is intended to be run after any non-trvial maintenance has ''' been performed on the Point class.</para> ''' </summary> Public Shared Sub Main() ' class test code goes here End Sub
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
515
4.3
Inheritance
Here is the output produced when given the source code for class Point, shown above:
<?xml version="1.0"?> <doc> <assembly> <name>Point</name> </assembly> <members> <member name="T:Graphics.Point"> <remarks>Class <c>Point</c> models a point in a two-dimensional plane. </remarks> </member> <member name="F:Graphics.Point.x"> <summary>Instance variable <c>x</c> represents the point's x-coordinate.</summary> </member> <member name="F:Graphics.Point.y"> <summary>Instance variable <c>y</c> represents the point's y-coordinate.</summary> </member> <member name="M:Graphics.Point.#ctor"> <summary>This constructor initializes the new Point to (0,0).</summary> </member> <member name="M:Graphics.Point.#ctor(System.Int32,System.Int32)"> <summary>This constructor initializes the new Point to (<paramref name="x"/>,<paramref name="y"/>).</summary> <param><c>x</c> is the new Point's x-coordinate.</param> <param><c>y</c> is the new Point's y-coordinate.</param> </member> <member name="M:Graphics.Point.Move(System.Int32,System.Int32)"> <summary>This method changes the point's location to the given coordinates.</summary> <param><c>x</c> is the new x-coordinate.</param>
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
516
4.3
Inheritance
<param><c>y</c> is the new y-coordinate.</param> <see cref= "M:Graphics.Point.Translate(System.Int32,System.Int32)"/> </member> <member name= "M:Graphics.Point.Translate(System.Int32,System.Int32)"> <summary>This method changes the point's location by the given x- and y-offsets. <example>For example: <code> Point p = new Point(3,5); p.Translate(-1,3); </code> results in <c>p</c>'s having the value (2,8). </example> </summary> <param><c>x</c> is the relative x-offset.</param> <param><c>y</c> is the relative y-offset.</param> <see cref="M:Graphics.Point.Move(System.Int32,System.Int32)"/> </member> <member name="M:Graphics.Point.Equals(System.Object)"> <summary>This method determines whether two Points have the same location.</summary> <param><c>o</c> is the object to be compared to the current object.</param> <returns>True if the Points have the same location and they have the exact same type; otherwise, false.</returns> <seealso cref= "M:Graphics.Point.op_Equality(Graphics.Point,Graphics.Point)" /> <seealso cref= "M:Graphics.Point.op_Inequality(Graphics.Point,Graphics.Point)" /> </member> <member name="M:Graphics.Point.ToString"> <summary>Report a point's location as a string.</summary>
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
517
4.3
Inheritance
<returns>A string representing a point's location, in the form (x,y), without any leading, training, or embedded whitespace.</returns> </member> <member name= "M:Graphics.Point.op_Equality(Graphics.Point,Graphics.Point)"> <summary>This operator determines whether two Points have the same location.</summary> <param><c>p1</c> is the first Point to be compared.</param> <param><c>p2</c> is the second Point to be compared.</param> <returns>True if the Points have the same location and they have the exact same type; otherwise, false.</returns> <seealso cref="M:Graphics.Point.Equals(System.Object)"/> <seealso cref= "M:Graphics.Point.op_Inequality(Graphics.Point,Graphics.Point)" /> </member> <member name= "M:Graphics.Point.op_Inequality(Graphics.Point,Graphics.Point)"> <summary>This operator determines whether two Points have the same location.</summary> <param><c>p1</c> is the first Point to be compared.</param> <param><c>p2</c> is the second Point to be compared.</param> <returns>True if the Points do not have the same location and the exact same type; otherwise, false.</returns> <seealso cref="M:Graphics.Point.Equals(System.Object)"/> <seealso cref= "M:Graphics.Point.op_Equality(Graphics.Point,Graphics.Point)" /> </member> <member name="M:Graphics.Point.Main"> <summary>This is the entry point of the Point class testing program. <para>This program tests each method and operator, and is intended to be run after any non-trvial maintenance has been performed on the Point class.</para>
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
518
4.3
Inheritance
</summary> </member> <member name="P:Graphics.Point.X"> <value>Property <c>X</c> represents the point's x-coordinate.</value> </member> <member name="P:Graphics.Point.Y"> <value>Property <c>Y</c> represents the point's y-coordinate.</value> </member> </members> </doc>
519
Grammar Summary
This section summarizes the Visual Basic language grammar. For information on how to read the grammar, see Grammar Notation.
520
4.3
Inheritance
SingleQuoteCharacter ::= ' | < Unicode left single-quote character (0x2018) > | < Unicode right single-quote character (0x2019) > 1.103.2 Identifiers Identifier ::= NonEscapedIdentifier [ TypeCharacter ] | Keyword TypeCharacter | EscapedIdentifier NonEscapedIdentifier ::= < IdentifierName but not Keyword > EscapedIdentifier ::= [ IdentifierName ] IdentifierName ::= IdentifierStart [ IdentifierCharacter+ ] IdentifierStart ::= AlphaCharacter | UnderscoreCharacter IdentifierCharacter IdentifierCharacter ::= UnderscoreCharacter | AlphaCharacter | NumericCharacter | CombiningCharacter | FormattingCharacter AlphaCharacter ::= < Unicode alphabetic character (classes Lu, Ll, Lt, Lm, Lo, Nl) > NumericCharacter ::= < Unicode decimal digit character (class Nd) > CombiningCharacter ::= < Unicode combining character (classes Mn, Mc) > FormattingCharacter ::= < Unicode formatting character (class Cf) > UnderscoreCharacter ::= < Unicode connection character (class Pc) >
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
521
4.3
Inheritance
IdentifierOrKeyword ::= Identifier | Keyword TypeCharacter ::= IntegerTypeCharacter | LongTypeCharacter | DecimalTypeCharacter | SingleTypeCharacter | DoubleTypeCharacter | StringTypeCharacter IntegerTypeCharacter ::= % LongTypeCharacter ::= & DecimalTypeCharacter ::= @ SingleTypeCharacter ::= ! DoubleTypeCharacter ::= # StringTypeCharacter ::= $ 1.103.3 Keywords Keyword ::= < member of keyword table in 2.3 > 1.103.4 Literals Literal ::= BooleanLiteral | IntegerLiteral | FloatingPointLiteral | StringLiteral | CharacterLiteral | DateLiteral | Nothing BooleanLiteral ::= True | False IntegerLiteral ::= IntegralLiteralValue [ IntegralTypeCharacter ]
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
522
4.3
Inheritance
IntegralLiteralValue ::= IntLiteral | HexLiteral | OctalLiteral IntegralTypeCharacter ::= ShortCharacter | UnsignedShortCharacter | IntegerCharacter | UnsignedIntegerCharacter | LongCharacter | UnsignedLongCharacter | IntegerTypeCharacter | LongTypeCharacter ShortCharacter ::= S UnsignedShortCharacter ::= US IntegerCharacter ::= I UnsignedIntegerCharacter ::= UI LongCharacter ::= L UnsignedLongCharacter ::= UL IntLiteral ::= Digit+ HexLiteral ::= & H HexDigit+ OctalLiteral ::= & O OctalDigit+ Digit ::= 0 | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 HexDigit ::= 0 | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | A | B | C | D | E | F OctalDigit ::= 0 | 1 | 2 | 3 | 4 | 5 | 6 | 7 FloatingPointLiteral ::= FloatingPointLiteralValue [ FloatingPointTypeCharacter ] | IntLiteral FloatingPointTypeCharacter
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
523
4.3
Inheritance
FloatingPointTypeCharacter ::= SingleCharacter | DoubleCharacter | DecimalCharacter | SingleTypeCharacter | DoubleTypeCharacter | DecimalTypeCharacter SingleCharacter ::= F DoubleCharacter ::= R DecimalCharacter ::= D FloatingPointLiteralValue ::= IntLiteral . IntLiteral [ Exponent ] | . IntLiteral [ Exponent ] | IntLiteral Exponent Exponent ::= E [ Sign ] IntLiteral Sign ::= + | StringLiteral ::= DoubleQuoteCharacter [ StringCharacter+ ] DoubleQuoteCharacter DoubleQuoteCharacter ::= " | < Unicode left double-quote character (0x201C) > | < Unicode right double-quote character (0x201D) > StringCharacter ::= < Character except for DoubleQuoteCharacter > | DoubleQuoteCharacter DoubleQuoteCharacter CharacterLiteral ::= DoubleQuoteCharacter StringCharacter DoubleQuoteCharacter C DateLiteral ::= # [ Whitespace+ ] DateOrTime [ Whitespace+ ] #
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
524
4.3
Inheritance
DateOrTime ::= DateValue Whitespace+ TimeValue | DateValue | TimeValue DateValue ::= MonthValue / DayValue / YearValue | MonthValue - DayValue - YearValue TimeValue ::= HourValue : MinuteValue [ : SecondValue ] [ WhiteSpace+ ] [ AMPM ] | HourValue [ WhiteSpace+ ] AMPM MonthValue ::= IntLiteral DayValue ::= IntLiteral YearValue ::= IntLiteral HourValue ::= IntLiteral MinuteValue ::= IntLiteral SecondValue ::= IntLiteral AMPM ::= AM | PM ElseIf ::= ElseIf | Else If Nothing ::= Nothing Separator ::= ( | ) | { | } | ! | # | , | . | : | ? Operator ::= & | * | + | - | / | \ | ^ | < | = | >
525
4.3
Inheritance
CCStatement ::= CCConstantDeclaration | CCIfGroup | LogicalLine CCExpression ::= LiteralExpression | CCParenthesizedExpression | CCSimpleNameExpression | CCCastExpression | CCOperatorExpression | CCConditionalExpression CCParenthesizedExpression ::= ( CCExpression ) CCSimpleNameExpression ::= Identifier CCCastExpression ::= DirectCast ( CCExpression , TypeName ) | TryCast ( CCExpression , TypeName ) | CType ( CCExpression , TypeName ) | CastTarget ( CCExpression ) CCOperatorExpression ::= CCUnaryOperator CCExpression | CCExpression CCBinaryOperator CCExpression CCUnaryOperator ::= + | - | Not CCBinaryOperator ::= + | - | * | / | \ | Mod | ^ | = | < > | < | > | < = | > = | & | And | Or | Xor | AndAlso | OrElse | < < | > > CCConditionalExpression ::= If ( CCExpression , CCExpression , CCExpression ) | If ( CCExpression , CCExpression ) CCConstantDeclaration ::= # Const Identifier = CCExpression LineTerminator
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
526
4.3
Inheritance
CCIfGroup ::= # I f CCExpression [ Then ] LineTerminator [ CCStatement+ ] [ CCElseIfGroup+ ] [ CCElseGroup ] # End If LineTerminator CCElseIfGroup ::= # ElseIf CCExpression [ Then ] LineTerminator [ CCStatement+ ] CCElseGroup ::= # Else LineTerminator [ CCStatement+ ] 1.104.2 External Source Directives Start ::= [ ExternalSourceStatement+ ] ExternalSourceStatement ::= ExternalSourceGroup | LogicalLine ExternalSourceGroup ::=
# ExternalSource ( StringLiteral , IntLiteral ) LineTerminator
[ LogicalLine+ ]
# End ExternalSource LineTerminator
1.104.3 Region Directives Start ::= [ RegionStatement+ ] RegionStatement ::= RegionGroup | LogicalLine RegionGroup ::= # Region StringLiteral LineTerminator [ RegionStatement+ ] # End Region LineTerminator
527
4.3
Inheritance
1.104.4 External Checksum Directives Start ::= [ ExternalChecksumStatement+ ] ExternalChecksumStatement ::= # ExternalChecksum ( StringLiteral , StringLiteral , StringLiteral ) LineTerminator
528
4.3
Inheritance
AttributeBlock ::= [ LineTerminator ] < AttributeList [ LineTerminator ] > [ LineTerminator ] AttributeList ::= Attribute | AttributeList Comma Attribute Attribute ::= [ AttributeModifier : ] SimpleTypeName [ OpenParenthesis [ AttributeArguments ] CloseParenthesis ] AttributeModifier ::= Assembly | Module AttributeArguments ::= AttributePositionalArgumentList | AttributePositionalArgumentList Comma VariablePropertyInitializerList | VariablePropertyInitializerList AttributePositionalArgumentList ::= AttributeArgumentExpression | AttributePositionalArgumentList Comma AttributeArgumentExpression VariablePropertyInitializerList ::= VariablePropertyInitializer | VariablePropertyInitializerList Comma VariablePropertyInitializer VariablePropertyInitializer ::= IdentifierOrKeyword ColonEquals AttributeArgumentExpression 1.105.2 AttributeArgumentExpression ::= ConstantExpression | GetTypeExpression | ArrayExpressionSource Files and Namespaces Start ::= [ OptionStatement+ ] [ ImportsStatement+ ]
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
529
4.3
Inheritance
[ AttributesStatement+ ] [ NamespaceMemberDeclaration+ ] StatementTerminator ::= LineTerminator | : AttributesStatement ::= Attributes StatementTerminator OptionStatement ::= OptionExplicitStatement | OptionStrictStatement | OptionCompareStatement | OptionInferStatement OptionExplicitStatement ::= Option Expl i c i t[ OnOff ] StatementTerminator OnOff ::= On | Off OptionStrictStatement ::= Option Str i c t [ OnOff ] StatementTerminator OptionCompareStatement ::= Option Compare CompareOption StatementTerminator CompareOption ::= Binary | Text OptionInferStatement ::= Option Infer [ OnOff ] StatementTerminator ImportsStatement ::= Imports ImportsClauses StatementTerminator ImportsClauses ::= ImportsClause | ImportsClauses Comma ImportsClause ImportsClause ::= AliasImportsClause | MembersImportsClause | XMLNamespaceImportsClause AliasImportsClause ::= Identifier Equals TypeName
530
4.3
Inheritance
MembersImportsClause ::= TypeName XMLNamespaceImportsClause ::= < XMLNamespaceAttributeName [ XMLWhitespace ] Equals [ XMLWhitespace ] XMLNamespaceValue > XMLNamespaceValue ::= DoubleQuoteCharacter [ XMLAttributeDoubleQuoteValueCharacter+ ] DoubleQuoteCharacter | SingleQuoteCharacter [ XMLAttributeSingleQuoteValueCharacter+ ] SingleQuoteCharacter NamespaceDeclaration ::= Namespace NamespaceName StatementTerminator [ NamespaceMemberDeclaration+ ] End Namespace StatementTerminator NamespaceName ::= Identifier | NamespaceName Period IdentifierOrKeyword NamespaceMemberDeclaration ::= NamespaceDeclaration | TypeDeclaration TypeDeclaration ::= ModuleDeclaration | NonModuleDeclaration NonModuleDeclaration ::= EnumDeclaration | StructureDeclaration | InterfaceDeclaration | ClassDeclaration | DelegateDeclaration
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
531
4.3
Inheritance
1.105.3 Types TypeName ::= ArrayTypeName | NonArrayTypeName NonArrayTypeName ::= SimpleTypeName | NullableTypeName SimpleTypeName ::= QualifiedTypeName | BuiltInTypeName QualifiedTypeName ::= Identifier [ TypeArguments ] | Global Period IdentifierOrKeyword [ TypeArguments ] | QualifiedTypeName Period IdentifierOrKeyword [ TypeArguments ] TypeArguments ::= OpenParenthesis Of TypeArgumentList CloseParenthesis TypeArgumentList ::= TypeName | TypeArgumentList Comma TypeName BuiltInTypeName ::= Object | PrimitiveTypeName TypeModifier ::= AccessModifier | Shadows IdentifierModifiers ::= [ NullableNameModifier ] [ ArrayNameModifier ] NullableTypeName ::= NonArrayTypeName ? NullableNameModifier ::= ? TypeImplementsClause ::= Implements TypeImplements StatementTerminator TypeImplements ::= NonArrayTypeName | TypeImplements Comma NonArrayTypeName
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
532
4.3
Inheritance
PrimitiveTypeName ::= NumericTypeName | Boolean | Date | Char | String NumericTypeName ::= IntegralTypeName | FloatingPointTypeName | Decimal IntegralTypeName ::= Byte | SByte | UShort | Short | UInteger | Integer | ULong | Long FloatingPointTypeName ::= Single | Double EnumDeclaration ::= [ Attributes ] [ TypeModifier+ ] Enum Identifier [ As NonArrayTypeName ] StatementTerminator EnumMemberDeclaration+ End Enum StatementTerminator EnumMemberDeclaration ::= [ Attributes ] Identifier [ Equals ConstantExpression ] StatementTerminator ClassDeclaration ::= [ Attributes ] [ ClassModifier+ ] Class Identifier [ TypeParameterList ] StatementTerminator [ ClassBase ] [ TypeImplementsClause+ ] [ ClassMemberDeclaration+ ] End Class StatementTerminator ClassModifier ::= TypeModifier | MustInherit | NotInheritable | Partial ClassBase ::= Inherits NonArrayTypeName StatementTerminator ClassMemberDeclaration ::= NonModuleDeclaration | EventMemberDeclaration | VariableMemberDeclaration | ConstantMemberDeclaration | MethodMemberDeclaration | PropertyMemberDeclaration | ConstructorMemberDeclaration | OperatorDeclaration
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
533
4.3
Inheritance
StructureDeclaration ::= [ Attributes ] [ StructureModifier+ ] Structure Identifier [ TypeParameterList ] StatementTerminator [ TypeImplementsClause+ ] [ StructMemberDeclaration+ ] End Structure StatementTerminator StructureModifier ::= TypeModifier | Partial StructMemberDeclaration ::= NonModuleDeclaration | VariableMemberDeclaration | ConstantMemberDeclaration | EventMemberDeclaration | MethodMemberDeclaration | PropertyMemberDeclaration | ConstructorMemberDeclaration | OperatorDeclaration ModuleDeclaration ::= [ Attributes ] [ TypeModifier+ ] Module Identifier StatementTerminator [ ModuleMemberDeclaration+ ] End Module StatementTerminator ModuleMemberDeclaration ::= NonModuleDeclaration | VariableMemberDeclaration | ConstantMemberDeclaration | EventMemberDeclaration | MethodMemberDeclaration | PropertyMemberDeclaration | ConstructorMemberDeclaration InterfaceDeclaration ::= [ Attributes ] [ TypeModifier+ ] Interface Identifier [ TypeParameterList ] StatementTerminator
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
534
4.3
Inheritance
[ InterfaceBase+ ] [ InterfaceMemberDeclaration+ ] End Inter faceStatementTerminator InterfaceBase ::= Inher i t s InterfaceBases StatementTerminator InterfaceBases ::= NonArrayTypeName | InterfaceBases Comma NonArrayTypeName InterfaceMemberDeclaration ::= NonModuleDeclaration | InterfaceEventMemberDeclaration | InterfaceMethodMemberDeclaration | InterfacePropertyMemberDeclaration ArrayTypeName ::= NonArrayTypeName ArrayTypeModifiers ArrayTypeModifiers ::= ArrayTypeModifier+ ArrayTypeModifier ::= OpenParenthesis [ RankList ] CloseParenthesis RankList ::= Comma | RankList Comma ArrayNameModifier ::= ArrayTypeModifiers | ArraySizeInitializationModifier DelegateDeclaration ::= [ Attributes ] [ TypeModifier+ ] Delegate MethodSignature StatementTerminator MethodSignature ::= SubSignature | FunctionSignature 1.105.4 Type Members ImplementsClause ::= [ Implements ImplementsList ]
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
535
4.3
Inheritance
ImplementsList ::= InterfaceMemberSpecifier | ImplementsList Comma InterfaceMemberSpecifier InterfaceMemberSpecifier ::= NonArrayTypeName Period IdentifierOrKeyword MethodMemberDeclaration ::= MethodDeclaration | ExternalMethodDeclaration InterfaceMethodMemberDeclaration ::= InterfaceMethodDeclaration MethodDeclaration ::= SubDeclaration | MustOverrideSubDeclaration | FunctionDeclaration | MustOverrideFunctionDeclaration InterfaceMethodDeclaration ::= InterfaceSubDeclaration | InterfaceFunctionDeclaration SubSignature ::= Sub Identifier [ TypeParameterList ] [ OpenParenthesis [ ParameterList ] CloseParenthesis ] FunctionSignature ::= Funct ion Identifier [ TypeParameterList ] [ OpenParenthesis [ ParameterList ] CloseParenthesis ] [ As [ Attributes ] TypeName ] SubDeclaration ::= [ Attributes ] [ ProcedureModifier+ ] SubSignature [ HandlesOrImplements ] LineTerminator Block End Sub StatementTerminator MustOverrideSubDeclaration ::= [ Attributes ] MustOverrideProcedureModifier+ SubSignature [ HandlesOrImplements ] StatementTerminator
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
536
4.3
Inheritance
InterfaceSubDeclaration ::= [ Attributes ] [ InterfaceProcedureModifier+ ] SubSignature StatementTerminator FunctionDeclaration ::= [ Attributes ] [ ProcedureModifier+ ] FunctionSignature [ HandlesOrImplements ] LineTerminator Block End Funct ion StatementTerminator MustOverrideFunctionDeclaration ::= [ Attributes ] MustOverrideProcedureModifier+ FunctionSignature [ HandlesOrImplements ] StatementTerminator InterfaceFunctionDeclaration ::= [ Attributes ] [ InterfaceProcedureModifier+ ] FunctionSignature StatementTerminator ProcedureModifier ::= AccessModifier | Shadows | Shared | Overridable | NotOverridable | Overrides | Overloads |
Partial
MustOverrideProcedureModifier ::= ProcedureModifier | MustOverride InterfaceProcedureModifier ::= Shadows | Overloads HandlesOrImplements ::= HandlesClause | ImplementsClause
537
4.3
Inheritance
ExternalMethodDeclaration ::= ExternalSubDeclaration | ExternalFunctionDeclaration ExternalSubDeclaration ::= [ Attributes ] [ ExternalMethodModifier+ ] Declare [ CharsetModifier ] Sub Identifier LibraryClause [ AliasClause ] [ OpenParenthesis [ ParameterList ] CloseParenthesis ] StatementTerminator ExternalFunctionDeclaration ::= [ Attributes ] [ ExternalMethodModifier+ ] Declare [ CharsetModifier ] Funct ion Identifier LibraryClause [ AliasClause ] [ OpenParenthesis [ ParameterList ] CloseParenthesis ] [ As [ Attributes ] TypeName ] StatementTerminator ExternalMethodModifier ::= AccessModifier | Shadows | Overloads CharsetModifier ::= Ansi | Unicode | Auto LibraryClause ::= Lib StringLiteral AliasClause ::= Alias StringLiteral ParameterList ::= Parameter | ParameterList Comma Parameter Parameter ::= [ Attributes ] [ ParameterModifier+ ] ParameterIdentifier [ As TypeName ] [ Equals ConstantExpression ] ParameterModifier ::= ByVal | ByRef | Optional | ParamArray ParameterIdentifier ::= Identifier IdentifierModifiers HandlesClause ::= [ Handles EventHandlesList ]
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
538
4.3
Inheritance
EventHandlesList ::= EventMemberSpecifier | EventHandlesList Comma EventMemberSpecifier EventMemberSpecifier ::= Identifier Period IdentifierOrKeyword | MyBase Period IdentifierOrKeyword | Me Period IdentifierOrKeyword ConstructorMemberDeclaration ::= [ Attributes ] [ ConstructorModifier+ ] Sub New [ OpenParenthesis [ ParameterList ] CloseParenthesis ] LineTerminator [ Block ] End Sub StatementTerminator ConstructorModifier ::= AccessModifier | Shared EventMemberDeclaration ::= RegularEventMemberDeclaration | CustomEventMemberDeclaration RegularEventMemberDeclaration ::= [ Attributes ] [ EventModifiers+ ] Event Identifier ParametersOrType [ ImplementsClause ] StatementTerminator InterfaceEventMemberDeclaration ::= [ Attributes ] [ InterfaceEventModifiers+ ] Event Identifier ParametersOrType StatementTerminator ParametersOrType ::= [ OpenParenthesis [ ParameterList ] CloseParenthesis ] | As NonArrayTypeName EventModifiers ::= AccessModifier | Shadows | Shared InterfaceEventModifiers ::= Shadows
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
539
4.3
Inheritance
CustomEventMemberDeclaration ::= [ Attributes ] [ EventModifiers+ ] Custom Event Identifier As TypeName [ ImplementsClause ] StatementTerminator EventAccessorDeclaration+ End Event StatementTerminator EventAccessorDeclaration ::= AddHandlerDeclaration | RemoveHandlerDeclaration | RaiseEventDeclaration AddHandlerDeclaration ::= [ Attributes ] AddHandler OpenParenthesis ParameterList CloseParenthesis LineTerminator [ Block ] End AddHandler StatementTerminator RemoveHandlerDeclaration ::= [ Attributes ] RemoveHandler OpenParenthesis ParameterList CloseParenthesis LineTerminator [ Block ] End RemoveHandler StatementTerminator RaiseEventDeclaration ::= [ Attributes ] RaiseEvent OpenParenthesis ParameterList CloseParenthesis LineTerminator [ Block ] End RaiseEvent StatementTerminator ConstantMemberDeclaration ::= [ Attributes ] [ ConstantModifier+ ] Const ConstantDeclarators StatementTerminator ConstantModifier ::= AccessModifier | Shadows
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
540
4.3
Inheritance
ConstantDeclarators ::= ConstantDeclarator | ConstantDeclarators Comma ConstantDeclarator ConstantDeclarator ::= Identifier [ As TypeName ] Equals ConstantExpression StatementTerminator VariableMemberDeclaration ::= [ Attributes ] VariableModifier+ VariableDeclarators StatementTerminator VariableModifier ::= AccessModifier | Shadows | Shared | ReadOnly | WithEvents |
Dim
VariableDeclarators ::= VariableDeclarator | VariableDeclarators Comma VariableDeclarator VariableDeclarator ::= VariableIdentifiers As ObjectCreationExpression | VariableIdentifiers [ As TypeName ] [ Equals Expression ] VariableIdentifiers ::= VariableIdentifier | VariableIdentifiers Comma VariableIdentifier VariableIdentifier ::= Identifier IdentifierModifiers ArraySizeInitializationModifier ::= OpenParenthesis BoundList CloseParenthesis [ ArrayTypeModifiers ] BoundList::= Bound | BoundList Comma Bound
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
541
4.3
Inheritance
Bound ::= Expression | 0 To Expression PropertyMemberDeclaration ::= RegularPropertyMemberDeclaration | MustOverridePropertyMemberDeclaration | AutoPropertyMemberDeclaration PropertySignature ::= Property Identifier [ OpenParenthesis [ ParameterList ] CloseParenthesis ] [ As [ Attributes ] TypeName ] RegularPropertyMemberDeclaration ::= [ Attributes ] [ PropertyModifier+ ] PropertySignature [ ImplementsClause ] LineTerminator PropertyAccessorDeclaration+ End Property StatementTerminator MustOverridePropertyMemberDeclaration ::= [ Attributes ] MustOverridePropertyModifier+ PropertySignature [ ImplementsClause ] StatementTerminator AutoPropertyMemberDeclaration ::= [ Attributes ] [ AutoPropertyModifier+ ] Property Identifier [ OpenParenthesis [ ParameterList ] CloseParenthesis ] [ As [ Attributes ] TypeName ] [ Equals Expression ] [ ImplementsClause ] LineTerminator | [ Attributes ] [ AutoPropertyModifier+ ] Property Identifier [ OpenParenthesis [ ParameterList ] CloseParenthesis ] As [ Attributes ] New [ NonArrayTypeName [ OpenParenthesis [ ArgumentList ] CloseParenthesis ] ] [ ObjectCreationExpressionInitializer ] [ ImplementsClause ] LineTerminator
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
542
4.3
Inheritance
InterfacePropertyMemberDeclaration ::= [ Attributes ] [ InterfacePropertyModifier+ ] PropertySignature StatementTerminator AutoPropertyModifier ::= AccessModifier | Shadows | Shared | Overridable | NotOverridable | Overrides |
Overloads
MustOverridePropertyModifier ::= PropertyModifier | MustOverride InterfacePropertyModifier ::= Shadows | Overloads | Default | ReadOnly |
WriteOnly
PropertyAccessorDeclaration ::= PropertyGetDeclaration | PropertySetDeclaration PropertyGetDeclaration ::= [ Attributes ] [ AccessModifier ] Get LineTerminator [ Block ] End Get StatementTerminator PropertySetDeclaration ::= [ Attributes ] [ AccessModifier ] Set [ OpenParenthesis [ ParameterList ]
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
543
4.3
Inheritance
CloseParenthesis ] LineTerminator [ Block ] End Set StatementTerminator OperatorDeclaration ::= [ Attributes ] [ OperatorModifier+ ] Operator OverloadableOperator OpenParenthesis ParameterList CloseParenthesis [ As [ Attributes ] TypeName ] LineTerminator [ Block ] End Operator StatementTerminator OperatorModifier ::= Public | Shared | Overloads | Shadows | Widening | Narrowing OverloadableOperator ::= + | - | * | / | \ | & | Like | Mod | And | Or | Xor | ^ | < < | > > | = | < > | > | < | > = | < = | Not | IsTrue | IsFalse | CType 1.105.5 Statements Statement ::= LabelDeclarationStatement | LocalDeclarationStatement | WithStatement | SyncLockStatement | EventStatement | AssignmentStatement | InvocationStatement | ConditionalStatement | LoopStatement | ErrorHandlingStatement | BranchStatement | ArrayHandlingStatement | UsingStatement Block ::= [ Statements+ ] LabelDeclarationStatement ::= LabelName :
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
544
4.3
Inheritance
LabelName ::= Identifier | IntLiteral Statements ::= [ Statement ] | Statements : [ Statement ] LocalDeclarationStatement ::= LocalModifier VariableDeclarators StatementTerminator LocalModifier ::= Stat i c | Dim | Const WithStatement ::= With Expression StatementTerminator [ Block ] End With StatementTerminator SyncLockStatement ::= SyncLock Expression StatementTerminator [ Block ] End SyncLock StatementTerminator EventStatement ::= RaiseEventStatement | AddHandlerStatement | RemoveHandlerStatement RaiseEventStatement ::= RaiseEvent IdentifierOrKeyword [ OpenParenthesis [ ArgumentList ] CloseParenthesis ] StatementTerminator AddHandlerStatement ::= AddHandler Expression Comma Expression StatementTerminator RemoveHandlerStatement ::= RemoveHandler Expression Comma Expression StatementTerminator AssignmentStatement ::= RegularAssignmentStatement |
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
545
4.3
Inheritance
CompoundAssignmentStatement | MidAssignmentStatement RegularAssignmentStatement ::= Expression Equals Expression StatementTerminator CompoundAssignmentStatement ::= Expression CompoundBinaryOperator [ LineTerminator ] Expression StatementTerminator CompoundBinaryOperator ::= ^ = | * = | / = | \ = | + = | - = | & = | < < = |
> > =
MidAssignmentStatement ::= Mid [ $ ] OpenParenthesis Expression Comma Expression [ Comma Expression ] CloseParenthesis Equals Expression StatementTerminator InvocationStatement ::= [ Call ] InvocationExpression StatementTerminator ConditionalStatement ::= IfStatement | SelectStatement IfStatement ::= BlockIfStatement | LineIfThenStatement BlockIfStatement ::= If BooleanExpression [ Then ] StatementTerminator [ Block ] [ ElseIfStatement+ ] [ ElseStatement ] End If StatementTerminator ElseIfStatement ::= ElseIf BooleanExpression [ Then ] StatementTerminator [ Block ] ElseStatement ::= Else StatementTerminator [ Block ]
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
546
4.3
Inheritance
LineIfThenStatement ::= I f BooleanExpression Then Statements [ Else Statements ] StatementTerminator SelectStatement ::= Select [ Case ] Expression StatementTerminator [ CaseStatement+ ] [ CaseElseStatement ] End Select StatementTerminator CaseStatement ::= Case CaseClauses StatementTerminator [ Block ] CaseClauses ::= CaseClause | CaseClauses Comma CaseClause CaseClause ::= [ Is [ LineTerminator ] ] ComparisonOperator [ LineTerminator ] Expression | Expression [ To Expression ] ComparisonOperator ::= = | < > | < | > | > = | < = CaseElseStatement ::= Case Else StatementTerminator [ Block ] LoopStatement ::= WhileStatement | DoLoopStatement | ForStatement | ForEachStatement WhileStatement ::= While BooleanExpression StatementTerminator [ Block ] End While StatementTerminator
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
547
4.3
Inheritance
DoLoopStatement ::= DoTopLoopStatement | DoBottomLoopStatement DoTopLoopStatement ::= Do [ WhileOrUntil BooleanExpression ] StatementTerminator [ Block ] Loop StatementTerminator DoBottomLoopStatement ::= Do StatementTerminator [ Block ] Loop WhileOrUntil BooleanExpression StatementTerminator WhileOrUntil ::= While | Unti l ForStatement ::= For LoopControlVariable Equals Expression To Expression [ Step Expression ] StatementTerminator [ Block ] [ Next [ NextExpressionList ] StatementTerminator ] LoopControlVariable ::= Identifier [ IdentifierModifiers As TypeName ] | Expression NextExpressionList ::= Expression | NextExpressionList Comma Expression ForEachStatement ::= For Each LoopControlVariable In [ LineTerminator ] Expression StatementTerminator [ Block ] [ Next [ NextExpressionList ] StatementTerminator ] ErrorHandlingStatement ::= StructuredErrorStatement | UnstructuredErrorStatement
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
548
4.3
Inheritance
StructuredErrorStatement ::= ThrowStatement | TryStatement TryStatement ::= Try StatementTerminator [ Block ] [ CatchStatement+ ] [ FinallyStatement ] End Try StatementTerminator FinallyStatement ::= Finally StatementTerminator [ Block ] CatchStatement ::= Catch [ Identifier [ As NonArrayTypeName ] ] [ When BooleanExpression ] StatementTerminator [ Block ] ThrowStatement ::= Throw [ Expression ] StatementTerminator UnstructuredErrorStatement ::= ErrorStatement | OnErrorStatement | ResumeStatement ErrorStatement ::= Error Expression StatementTerminator OnErrorStatement ::= On Error ErrorClause StatementTerminator ErrorClause ::=
GoTo - 1 | GoTo 0 |
GotoStatement |
Resume Next
549
4.3
Inheritance
ResumeClause ::= Next | LabelName BranchStatement ::= GotoStatement | ExitStatement | ContinueStatement | StopStatement | EndStatement | ReturnStatement GotoStatement ::= GoTo LabelName StatementTerminator ExitStatement ::= Exi t ExitKind StatementTerminator ExitKind ::= Do | For | While | Select | Sub | Function | Property | Try ContinueStatement ::= Continue ContinueKind StatementTerminator ContinueKind ::= Do | For | While StopStatement ::= Stop StatementTerminator EndStatement ::= End StatementTerminator ReturnStatement ::= Return [ Expression ] StatementTerminator ArrayHandlingStatement ::= RedimStatement | EraseStatement RedimStatement ::= ReDim [ Preserve ] RedimClauses StatementTerminator RedimClauses ::= RedimClause | RedimClauses Comma RedimClause RedimClause ::= Expression ArraySizeInitializationModifier EraseStatement ::= Erase EraseExpressions StatementTerminator
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
550
4.3
Inheritance
EraseExpressions ::= Expression | EraseExpressions Comma Expression UsingStatement ::= Using UsingResources StatementTerminator [ Block ] End Using StatementTerminator UsingResources ::= VariableDeclarators | Expression 1.105.6 Expressions Expression ::= SimpleExpression | TypeExpression | MemberAccessExpression | DictionaryAccessExpression | InvocationExpression | IndexExpression | NewExpression | CastExpression | OperatorExpression | ConditionalExpression | LambdaExpression | QueryExpression | XMLLiteralExpression | XMLMemberAccessExpression ConstantExpression ::= Expression SimpleExpression ::= LiteralExpression | ParenthesizedExpression | InstanceExpression |
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
551
4.3
Inheritance
SimpleNameExpression | AddressOfExpression LiteralExpression ::= Literal ParenthesizedExpression ::= OpenParenthesis Expression CloseParenthesis InstanceExpression ::= Me | MyClass | MyBase SimpleNameExpression ::= Identifier [ OpenParenthesis Of TypeArgumentList CloseParenthesis ] AddressOfExpression ::= AddressOf Expression TypeExpression ::= GetTypeExpression | TypeOfIsExpression | IsExpression | GetXmlNamespaceExpression GetTypeExpression ::= GetType OpenParenthesis GetTypeTypeName CloseParenthesis GetTypeTypeName ::= TypeName | OpenTypeName OpenTypeName ::= QualifiedOpenTypeName | NullableOpenTypeName QualifiedOpenTypeName ::= Identifier [ TypeArityList ] | Global Period IdentifierOrKeyword [ TypeArityList ] | QualifiedOpenTypeName Period IdentifierOrKeyword [ TypeArityList ] TypeArityList ::= ( Of [ CommaList ] ) CommaList ::= Comma | CommaList CommaNullableOpenTypeName ::= QualifiedOpenTypeName ?
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
552
4.3
Inheritance
TypeOfIsExpression ::= TypeOf Expression Is [ LineTerminator ] TypeName IsExpression ::= Expression Is [ LineTerminator ] Expression | Expression IsNot [ LineTerminator ] Expression GetXmlNamespaceExpression ::= GetXmlNamespace XMLNamespaceName ] CloseParenthesis OpenParenthesis [
MemberAccessExpression ::= [ MemberAccessBase ] Period IdentifierOrKeyword [ OpenParenthesis Of TypeArgumentList CloseParenthesis ] MemberAccessBase ::= Expression | BuiltInTypeName | Global | MyClass |
MyBase
DictionaryAccessExpression ::= [ Expression ] ! IdentifierOrKeyword InvocationExpression ::= Expression [ OpenParenthesis [ ArgumentList ] CloseParenthesis ] ArgumentList ::= PositionalArgumentList | PositionalArgumentList Comma NamedArgumentList | NamedArgumentList PositionalArgumentList ::= [ Expression ] | PositionalArgumentList Comma [ Expression ] NamedArgumentList ::= IdentifierOrKeyword ColonEquals Expression | NamedArgumentList Comma IdentifierOrKeyword ColonEquals Expression
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
553
4.3
Inheritance
IndexExpression ::= Expression OpenParenthesis [ ArgumentList ] CloseParenthesis NewExpression ::= ObjectCreationExpression | ArrayExpression | AnonymousObjectCreationExpression AnonymousObjectCreationExpression ::= New ObjectMemberInitializer ObjectCreationExpression ::= New NonArrayTypeName [ OpenParenthesis [ ArgumentList ] CloseParenthesis ] [ ObjectCreationExpressionInitializer ] ObjectCreationExpressionInitializer ::= ObjectMemberInitializer | ObjectCollectionInitializer ObjectMemberInitializer ::= With OpenCurlyBrace FieldInitializerList CloseCurlyBrace FieldInitializerList ::= FieldInitializer | FieldInitializerList , FieldInitializer FieldInitializer ::= [ [ Key ] . IdentifierOrKeyword = ] Expression ObjectCollectionInitializer ::= From CollectionInitializer CollectionInitializer ::= OpenCurlyBrace [ CollectionElementList ] CloseCurlyBrace CollectionElementList ::= CollectionElement | CollectionElementList Comma CollectionElement CollectionElement ::= Expression | CollectionInitializer
554
4.3
Inheritance
ArrayExpression ::= ArrayCreationExpression | ArrayLiteralExpression ArrayCreationExpression ::= New NonArrayTypeName ArrayNameModifier CollectionInitializerArrayLiteralExpression ::= CollectionInitializer CastExpression ::= DirectCast OpenParenthesis Expression Comma TypeName CloseParenthesis | TryCast OpenParenthesis Expression Comma TypeName CloseParenthesis | CType OpenParenthesis Expression Comma TypeName CloseParenthesis | CastTarget OpenParenthesis Expression CloseParenthesis CastTarget ::= CBool | CByte | CChar | CDate | CDec | CDbl | CInt | CLng | CObj | CSByte | CShort | CSng | CStr | CUInt | CULng | CUShort OperatorExpression ::= ArithmeticOperatorExpression | RelationalOperatorExpression | LikeOperatorExpression | ConcatenationOperatorExpression | ShortCircuitLogicalOperatorExpression | LogicalOperatorExpression | ShiftOperatorExpression ArithmeticOperatorExpression ::= UnaryPlusExpression | UnaryMinusExpression | AdditionOperatorExpression | SubtractionOperatorExpression | MultiplicationOperatorExpression | DivisionOperatorExpression |
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
555
4.3
Inheritance
ModuloOperatorExpression | ExponentOperatorExpression UnaryPlusExpression ::= + Expression UnaryMinusExpression ::= - Expression AdditionOperatorExpression ::= Expression + [ LineTerminator ] Expression SubtractionOperatorExpression ::= Expression - [ LineTerminator ] Expression MultiplicationOperatorExpression ::= Expression * [ LineTerminator ] Expression DivisionOperatorExpression ::= FPDivisionOperatorExpression | IntegerDivisionOperatorExpression FPDivisionOperatorExpression ::= Expression / [ LineTerminator ] Expression IntegerDivisionOperatorExpression ::= Expression \ [ LineTerminator ] Expression ModuloOperatorExpression ::= Expression Mod [ LineTerminator ] Expression ExponentOperatorExpression ::= Expression ^ [ LineTerminator ] Expression RelationalOperatorExpression ::= Expression = [ LineTerminator ] Expression | Expression < > [ LineTerminator ] Expression | Expression < [ LineTerminator ] Expression | Expression > [ LineTerminator ] Expression | Expression < = [ LineTerminator ] Expression | Expression > = [ LineTerminator ] Expression LikeOperatorExpression ::= Expression Like [ LineTerminator ] Expression ConcatenationOperatorExpression ::= Expression & [ LineTerminator ] Expression LogicalOperatorExpression ::= Not Expression | Expression And [ LineTerminator ] Expression |
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
556
4.3
Inheritance
Expression Or [ LineTerminator ] Expression | Expression Xor [ LineTerminator ] Expression ShortCircuitLogicalOperatorExpression ::= Expression AndAlso [ LineTerminator ] Expression | Expression OrElse [ LineTerminator ] Expression ShiftOperatorExpression ::= Expression < < [ LineTerminator ] Expression | Expression > > [ LineTerminator ] Expression BooleanExpression ::= Expression LambdaExpression ::= ExpressionLambda | StatementLambda ExpressionLambda ::= Function [ OpenParenthesis [ ParametertList ] CloseParenthesis ] Expression | Sub [ OpenParenthesis [ ParametertList ] CloseParenthesis ] Expression StatementLambda ::= FunctionStatementLambda | SubStatementLambda FunctionStatementLambda ::= Function [ OpenParenthesis [ ParametertList ] CloseParenthesis ] [ As TypeName ] LineTerminator Block
End Function
557
4.3
Inheritance
QueryExpression ::= FromOrAggregateQueryOperator | QueryExpression QueryOperator FromOrAggregateQueryOperator ::= FromQueryOperator | AggregateQueryOperator JoinOrGroupJoinQueryOperator := JoinQueryOperator | GroupJoinQueryOperator QueryOperator ::= FromQueryOperator | AggregateQueryOperator | SelectQueryOperator | DistinctQueryOperator | WhereQueryOperator | OrderByQueryOperator | PartitionQueryOperator | LetQueryOperator | GroupByQueryOperator | JoinOrGroupJoinQueryOperator CollectionRangeVariableDeclarationList ::= CollectionRangeVariableDeclaration | CollectionRangeVariableDeclarationList Comma CollectionRangeVariableDeclaration CollectionRangeVariableDeclaration ::= Identifier [ As TypeName ] In [ LineTerminator ] Expression ExpressionRangeVariableDeclarationList ::= ExpressionRangeVariableDeclaration | ExpressionRangeVariableDeclarationList Comma ExpressionRangeVariableDeclaration ExpressionRangeVariableDeclaration ::= Identifier [ As TypeName ] Equals Expression
558
4.3
Inheritance
FromQueryOperator ::= [ LineTerminator ] From [ LineTerminator ] CollectionRangeVariableDeclarationList JoinQueryOperator ::= [ LineTerminator ] Jo in [ LineTerminator ] CollectionRangeVariableDeclaration [ JoinOrGroupJoinQueryOperator ] [ LineTerminator ] On [ LineTerminator ] JoinConditionList JoinConditionList ::= JoinCondition | JoinConditionList And [ LineTerminator ] JoinCondition JoinCondition ::= Expression Equals [ LineTerminator ] Expression LetQueryOperator ::= [ LineTerminator ] Let [ LineTerminator ] ExpressionRangeVariableDeclarationList SelectQueryOperator ::= [ LineTerminator ] Select [ LineTerminator ] ExpressionRangeVariableDeclarationList DistinctQueryOperator ::= [ LineTerminator ] Dist inc t [ LineTerminator ] WhereQueryOperator ::= [ LineTerminator ] Where [ LineTerminator ] BooleanExpression PartitionQueryOperator ::= [ LineTerminator ] Take [ LineTerminator ] Take [ LineTerminator ] Skip [ LineTerminator ] Skip [ LineTerminator ] Expression | While [ LineTerminator ] BooleanExpression | [ LineTerminator ] Expression | While [ LineTerminator ] BooleanExpression
559
4.3
Inheritance
OrderExpressionList ::= OrderExpression | OrderExpressionList Comma OrderExpression OrderExpression ::= Expression [ Ordering ] Ordering ::= Ascending | Descending GroupByQueryOperator ::= [ LineTerminator ] Group [ [ LineTerminator ] ExpressionRangeVariableDeclarationList ] [ LineTerminator ] By [ LineTerminator ] ExpressionRangeVariableDeclarationList [ LineTerminator ] Into [ LineTerminator ] ExpressionRangeVariableDeclarationList AggregateQueryOperator ::= [ LineTerminator ] Aggregate [ LineTerminator ] CollectionRangeVariableDeclaration [ QueryOperator+ ] [ LineTerminator ] Into [ LineTerminator ] ExpressionRangeVariableDeclarationList GroupJoinQueryOperator ::= [ LineTerminator ] Group Join [ LineTerminator ] CollectionRangeVariableDeclaration [ JoinOrGroupJoinQueryOperator ] [ LineTerminator ] On [ LineTerminator ] JoinConditionList [ LineTerminator ] Into [ LineTerminator ] ExpressionRangeVariableDeclarationList ConditionalExpression ::= If OpenParenthesis BooleanExpression Comma Expression Comma Expression CloseParenthesis | If OpenParenthesis Expression Comma Expression CloseParenthesis
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
560
4.3
Inheritance
XMLLiteralExpression ::= XMLDocument | XMLElement | XMLProcessingInstruction | XMLComment | XMLCDATASection XMLCharacter ::= < Unicode tab character (0x0009) > | < Unicode linefeed character (0x000A) > | < Unicode carriage return character (0x000D) > | < Unicode characters 0x0020 0xD7FF > | < Unicode characters 0xE000 0xFFFD > | < Unicode characters 0x10000 0x10FFFF > XMLString ::= XMLCharacter+ XMLWhitespace ::= XMLWhitespaceCharacter+ XMLWhitespaceCharacter ::= < Unicode carriage return character (0x000D) > | < Unicode linefeed character (0x000A) > | < Unicode space character (0x0020) > | < Unicode tab character (0x0009) > XMLNameCharacter ::= XMLLetter | XMLDigit | . | - | _ | : | XMLCombiningCharacter | XMLExtender XMLNameStartCharacter ::= XMLLetter | _ | : XMLName ::= XMLNameStartCharacter [ XMLNameCharacter+ ] XMLLetter ::= < Unicode character as defined in the Letter production of the XML 1.0 specification > XMLDigit ::= < Unicode character as defined in the Digit production of the XML 1.0 specification > Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved. 561
4.3
Inheritance
XMLCombiningCharacter ::= < Unicode character as defined in the CombiningChar production of the XML 1.0 specification > XMLExtender ::= < Unicode character as defined in the Extender production of the XML 1.0 specification > XMLEmbeddedExpression ::= < % = [ LineTerminator ] Expression [ LineTerminator ] % > XMLDocument ::= XMLDocumentPrologue [ XMLMisc+ ] XMLDocumentBody [ XMLMisc+ ] XMLDocumentPrologue ::= < ? xml XMLVersion [ XMLEncoding ] [ XMLStandalone ] [ XMLWhitespace ] ?
>
XMLVersion ::= XMLWhitespace version [ XMLWhitespace ] = [ XMLWhitespace ] XMLVersionNumberValue XMLVersionNumberValue ::= SingleQuoteCharacter 1 . 0 SingleQuoteCharacter | DoubleQuoteCharacter 1 . 0 DoubleQuoteCharacter XMLEncoding ::= XMLWhitespace encoding [ XMLWhitespace ] = [ XMLWhitespace ] XMLEncodingNameValue XMLEncodingNameValue ::= SingleQuoteCharacter XMLEncodingName SingleQuoteCharacter | DoubleQuoteCharacter XMLEncodingName DoubleQuoteCharacter XMLEncodingName ::= XMLLatinAlphaCharacter [ XMLEncodingNameCharacter+ ] XMLEncodingNameCharacter ::= XMLUnderscoreCharacter |
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
562
4.3
Inheritance
XMLLatinAlphaCharacter | XMLNumericCharacter | XMLPeriodCharacter | XMLDashCharacter XMLLatinAlphaCharacter ::= < Unicode Latin alphabetic character (0x0041-0x005a, 0x0061-0x007a) > XMLNumericCharacter ::= < Unicode digit character (0x0030-0x0039) > XMLHexNumericCharacter ::= XMLNumericCharacter | < Unicode Latin hex alphabetic character (0x0041-0x0046, 0x0061-0x0066) > XMLPeriodCharacter ::= < Unicode period character (0x002e) > XMLUnderscoreCharacter ::= < Unicode underscore character (0x005f) > XMLDashCharacter ::= < Unicode dash character (0x002d) > XMLStandalone ::= XMLWhitespace standalone [ XMLWhitespace ] = [ XMLWhitespace ] XMLYesNoValue XMLYesNoValue ::= SingleQuoteCharacter XMLYesNo SingleQuoteCharacter | DoubleQuoteCharacter XMLYesNo DoubleQuoteCharacter XMLYesNo ::= yes | no XMLMisc ::= XMLComment | XMLProcessingInstruction | XMLWhitespace XMLDocumentBody ::= XMLElement | XMLEmbeddedExpression
563
4.3
Inheritance
XMLElement ::= XMLEmptyElement | XMLElementStart XMLContent XMLElementEnd XMLEmptyElement ::= < XMLQualifiedNameOrExpression [ XMLAttribute+ ] [ XMLWhitepace ] / > XMLElementStart ::= < XMLQualifiedNameOrExpression [ XMLAttribute+ ] [ XMLWhitespace ] > XMLElementEnd ::= < / > | < / XMLQualifiedName [ XMLWhitespace ] > XMLContent ::= [ XMLCharacterData ] [ XMLNestedContent [ XMLCharacterData ] ]+ XMLCharacterData ::= < Any XMLCharacterDataString that does not contain the string "]]>" > XMLCharacterDataString ::= < Any Unicode character except < or & >+ XMLNestedContent ::= XMLElement | XMLReference | XMLCDATASection | XMLProcessingInstruction | XMLComment | XMLEmbeddedExpression XMLAttribute ::= XMLWhitespace XMLAttributeName [ XMLWhitespace ] = [ XMLWhitespace ] XMLAttributeValue | XMLWhitespace XMLEmbeddedExpression
564
4.3
Inheritance
XMLAttributeName ::= XMLQualifiedNameOrExpression | XMLNamespaceAttributeName XMLAttributeValue ::= DoubleQuoteCharacter [ XMLAttributeDoubleQuoteValueCharacter+ ] DoubleQuoteCharacter | SingleQuoteCharacter [ XMLAttributeSingleQuoteValueCharacter+ ] SingleQuoteCharacter | XMLEmbeddedExpression XMLAttributeDoubleQuoteValueCharacter ::= < Any XMLCharacter except <, &, or DoubleQuoteCharacter > | XMLReference XMLAttributeSingleQuoteValueCharacter ::= < Any XMLCharacter except <, &, or SingleQuoteCharacter > | XMLReference XMLReference ::= XMLEntityReference | XMLCharacterReference XMLEntityReference ::= & XMLEntityName ; XMLEntityName ::= l t | gt | amp | apos | quot XMLCharacterReference ::= & # XMLNumericCharacter+ ; | & # x XMLHexNumericCharacter+ ; XMLNamespaceAttributeName ::= XMLPrefixedNamespaceAttributeName | XMLDefaultNamespaceAttributeName XMLPrefixedNamespaceAttributeName ::= xmlns : XMLNamespaceName
565
4.3
Inheritance
XMLDefaultNamespaceAttributeName ::=
xmlns
XMLNamespaceName ::= XMLNamespaceNameStartCharacter [ XMLNamespaceNameCharacter+ ] XMLNamespaceNameStartCharacter ::= < Any XMLNameCharacter except : > XMLNamespaceNameCharacter ::= XMLLetter | _ XMLQualifiedNameOrExpression ::= XMLQualifiedName | XMLEmbeddedExpression XMLQualifiedName ::= XMLPrefixedName | XMLUnprefixedName XMLPrefixedName ::= XMLNamespaceName : XMLNamespaceName XMLUnprefixedName ::= XMLNamespaceName XMLProcessingInstruction ::= < ? XMLProcessingTarget [ XMLWhitespace [ XMLProcessingValue ] ] ? > XMLProcessingTarget ::= < Any XMLName except a casing permutation of the string "xml" > XMLProcessingValue ::= < Any XMLString that does not contain the string "?>" > XMLComment ::= < ! - - [ XMLCommentCharacter+ ] - - > XMLCommentCharacter ::= < Any XMLCharacter except dash (0x002D) > | - < Any XMLCharacter except dash (0x002D) > XMLCDATASection ::=
< ! [ CDATA [ [ XMLCDATASectionString ] ] ] >
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
566
4.3
Inheritance
XMLCDATASectionString ::= < Any XMLString that does not contain the string "]]>" > XMLMemberAccessExpression ::= Expression . [ LineTerminator ] < XMLQualifiedName > | Expression . @ [ LineTerminator ] < XMLQualifiedName > | Expression . @ [ LineTerminator ] IdentifierOrKeyword | Expression . . . [ LineTerminator ] < XMLQualifiedName >
567
Change List
The following sections list of changes made to the specification between versions. The sections affected are listed after each change.
568
4.3
Inheritance
Added default instances. [11.6.2, 11.6.2.1, 11.6.2.2] Added generic types and methods. [2.3, 2.4.7, 4.1.1, 4.4.1, 4.5.1, 4.6, 4.7.1, 4.7.2, 4.9, 4.9.1, 4.9.2, 5.1, 5.2.2, 6.1, 6.3.1, 6.3.2, 7, 7.2, 7.5, 7.5.1, 7.6, 7.8, 7.8.1, 7.9, 7.11, 7.12, 7.12.1, 8.6, 8.10, 9.1, 9.2.1, 9.2.2, 9.3.2, 9.4, 9.6, 9.8.1, 9.8.2, 9.8.3, 10.2, 10.9.3, 10.10.1.2, 11.1, 11.4.4, 11.4.5, 11.5.1, 11.5.2, 11.5.3, 11.6, 11.6.2, 11.8, 11.8.2, 11.8.5, 11.10.1, 11.10.2, 11.10.3, 12.2.16, 12.3] 1.106.2 Minor changes Binary string comparisons are always used for conditional compilation. Otherwise, string comparisons would not work because text string comparisons depend on the run-time culture. [3.1.2] Types cannot inherit from a type that is directly or indirectly contained within it. Also clarified the examples. [4.3] Changed the grammar and spec so that enumerated types can use the System equivalents of the fundamental types as an underlying type. [7.4] We now allow a conversion between an array of an enumerated type and an array of the underlying type of the enumeration. [8.5, 8.8. 8.9] When overriding a method, you can override it with a MustOverr ide method, causing it to become abstract. [4.5.1] A type member can handle an event in its own class using Handles. [9.2.6] Methods and properties that are declared Overr idesnow assume Overloads which is more , logical than assuming Shadows. [4.3.3] Fields and local variables are allowed to initialize multiple variables at once using the As New syntax. [9.6] Removed the restriction that an inner Catch block cannot branch into an outer Try block. [10.10.1.2] Classes cannot inherit from System.MulticastDelegate. [7.5.1] Shared variables in structures can have initializers. [9.6.3]
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
569
4.3
Inheritance
Added a rule that numeric types are preferred over enumerated types when doing overload resolution against the literal 0. [11.8.1] Array size initializers can explicitly state a lower bound of zero. [9.6.3.3] Added an external checksum directive. [3.4] Array-size initializers on fields are allowed to be non-constant expressions. [9.6.3.3] Keywords with type characters are now treated as identifiers. [2.2] Constants, fields, properties, locals and parameters that have the same name as their type can be interpreted either as the member or the type for the purposes of member lookup. [11.6.1] Added a section talking about types restricted by the .NET Framework and moved discussion of System.Void to that section. [7, 7.13] When dealing with a project and a referenced assembly that define the same fully-qualified name, the projects type is preferred, otherwise the name is ambiguous. [4.7.2, 11.4.4]
On Error statements do not extend over the call to New at the beginning of a constructor.
[10.10.2] Resolving an overloaded call to a late bound call is disallowed if the containing type is an interface. [11.8.1]
Overloads and Shadows are not allowed in a standard module. [4.3.3]
When looking up in interfaces, a name shadowed in one path through the hierarchy is shadowed in all paths through the hierarchy. Previously, we would give an ambiguity error. [4.3.2] When binding through imports, types and type members are given preference over namespaces. [4.7.2, 11.6] Changing the default property of a type no longer requires Shadows. [9.7.3] Unreserved the contextual keywords Assembly, Ansi, Auto, Preserve, Unicode and Until. [2.3]
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
570
4.3
Inheritance
1.106.3 Clarifications/Errata Added a note that full-width/half-width equivalence only works on a whole-token basis (i.e. you cant mix it within a token.) [1.1] There are places in the language that allow regular type names but do not allow array type names. Added clarification to the grammar to call these out. [5.2, 7, 7.2, 7.5.1, 7.8.1, 7.9, 9.1, 9.2.5, 10.10.1.2, 11.10.1, 11.10.2, 11.10.3] The spec incorrectly states that colons can only be used as separators at the statement level. In fact, they can be used almost anywhere. Clarified the grammar and spec. [6, 6.2.1, 6.2.2, 6.2.3, 6.3, 6.4.1, 7.2, 7.4, 7.4.1, 7.5, 7.5.1, 7.6, 7.7, 7.8, 7.8.1, 9.2.1, 9.2.2, 9.3, 9.4, 9.5, 9.6, 9.7, 9.7.1, 9.7.2, 10] As a part of the previous bullet point, made labels a statement unto themselves and fixed up the grammar. [10, 10.1] Narrowed the scope of the grammar for delegate declarations to be more accurate. [7.10] Array covariance also includes interface implementation in addition to inheritance. [8.5] When implementing an interface or handling an event, identifiers after a period can match keywords. [9.1, 9.2.6] Split MustOverr ide declarations out from regular method and property declarations. [9.2.1, 9.7] The grammar was wrong for variable declarations Dim is a regular modifier, and not a separate element. [9.6] Spelled out how attributes on WithEvents fields are transmitted to the underlying synthetic members. [9.6.2] Invoking an event wont throw an exception if there are no listeners. [10.5.1] Expanded the RaiseEvent section to cover the avoidance of race conditions on raising an event. [10.5.1] Clarified that RaiseEvent takes an identifier, not an expression. [10.5.1]
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
571
4.3
Inheritance
Covered a corner case with Mid assignment. [10.6.3] Statements after a line I f are not optional. [10.8.1] Expanded the DoLoop grammar to make it more explicit. [10.9.1] Loop control variables in For loops can be of an array type. [10.9.2] The size modifier is not optional in an array creation expression. [11.10.2] Property return types can have attributes [9.7] Split out the interface versions of event, property and method declarations for grammatical clarity. [7.8.2, 9.2, 9.2.1, 9.4, 9.7]
MyClass and MyBase cannot stand alone and are moved to the qualified expression
A Handles clause can have more than one identifier, but only two identifiers are legal. [9.2.6] Conditional compilation only supports a subset of constant expressions. [3.1] Corrected references to System.Monitor to System.Threading.Moni tor[10.4] . Clarified that a compiler can put declarations into a particular namespace by default. [6.4] Re-throwing an exception (Throw with no argument) cannot occur inside of a Fina l l y block. [10.10.1.3] A conversion from a type to itself is considered widening. [8.8] Tried to clarify the exact behavior of DirectCasta bit better than before. [11.11] The element type of a collection in a For Each statement does not have to have an implicit conversion to the loop control variable/expression: the conversion can be of any kind. [10.9.3] Clarified decimal division behavior. [11.13.6]
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
572
4.3
Inheritance
Clarified that overloaded operators are not considered when converting Nothing to an empty string for the & operator. Also clarified the same behavior applies to the Like operator. [11.15, 11.16] Clarified that operations that have Object parameters might result in something other than Integer. [11.12.2] Added explicit operator type tables. [11.12.3, 11.13.1, 11.13.2, 11.13.3, 11.13.4, 11.13.5, 11.13.6, 11.13.7, 11.13.8, 11.14, 11.15, 11.16, 11.17, 11.17.1, 11.18] Tried to make the most specific rules more clear in intent. [11.8.1] Clarified that shadowing a ParamArray method by name and signature hides only that signature, even if the shadowing method matches the unexpanded signature of the ParamArray method. [4.3.3] Moved the rule about preferring fewer paramarray matches over more during overload resolution earlier in the process to match compiler (and desired) behavior. [11.8.1, 11.8.2] Array-size initializers and array-element initializers cannot be combined. [9.6.3.3] When specifying bounds on an array creation expression and supplying an array-element initializer, the bounds must be specified using constant expressions. [9.6.3.4] Added a discussion of boxing and unboxing, as well as limitations to our anti-aliasing of boxed types design. [8.6] Tried to clarify an obscure rule about enumerated types and For loops with Object loop control variables. [10.9.2] Clarified that mucking with the loop control variable during an Object loop doesnt change the type of the loop. [10.9.2] Noted that delegates and external methods can use duplicate parameter names. [9.2.5] Interfaces have a widening conversion to Object. [8.4, 8.8] Interfaces do not inherit from Object. [7.8.1]
573
4.3
Inheritance
Object is reference type. It is not a type that is neither a reference type nor a value type.
[7] Noted the position of System.ValueTypeand System.Enum in the value type hierarchy. [7.1] Called out the primitive type conversions we allow when boxed as Object. [8.6] Expanded the explanation of a delegates members. [7.10] Expanded discussion of implicit locals. [10.1.1, 10.2.1] Clarified how static initializers work. [10.2] Noted which synthetic names are ignored by name binding. [9.4, 9.4.1, 9.7.1, 9.7.2] Exceptions caught in TryCatch blocks store their exceptions in the Err object. [10.10.1.2] Called out the presence of the identity conversion. [8] Clarified how late-bound accesses are treated in expression contexts. [11.1, 11.1.1, 11.3, 11.6, 11.8.1, 11.9] Corrected rules on when shared constructors execute. [9.3.2] 1.106.4 Miscellaneous Changed references to the name of the language from Microsoft Visual Basic .NET to Microsoft Visual Basic. The official name of the language itself is simply Visual Basic. Moved multi-token punctuators and operators such as <= or >>= into the lexical grammar for clarity. [2.5, 2.6, 5.2.2, 10.6.2, 10.8.2, 11.14] Removed the NamespaceOrTypeName production because it wasnt really needed. [4.7, 6.3.1, 6.3.2, 7] Removed the local variable productions because they were superfluous. [10.2] Consolidated all the productions that just included access modifiers and Shadows into a single production. [7, 7.4, 7.5, 7.6, 7.7, 7.8, 7.10] With the advent of a default response file for the command-line, all projects will import System by default, so I removed it from all examples. [Too many to count.]
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
574
4.3
Inheritance
Changed the suffix for preprocessing statement productions from Element to Statement and the prefix for conditional compilation statements from Conditional to CC. [3.1, 3.1.1, 3.1.2, 3.2, 3.3] Corrected the example in the Resume statement section. [10.10.2.3] Added Protected Friend to the modifier production. [4.6] Added some examples to array creation expressions. [11.10.2]
The default constructor for a MustInherit class is Protected, not Public. [9.3.1]
TypeOfIs will give a compile-time error if the expression can never be of the specified
type. [11.5.2] Default constructors will call InitializeComponent if emitted into a designer-generated class. [9.3.1] Accessing members of value types through type constraints is not allowed. [4.9.2] Added special boxing/unboxing behavior of System.Nullable(Of T). [8.6] A method can be more applicable than another method even if parameter types dont convert to each other, as long at least one parameter type is more applicable. [11.8.1]
System.ValueType is searched for members if a Structure constraint is specified. [4.9.2]
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
575
4.3
Inheritance
Circularity in structure definition is not allowed. [7.6.1] 1.107.2 Clarifications/Errata Updated SubSignatureand Funct ionSignature include Sub and Function keywords so that to they appear in delegate productions. [9.2.1] Added missing InvocationExpression to the Expression production. [11] Removed duplicate VariableInitializer production. [9.6.3.4] Removed incorrect statement that Catch statements can now specify Object as the exception type in a catch. [10.10.1.2] Corrected language about interaction between generic types and Protected members. [4.6] Corrected TypeArityList production. [11.5.1] Clarified that boxed value types are not copied when they are within any type, not just a boxed value type. [8.6] Corrected ForEachStatement production. [10.9.3] Corrected ConversionOperatorModifier production. [9.8.3] Corrected MemberAccessExpression production to include possible generic parameters. [11.6] External method offsets are specified using #, not @. [9.2.2] Removed unnecessary RegularInitializer production. [9.6.3, 9.6.3.1] Collapsed the operator syntactic productions into a single production, as the actual syntax is more flexible than implied. [9.8, 9.8.1, 9.8.2, 9.8.3] Clarified the use of Boolean expressions in the language. [10.8.1, 10.9.1, 10.9.2, 10.10.1.2] Noted that Protected constructors can only be called from the types constructors or derived types constructors. [4.6] Noted that default instances can only be created by compiler-generated code. [11.6.2.2]
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
576
4.3
Inheritance
Simplified Imports syntax productions for simplicity. [6.3, 6.3.1, 6.3.2] The rules about special types not being on the heap includes calls to instance methods of Object and System.ValueType [7.13] . Clarified that ParamArray signatures are not expanded when doing name hiding. [4.3.3] EnumDeclaration takes a NonArrayTypeName rather than just a QualifiedName. [7.4] Clarified the way that overloaded operators are mapped in and out of Visual Basic. [9.8.1, 9.8.2, 9.8.4] Clarified how binding works with case sensitive languages and declarations distinguished only by case. [4.1] 1.107.3 Miscellaneous All code samples were reviewed for correctness and updated as needed.
577
4.3
Inheritance
Allowed delegate-creation expressions to create delegates over signatures that do not exactly match. [11.10.3] Added nullable value types. [2.5, 7, 7.1.1, 7.13, 8.6.1, 8.8, 8.9, 8.11, 9.2.5, 9.6, 9.6.3.2, 9.8, 9.8.3, 10.2, 10.9.2, 11.5.3, 11.10.3, 11.12.3, 11.16, 11.17, 11.19, 11.20] Added query expressions. [11, 11.21, 11.21.1, 11.21.2, 11.21.3, 11.21.4, 11.21.5, 11.21.6, 11.21.7, 11.21.8, 11.21.9, 11.21.10, 11.21.11, 11.21.12, 11.21.13, 11.21.14] Added the conditional operator. [3.1, 11, 11.2, 11.22] Added XML namespace imports. [6.3, 6.3.3, 11.5, 11.5.4] Added XML literal expressions. [2, 2.3, 11, 11.23, 11.23.1, 11.23.2, 11.23.3, 11.23.4, 11.23.5, 11.23.6, 11.23.7, 11.23.8] Added XML member access expressions. [11, 11.10.4, 11.24] 1.108.2 Minor Changes Added a section clarifying the role of native conversions. [8.12, 11.11] The language now takes into account the Interna lsV i s i b l eToAtt r i bu te attribute. [4.5.1, 4.6, 7.5.2, 7.7.1, 7.12, 9.2.3] Group classes can now collect from multiple types, including generic types. [11.6.2.2] The members of Object are available on instances of interfaces, but are implicitly shadowed by the interfaces members. [7.8.2] Delegate types can now be compared for specificity based on return types in overload resolution. [11.8.1] Type arguments, whether inferred or not, must satisfy constraints for a method to be applicable in overload resolution. [11.8.2] Type argument inference is extended to allow multiple inferences for a type argument to be resolved. [8.13, 11.8.5] Type arguments can be inferred from the return type of an AddressOf expression. [11.8.5]
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
578
4.3
Inheritance
1.108.3 Clarifications/Errata Changed the terminology of the target of a method/property group to be target instead of the more confusing associated instance expression. [10.6.1, 11.1, 11.1.1, 11.3, 11.4.5, 11.6, 11.8, 11.9, 11.10.3] Clarified that type inference is done before overload resolution. [11.8.1] Clarified how paramarrays participate in overload resolution. [11.8.1, 11.8.2] Added missing rule that prefers more derived over more base methods with identical signatures in overload resolution. [11.8.1] Fixed incorrect rules for choosing less generic methods in overload resolution. [11.8.1] Clarified that regular initializers can initialize only on variable at a time. [9.6.3.1] Fixed simple name and qualified name binding to have correct behavior in the presence of type arguments. [4.7.1, 4.7.2, 11.4.4, 11.6] Removed the spurious rule that delegates cannot be created over methods declared MustOverr ide [11.10.3] . Clarified the terminology around late binding and AddressOf. [11.4.5] Fixed errors in the handling of arrays in type argument inference. [11.8.5] Clarified type argument inference using constructed types. [11.8.5] Clarified rules around reference conversions. [8.4] Clarified/corrected rules regarding type parameter constraints. [8.8, 8.9, 8.10, 8.12] Clarified that a variable name with an object initializer cannot have an array type modifier. [9.6.3.2] Clarified the behavior of the concatenation operator. [11.16] Clarified/corrected the behavior of Boolean expressions. [11.19] Corrected the incorrect result type of operators on Boolean values. [11.13.1, 11.13.2, 11.13.3, 11.13.4, 11.13.5, 11.13.6, 11.13.7, 11.18]
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
579
4.3
Inheritance
Added IsNot to the operator precedence table, it has the same precedence as Is. [11.12.1] Fixed CatchStatement production to indicate the As clause can be optional. [10.10.1.2] Fixed MemberAccessExpression production so that the period is not optional. [11.6] Corrected the grammar for conditional compilation expressions. [3.1] Reworked type constraints section, clarifying and allowing array covariance in satisfying type constraints. [4.9.2] Corrected that = and <> cannot be used on Object with Option Strict, but IsNot can. [6.2.2] Noted that Date can be used in constant expressions. [11.2] 1.108.4 Miscellaneous Removed unneeded DelegateCreationExpression production. [11.10, 11.10.3] Changed term from predefined to intrinsic. [8.11, 8.11.1, 8.11.2, 11.12.3, 11.15] Removed all references to foo, bar, and baz in examples.
4.3
Inheritance
Added multi-line lambda expressions that can contain statements and single-line statement lambdas. [11.20] Added generic variance. [4.9.1, 4.9.3, 7.11, 8.4.1, 8.8, 8.9] 1.109.2 Minor Changes When binding type names, inaccessible types are ignored. [4.7.1, 4.7.2, 11.4.4] Structures are no longer required to have at least one instance variable. [7.6.1] Optional parameters are now allowed to be structures, unconstrained type parameters and reference types. [9.2.5.3] The special application of parenthesis to the return value of a method is further restricted to the situation where there is only one accessible method. [11.8] Loop control variables with an identifier will define a new local identifier if local type inference is being used and the identifier is either undefined or resolves to a type [10.9.2, 10.9.3, 11.4.4] 1.109.3 Clarifications/Errata Moved multi-token punctuators and operators back into syntactic grammar, as whitespace rules apply to them. [2.5, 2.6, 3.1, 5.2.2, 9.8, 10.6.2, 10.8.2, 11.8, 11.14, 11.18] Fixed production for comparison operators in Select statements. [10.8.2] Fixed a number of errors in the productions for properties. [9.7] Fixed error in production for date literals. [2.4.6] Fixed error in XML character reference production. [11.23.4] Corrected assertion that property accessors can have the same access as their containing property. [9.7] Fixed spelling of Wend and Variantin the keyword table. [2.3] Clarified that reclassification of method pointers and lambda methods takes place anywhere a conversion is made. [11.1.1]
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
581
4.3
Inheritance
Fixed assertion that lambda methods without a contextual type become Funcinstead they become an anonymous lambda type which can be converted to types like Func. [11.1.1] Removed spurious XMLFragment and XmlNames productions. [11.23, 11.23.1] Corrected XMLAttr ibuteproduction to start with whitespace. [11.23.4] Clarified that implicitly declared locals in lambda expressions are declared in the method containing the expression, not the expression itself. [10.2.1] Corrected type name productions so that generic type names are handled properly. [4.7, 6.4.1, 7, 7.12, 9.2.6, 11.5.1] Cleaned up conversion descriptions and classifications. [8, 8.1, 8.3, 8.4, 8.5, 8.8, 8.9, 8.12, 11.5.2] Corrected the fact that array covariance doesnt just work for inheritance and implementation. [8.5, 8.8, 8.9] Fixed XMLEncodingNameCharacter production to include the dash character. [11.23.3] Clarified that string interning may occur (but may not). [2.4.4] Added a section on anonymous delegate conversions. [8.4.2, 8.8, 8.9] Defined the phrase intrinsic conversion. [8.11] Removed incorrect statement that lines excluded by conditional compilation must be lexically valid. [3.1] Clarified that the return types of an overridden method and the overriding method must match. [4.5.1] Removed incorrect assertion that Protectedconstructors can only be called from within a types constructors or in constructors of derived types. [4.6] Clarified that structures satisfy a New constraint. [4.9.1] Clarified that arrays implement IList(Of T) for types that their element type has a widening identity or reference conversion to. [7.9]
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
582
4.3
Inheritance
Clarified that boxed enumerated types can be unboxed to their underlying type. [8.6] Clarified that only intrinsic conversions can be used for optional parameter values. [9.2.5.3] Clarified that local declarations are made implicitly if the name does not resolve to anything else. [10.2.1] Corrected the grammar for argument lists. [5.2.2, 11.8] Clarified the use of IDispatchConstantAtt r i bu te IUnknownConstantAtt r i bu te argument list and in matching. [11.8.2.] Corrected the direction of conversion for the parameter types when matching signatures in delegate-creation expressions. [11.10.3] Attempted to clarify the position of the null value in the language. [7.1, 7.3, 8.4, 8.6, 10.10.1.3, 10.12.1, 11.1, 11.1.1, 11.4.1, 11.8.2, 11.14] The type of the expression in a TryCast does not have to be a value type. [11.11] Else If allows a space. [2.5, 3.2, 10.8.1] Whitespace is allowed by the preprocessor. [3.1] Constant string expressions are evaluated at compile-time. [2.4.4] Clarified nesting of Region directives. [3.3] Clarified that implementing methods must match the return type. [4.4.1] Clarified that enumerated types cannot declare type parameters. [4.9] Clarified which inconsistent combinations of type constraints are disallowed. [4.9.2] Attribute class need not have explicit AttributeUsage. [5.1] Positional attribute parameters may be Optional or ParamArray. [5.2] IdentifierModifiers ? and () may be combined. [7, 9.2.5, 9.6, 10.9.2] Delegate declarations may have attributes and modifiers. [7.10] Nested partial type declarations can occur in non-nested containers. [7.12]
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
583
4.3
Inheritance
Clarified the intrinsic Boolean conversions. [8.2] Handles clauses allow zero-argument delegate relaxation, and allows MyClass and MyBase when specifing event. [9.2.6, 11.4.3] Statements must start with a keyword or identifier. [10] Inference is always be used for local constant declarations which lack explicit types. [10.2] Clarified treatment of Boolean Expressions. [10.8.1, 10.9.1, 10.10.1.2, 11.19] Multiple Next statements are combined into one. [10.9.2, 10.9.3] The collection pattern matches extension methods. [10.9.3] The expression "Nothing" has classification "default value" and may be reclassified. [11.1.1, 11.4.1] There are limits on what kind of lambdas may be turned into expression trees. [11.1.1] Object is a valid type for a constant expression. Coercions to and from the null string are allowed in constants. Clarified that operations on constants must result in a valid type for a constant.. [11.2] Clarified the order of evaluation of invocation arguments. [11.3, 11.8] TypeArityList requires parentheses and "Of". [11.5.1] Member access expressions work on nullable types. [11.6] E.New is allowed when E is an enumeration. [11.6] Equality and inequality operators are not allowed on Object with Option Strict On. [11.12.2] Clarified behavior of AndAlso and OrElse for nullable booleans. [11.17.1] Join and GroupJoin query operators allow nested Join and GroupJoin query operators. [11.21, 11.21.5, 11.21.14] Only the Let query operator may specify the type of its expression range variable. [11.21] The "Group" clause of the "Group By" query operator is optional. [11.21.12]
Confidential Material Copyright Microsoft Corporation 2012. All Rights Reserved.
584
4.3
Inheritance
585