Wassup Buddies!
'Cloning' is the topic we will be cloning down for this blogpost.
To save time and memory, coders often prefer to re-use already existing objects. Commonly, a new reference pointing to an existing object, or talking about veterans of Java, techniques like cloning are used.
Digging deeper in this context, we will understand what are the repercussions and whats are the various ways of cloning. Core java interviewers find it an interesting topic to engulf the candidates by asking various pros and cons.
I have placed the sample Java class that I created for my understanding and the differentiation of types, here.
Please feel free to get your hands dirty. Many of our readers might find the topic repetitive, but I hope they will find a new perspective of seeing or explaining the concepts or would love to share their thoughts and suggestions for improving the knowledge of others.
A brief primer for better understanding of the code is as follows:
Scenario 1: Assigning the reference.
Test ob1 = new Test();
Where Test.java is -
class Test {int var1, var2;Dummy dummy = new Dummy();Test() {var1 = 10;var2 = 20;dummy.aVar = 30;dummy.bVar = 40;}}
Simply straight forward, an object is having its reference.
ob1 original -- 10 20 30 40
Scenario 2: Creating a new reference for the same object, using the earlier created reference.
Test ob2 = ob1;
Now make some changes using the second reference to see the impact on the object created.
ob2.var1 = 100;ob2.dummy.aVar = 1000;
What do we get?
ob1 original -- 10 20 30 40
ob1 changed -- 100 20 1000 40
ob2 original -- 100 20 1000 40
So clearly, as expected, reference will impact the original object.
Scenario 3: Shallow Cloning
CloneableTest cloneableTest = new CloneableTest();
Where CloneableTest.java is -
class CloneableTest implements Cloneable {int var1, var2;Dummy dummy = new Dummy();CloneableTest() {var1 = 10;var2 = 20;dummy.aVar = 30;dummy.bVar = 40;}public Object clone()throws CloneNotSupportedException {
return super.clone();}}
Note: For using cloneableTest.clone(), we need to have the class implementing Cloneable interface, else we will get ''CloneNotSupportedException".
Bringing in a new reference for testing -
CloneableTest newCloneable = null;try {newCloneable = (CloneableTest) cloneableTest.clone();newCloneable.var1 = 99;newCloneable.var2 = 99;newCloneable.dummy.aVar = 99;newCloneable.dummy.bVar = 99;} catch (CloneNotSupportedException e) {// e.printStackTrace();System.err.println("Exception since Cloneable interface is not implemented!!");}
Changes have been made, but what about the original object? Let's check!
Original Before : 10 & 20 & 30 & 40
Cloned : 99 & 99 & 99 & 99
Original After : 10 & 20 & 99 & 99
Hence take away from this example - original primitive attributes won't change, but non primitives will change.
Still the cloned object is not yet completely change proof.
Scenario 4: Deep Cloning
CloneableDeepTest cloneableDeepTest = new CloneableDeepTest();
Where CloneableDeepTest.java is -
class CloneableDeepTest implements Cloneable {int var1, var2;Dummy dummy = new Dummy();CloneableDeepTest() {var1 = 10;var2 = 20;dummy.aVar = 30;dummy.bVar = 40;}public Object clone()throws CloneNotSupportedException {CloneableDeepTest cloneableDeepTest= (CloneableDeepTest) super.clone();cloneableDeepTest.dummy = new Dummy();return cloneableDeepTest;}
}
Bringing in new reference again -
CloneableDeepTest cloneableDeepTest2 = null;try {cloneableDeepTest2 =(CloneableDeepTest) cloneableDeepTest.clone();cloneableDeepTest2.var1 = 55;cloneableDeepTest2.var2 = 66;cloneableDeepTest2.dummy.aVar = 77;cloneableDeepTest2.dummy.bVar = 88;} catch (CloneNotSupportedException e) {// e.printStackTrace();System.err.println("Exception since Cloneable interface is not implemented!!");}
What should be the impact on the original object now? Let's check!
Original Before : 10 & 20 & 30 & 40
Cloned : 55 & 66 & 77 & 88
Original After : 10 & 20 & 30 & 40
Sigh! The cloned object still retains the expected original values for both primitive and non primitive attributes.
Major difference between the implementation of shallow and deep clonings can be seen in the overridden clone() methods -
For shallow we simply call super.clone() as follows -
public Object clone()throws CloneNotSupportedException {return super.clone();}
While for deep cloning, we need to go a bit further and control the way how inner objects are being generated as follows -
public Object clone()throws CloneNotSupportedException {CloneableDeepTest cloneableDeepTest =(CloneableDeepTest) super.clone();cloneableDeepTest.dummy = new Dummy();return cloneableDeepTest;}
That's all I have for now.
Happy Coding!!
No comments:
Post a Comment