On Sun, Feb 24, 2008 at 3:00 PM, Aristotle Pagaltzis <[EMAIL PROTECTED]> wrote:
>  Something like
>
>     path { $app_base_dir / $conf_dir / $foo_cfg . $cfg_ext }
>
>  where the operators in that scope are overloaded irrespective of
>  the types of the variables (be they plain scalar strings,
>  instances of a certain class, or whatever).

This is excellent.  I've long supported the one symbol-one meaning
idea.  There are some trade-offs, though, which I'll describe.  I'll
use Haskell as my object language, since it does this sort of operator
defining rather than operator overloading (only for infix binary
operators though, not full syntactic constructs).

Recently I implemented a Vector class in Haskell.  Vectors have
addition and multiplication, of a sort, so it makes sense to use the +
and * operators.   In Haskell overloading comes in type classes, where
the operators you overload need to have specific types and you also
have an implicit contract to obey some laws about them when you
overload.  In the context of Perl, types don't mean squat, but the
implicit laws are still important, so this argument still applies.
The Num class for these operators looks like this:

  class Num a where
    (+) :: a -> a -> a   -- take 2 arguments of type a and return 1 of type a
    (*) :: a -> a -> a
    ...

Which means that if I wanted to overload (+) to work on vectors, my
(+) would have to have type Vector -> Vector -> Vector.  That's fine,
it is that type.  And it's associative and commutative as the class
expects.

The trouble is with (*).  There are three types of "multiplication" on
vectors, with these types:

    Double -> Vector -> Vector
    Vector -> Double -> Vector
    Vector -> Vector -> Double  -- inner product

None of which is Vector -> Vector -> Vector as Num is expecting.

Now, not letting me overload (*) was a good idea on Num's part.
Normally if you have some v in Num, you can say (v*v) + v and it will
be legal.  But if v were a vector, then this wouldn't work, you'd end
up trying to add a scalar and a vector.  It's very simple, Vectors are
not Nums in the sense that the class requires

The way I solved this was to create my own class with special vector
operations, so the user of the module had to differentiate between
vector operations and scalar operations.

  class Vector v where
    (^+^) :: v -> v -> v
    (*^)  :: Double -> v -> v
    (^*^) :: v -> v -> Double

  x ^* y  = y *^ x
  x ^-^ y = x ^+^ (-1) *^ y

Which could be construed as annoying.

This problem could have been ameliorated in another way, namely to
have designed Num differently.  If Num separated the notions of
addition and multiplication, or even had been designed as a Vector
space in the first place, then I could have used + and * like I wanted
to.  But it wasn't, so I couldn't.  More below.

>  I hope it's obvious how such a thing would me implemented. Now,
>  if you used type-bound overloading, then the following two
>  expressions cannot yield the same result:
>
>     ( 2 / 3 ) * $x
>     2 * $x / 3
>
>  But if overloading was scope-bound, they would!

And here is why you need ad-hoc overloading in addition to scope-based
overloading.  There are times when I mean 2/3 < 4 to mean the symbolic
data structure representing that condition, and there are other times
when I just want to check whether 2/3 is less than 4!  Here's a
contrived example:

  my $expr = 1;
  my $count = 1;
  while ($count < 10) {
    print "$count> $expr\n";
    $expr = $expr + $expr;
    $count = $count + 1;
  }

With the intention of printing something like:

  1> 1
  2> 1 + 1
  3> (1 + 1) + (1 + 1)
  ...

Which is, of course, broken.  I would have to do something crazy with scopes:

  while ($count < 10) {
    print "$count> $expr\n";
    $expr = do { use Math::Symbolic; $expr + $expr };
    $count = $count + 1;
  }

Which could also be construed as annoying.

However, here $count + 1 and $expr + $expr are the same +.  They both
mean add.  Why should I have to play with scopes for this.  Contrast
this with Java, where 3 + 4 and "hello " + "world" are different +s
(unless you're used to thinking about monoids).  Getting those two to
coexist is the thing that should require playing with scopes (or
better yet, using diffrent symbols for the two of them).

I do think the best solution is a combination of overloading and
scoping.  That is, allow operator overloading, but not overloading the
name "+", rather overloading a specific +, such as &Math::infix:<+>.

This still allows poor usage as we've commonly seen with operator
overloading.  But it also allows well-behaved usage, which was
previously forbidden.

Luke

Reply via email to