NoSuchMethodError: The method 'forEach' was called on null

NoSuchMethodError: The method 'forEach' was called on null

用 Dio 获取数据后出现错误。

Exception has occurred.
NoSuchMethodError (NoSuchMethodError: The method 'forEach' was called on null.
Receiver: null
Tried calling: forEach(Closure: (String, dynamic) => Null))

我在这一行得到上面的错误jsonResponse.forEach((key, value) {

这就是我从 json:

获取数据的方式
    Future<List<AirQualityModel>> _getAirQuality() async {
var dio = Dio();
var cookieJar;
Response dioResponse;
Directory appDocDir = await getApplicationDocumentsDirectory();
String appDocPath = appDocDir.path;
try {
  cookieJar = PersistCookieJar(dir: appDocPath + "/.cookies/");
  dio.interceptors.add(CookieManager(cookieJar));
  dioResponse = await dio.get(
      "https://www.airvisual.com/api/v2/node/5ded3e13994dfe107f7013a0");
  debugPrint('dioResponse: ' + dioResponse.data.toString());
  debugPrint("response: " + dioResponse.statusMessage.toString());
  jsonResponse = json.decode(dioResponse.data);
  airqualityList = List<AirQualityModel>();
  jsonResponse?.forEach((key, value) {
    airqualityList = (jsonResponse['current'] as List)
        .map<AirQualityModel>((j) => AirQualityModel.fromJson(j))
        .toList();
  });
  debugPrint('List: $airqualityList');
  return airqualityList;
} catch (e) {
  print('catch error: $e');
}

}

这是我的 airquality_model.dart

class AirQualityModel {
final int pm25;
final int co2;
final int humidity;
final int temperature;

AirQualityModel(this.pm25, this.co2, this.humidity, this.temperature);

AirQualityModel.fromJson(Map<String, dynamic> json)
  : pm25 = json['p2'],
    co2 = json['co'],
    humidity = json['hm'],
    temperature=json['tp'];
}

这是我的调试控制台:

我已经更新了我的代码,我在解码 json 的地方设置了一个状态,现在我的调试控制台看起来像这样:

The following RangeError was thrown building:
RangeError (index): Invalid value: Valid value range is empty: 0

我该如何解决这个错误?

jsonResponse 可能为 null,因此在使用之前检查它,像这样 jsonResponse?.forEach((key, value)

您可以复制粘贴运行下面的完整代码
第 1 步:dioResponse.dataMap ,你不需要再做 json.decode
第二步:dioResponse.data["current"]不是List,可以直接用Future<AirQualityModel>

代码片段

Future<AirQualityModel> _getAirQuality() async {
    var dio = Dio();
    var cookieJar;
    Response dioResponse;
    Directory appDocDir = await getApplicationDocumentsDirectory();
    String appDocPath = appDocDir.path;
    try {
      cookieJar = PersistCookieJar(dir: appDocPath + "/.cookies/");
      dio.interceptors.add(CookieManager(cookieJar));
      dioResponse = await dio.get(
          "https://www.airvisual.com/api/v2/node/5ded3e13994dfe107f7013a0");

      return AirQualityModel.fromJson(dioResponse.data["current"]);
    } catch (e) {
      print('catch error: $e');
    }
  }

工作演示

完整代码

import 'dart:convert';
import 'dart:io';

import 'package:cookie_jar/cookie_jar.dart';
import 'package:dio/dio.dart';
import 'package:dio_cookie_manager/dio_cookie_manager.dart';
import 'package:flutter/material.dart';
import 'package:path_provider/path_provider.dart';

class AirQualityModel {
  final double pm25;
  final double co2;
  final double humidity;
  final double temperature;

  AirQualityModel(this.pm25, this.co2, this.humidity, this.temperature);

  AirQualityModel.fromJson(Map<String, dynamic> json)
      : pm25 = json['p2'].toDouble(),
        co2 = json['co'].toDouble(),
        humidity = json['hm'].toDouble(),
        temperature = json['tp'].toDouble();
}

void main() {
  runApp(MyApp());
}

class MyApp extends StatelessWidget {
  @override
  Widget build(BuildContext context) {
    return MaterialApp(
      title: 'Flutter Demo',
      theme: ThemeData(
        primarySwatch: Colors.blue,
        visualDensity: VisualDensity.adaptivePlatformDensity,
      ),
      home: MyHomePage(title: 'Flutter Demo Home Page'),
    );
  }
}

class MyHomePage extends StatefulWidget {
  MyHomePage({Key key, this.title}) : super(key: key);

  final String title;

  @override
  _MyHomePageState createState() => _MyHomePageState();
}

class _MyHomePageState extends State<MyHomePage> {
  Future<AirQualityModel> _future;

  Future<AirQualityModel> _getAirQuality() async {
    var dio = Dio();
    var cookieJar;
    Response dioResponse;
    Directory appDocDir = await getApplicationDocumentsDirectory();
    String appDocPath = appDocDir.path;
    try {
      cookieJar = PersistCookieJar(dir: appDocPath + "/.cookies/");
      dio.interceptors.add(CookieManager(cookieJar));
      dioResponse = await dio.get(
          "https://www.airvisual.com/api/v2/node/5ded3e13994dfe107f7013a0");

      return AirQualityModel.fromJson(dioResponse.data["current"]);
    } catch (e) {
      print('catch error: $e');
    }
  }

  @override
  void initState() {
    _future = _getAirQuality();
    super.initState();
  }

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(
        title: Text(widget.title),
      ),
      body: FutureBuilder(
          future: _future,
          builder: (context, AsyncSnapshot<AirQualityModel> snapshot) {
            switch (snapshot.connectionState) {
              case ConnectionState.none:
                return Text('none');
              case ConnectionState.waiting:
                return Center(child: CircularProgressIndicator());
              case ConnectionState.active:
                return Text('');
              case ConnectionState.done:
                if (snapshot.hasError) {
                  return Text(
                    '${snapshot.error}',
                    style: TextStyle(color: Colors.red),
                  );
                } else {
                  return Card(
                      elevation: 6.0,
                      child: Padding(
                        padding: const EdgeInsets.only(
                            top: 6.0, bottom: 6.0, left: 8.0, right: 8.0),
                        child: Row(
                          crossAxisAlignment: CrossAxisAlignment.start,
                          children: <Widget>[
                            Text("co2  ${snapshot.data.co2.toString()}"),
                            Spacer(),
                            Text(
                              "pm25 ${snapshot.data.pm25.toString()}",
                            ),
                          ],
                        ),
                      ));
                  ;
                }
            }
          }),
    );
  }
}