Sqflite выдает ошибку стека при попытке вставить данные

Я просто пытаюсь сохранить пользователя в моей базе данных. Следующее - мой класс модели пользователя

class User {
  int _id;
  String _userId;
  String _mobileNumber;

  User([this._userId, this._mobileNumber]);

  User.withId(this._id, this._userId, this._mobileNumber);

  int get id => _id;

  String get userId => _userId;

  String get mobileNumber => _mobileNumber;

  set userId(String newUserId) {
    this.userId = newUserId;
  }

  set mobileNumber(String newMobileNumber) {
    this.mobileNumber = newMobileNumber;
  }

  Map<String, dynamic> toMap() {
    var map = Map<String, dynamic>();
    if (id != null) {
      map["id"] = _id;
    }
    map["userId"] = _userId;
    map["mobileNumber"] = _mobileNumber;

    return map;
  }

  User.fromMapObject(Map<String, dynamic> map) {
    this._id = map["id"];
    this._mobileNumber = map["userId"];
    this._userId = map["mobileNumber"];
  }
}

Вот мой класс репозитория

class UserRepository {
  static UserRepository _userRepository;
  static Database _database;

  String userTable = "user";
  String colId = "id";
  String colUserId = "userId";
  String colMobileNumber = "mobileNumber";
  String databaseName = "dice.db";

  UserRepository._createInstance();

  factory UserRepository() {
    if (_userRepository == null) {
      _userRepository = UserRepository._createInstance();
    }
    return _userRepository;
  }

  Future<Database> get database async {
    if (_database == null) {
      _database = await initializeDatabase();
    }
    return _database;
  }

  Future<Database> initializeDatabase() async {
    Directory directory = await getApplicationDocumentsDirectory();
    String path = directory.path + databaseName;

    var userDatabase =
        await openDatabase(path, version: 1, onCreate: _createDb);
    return userDatabase;
  }

  void _createDb(Database db, int newversion) async {
    await db.execute(
        "CREATE TABLE $userTable($colId INTEGER PRIMARY KEY AUTOINCREMENT,$colUserId TEXT, $colMobileNumber TEXT)");
  }

  Future<List<Map<String, dynamic>>> getUserMapList() async {
    Database db = await this.database;
    return await db.query(userTable);
  }

  Future<int> insertUser(User user) async {
    Database db = await this.database;
    return await db.insert(userTable, user.toMap());
  }

  Future<int> updateNote(User user) async {
    var db = await this.database;
    return await db.update(userTable, user.toMap(),
        where: "$colId = ?", whereArgs: [user.id]);
  }

  Future<List<User>> getUserList() async {
    var userMapList = await getUserMapList();

    List<User> userList = List<User>();
    //We have only one user so i am only getting user at 0th position
    userList.add(User.fromMapObject(userMapList[0]));
    return userList;
  }
}

var userRepository = UserRepository();

Теперь в моем основном классе по нажатию кнопки я пишу следующий код, чтобы сохранить пользователя

  User user = User();
    user.mobileNumber = mobileData.data.mobile.toString();
    user.userId = mobileData.data.userid.toString();
    userRepository.insertUser(user).then((result) {
      if (result != 0) {
        print("Successfully saved to database");
      } else {
        print("Error saving to database");
      }
    });

Ниже приводится ошибка, когда я нажимаю кнопку

[ERROR:flutter/shell/common/shell.cc(184)] Dart Error: Unhandled exception:
E/flutter (23914): Stack Overflow
E/flutter (23914): #0      User.mobileNumber= (package:dice_clutter/repository/User.dart:20:3)
E/flutter (23914): #1      User.mobileNumber= (package:dice_clutter/repository/User.dart:21:10)

RootZone.runUnary (dart:async/zone.dart:1379:54)
E/flutter (23914): #19311  _FutureListener.handleValue (dart:async/future_impl.dart:129:18)
E/flutter (23914): #19312  Future._propagateToListeners.handleValueCallback (dart:async/future_impl.dart:642:45)
E/flutter (23914): #19313  Future._propagateToListeners (dart:async/future_impl.dart:671:32)
E/flutter (23914): #19314  Future._complete (dart:async/future_impl.dart:476:7)
E/flutter (23914): #19315  _SyncCompleter.complete (dart:async/future_impl.dart:51:12)
E/flutter (23914): #19316  _AsyncAwaitCompleter.complete (dart:async/runtime/libasync_patch.dart:28:18)
E/flutter (23914): #19317  _completeOnAsyncReturn (dart:async/runtime/libasync_patch.d          

Я создаю только одну ссылку из своего класса репозитория. Любая помощь будет оценена.

2 ответа

Решение

У вас есть стекопоток, потому что вы используете имя вашего сеттера

Изменить это:

set userId(String newUserId) {
  this.userId = newUserId;
}

set mobileNumber(String newMobileNumber) {
  this.mobileNumber = newMobileNumber;
}

К этому:

  set userId(String newUserId) {
     _userId = newUserId;
  }

  set mobileNumber(String newMobileNumber) {
    _mobileNumber = newMobileNumber;
   }

Похоже, проблема вызвана этим сеттером:

  set mobileNumber(String newMobileNumber) {
    this.mobileNumber = newMobileNumber;
  }

настройка this.mobileNumber приведет к запуску установщика, поэтому вызов внутри метода установщика вызовет бесконечный цикл и исключение переполнения стека.

Чтобы исправить это, установите приватную переменную:

  set mobileNumber(String newMobileNumber) {
    _mobileNumber = newMobileNumber;
  }

Вы должны сделать то же самое на своем установщике userId, поскольку это имеет ту же проблему.

Другие вопросы по тегам