Flutter: метод Hive 'get' метод вызван на нуль - PullRequest
1 голос
/ 14 апреля 2020

Я пытаюсь использовать Hive в своем приложении в качестве альтернативы Shared Preferences во Flutter. Тем не менее, я получаю сообщение об ошибке:

I/flutter ( 4004): The method 'get' was called on null.
I/flutter ( 4004): Receiver: null
I/flutter ( 4004): Tried calling: get("counter", defaultValue: 0)

E/flutter ( 4004): [ERROR:flutter/lib/ui/ui_dart_state.cc(157)] 
Unhandled Exception: HiveError: This should not happen. Please open an 
issue on GitHub.

Я выполнил все шаги, показанные в документации на pub.dev, однако, я не использую TypeAdapters, и я просто пытаюсь счетчик это инт. Это моя реализация:

var box = Hive.box('box');
int counter;

 void initHive() async {
 await openBox();
 getCounter();  //Updated code
 }

Future openBox() async {
var dir = await getApplicationDocumentsDirectory();
Hive.init(dir.path);
box = await Hive.openBox('box');
return;
}

void getCounter() { 
counter1 = box.get('counter1', defaultValue: 0);
// I am not storing any value initially, hence as it is null I want it 
//to return the value 0 but the 'get' method keeps getting called on 
//null.
}

void initState() {
initHive();
super.initState();
}

Я не уверен насчет нескольких вещей в Hive:

  • Когда я использую метод put(), сохраняется ли значение моих счетчиков?
  • Я инициализирую поле в одном файле дротика и вызываю это как глобальную переменную в моем приложении: это вызывает ошибку?
  • Кроме того, я только добавил зависимость от улья, потому что я не нужен отдых. Может ли это быть причиной проблемы?
  • Должен ли я открывать коробку каждый раз при выполнении операции get & put.

1 Ответ

1 голос
/ 15 апреля 2020

Редактировать
Вы можете инициализировать Hive box в main

Box box;

Future<void> main() async{
  WidgetsFlutterBinding.ensureInitialized();
  var dir = await getApplicationDocumentsDirectory();
  Hive.init(dir.path);
  box = await Hive.openBox('box');

  runApp(MyApp());
}

Редактировать
Проблема последовательность выполнения кода
Чтобы ваш обновленный код работал, вы должны поместить getCounter() в initHive()

Причина : Hive box не готова, потому что я / O нужно время, и когда выполнение getCounter() box все еще null
Если вы встретите The method 'get/put' was called on null., это означает, что ваш box еще не готов
Вы должны проверить async await и выполнение кода последовательность снова
И вам не нужно открывать box каждый раз

фрагмент кода

void initHive() async {
    await openBox();
    getCounter();
  } 

Вы можете скопировать и вставить полный код кода ниже
Шаг 1: Вам нужно await _openBox(), и для этого в initState() вы можете использовать функцию hiveOperation(), чтобы сделать async await
фрагмент кода

 void hiveOperation() async{
    await _openBox();
    updateInt();
  }

  @override
  void initState() {
    hiveOperation();
    super.initState();
  }

Шаг 2: https://pub.dev/packages/hive#usage, вы можете использовать Hive, как map. Не надо await Futures.
Вам не нужно await newBox.put('updateInt', updateInt);, просто newBox.put('updateInt', updateInt); будет работать

вывод

I/flutter ( 5675): 30

полный код

import 'package:flutter/material.dart';
import 'package:hive/hive.dart';
import 'package:path_provider/path_provider.dart';

Box box;

Future<void> main() async{
  WidgetsFlutterBinding.ensureInitialized();
  var dir = await getApplicationDocumentsDirectory();
  Hive.init(dir.path);
  box = await Hive.openBox('box');

  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> {
  int _counter = 0;
  int counter1;

  void _incrementCounter() {
    setState(() {
      _counter++;
    });
  }

  void getCounter() {
    counter1 = box.get('counter1', defaultValue: 0);
    _counter = box.get('counter1', defaultValue: 0);
    print(counter1);
    print(_counter);
// I am not storing any value initially, hence as it is null I want it
//to return the value 0 but the 'get' method keeps getting called on
//null.
  }

  @override
  void initState() {
    //initHive();
    getCounter();
    super.initState();
  }

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(
        title: Text(widget.title),
      ),
      body: Center(
        child: Column(
          mainAxisAlignment: MainAxisAlignment.center,
          children: <Widget>[
            Text(
              'You have pushed the button this many times:',
            ),
            Text(
              '$_counter',
              style: Theme.of(context).textTheme.headline4,
            ),
          ],
        ),
      ),
      floatingActionButton: FloatingActionButton(
        onPressed: _incrementCounter,
        tooltip: 'Increment',
        child: Icon(Icons.add),
      ),
    );
  }
}
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...