mirror of
https://github.com/dart-lang/sdk
synced 2024-09-16 03:56:57 +00:00
c4de71ede2
Original description: Change front_end handling of callable classes. This CL is the first in a series of CLs to change the handling of callable classes in Dart 2.0. For purposes of this description, a "callable class" is a class whose interface contains a `.call` method. In Dart 1.0, a callable class was considered to be a subtype of the `.call` method's function type. This allowed the user to create custom objects with similar behavior to closures, but with additional fields and methods. In Dart 2.0, a callable class is just an ordinary class, with no subtype relation to any particular function type. (Note however that it is still permissible for a class to declare that it "implements Function"). To reduce the amount of code broken by this change, a piece of syntactic sugar is being added: if an expression whose static type is a callable class appears where a function type is expected, an implicit tear-off of the `.call` method is inserted. Note that it is still possible at compile time to invoke an expression whose static type is a callable class, and it is still possible at runtime to invoke an expression whose runtime type is a callable class; in both cases, this is considered an implicit invocation of the class's `.call` method. This is unchanged from Dart 1.0 behavior. This CL introduces test cases for the new behavior, and implements the implicit tear-off of `.call` in the front end. Still to be implemented in future CLs: - Spec text needs to be written. - DDC/analyzer code needs to be written to perform implicit tear-offs of `.call`. - The subtyping algorithm in DDC, analyzer, VM, and dart2js needs to be changed so that callable classes are no longer considered subtypes of any particular function type. - A small corner case involving type parameters still needs to be addressed (see TODO in pkg/front_end/lib/src/fasta/type_inference/type_inferrer.dart). Fixes #32064. Change-Id: I0d397c608321e25ba42cc02aa8a516aa77aa7c9d Reviewed-on: https://dart-review.googlesource.com/41280 Reviewed-by: Vyacheslav Egorov <vegorov@google.com> Commit-Queue: Paul Berry <paulberry@google.com>
57 lines
1.5 KiB
Dart
57 lines
1.5 KiB
Dart
// Copyright (c) 2018, the Dart project authors. Please see the AUTHORS file
|
|
// for details. All rights reserved. Use of this source code is governed by a
|
|
// BSD-style license that can be found in the LICENSE file.
|
|
// Dart test program to test arithmetic operations.
|
|
|
|
import "package:expect/expect.dart";
|
|
|
|
class C1 {
|
|
int call(int i) => 2 * i;
|
|
}
|
|
|
|
class C2 implements Function {
|
|
int call(int i) => 2 * i;
|
|
}
|
|
|
|
class D {
|
|
static C1 c1 = new C1();
|
|
static dynamic d1 = new C1();
|
|
static C2 c2 = new C2();
|
|
static dynamic d2 = new C2();
|
|
|
|
void test1() {
|
|
// Implicitly invokes c1.call(1)
|
|
Expect.equals(c1(1), 2); //# 01: ok
|
|
// Implicitly invokes d1.call(1)
|
|
Expect.equals(d1(1), 2); //# 02: ok
|
|
// Implicitly invokes c2.call(1)
|
|
Expect.equals(c2(1), 2); //# 03: ok
|
|
// Implicitly invokes d2.call(1)
|
|
Expect.equals(d2(1), 2); //# 04: ok
|
|
}
|
|
|
|
static void test2() {
|
|
// Implicitly invokes c1.call(1)
|
|
Expect.equals(c1(1), 2); //# 05: ok
|
|
// Implicitly invokes d1.call(1)
|
|
Expect.equals(d1(1), 2); //# 06: ok
|
|
// Implicitly invokes c2.call(1)
|
|
Expect.equals(c2(1), 2); //# 07: ok
|
|
// Implicitly invokes d2.call(1)
|
|
Expect.equals(d2(1), 2); //# 08: ok
|
|
}
|
|
}
|
|
|
|
main() {
|
|
new D().test1();
|
|
D.test2();
|
|
// Implicitly invokes D.c1.call(1)
|
|
Expect.equals(D.c1(1), 2); //# 09: ok
|
|
// Implicitly invokes D.d1.call(1)
|
|
Expect.equals(D.d1(1), 2); //# 10: ok
|
|
// Implicitly invokes D.c2.call(1)
|
|
Expect.equals(D.c2(1), 2); //# 11: ok
|
|
// Implicitly invokes D.d2.call(1)
|
|
Expect.equals(D.d2(1), 2); //# 12: ok
|
|
}
|