FutureBuilder Flutter 中未处理的异常

Sam*_*man 4 dart flutter

我尝试使用FutureBuilder小部件flutter从网络中获取一些数据。为此,我使用以下代码:

Future<List<Product>> getWishList() async {
http.Response response = await http.post(
  MY_URL,
  headers: {
    HttpHeaders.acceptHeader: 'application/json',
    HttpHeaders.contentTypeHeader: 'application/json; charset=utf-8'
  },
);
if (response.statusCode == 200) {
  List<Product> ret = List();
  Map<String, dynamic> result;
  try {
    result = json.decode(response.body);
    for (var i = 0; i < result['data'].length; i++) {
        ret.add(Product.fromJson(result['data'][i]));
    }
    return ret;
  } catch (e) {
    return throw Exception("Json parse error");
  }
} else {
  return throw Exception("network connection failed");
}
}
Run Code Online (Sandbox Code Playgroud)

和:

FutureBuilder(
            future: getWishList(),
            builder: (BuildContext context, AsyncSnapshot snapshot) {
              if(snapshot.connectionState == ConnectionState.done){
                  if(snapshot.hasError){
                    controller.forward(from: 0.0);
                    return Material(
                      child: Container(
                        margin: const EdgeInsets.only(top: 36.0),
                        width: double.infinity,
                        child: FadeTransition(
                          opacity: animation,
                          child: PButton(
                            onPressed: (){
                              setState(() {
                                getWishList();
                              });
                            },
                            child: Column(
                              children: <Widget>[
                                Icon(Icons.perm_scan_wifi,color: Colors.black,size: 76.0,),
                                SizedBox(height:24.0),
                                Text("Try again",style: TextStyle(fontSize: 16.0,color: const Color(0XFF222222)),),

                              ],
                            ),
                          ),
                        ),
                      ),
                    );
                  }else{
                    return new ListView(
                        children: <Widget>[
                          GestureDetector(
                              onTap:(){
                                setState(() {
                                  getWishList();
                                });
                              },
                              child: new Text("Every thing ok"))
                        ]);
                  }
              }else{
                return Center(
                  child: Container(
                      margin: const EdgeInsets.only(top: 36.0),
                      child: CircularProgressIndicator()),
                );
              }
            })
Run Code Online (Sandbox Code Playgroud)

现在,如果 http 响应第一次返回错误,一切都很好,但单击后Try again再次出错,此消息将显示在控制台上:

[VERBOSE-2:shell.cc(184)] Dart 错误:未处理的异常:异常:网络连接失败 _WishListState.getWishList (package:parchino/screen/screen_wish_list.dart:127:14) _WishListState.build... (package: parchino/screen/screen_wish_list.dart:65:33) State.setState (package:flutter/src/widgets/framework.dart:1130:30) _WishListState.build.. (package:parchino/screen/screen_wish_list.dart:64: 31) GestureRecognizer.invokeCallback (package:flutter/src/gestures/recognizer.dart:102:24) TapGestureRecognizer._checkUp (package:flutter/src/gestures/tap.dart:242:9) TapGestureRecognizer.handlePrimaryPointer(package:flutter/src/gestures/tap.dart: src/gestures/tap.dart:175:7) PrimaryPointerGestureRecognizer.handleEvent (package:flutter/src/gestures/recognizer.dart:315:9) PointerRouter._dispatch (package:flutter/src/gestures/pointer_router.dart<…>

Mul*_*dec 5

当您调用 时setState,您将小部件标记为 Dirty 并基本上告诉框架重建它,但只有在getWishList被调用之后(内部的那个setState)。由于它是一种async方法,因此它可以快速启动并重建小部件。

通过重新构建小部件,您可以重新构建FutureBuilder试图评估其未来的 。由于 future 是一个函数,它会调用它并对getWishList.

这使得两次调用相同的方法,因此两次调用 http 服务器的速度非常快。这些调用可能存在冲突,并引发错误。

您不应Future直接在 中调用 a ,FutureBuilder而应使用先前获得的 a Future

Future<List<Product>> myFuture;

@override
void initState() {
  myFuture = getWishList();
  super.initState();
}

Future<List<Product>> getWishList() async {
 //Do your stuff
}
Run Code Online (Sandbox Code Playgroud)

然后在您的构建中,设置myFuture为 的未来,FutureBuilder在您的 中setStatemyFuture再次设置:

FutureBuilder(
  future: myFuture,
  builder: (BuildContext context, AsyncSnapshot snapshot) {
      //...
      setState(() {
        myFuture = getWishList();
      });
      //...
  }
);
Run Code Online (Sandbox Code Playgroud)

这将使setState集合成为一个新的未来,myFuture并要求小部件重建自身。作为FutureBuilder重建,它会评估myFuture而不是再次调用 http。