The Artima Developer Community
Sponsored Link

.NET Buzz Forum
Obscure language bug #29: Intrinsics and operator overloading

0 replies on 1 page.

Welcome Guest
  Sign In

Go back to the topic listing  Back to Topic List Click to reply to this topic  Reply to this Topic Click to search messages in this forum  Search Forum Click for a threaded view of the topic  Threaded View   
Previous Topic   Next Topic
Flat View: This topic has 0 replies on 1 page
Paul Vick

Posts: 783
Nickname: paulv
Registered: Aug, 2003

Paul Vick is a Tech Lead on Visual Basic at Microsoft Corp.
Obscure language bug #29: Intrinsics and operator overloading Posted: Apr 25, 2006 3:12 PM
Reply to this message Reply

This post originated from an RSS feed registered with .NET Buzz by Paul Vick.
Original Post: Obscure language bug #29: Intrinsics and operator overloading
Feed Title: Panopticon Central
Feed URL: /error.aspx?aspxerrorpath=/rss.aspx
Feed Description: a blog on Visual Basic, .NET and other stuff
Latest .NET Buzz Posts
Latest .NET Buzz Posts by Paul Vick
Latest Posts From Panopticon Central

Advertisement

At the end of his entry on the concatenation operator, Bill observes some strange behavior around types that have a user-defined conversion to String but can’t be used with the concatenation operator. This is actually a design bug in our operator overloading resolution logic that we’re planning on fixing, since it won’t impact compatibility at all. To quote almost verbatim from the preliminary spec I wrote on it:

Consider the following program:

Module Module1

    Sub Main()

        Dim CI As New C()

 

        ' Error: '&' is not defined for 'String' and 'C'

        Console.WriteLine(CI & "world")

 

        Dim S As String = CI

 

        ' Works

        Console.WriteLine(S & "world")

    End Sub

End Module

 

Public Class C

    Public Shared Widening Operator CType(ByVal x As C) As String

        Return "Hello"

    End Operator

End Class

The first expression fails because of a problem in the way that the operators that participate in operator resolution are chosen. Given an operation on two types, X and Y, the operator that is used is resolved as follows:

  • If X and Y are both intrinsic types, the operator to use is determined by the operator tables in the language specification.
  • Otherwise, the user-defined operators in X and the user-defined operators in Y are collected together and overload resolution is used to determine what the best fit, if any, is.

When you mix user-defined and intrinsic types, however, the algorithm can fail unexpectedly. When interpreting the expression C & String, the second bullet point is applied since both types aren’t intrinsic. Since neither type has a user-defined & operator, a compile-time error is given.

(C# doesn’t have this problem because they don’t distinguish between user-defined operators and intrinsic operators: they collect all operators together and throw them into overload resolution. We can’t do this because our intrinsic operator resolution rules do not conform to standard overload resolution rules. According to standard overload resolution, 1 / 2 should result in a Decimal value, but we change the rules to make this result in a Double value. This means we can’t just throw the standard operators into overload resolution into the second bullet point above, because then something like X / Integer would result in Decimal while CInt(X) / Integer would result in Double.)

To solve this problem, operator resolution will pre-select a single intrinsic operator to participate in overload resolution. Thus, the rules for operator resolution will be expanded to:

  • If X and Y are both intrinsic types, look up the result type in our operator tables and use that.
  • If X is an intrinsic type, then
    • Collect all of the intrinsic types that Y converts to.
    • Choose the most encompassed type, T, from the list. If there is no single most encompassed type, then we don’t consider an intrinsic operator.
    • Lookup up the intrinsic operator for X and T, and call it O. If there is no intrinsic operator defined for those two types, then we don’t consider an intrinsic operator.
    • The set of operators to be considered is all the user-defined operators in Y, plus O, if it exists.
  • If Y is an intrinsic type, then perform the same steps as for X (obviously, both can’t be intrinsic types at this point).
  • Do overload resolution on the set of operators to be considered.

This algorithm would appear to give us what we want. It’s the same basic steps we go through today, with the addition of the “narrowest” intrinsic operator possible if one of the types is intrinsic. It should also solve Bill’s problem.

Read: Obscure language bug #29: Intrinsics and operator overloading

Topic: Codus featured on Scott Hanselman's HanselMinutes Previous Topic   Next Topic Topic: Evolution of WCM in MOSS 2007 - Webcast recording

Sponsored Links



Google
  Web Artima.com   

Copyright © 1996-2019 Artima, Inc. All Rights Reserved. - Privacy Policy - Terms of Use