classをcase classに変換するscala.metaなライブラリ
練習を兼ねて、scala.metaを使って普通のclass
をcase class
に進化させるライブラリを作った。
case class
にすると自動で生成される以下のメソッド群をscalametaで模倣している。
toString
copy
equals
apply
unapply
hashCode
はめんどくさいのでスキップ…。
@Case class Hoge(val n: Int)
とか書くとだいたい
case class Hoge(n: Int)
と同じ感じになる、ということ。
使い方
mavenにあげているので、依存の追加は簡単。
libraryDependencies += "net.petitviolet" %% "acase" % "<latest-version>"
でもマクロを有効にするための設定がいろいろ必要となる。
手順はここに書いた。setup
これがどれくらい必要なのかわからないが、とりあえずこれを書いていれば動く。
もうちょっと詳しく
publicなコンストラクタフィールド(?)しか持たないclassに@Case
をつける。
@Case class CaseApp(val n: Int, val s: String)
これが、おおよそ以下のように展開される。
class CaseApp(val n: Int, val s: String) { override def toString: String = { "CaseApp" + "(" + ("n" + ": " + n.toString + ", " + "s" + ": " + s.toString) + ")" } override def equals(obj: Any): Boolean = { if (!obj.isInstanceOf[CaseApp]) false else { val other = obj.asInstanceOf[CaseApp] this.n == other.n && this.s == other.s } } def copy(n = this.n, s = this.s) = new CaseApp(n, s) } object CaseApp { def unapply(arg: CaseApp): Option[(Int, String)] = { Some((arg.n, arg.s)) } def apply(n: Int, s: String): CaseApp = new CaseApp(n, s) }
とはいえ
case class
にしない理由があるのにとりあえず全部生やすのも意味ないので、別々にアノテーションとして実装してある。
@ToString
@Copy
@Equals
@Apply
@Unapply
コンストラクタを隠したい、みたいな場合には欲しいアノテーションだけをごてごてとつければ良い。
@Equals @ToString @Copy @Unapply class Hoge(val n: Int, val s: String)
private
なコンストラクタがあるとUnapply
とかを実装出来ないので全てpublicである必要がある。
もちろんリフレクション使えば出来るんだけど…。
感想
インスタンスメソッドを差し込むパターンとコンパニオンオブジェクトに差し込むパターンもあり、 何となくscalametaに慣れてきたところ。
とはいえアノテーション書きまくるの、Javaっぽいしちょっと嫌だなぁというのがまだ拭えないのでdef macro来てくれるの待ってる。
@petitviolet https://t.co/aWzeRLsh3f の中でdef macro今作ってますみたいなことが書かれてます
— Toshiyuki Takahashi (@tototoshi) 2017年3月17日