>  기사  >  Java  >  Java 객체 복사에 대한 일반적인 인터뷰 질문

Java 객체 복사에 대한 일반적인 인터뷰 질문

(*-*)浩
(*-*)浩원래의
2020-01-03 13:39:191840검색

Java 객체 복사에 대한 일반적인 인터뷰 질문

복제를 사용하는 이유는 무엇인가요? (추천 학습: v Java 공통 시험 문제 )

객체를 처리하고 다음 작업을 위해 원본 데이터를 유지하려면 Java 언어의 복제가 클래스 인스턴스를 목표로 해야 합니다.

객체 복제를 구현하는 방법은 무엇입니까?

두 가지 방법이 있습니다:

Cloneable 인터페이스를 구현하고 Object 클래스의 clone() 메서드를 재정의합니다. Serialized 인터페이스를 구현하고 객체 직렬화 및 역직렬화를 통해 복제를 구현합니다. 이는 진정한 심층 복제를 달성할 수 있는 코드입니다.

import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.io.Serializable;

public class MyUtil {

    private MyUtil() {
        throw new AssertionError();
    }

    @SuppressWarnings("unchecked")
    public static <T extends Serializable> T clone(T obj) throws Exception {
        ByteArrayOutputStream bout = new ByteArrayOutputStream();
        ObjectOutputStream oos = new ObjectOutputStream(bout);
        oos.writeObject(obj);

        ByteArrayInputStream bin = new ByteArrayInputStream(bout.toByteArray());
        ObjectInputStream ois = new ObjectInputStream(bin);
        return (T) ois.readObject();

        // 说明:调用ByteArrayInputStream或ByteArrayOutputStream对象的close方法没有任何意义
        // 这两个基于内存的流只要垃圾回收器清理对象就能够释放资源,这一点不同于对外部资源(如文件流)的释放
    }
}

다음은 테스트 코드입니다.

import java.io.Serializable;

/**
 * 人类
 * @author nnngu
 *
 */
class Person implements Serializable {
    private static final long serialVersionUID = -9102017020286042305L;

    private String name;    // 姓名
    private int age;        // 年龄
    private Car car;        // 座驾

    public Person(String name, int age, Car car) {
        this.name = name;
        this.age = age;
        this.car = car;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = age;
    }

    public Car getCar() {
        return car;
    }

    public void setCar(Car car) {
        this.car = car;
    }

    @Override
    public String toString() {
        return "Person [name=" + name + ", age=" + age + ", car=" + car + "]";
    }

}
/**
 * 小汽车类
 * @author nnngu
 *
 */
class Car implements Serializable {
    private static final long serialVersionUID = -5713945027627603702L;

    private String brand;       // 品牌
    private int maxSpeed;       // 最高时速

    public Car(String brand, int maxSpeed) {
        this.brand = brand;
        this.maxSpeed = maxSpeed;
    }

    public String getBrand() {
        return brand;
    }

    public void setBrand(String brand) {
        this.brand = brand;
    }

    public int getMaxSpeed() {
        return maxSpeed;
    }

    public void setMaxSpeed(int maxSpeed) {
        this.maxSpeed = maxSpeed;
    }

    @Override
    public String toString() {
        return "Car [brand=" + brand + ", maxSpeed=" + maxSpeed + "]";
    }

}
class CloneTest {

    public static void main(String[] args) {
        try {
            Person p1 = new Person("郭靖", 33, new Car("Benz", 300));
            Person p2 = MyUtil.clone(p1);   // 深度克隆
            p2.getCar().setBrand("BYD");
            // 修改克隆的Person对象p2关联的汽车对象的品牌属性
            // 原来的Person对象p1关联的汽车不会受到任何影响
            // 因为在克隆Person对象时其关联的汽车对象也被克隆了
            System.out.println(p1);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

참고: 직렬화 및 역직렬화 기반 복제는 심층 복제일 뿐만 아니라 더 중요한 것은 일반 검증을 통해 복제 여부를 확인할 수 있습니다. 객체는 직렬화를 지원하며 이 검사는 컴파일러에 의해 수행되며 런타임에 예외를 발생시키지 않습니다. 이 솔루션은 분명히 객체를 복제하기 위해 Object 클래스의 복제 메서드를 사용하는 것보다 낫습니다. 문제를 런타임에 그대로 두는 것보다 컴파일 타임에 문제를 노출시키는 것이 항상 더 좋습니다.

깊은 복사와 얕은 복사의 차이점은 무엇인가요?

얕은 복사는 개체의 참조 주소만 복사합니다. 두 개체는 동일한 메모리 주소를 가리키므로 값이 수정되면 다른 값도 그에 따라 변경됩니다(예: 할당()). and deep copy 객체와 해당 값을 복사하는 것입니다. 두 객체 중 하나라도 수정되면 다른 값은 변경되지 않습니다(예: JSON.parse() 및 JSON.stringify() ), 그러나 이 방법은 함수 유형을 복사할 수 없습니다)

위 내용은 Java 객체 복사에 대한 일반적인 인터뷰 질문의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

성명:
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.