目录

avoid_dynamic_calls

避免在 dynamic 目标上调用方法或访问属性。

此规则从 Dart 2.12 开始可用。

详细信息

#

应该避免在显式或隐式静态类型为 dynamic 的对象上调用方法或访问属性。动态调用在每个运行时环境和编译器中的处理方式略有不同,但大多数生产模式(甚至某些开发模式)都具有与动态调用相关的编译大小和运行时性能损失。

此外,类型为 dynamic 的目标会禁用大多数静态分析,这意味着与正确静态类型化的 Dart 代码相比,更容易导致运行时 NoSuchMethodErrorTypeError

对于 Object? 上存在的方法和属性,存在例外情况

  • a.hashCode
  • a.runtimeType
  • a.noSuchMethod(someInvocation)
  • a.toString()

... 这些成员在基于 Web 的运行时中动态调度,但在基于 VM 的运行时中不是。此外,它们非常常见,因此禁止使用 any.toString()any == true 等操作将会非常不利。

请注意,尽管 Function 是一种类型,但其语义与 dynamic 非常相似,并且调用类型为 Function 的对象也会触发此 lint。

允许在强制转换表达式(as dynamicas Function)上进行动态调用。

错误

dart
void explicitDynamicType(dynamic object) {
  print(object.foo());
}

void implicitDynamicType(object) {
  print(object.foo());
}

abstract class SomeWrapper {
  T doSomething<T>();
}

void inferredDynamicType(SomeWrapper wrapper) {
  var object = wrapper.doSomething();
  print(object.foo());
}

void callDynamic(dynamic function) {
  function();
}

void functionType(Function function) {
  function();
}

正确

dart
void explicitType(Fooable object) {
  object.foo();
}

void castedType(dynamic object) {
  (object as Fooable).foo();
}

abstract class SomeWrapper {
  T doSomething<T>();
}

void inferredType(SomeWrapper wrapper) {
  var object = wrapper.doSomething<Fooable>();
  object.foo();
}

void functionTypeWithParameters(Function() function) {
  function();
}

用法

#

要启用 avoid_dynamic_calls 规则,请在 analysis_options.yaml 文件中的 linter > rules 下添加 avoid_dynamic_calls

analysis_options.yaml
yaml
linter:
  rules:
    - avoid_dynamic_calls