ITEM 13: overriding clone judiciously

public interface Cloneable {
}

λ©”μ„œλ“œ ν•˜λ‚˜ μ—†λŠ” Cloneable μΈν„°νŽ˜μ΄μŠ€λŠ” Object의 protected λ©”μ„œλ“œμΈ clone의 λ™μž‘λ°©μ‹μ„ κ²°μ •ν•œλ‹€. Cloneable을 κ΅¬ν˜„ν•œ 클래슀의 μΈμŠ€ν„΄μŠ€μ—μ„œ clone을 ν˜ΈμΆœν•˜λ©΄ κ·Έ 객체의 ν•„λ“œλ“€μ„ ν•˜λ‚˜ν•˜λ‚˜ λ³΅μ‚¬ν•œ 객체λ₯Ό λ°˜ν™˜ν•˜λ©°, κ΅¬ν˜„ν•˜μ§€ μ•Šμ€ 클래슀의 μΈμŠ€ν„΄μŠ€μ—μ„œ ν˜ΈμΆœν•˜λ©΄ CloneNotSupportException을 λ˜μ§„λ‹€.

μΈν„°νŽ˜μ΄μŠ€λ₯Ό κ΅¬ν˜„ν•œλ‹€λŠ” 것은 일반적으둜 ν•΄λ‹Ή ν΄λž˜μŠ€κ°€ κ·Έ μΈν„°νŽ˜μ΄μŠ€μ—μ„œ μ •μ˜ν•œ κΈ°λŠ₯을 μ œκ³΅ν•œλ‹€κ³  μ„ μ–Έν•˜λŠ” ν–‰μœ„μΈλ°, Cloneable의 κ²½μš°μ—λŠ” μƒμœ„ ν΄λž˜μŠ€μ— μ •μ˜λœ λ™μž‘ 방식을 λ³€κ²½ν•œ κ²ƒμ΄λ―€λ‘œ λ”°λΌν•˜λŠ” 것은 μ’‹μ§€ μ•Šλ‹€.

λΆˆλ³€ ν΄λž˜μŠ€λŠ” ꡳ이 clone을 μ œκ³΅ν•˜μ§€ μ•ŠλŠ” 것이 μ’‹λ‹€.

public final class PhoneNumber implements Cloneable{
    private final short areaCode, prefix, lineNum;
    private int hashCode;

    @Override
    public PhoneNumber clone(){
        try{
            // ν˜•λ³€ν™˜
            return (PhoneNumber) super.clone();
        }catch (CloneNotSupportedException e){
            throw new AssertionError();
        }
    }
}
public class Object {

    @HotSpotIntrinsicCandidate
    protected native Object clone() throws CloneNotSupportedException;

try-catch둜 κ΅¬ν˜„ν•œ μ΄μœ λŠ” Object의 clone()이 CloneNotSupportedException을 던져주고 있기 λ•Œλ¬Έμ΄λ©°, Cloneable을 κ΅¬ν˜„ν•˜λ©΄ CloneNotSupportedException이 λ°œμƒν•˜μ§€ μ•Šμ„ 것을 μ•ˆλ‹€.

Object λ©”μ„œλ“œλŠ” CloneNotSupportedException을 λ˜μ§„λ‹€κ³  μ„ μ–Έν–ˆμ§€λ§Œ, μž¬μ •μ˜ν•œ λ©”μ†Œλ“œμ—μ„œλŠ” throwsμ ˆμ„ μ—†μ• λŠ” 것이 μ’‹λ‹€. 검사 μ˜ˆμ™Έλ₯Ό λ˜μ§€μ§€ μ•Šμ•„μ•Ό κ·Έ λ©”μ„œλ“œλ₯Ό μ‚¬μš©ν•˜κΈ° νŽΈν•˜κΈ° λ•Œλ¬Έμ΄λ‹€.(비검사 μ˜ˆμ™Έ-item71)

clone()은 원본 객체에 μ•„λ¬΄λŸ° ν•΄λ₯Ό λΌμΉ˜μ§€ μ•ŠλŠ” λ™μ‹œμ— 볡제된 객체의 λΆˆλ³€μ‹μ„ 보μž₯ν•΄μ•Όν•œλ‹€.

public class Stack {
    private Object[] elements;
    private int size = 0;
    private static final int DEFAULT_CAPACITY = 16;

    public Stack() {
        elements = new Object[DEFAULT_CAPACITY];
    }

    public void push(Object e){
        ensureCapacity();
        elements[size++] = 0;
    }

    public Object pop(){
        if(size == 0){
            throw new EmptyStackException();
        }
        return elements[--size];
    }

    // μ›μ†Œλ₯Ό μœ„ν•œ 곡간을 적어도 ν•˜λ‚˜ 이상 μ—¬μœ λ₯Ό 두며, λŠ˜λ €μ•Όν•˜λŠ” 경우 두배 이상 λŠ˜λ¦°λ‹€.
    private void ensureCapacity(){
        if(elements.length == size){
            elements = Arrays.copyOf(elements, 2*size+1);
        }
    }

}

μœ„ stack ν΄λž˜μŠ€λŠ” 가변객체λ₯Ό μ°Έμ‘°ν•˜κ³  μžˆλ‹€.

    @Override
    public Stack clone() {
        try{
            Stack result = (Stack) super.clone();
            result.elements = elements.clone();
            return result;
        }catch (CloneNotSupportedException e){
            throw new AssertionError();
        }
    }

원본 객체λ₯Ό μœ μ§€ν•˜λ©΄μ„œ 볡제된 객체의 λΆˆλ³€μ„ μœ μ§€ν•˜λŠ” κ°€μž₯ μ‰¬μš΄ 방법은 elements λ°°μ—΄μ˜ clone을 μž¬κ·€μ μœΌλ‘œ ν˜ΈμΆœν•΄μ£ΌλŠ” 것이닀 .μ΄λ•Œ, λ°°μ—΄μ˜ clone은 λŸ°νƒ€μž„ νƒ€μž…κ³Ό μ»΄νŒŒμΌνƒ€μž„ νƒ€μž… λͺ¨λ‘κ°€ 원본 λ°°μ—΄κ³Ό λ˜‘κ°™μ€ 배열을 λ°˜ν™˜ν•˜λ―€λ‘œ λ³„λ„λ‘œ ν˜•λ³€ν™˜μ„ 해쀄 ν•„μš”λŠ” μ—†λ‹€. λ”°λΌμ„œ 배열을 λ³΅μ œν•  λ•ŒλŠ” λ°°μ—΄μ˜ clone λ©”μ„œλ“œλ₯Ό μ‚¬μš©ν•˜λŠ” 것을 ꢌμž₯ν•œλ‹€.

ν•œνŽΈ μœ„ Stack 클래슀의 elements ν•„λ“œκ°€ finalμ΄μ—ˆλ‹€λ©΄ final ν•„λ“œλŠ” μƒˆλ‘œμš΄ 값을 ν• λ‹Ήν•  수 μ—†κΈ°λ•Œλ¬Έμ— μœ„ clone() λ©”μ„œλ“œ 방식은 μž‘λ™ν•˜μ§€ μ•ŠλŠ”λ‹€.

Cloneable μ•„ν‚€ν…μ²˜λŠ” κ°€λ³€ 객체λ₯Ό μ°Έμ‘°ν•˜λŠ” ν•„λ“œλŠ” final둜 μ„ μ–Έν•˜λΌλŠ” 일반 μš©λ²•κ³Ό μΆ©λŒν•œλ‹€. 원본과 볡제된 객체가 κ°€λ³€ 객체λ₯Ό κ³΅μœ ν•΄λ„ λœλ‹€λ©΄ μƒκ΄€μ—†μ§€λ§Œ, λ³΅μ œν•  수 μžˆλŠ” 클래슀λ₯Ό λ§Œλ“€κΈ° μœ„ν•΄ 일뢀 ν•„λ“œμ—μ„œ final ν•œμ •μžλ₯Ό μ œκ±°ν•΄μ•Όν•  μˆ˜λ„ μžˆλ‹€.

public class HashTable implements Cloneable{
    private Entry[] buckets = new Entry[10];

    private static class Entry {
        final Object key;
        Object value;
        Entry next;

        Entry(Object key, Object value, Entry next){
            this.key = key;
            this.value = value;
            this.next = next;
        }


        Entry deepCopy(){
            // ν•΄λ‹Ή μ—”νŠΈλ¦¬κ°€ κ°€λ¦¬ν‚€λŠ” μ—°κ²° 리슀트λ₯Ό μž¬κ·€μ μœΌλ‘œ 볡사
            // μž¬κ·€ 호좜이기 λ•Œλ¬Έμ—, 리슀트의 μ›μ†Œ 수 만큼 μŠ€νƒ ν”„λ ˆμž„μ„ μ†ŒλΉ„ν•΄ λ¦¬μŠ€νŠΈκ°€ κΈΈλ©΄ μŠ€νƒ μ˜€λ²„ν”Œλ‘œμš° λ°œμƒ μœ„ν—˜ 있음.
            // return new Entry(key, value, next == null ? null : next.deepCopy());

            // μŠ€νƒμ˜€λ²„ν”Œλ‘œμš° 문제λ₯Ό ν”Όν•˜κΈ° μœ„ν•΄ 반볡자λ₯Ό μ‚¬μš©
            Entry result = new Entry(key,value, next);
            for(Entry p = result; p.next != null; p = p.next)
                p.next = new Entry(p.next.key, p.next.value, p.next.next);
            return result;
        }
    }

    /**
     * 잘λͺ»λœ Clone
     * 원본과 같은 μ—°κ²° 리슀트λ₯Ό μ°Έμ‘° ν•΄ 원본과 볡제본 λͺ¨λ‘ 예기치 μ•Šκ²Œ λ™μž‘ν•  κ°€λŠ₯μ„± 생김
     * @return
     * @Override
     *     public  HashTable clone(){
     *         try{
     *             HashTable result = (HashTable) super.clone();
     *             result.buckets = buckets.clone();
     *             return result;
     *         }catch (CloneNotSupportedException e){
     *             throw new AssertionError();
     *         }
     *     }
     */


    /**
     *
     * @return
     */
    @Override
    public  HashTable clone(){
        try{
            HashTable result = (HashTable) super.clone();
            result.buckets = new Entry[buckets.length];
            for(int i =0 ; i < buckets.length; i++){
                if(buckets[i] != null){
                    result.buckets[i] = buckets[i].deepCopy();
                }
            }
            return result;
        }catch (CloneNotSupportedException e){
            throw new AssertionError();
        }
    }
}

HashTable 클래슀λ₯Ό 보면, Stackκ³Ό λ™μΌν•œ λ°©λ²•μœΌλ‘œ clone()을 κ΅¬ν˜„ν•˜λ©΄ 원본과 같은 μ—°κ²° 리슀트λ₯Ό μ°Έμ‘° ν•΄ 원본과 볡제본 λͺ¨λ‘ 예기치 μ•Šκ²Œ λ™μž‘ν•  κ°€λŠ₯μ„± 생길 수 μžˆκΈ°λ•Œλ¬Έμ—, deepCopy()λ₯Ό λ³„λ„λ‘œ κ΅¬ν˜„ν•˜μ—¬ ν•΄λ‹Ή μ—”νŠΈλ¦¬κ°€ κ°€λ¦¬ν‚€λŠ” μ—°κ²° 리슀트λ₯Ό μž¬κ·€μ μœΌλ‘œ λ³΅μ‚¬ν•΄μ£Όμ—ˆλ‹€.

clone() λ©”μ„œλ“œλŠ” μž¬μ •μ˜λ  수 μžˆλŠ” λ©”μ„œλ“œλ₯Ό ν˜ΈμΆœν•΄μ„œλŠ” μ•ˆλœλ‹€. λ§Œμ•½ clone()이 ν•˜μœ„ ν΄λž˜μŠ€μ—μ„œ μž¬μ •μ˜ν•œ λ©”μ„œλ“œλ₯Ό ν˜ΈμΆœν•˜λ©΄, ν•˜μœ„ ν΄λž˜μŠ€λŠ” 볡제 κ³Όμ •μ—μ„œ μžμ‹ μ˜ μƒνƒœλ₯Ό ꡐ정할 수 μžˆλŠ” 기회λ₯Ό μΌκ²Œλ˜μ–΄ 원본과 μƒνƒœκ°€ λ‹¬λΌμ§ˆ κ°€λŠ₯성이 μ»€μ§€κ²Œλœλ‹€.

μš”μ•½ν•˜μžλ©΄, Cloneable을 κ΅¬ν˜„ν•˜λŠ” λͺ¨λ“  ν΄λž˜μŠ€λŠ” clone()을 μž¬μ •μ˜ν•΄μ•Ό ν•œλ‹€. μ΄λ•Œ μ ‘κ·Ό μ œν•œμžλŠ” public으둜, λ°˜ν™˜ νƒ€μž…μ€ 클래슀 μžμ‹ μœΌλ‘œ λ³€κ²½ν•œλ‹€. κ°€μž₯ λ¨Όμ € super.clone 을 ν˜ΈμΆœν•œ ν›„ ν•„μš”ν•œ ν•„λ“œλ₯Ό μ „λΆ€ 적절히 μˆ˜μ •ν•΄μ€˜μ•Ό ν•œλ‹€. 일반적으둜 'κΉŠμ€ ꡬ쑰'에 μˆ¨μ–΄ μžˆλŠ” λͺ¨λ“  κ°€λ³€ 객체λ₯Ό λ³΅μ‚¬ν•˜κ³ , 볡제본이 κ°€μ§„ 객체 μ°Έμ‘° λͺ¨λ‘κ°€ λ³΅μ‚¬λœ 객체듀을 κ°€λ¦¬ν‚€κ²Œ ν•΄μ•Όν•œλ‹€. κΈ°λ³Έ νƒ€μž… ν•„λ“œμ™€ λΆˆλ³€ 객체 μ°Έμ‘°λ§Œμ„ κ°–λŠ” 클래슀라면 아무 ν•„λ“œλ„ μˆ˜μ •ν•  ν•„μš”λŠ” μ—†λ‹€.

Cloneable을 κ΅¬ν˜„ν•˜μ§€ μ•Šμ€ 클래슀라면 볡사 μƒμ„±μžμ™€ 볡사 νŒ©ν„°λ¦¬λΌλŠ” 더 λ‚˜μ€ 객체 볡사 방식을 μ œκ³΅ν•  수 μžˆλ‹€.

// λ³΅μ‚¬μƒμ„±μž
public Item(Item item){...};
// 볡사 νŒ©ν„°λ¦¬
public static Item newInstance(Item item){...};

볡사 μƒμ„±μžλž€ λ‹¨μˆœνžˆ μžμ‹ κ³Ό 같은 클래슀의 μΈμŠ€ν„΄μŠ€λ₯Ό 인수둜 λ°›λŠ” μƒμ„±μžλ₯Ό λ§ν•œλ‹€.

볡사 μƒμ„±μž/νŒ©ν„°λ¦¬λŠ”

  • μƒμ„±μžλ₯Ό μ“°μ§€ μ•ŠλŠ” λ°©μ‹μ˜ 객체 생성 λ©”μ»€λ‹ˆμ¦˜μ„ μ‚¬μš©ν•˜μ§€ μ•ŠλŠ”λ‹€.

  • μ—‰μ„±ν•˜κ²Œ λ¬Έμ„œν™”λœ κ·œμ•½μ— κΈ°λŒ€μ§€ μ•Šκ³ , 정상적인 final ν•„λ“œ μš©λ²•κ³Όλ„ μΆ©λŒν•˜μ§€ μ•ŠλŠ”λ‹€.

  • λΆˆν•„μš”ν•œ 검사 μ˜ˆμ™Έλ₯Ό λ˜μ§€μ§€ μ•Šκ³ , ν˜•λ³€ν™˜λ„ ν•„μš”μΉ˜ μ•ŠλŠ”λ‹€.

  • ν•΄λ‹Ή ν΄λž˜μŠ€κ°€ κ΅¬ν˜„ν•œ μΈν„°νŽ˜μ΄μŠ€ νƒ€μž…μ˜ μΈμŠ€ν„΄μŠ€λ₯Ό 인수둜 받을 수 μžˆλ‹€.

λͺ¨λ“  λ²”μš© μ»¬λ ‰μ…˜ κ΅¬ν˜„μ²΄λŠ” Collection μ΄λ‚˜ Map νƒ€μž…μ„ λ°›λŠ” μƒμ„±μžλ₯Ό μ œκ³΅ν•œλ‹€. 이λ₯Ό μ΄μš©ν•˜λ©΄ ν΄λΌμ΄μ–ΈνŠΈλŠ” μ›λ³Έμ˜ κ΅¬ν˜„ νƒ€μž…μ— 얽맀이지 μ•Šκ³  볡제본의 νƒ€μž…μ„ 직접 선택할 수 μžˆλ‹€. (HashSet을 TreeSet νƒ€μž…μœΌλ‘œ 볡제 κ°€λŠ₯)

new TreeSet<>(s);

Last updated

Was this helpful?