link もっと前
   2008年 2月 7日 -
      2008年 2月 7日  
link もっと後

link 未来から過去へ表示(*)
link 過去から未来へ表示

日々

link permalink

Generics と clone メソッド

このキーワードでピンと来た方はこの日記を読む必要はないと思います。

Generics なクラスに clone メソッドを作ろうとしてはまりました。以下のようなコードを書いたわけです。

Generics なクラスでディープコピー(MyVector.java)

class MyVector<T extends Cloneable> implements Cloneable {
    T[] items;
    
    public Object clone() {
        MyVector<T> copy = new MyVector<T>();
        
        for (int i = 0; i < items.length; i++) {
            copy.items[i] = (T)(items[i].clone()); //error!!
        }
        return copy;
    }
}

Cloneable を継承する型だと宣言したところで Cloneable インタフェースには clone() の定義がありません。コンパイラはこのコードを見ても T が public メソッドの clone() を持っているのか、いないのかコンパイル時に判断できません。

従ってこのコードを実行すると基底クラスの Object クラスの clone() メソッド(private メソッド)が呼ばれる可能性があります。そのためコンパイラは「(Object の)clone() は public じゃないから呼べません」とエラーを出すのです。

clone() を呼ばずに、以下のようなコードに直せばコンパイルが通ります。しかしこのコードで生成される T のコピーはあくまでも「シャローコピー」ですから、T がシャローコピーでコピーしきれない要素を持っていたときに破綻します。


copy.add(get(i)); //OK

配列の要素まで「ディープコピー」をするには、T に clone() 関数の定義を強制させる必要があります。これに NewCloneable とでも名前を付けますと、以下のようになります。

Clone を強制する(MyVector2.java)

public interface NewCloneable extends Cloneable {
    public Object clone();
}

class MyVector2<T extends NewCloneable> implements Cloneable {
    T[] items;
    
    public Object clone() {
        MyVector2<T> copy = new MyVector2<T>();
        
        for (int i = 0; i < items.length; i++) {
            copy.items[i] = (T)(items[i].clone()); //OK
        }
        return copy;
    }
}

新しく作った MyVector2 クラスは「ディープコピー」ができるようになりました。しかしこのクラスは全く使い物になりません。なぜなら NewCloneable を継承するクラスは標準クラスライブラリに存在しないからです。つまり何を言っているのかというと、


MyVector<Integer> v1;  //OK
MyVector2<Integer> v2; //error!!

MyVector と同じ乗りで MyVector2 を使おうとすると「Integer は NewCloneable を実装してないからダメ」と怒られます。せっかくの Generics なのに自分で作ったクラスしかぶち込めないんじゃ、ちょっとあんまりですよね。

それならと、Integer を継承して NewCloneable を実装したクラス NewInteger を定義しよう!と思っても Integer は final 宣言されていて継承できません。MyVector2 はますます役立たずです。

結局はシャローコピーで我慢せよってことですかね。もしくは clone() なんか実装すんじゃねーよってことでしょうか…。

[編集者: すずき]
[更新: 2008年 2月 15日 01:09]
link 編集する

コメント一覧

  • コメントはありません。
open/close この記事にコメントする



link もっと前
   2008年 2月 7日 -
      2008年 2月 7日  
link もっと後

管理用メニュー

link 記事を新規作成

合計:  counter total
本日:  counter today

link About www.katsuster.net
RDF ファイル RSS 1.0
QR コード QR コード

最終更新: 9/17 20:03

カレンダー

<2008>
<<<02>>>
-----12
3456789
10111213141516
17181920212223
242526272829-

最近のコメント 5件

  • link 18年09月07日
    すずき 「ありがとう!\nこちらこそ、楽しみにして...」
    (更新:09/11 19:30)
  • link 18年09月07日
    よしだあ 「おつかれさまでした!\nまた仕事できるの...」
    (更新:09/11 19:17)
  • link 18年08月15日
    すずき 「うーん、なんか暴走したり、動かなかったり...」
    (更新:08/15 10:52)
  • link 18年08月15日
    すずき 「実行できた。あと実行ファイルパスについて...」
    (更新:08/15 10:42)
  • link 18年08月15日
    すずき 「さすがに x86_64 と arm のク...」
    (更新:08/15 10:35)

最近の記事 3件

link もっとみる
  • link 18年09月13日
    すずき 「[府民から都民へ] 家が決まりました。今月末から東京都民です。さよ...」
    (更新:09/17 20:03)
  • link 18年09月11日
    すずき 「[エアコン浄化] 今年の 7月に(2018年 7月 17日の日記参...」
    (更新:09/17 19:38)
  • link 18年09月10日
    すずき 「[引っ越し準備] 引っ越し用の新品の段ボールが 50箱以上届き、家...」
    (更新:09/17 19:32)

こんてんつ

open/close wiki
open/close Java API

過去の日記

open/close 2002年
open/close 2003年
open/close 2004年
open/close 2005年
open/close 2006年
open/close 2007年
open/close 2008年
open/close 2009年
open/close 2010年
open/close 2011年
open/close 2012年
open/close 2013年
open/close 2014年
open/close 2015年
open/close 2016年
open/close 2017年
open/close 2018年
open/close 過去日記について

その他の情報

open/close アクセス統計
open/close サーバ一覧
open/close サイトの情報