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()コンストラクタまで変更しないといけないのであまりスマートではない気もする。



最後に


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

最新記事

すべて表示

【Flutter/Dart】TextFieldで文字列をフォーマットする

やりたいこと TextFieldで入力フォームを作りたい。 例えば入力内容が金額の場合、3桁区切りで頭に¥を付けた表記にしたい。 ただしユーザにこれらを入力させるのではなく、ユーザはあくまで数字を入力するだけで、アプリ側で自動でフォーマットしたい。 方法...

【Flutter/dart】late変数が初期化されたかのチェック

やりたいこと 初期値さえ決まればあとは不変な変数がある ただし、コンストラクタ起動時にはまだ決定できない このような変数について late finalで変数を定義 (何らかのタイミングで)初期化されたかどうかをチェックし、されていなければ値を入れる(チェックしないとfina...

Comments


カテゴリ

あなたの買い物をサポートする
アプリ 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