Contents
- Type System and Soundness
- Null Safety and Variable Initialization
- Error and Exception Handling
- Workflow: Resolving Static Analysis and Runtime Failures
- Examples
Type System and Soundness
Dart enforces a sound type system to ensure that variables match their compile-time types at runtime. To prevent runtime type errors:
- Method Overrides: Ensure subclass overrides maintain covariant return types and contravariant parameter types. Never tighten parameter types in a subclass unless explicitly using the
covariantkeyword to override safety checks. - Generics and Collection Types: Avoid assigning untyped collections (like
List<dynamic>) directly to strongly typed targets. Always provide explicit generic parameters (e.g.List<String>) during list or map initialization. - Explicit Downcasting: Avoid implicit downcasts from
dynamicvalues. Use explicit type casting (e.g.as List<Map<String, dynamic>>) only when the underlying structure is guaranteed to match. - Strict Configuration: Enable strict casting configurations in your project’s
analysis_options.yamlto enforce explicit casts and catch dynamic typing bugs during code analysis:analyzer: language: strict-casts: true
Null Safety and Variable Initialization
Properly manage variable states to avoid compile-time analyzer errors and runtime null pointer exceptions:
- Nullable Indicators: Append
?to types that are permitted to hold null values. Use!only when asserting that a value is definitively non-null. - Late Initialization: Apply the
latekeyword only when the variable is guaranteed to be initialized before its first access. If a late variable is accessed before initialization, aLateInitializationErroris thrown at runtime. - Wildcard Variables: Use a single underscore
_(supported in modern Dart versions) to declare local parameters or variables that are intentionally unread, satisfying unused variable lints.
Error and Exception Handling
Distinguish between recoverable app failures (Exceptions) and unrecoverable bugs (Errors):
- Catching Exceptions: Implement try-catch blocks to catch and handle subclasses of
Exception(e.g.FormatException,SocketException). These represent expected runtime failures that the application should handle gracefully. - Avoid Catching Errors: Do not catch
Erroror its subclasses (e.g.TypeError,RangeError). Errors represent coding bugs that must be corrected during development, not suppressed at runtime. - Rethrowing: When catching an exception to perform logging or cleanup, always use the
rethrowkeyword instead of throwing the caught exception again.rethrowpreserves the original stack trace.
Workflow: Resolving Static Analysis and Runtime Failures
Follow this checklist to identify, correct, and verify static and runtime bugs:
- [ ] Run static analyzer: Execute
dart analyze . --fatal-infosin the project root. - [ ] Apply automated fixes: Preview and apply mechanical adjustments using:
dart fix --dry-run dart fix --apply - [ ] Identify manual fixes: Address complex warnings:
- If a nullability conflict is reported: Assess the business logic, specify defaults using the coalescing operator
??, or use the null-aware chain?.. - If a covariant mismatch is reported: Adjust subclass parameters or apply the
covariantmodifier.
- If a nullability conflict is reported: Assess the business logic, specify defaults using the coalescing operator
- [ ] Test execution: Run
dart testorflutter testto ensure runtime soundness. - [ ] Verify late state changes: Inspect stack traces if a
TypeErrororLateInitializationErroroccurs at runtime, correcting initialization flows.
Examples
Correcting Dynamic List Assignments
// Fails Static Analysis due to List<dynamic> assignment:
void printScores(List<int> scores) => print(scores);
void badMain() {
final scores = []; // Inferred as List<dynamic>
scores.add(90);
printScores(scores); // Error: List<dynamic> cannot be assigned to List<int>
}
// Corrected Implementation:
void goodMain() {
final scores = <int>[]; // Explicitly typed
scores.add(90);
printScores(scores); // Compiles and runs safely
}
Method Overrides and Covariants
class Worker {
void performTask(Task t) {}
}
// Fails Static Analysis: Tightening parameter type from Task to CodingTask
class SoftwareEngineer extends Worker {
@override
void performTask(CodingTask t) {}
}
// Corrected Implementation using the covariant keyword:
class LeadSoftwareEngineer extends Worker {
@override
void performTask(covariant CodingTask t) {}
}
Safely Managing Late Initializations
class DatabaseClient {
// Late keyword defers null safety checks to runtime
late final String connectionString;
void initialize(String url) {
connectionString = url;
}
void connect() {
// If connect() is called before initialize(), this throws a LateInitializationError
print('Connecting to $connectionString');
}
}