一畳のくつろぎタイム

このブログでは紹介する商品画像をAmazonアソシエイトより借りています。画像やリンクにはアフィリエイト広告が含まれる事があります

2018年9月5日水曜日

SwiftのOptionalがうざいところ

Swiftのオプショナルがうざい

なにがうざいかと言うと、なかったら作る処理が簡潔に書けない。

コードオンリーでUITableViewを作っていると、UITableViewCellを作るときにdequeueReusableCellメソッドがnilを返した場合に新しくインスタンスを作り、nilでなければメソッドより返された値を使う。
??演算子を使用すれば、それに近いことはできる。だがしかし、新たに作成したUITableViewCellの初期設定がしたいのだ。??の後ろが1行じゃ足りなくて、もう少し行が欲しい。
  1. let cell:UITableViewCell = tableView.dequeueReusableCell(withIdentifier: "hoge") ?? UITableViewCell(style: .default, reuseIdentifier: "hoge")
Objective-Cの場合、至ってシンプルである。nilだったら作って設定すればいい。
  1. - (nonnull UITableViewCell *)tableView:(nonnull UITableView *)tableView cellForRowAtIndexPath:(nonnull NSIndexPath *)indexPath {
  2. UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:@"hoge"];
  3. // 再利用セルがない場合は
  4. if( cell == nil ) {
  5. // 作成する
  6. cell = [[UITableViewCell alloc]initWithStyle:UITableViewCellStyleDefault reuseIdentifier:@"hoge"];
  7. // ここで作ったcellの共通設定
  8. }
  9. // ここで内容の更新処理
  10. return cell;
  11. }



Swiftの場合、guard文だと処理の中断が必須なため。guard文の中と外で2回内容の更新処理を書かなくてはならなくなる
  1. func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
  2. guard let cell = tableView.dequeueReusableCell(withIdentifier: "hoge") else {
  3.  
  4. let cell = UITableViewCell(style: .default, reuseIdentifier: "hoge")
  5. // 初期設定
  6.  
  7. // 内容の更新処理
  8. return cell
  9. }
  10.  
  11.  
  12. // 内容の更新処理
  13.  
  14. return cell
  15. }

オプショナルバインディングだと大した処理がないのに、代入のためだけのif-else両方が必要になる
  1. let cell:UITableViewCell
  2. if let unwrappedCell = tableView.dequeueReusableCell(withIdentifier: "hoge") {
  3. cell = unwrappedCell
  4. } else {
  5. cell = UITableViewCell(style: .default, reuseIdentifier: "hoge")
  6. // 初期設定
  7. }
  8.  
  9. // 内容の更新処理
  10.  
  11. return cell
外のスコープに変数を用意するぐらいだったら、オプショナルバインディングなどいらぬ。
以下のようにオプショナル構文を使わないで書いたって同じことだ。
  1. // セルの再利用可能か取得して見る
  2. let optCell:UITableViewCell? = tableView.dequeueReusableCell(withIdentifier: "hoge")
  3. let cell:UITableViewCell
  4. if optCell == nil {
  5. cell = UITableViewCell(style: .default, reuseIdentifier: "hoge")
  6. // 初期設定をする
  7. } else {
  8. // アンラップする
  9. cell = optCell!
  10. }
  11. // 内容の更新処理をする
  12. return cell
わずかではあるが、変数を2個用意しないで済むオプショナルバインディングを使った方が簡潔なのがなんか腹立たしい。

内容の更新処理をメソッド化すれば2回同じことは書かないで済むし、UITableViewCellのサブクラスを作れば??演算子でいけますが、だがObjective-Cと同じことができないのがなんかうざい。
??演算子の後ろにすぐ実行するUITableViewCellを戻り値とするクロージャ書いたらなんかできそうな気がするも、邪道な感じがして試したくもない。

とりあえず外のスコープの変数用意しておいて、オプショナルバインディグが正攻法かなと思う。