-
Notifications
You must be signed in to change notification settings - Fork 211
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Dart's odd problems #3637
Comments
Hi @seha-bot, thanks for the input! It's always good to have a debate about things that could work in a different way. Many of these points are well taken! In some other cases, however, I think it would be useful if you could be a bit more precise, as mentioned in some cases below. Now for some responses:
Dart does maintain heap soundness (that is, a variable with declared type What you're reporting in the linked issue dart-lang/sdk#55021 is that type inference may provide some type arguments that are determined by the context rather than the inferred expression itself. I don't think it's quite fair to say that this shows 'no real type safety'. A more factual phrasing could be "Dart type inference gives priority to the context type; please note that this can result in more general inferred type arguments than you expect". If you're not happy about the inferred type arguments you can always provide different type arguments manually.
This is a good one! It used to be a warning, but it was removed. Consider voting for dart-lang/sdk#59293 if you wish to restore the diagnostic message about this situation.
This is a very useful distinction. Specialization of a return type is a standard mechanism, and completely type safe (several languages support this), and there is no need to flag locations where this kind of specialization occurs. Specialization of a formal parameter type (which requires Note that using this mechanism is very similar to putting a type cast at the beginning of the method: class A { void m(num n) {...}}
class B extends A { void m(covariant int i) {...}
// Nearly the same thing:
class A { void m(num n) {...}}
class B extends A { void m(num n) {
int i = n as int; // Throws a type error unless `n` is an `int`.
...
} These two approaches are equally unsafe, but the version that uses
OK, The type The value of an expression of type Any object has a size (not that you can easily find it, but it will occupy a certain amount of storage in the heap), and that doesn't change just because that object is obtained as the result of evaluation of an expression of type
The fact that pattern matching exhaustiveness analysis doesn't recognize every situation where a switch is actually exhaustive is unavoidable. It may of course be possible to improve on the current algorithm, in particular in the example of dart-lang/sdk#58844, but we can't expect to achieve completeness in this area, for any language with pattern matching.
This is definitely working as intended. Do you actually want the semantics of |
Thank you @eernstg for your thorough reply. All of those cases have a valid reasoning for their existence. My goal is just to point them out because some may seem unintuitive for people who haven't come across them. I was wondering if you, perchance, have any odd cases to show me, so I can put them in my presentation? Thank you again! |
There is always the associativity of cascades and conditional expressions. And the ability to ignore Maybe the syntax for conditional imports: import "file1.dart" if (dart.library.io) "file2.dart"; (It's more readable when not on the same line.) Implicit const c = []; // No problem, value is automatically constant.
class C {
final c = []; // Error, value must be constant.
final x;
const C({this.x = []}); // Error, default value must be constant.
} (The default value not being implicitly constant was deliberate. The other one was probably a good idea, but was mostly due to being forgotten in the list of constant contexts.) |
Thank you so much! This is very helpful! |
😃 Here are some other cases: Web-numbers and native numbers differ: https://dart.dev/guides/language/numbers. The upper bound algorithm may choose a very general type in some cases: The fundamental properties of Some details about Compile-time constants (known as constant expressions): A tiny, very specialized sublanguage of Dart. Proposals about where and how to extend this sublanguage are coming up again and again, and it's very resource intensive to do it. So you may be surprised about what is expressible and what is not expressible as a constant expression, and if you can't express a specific thing then you may be stuck, with very little hope that it will be made possible in the next few versions of Dart. It is allowed to omit type arguments in type annotations (such as the declared type of a variable or a formal parameter, or a function return type), and they are then chosen by instantiation to bound. This means that
Singleton records need a comma, like |
I had no idea some of these existed 😮 Constant expressions should definitely be worked on some more as they are very useful. I have one question about Dart program termination: Consider these two examples: In this example all asynchronous tasks are finished before exiting: void main() {
print("start");
() async {
int i = 0;
while (i++ < 0xFFFFF) {
await null;
}
print("done async");
}();
print("done");
} In this example the program finishes without void main() async {
Future<void>? future;
future = () async {
await null;
await future;
}();
print("start");
await future;
print("done");
} This seems like a contradiction, but I might not understand how tasks are created and evaluated in Dart. Bonus question: Thank you!!!! PS. I will close this issue when I finish my presentation which is on Monday. |
In a browser: When you navigate away from the page. Native: When there's nothing left to do (in the main isolate). Which can lead to a surprising exit when you'd think the program was blocked at an async await. void main() async {
try {
await Completer().future;
} finally {
print("done");
}
} This looks like it would block at the await of a future that nobody will ever complete, but an await gives control back to the event loop, and the event loop is empty, so the program just exits. Your second example is like this. The future reopened by your async function call is awaited inside that function. But it's a passive deadlock, with nobody actually scheduled to do anything. Just a callback put on a future which never gets called, and the program ends because nobody is doing anything. (Also, you shouldn't get a stack overflow. Returning a future should await it inside the function, giving you a deadlock. It's not implemented correctly, though.) |
I just want to say that, while I've been using Dart for a month and run into tons of odd or unexpected behavior, I'm really happy that methods are checked with proper variance, following the Liskov Subsitution Principle (that is, the return type of a method should be a subtype of the overridden method's return type, while argument types should be supertypes). TypeScript, for all the increased strictness that has been added over the years, still does not, and cannot, do this checking on method overrides! It's odd to me to complain about type "unsafety" and then turn around and complain about, literally, type safety. The "covariant" keyword is shorthand for a downcast that is checked at runtime. It's just an extra feature (that I'm not sure if I'll ever use, but it's there). Oh, I'm also super grateful that Dart infers method argument types from the super method, because TypeScript does not do that, and it saves a bunch of typing! A huge limitation of Dart's generics that TypeScript doesn't have is that class type parameters are always covariant. So you can do this, with no compile-time error, even though it's not sound: abstract interface class Printer<T> {
void print(T t);
}
void printStringWithIntPrinter(Printer<int> p) {
final Printer<Object> q = p;
q.print("hello");
} (In fairness, TypeScript requires some coaxing sometimes when it comes to variance, but they've added In reference to dart-lang/sdk#55021, the fact that type parameters have runtime values (based on compile-time type inference) is pretty unusual in Dart, and I'm still getting used to it. In theory, it has some advantages, particularly around enforcing "runtime soundness." The inferred type parameters seem to be poor, in practice, because of "prioritizing" (only looking at) the context type. The fact that casting a List always requires allocating a new object is an example of something that's been hard to get my head around. For example, if I have a List of |
I believe type arguments are reified during runtime, so they're definitely different from other languages which erase them at compile time. As for variance, see dart-archive/linter#753 and dart-lang/sdk#57494 |
...
These are basically the two sides of one coin. We can defer some soundness checking to runtime because we have access to the reified type arguments of the object at runtime. (It also means that runtime type tests like |
First off I would like to apologise because this is not an issue with
invariant_collection
. I am writing here only because I don't know how to contact you ':-)Your attention to writing safe and readable code and your contribution to Dart made me reach out to you for assistance.
I am collecting examples of "odd" Dart code for my presentation.
What do I mean by "odd"?
Odd code is code which is either undefined or unclear. Undefined is self-explanatory, but I doubt there is anything undefined in the Dart standard (you tell me :D). Unclear is code which can confuse the reader and induce him to thinking it does thing X, when in reality it does thing Y.
Motivation
I am a member of a discord server focusing on embedded engineering and it features an "odd problems emporium" channel. People post odd code and without using a compiler you need to explain what that code does.
Example:
What does this program return? Solution is that this is UB as there is no sequence point.
What have I collected so far
No real type safety
dart-lang/sdk#55021
Default parameters are allowed to be overriden
the existence of
covariant
Return values of methods are allowed to be specialized, while for parameters you need
covariant
.I will present this as "odd" because most people will be confused after I show them these two examples in succession:
Slide 1:
Slide 2:
void is of type Null and has a size
Pattern matching is falsely non-exhaustive
#3633
Conditional member access operator discards the assignment operations
Note
I am not trying to get anyone to dislike Dart and all of these "issues" have a valid reasoning behind their existence.
Thank you for reading this and I hope you can provide some more examples of oddity in Dart. Also, would you like to say something so I can put it as a quote in the presentation?
Cheers
The text was updated successfully, but these errors were encountered: