问题
I was going through Copy Constructors, I have gone through the links in stack over flow and others as well. But i am not clear on the following points.
- Why do we need a Copy Constructor
- When do we need a Copy Constructor
I mean what is the exact situation or scenario we would need to use Copy Constructor. Can some one explain with an example or point out links so that i can go through and understand them in clear.
Following are the links i have gone through to get an understanding of what is a copy constructor.
http://www.programmerinterview.com/index.php/java-questions/how-copy-constructors-work/
https://deepeshdarshan.wordpress.com/2013/12/05/copy-constructors-in-java/
The second link explains \'why\' and \'where\' the copy constructor is to be used. But still i am not clear on it.
Below is my class Employee.java
package com.test;
/**
* @author avinashd
*
*/
public class Employee {
private String rollNo;
private String name;
//constructor
public Employee(String rollNo, String name){
this.rollNo = rollNo;
this.name = name;
}
//copy constructor
public Employee(Employee employee){
this.rollNo = employee.rollNo;
this.name = employee.name;
}
public String getRollNo() {
return rollNo;
}
public void setRollNo(String rollNo) {
this.rollNo = rollNo;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
}
Copy Constructor is used to create and exact copy of an object with the same values of an existing object.
Say for example we have an Employee with values as rollNo: 1
and name: avinash
. Copy Constructor would create a similar object with values as rollNo: 1
and name: avinash
. But both are 2 different objects and changes to the values of on object will not affect another object.
The Question here is
When we have a constructor such as
public Employee(String rollNo, String name){
this.rollNo = rollNo;
this.name = name;
}
to create an object. We can call the same constructor to create another object. But why do we need to call copy constructor.When do we need to call it ?. Please explain
回答1:
There are 2 good reasons for using a copy constructor instead of the constructor passing all parameters :
- when you have a complex object with many attributes it is much simpler to use the copy constructor
- if you add an attribute to your class, you just change the copy constructor to take this new attribute into account instead of changing every occurence of the other constructor
回答2:
Copy constructors, by convention, should provide a deep copy of objects. As already mentioned by other answers, the main convenience provided by copy constructors is when your object becomes too complex. Note that java.lang.Cloneable
provides an (almost) similar declaration.
But there are a number of advantages to using copy constructors over the Cloneable
interface.
Cloneable
as an interface does not actually provide any methods. For it to be effective, you still need to override theclone
method ofjava.lang.Object
. This is quite a counterintuitive use for an interface.clone
returns anObject
. For it to be of any use, you still need to typecast. This is awkward and could lead to runtime errors.The
clone
method is poorly documented. Forclone
, things can get messed up if you have final fields pointing to mutable objects.Most importantly, copy constructors can take in and deep copy instances of subclasses. IMO, this is where copy constructors really shine.
There are more advantages (see Joshua Bloch's Effective Java 2e) but these are the points which I have found most pertinent to what I have worked on so far.
[1] Nothing in the Java language actually provides a default construct for deep copying. At most, objects can just tell programmers that they can be deep copied by, say, implementing Cloneable
or providing a copy constructor.
回答3:
What if you want to have another Employee
instance with the exact same values as the one you already have?.
Will you call?
setName(oldEmployee.getName())..
setRollNumber(oldEmployee.getRollNumber())..
etc..
Instead of doing that, use this
Employee copyOfEmployeeOne=new Employee(employeeOneInstance);
// no need of a sequence of setters..
回答4:
Another case would be storing object "historical" values.
So you have single object but whenever you change its state you want to add it to ArrayList
or whatever data structure fits you best, instead of making manual copy of the data you just use "copy constructor" before further modification.
回答5:
Copy constructors give us many advantages over Object.clone() method because they
- Don’t force us to implement any interface or throw an exception.
- Don’t require any casts.
- Don’t require us to depend on an unknown object creation mechanism.
- Don’t require parent class to follow any contract or implement anything.
- Allow us to modify final fields.
- Allow us to have complete control over object creation, we can write our initialization logic in it.
Read more on Java Cloning - Copy Constructor versus Cloning
回答6:
Through Copy constructor We can do cloning with out using much complex stuff like implementing Cloneable interface and overwriting clone method. Also we no need to worry specially about deep cloning.
But points to be noted are :
1.When we implement Cloneable, it is an intimation to other classes/users that this class' objects can be cloneable. With out this, Other classes may not have explicit information about cloneable.
2.If we make our copy constructor as private , then we can restrict cloning of this class' object. Then this copy constructor can only be used to initialize newly created objects in local to class rather than for cloning purpose in other class.
3.When you do not want to make your class cloneable but if you have written copy constructor with access specifier public, then it leads to insecurity that other classes can create objects of your class.
回答7:
Copy Constructors implements both shallow and deep cloning mechanism, but main advantages of using copy constructor over cloning(using Cloneable interface) is:
- We don't require any type casing as of in using Object.clone()method.
- We will be able to modify the final fields for copying purpose unlike we are unable to modify or access final fields in case of Object.clone() mechanism.
- It allows us to have complete control over object creation, we can write our initialization logic in it.
- If we want to clone object of our class that holds reference variable of other dependent class we don't need to implement clone method that class. Simply by initializing our copy constructor, we can achieve that.
回答8:
Consider this example where super class has a copy constructor provided to implicitly force developer to manually copy fields over to parent class. This can be useful for Downcasting:
public class Employee {
private String name;
private int age;
// regular constructor
public Employee(String name, int age) {
this.name = name;
this.age = age;
}
// copy constructor
public Employee(Employee that) {
this.name = that.name;
this.age = that.age;
}
// getters & setters
}
public class SeniorMgmt extends Employee {
private boolean secureAccess;
public SeniorMgmt(Employee employee, boolean secureAccess) {
super(employee);
this.secureAccess = secureAccess;
}
// getters & setters
}
public class TestDrive {
public static void main(String[] args) {
Employee programmer = new Employee("John", 34);
SeniorMgmt promotedProgrammer = new SeniorMgmt(programmer, true);
}
}
来源:https://stackoverflow.com/questions/29362169/why-do-we-need-copy-constructor-and-when-should-we-use-copy-constructor-in-java