top of page

[dart]意図しないところで変数の値が変わる


状況


アプリの動作を規定するクラスがある。「設定変更」の画面は既存のインスタンスのコピーを引数にとり、画面上に設定値を表示。「保存」ボタンを押すとこのインスタンスのコピーを既存のインスタンスと入れ替える。「保存」ボタンを押さずに戻った場合は既存のインスタンスを保持する。


ところが、ある程度操作して「保存」ボタンを押さずに戻った場合でも、既存のインスタンスが変更されてしまっている。



原因


「設定変更」の画面を作る際に、インスタンスを=で代入していた。


dartでは(というかC#などほとんどの言語でそうだが)クラスを=で代入しても参照渡しになる。つまり、変数のアドレスが渡されるので、代入先の値を変更すると元の値も変更される。上記の例では、「設定変更」の画面上でインスタンスのコピーの値を変更しているつもりが、既定のインスタンスの値まで変更しており、「保存」ボタンを押してインスタンスを入れ替える処理をしなくても値が変わってしまっていた。



解決策


=で渡さずに、新しいインスタンスを作る必要がある。dartではクラスをまるごとコピーしてくれるメソッドはないっぽいので、元の値を引き継げるようなメソッドを用意しておくと便利。

class settingData{
  String name;
  int id;

  //メンバ変数をコピーして新しいインスタンスを作るコンストラクタ
  settingData.copyFrom(settingData source){
    this.name=source.name;
    this.id=source.id;
  }
}
class SettingPage extends StatefulWidget{
  final settingData data;
    
  SettingPage({this.data})
}

//~~~~何らかのクラス~~~~~~

Future<void> goToSettingPage(settingData data) async{
  await Navigator.of(context).push(new MaterialPageRoute(
    builder: (context) =>
    // SettingPage(data: data)   ダメな例
    SettingPage(data: settingData.copyFrom(data))  //こうする
  ));
}

ただし、このやり方だとsettingDataクラスに新しいメンバ変数を追加するたびにcopyFrom()コンストラクタまで変更しないといけないのであまりスマートではない気もする。



最後に


かなり初歩的なミスですが、上記のようなシチュエーションは少なくないとは思います(ユーザが承認するまで仮のデータを使う)。変数の受け渡しには注意しましょう。

最新記事

すべて表示

やりたいこと タイトル通り。 2つのWidgetを画面上で1列に並べたい 1つは画面中央、もう1つは画面右端 RowのmainAxisAligmentでやろうとしたら地味に難しかった 解決策 以下のようにする Row( mainAxisAlignment: MainAxisAlignment.spaceBetween, children: [ const Expanded(

現象 あるアイコンは長押し時に所定の動作を実行します これを実現するために、GestureDetectorのonLongPressに処理を登録していました しかしいざビルドしてみると、アイコンが表示されたタイミングで処理が実行されてしまいました 以下が該当部分のソースコードです。 さあ、どこが間違っているでしょう? return GestureDetector( child: Containe

前提 以下のようなケースを考えます。 アイテムの一覧がある ある1つのアイテムの詳細を表示するページがある 詳細ページではそのアイテムの削除ができる 削除したら他のページ(一覧ページなど)に戻る これはアプリではよくあるパターンだと思います。 課題 reduxパターンを用いている、より具体的にはProviderで状態を管理している場合、以下のような構成になっているのではないでしょうか? アプリ全体

あなたの買い物をサポートする
アプリ Shop Plan

iphone6.5p2.png

​いつ何を買うかの計画を立てられるアプリです。

google-play-badge.png
Download_on_the_App_Store_Badge_JP_RGB_blk_100317.png

靴を大切にしよう!靴管理アプリ SHOES_KEEP

納品:iPhone6.5①.png

靴の履いた回数、お手入れ回数を管理するアプリです。

google-play-badge.png
Download_on_the_App_Store_Badge_JP_RGB_blk_100317.png

「後で読む」を忘れないアプリ ArticleReminder

気になった​Webサイトを登録し、指定時刻にリマインダを送れるアプリです

google-play-badge.png
Download_on_the_App_Store_Badge_JP_RGB_blk_100317.png
bottom of page