【问题标题】:How can I satisfy '(int) => void' is not a subtype of type '((num) => void)?' exception?我怎样才能满足'(int)=> void'不是'((num)=> void)类型的子类型?例外?
【发布时间】:2021-06-15 06:36:02
【问题描述】:

我在我的 Flutter 项目中使用 pref 插件。我有两个滑块设置用于配置下限和上限。当下限大于上限或相反时,我想防止用户配置无效的组合设置。

不幸的是,PrefSlider 没有验证方法,所以我试图在 onChanged 处理程序中灭火:

      PrefSlider<int>(
        title: "Lower Limit",
        pref: "lower_limit",
        trailing: (num value) => Text("$value"),
        min: 0,
        max: 300,
        onChange: (value) {
          final upperLimit = PrefService.of(context).get<int>("upper_limit") ?? 300;
          if (value >= upperLimit) {
            PrefService.of(context).set<int>("lower_limit", upperLimit - 1);
          }
        },
      ),
      PrefSlider<int>(
        title: "Upper Limit",
        pref: "upper_limit",
        trailing: (num value) => Text("$value"),
        min: 0,
        max: 300,
        onChange: (value) {
          final lowerLimit = PrefService.of(context).get<int>("lower_limit") ?? 0;
          if (value <= lowerLimit) {
            PrefService.of(context).set<int>("upper_limit", lowerLimit + 1);
          }
        },
      ),

当我移动滑块并且 pref 尝试调用我的 onChanged 函数时,我得到了一个很大的异常:

E/flutter (21583): [ERROR:flutter/lib/ui/ui_dart_state.cc(199)] Unhandled Exception: type '(int) => void' is not a subtype of type '((num) => void)?'
E/flutter (21583): #0      _PrefSliderState._onChange (package:pref/src/slider.dart:90:18)
E/flutter (21583): #1      _SliderState._handleChanged (package:flutter/src/material/slider.dart:540:24)
E/flutter (21583): #2      _RenderSlider._startInteraction (package:flutter/src/material/slider.dart:1235:17)
E/flutter (21583): #3      _RenderSlider._handleTapDown (package:flutter/src/material/slider.dart:1296:5)
E/flutter (21583): #4      TapGestureRecognizer.handleTapDown.<anonymous closure> (package:flutter/src/gestures/tap.dart:580:61)
E/flutter (21583): #5      GestureRecognizer.invokeCallback (package:flutter/src/gestures/recognizer.dart:182:24)
E/flutter (21583): #6      TapGestureRecognizer.handleTapDown (package:flutter/src/gestures/tap.dart:580:11)
E/flutter (21583): #7      BaseTapGestureRecognizer._checkDown (package:flutter/src/gestures/tap.dart:287:5)
E/flutter (21583): #8      BaseTapGestureRecognizer.didExceedDeadline (package:flutter/src/gestures/tap.dart:258:5)
E/flutter (21583): #9      PrimaryPointerGestureRecognizer.didExceedDeadlineWithEvent (package:flutter/src/gestures/recognizer.dart:501:5)
E/flutter (21583): #10     PrimaryPointerGestureRecognizer.addAllowedPointer.<anonymous closure> (package:flutter/src/gestures/recognizer.dart:454:41)
E/flutter (21583): #11     _rootRun (dart:async/zone.dart:1346:47)
E/flutter (21583): #12     _CustomZone.run (dart:async/zone.dart:1258:19)
E/flutter (21583): #13     _CustomZone.runGuarded (dart:async/zone.dart:1162:7)
E/flutter (21583): #14     _CustomZone.bindCallbackGuarded.<anonymous closure> (dart:async/zone.dart:1202:23)
E/flutter (21583): #15     _rootRun (dart:async/zone.dart:1354:13)
E/flutter (21583): #16     _CustomZone.run (dart:async/zone.dart:1258:19)
E/flutter (21583): #17     _CustomZone.bindCallback.<anonymous closure> (dart:async/zone.dart:1186:23)
E/flutter (21583): #18     Timer._createTimer.<anonymous closure> (dart:async-patch/timer_patch.dart:18:15)
E/flutter (21583): #19     _Timer._runTimers (dart:isolate-patch/timer_impl.dart:395:19)
E/flutter (21583): #20     _Timer._handleMessage (dart:isolate-patch/timer_impl.dart:426:5)
E/flutter (21583): #21     _RawReceivePortImpl._handleMessage (dart:isolate-patch/isolate_patch.dart:184:12)

我就是不知道出了什么问题。不知何故,处理程序函数应该可以为空? '((num) =&gt; void)?' 中的问号是什么?这是插件尝试调用我的处理程序的代码部分:

  Future<void> _onChange(double value) async {
    final service = PrefService.of(context, listen: false);

    if (T == double) {
      service.set<double>(widget.pref, value);

      if (widget.onChange != null) {
        widget.onChange!(value);
      }
    } else if (T == int) { // for PrefSlider<int>
      service.set<int>(widget.pref, value.round());

      if (widget.onChange != null) {
        widget.onChange!(value.round()); // <- this is the call
      }
    } else if (T == num) {
      service.set<double>(widget.pref, value);

      if (widget.onChange != null) {
        widget.onChange!(value);
      }
    }
  }

从更广泛的角度来看,我想也许我可以将 minmax 字段转换为状态,并以某种方式在每次更改时重建,但这可能会破坏 UI。

【问题讨论】:

  • 该错误意味着它需要一个接受num 参数并且什么都不返回的函数(并且函数本身可能是null),但是您传递了一个接受int 的函数争论。这是不允许的,因为回调可能会使用 double 进行调用,而 void Function(int) 不会这样。但是,我无法解释为什么您会收到该错误(PrefSlider&lt;T&gt;onChangevoid Function(T),而您明确指定intT)。跨度>
  • 此外,widget.onChange!(value.round()) 似乎是错误的,因为它需要是 widget.onChange!(value.round() as T)(这是我尝试提炼示例时遇到的错误)。

标签: flutter dart callback sharedpreferences nullable


【解决方案1】:

将泛型类型改为num,而不是int

PrefSlider<num>

【讨论】:

  • 这可能会解决异常,但是这会导致分支到 service.set&lt;double&gt; 代码而不是 service.set&lt;int&gt; 就像它是 PrefSlider&lt;double&gt; 一样。
【解决方案2】:

看起来我可以通过为回调指定显式 num 类型来缓解异常:

  PrefSlider<int>(
    title: "Lower Limit",
    pref: "lower_limit",
    trailing: (num value) => Text("$value"),
    min: 0,
    max: 300,
    onChange: (num value) {
      final upperLimit = PrefService.of(context).get<int>("upper_limit") ?? 300;
      if (value >= upperLimit) {
        PrefService.of(context).set<int>("lower_limit", upperLimit - 1);
      }
    },
  ),
  PrefSlider<int>(
    title: "Upper Limit",
    pref: "upper_limit",
    trailing: (num value) => Text("$value"),
    min: 0,
    max: 300,
    onChange: (num value) {
      final lowerLimit = PrefService.of(context).get<int>("lower_limit") ?? 0;
      if (value <= lowerLimit) {
        PrefService.of(context).set<int>("upper_limit", lowerLimit + 1);
      }
    },
  ),

有趣的是,这些都是仅运行时的异常,如果我从 trailing: (num value) =&gt; Text("$value"), 中删除显式的 num,也会发生这种情况。

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 2022-10-17
    • 2022-10-31
    • 1970-01-01
    • 2020-01-14
    • 2019-03-27
    • 2021-06-21
    • 2021-10-20
    • 1970-01-01
    相关资源
    最近更新 更多