so as the title states, I am wondering how to clone objects, because the method clone() is protected so I cannot really use it.
thx in advance,
h3ckboy
 
      
    so as the title states, I am wondering how to clone objects, because the method clone() is protected so I cannot really use it.
thx in advance,
h3ckboy
java.lang.Cloneable interfaceclone() method.wouldnt I then have to know how to clone method works?
If you read the API you’ll see:
[quote=""]
So extend the class, add on the Clonable interface and override the clone method with a public version that calls up to the original.
You can also do this with reflection:
    public static <T> T clone(T object)
            throws IllegalArgumentException, IllegalAccessException, InstantiationException
    {
        final Class<T> klass = (Class<T>) object.getClass();
        final T copy = klass.newInstance();
        copyFields( object, copy );
        return copy;
    }
    public static <T> void copyFields(T source, T dest)
            throws IllegalArgumentException, IllegalAccessException
    {
        final Class<?> klass = source.getClass();
        
        for ( Field field : klass.getDeclaredFields() ) {
            field.set( dest, field.get(source) );
        }
    }
Should work, but that code is untested. It also presumes the class of the object being copied has a no argument constructor.
The problem with someClass.newInstance(); is that there must be a public (no-arg) constructor, and boy… if it has side-effects…
private and final fields are also a problem, not always fixable with setAccessible(true) because the JIT will optimize final fields and changing the actual field, might not always update the value used in calculations inside method bodies (ask: OrangyTang / search the JGO forum for the thread)
I agree, it’s vastly better to do this without reflection then with. But depending on what he wants it should be good enough.
You don’t even need to override clone(), you can just roll your own if you need more options.
Ex.
//Default initializer.
public MyClass()
{
    id = getNextID();
    x = 4;
    y = 1000;
}
public MyClass(MyClass cloner)
{
    x = cloner.getX();
    y = cloner.getY();
    id = cloner.getID();
}
public MyClass(MyClass cloner, int newID)
{
    x = cloner.getX();
    y = cloner.getY();
    id = newID;
}
eww! 
Don’t use public constructors to expose such functionality.
Create private constructors and public methods, or better, the FactoryPattern/BuilderPattern
ahh so, basically you recreate the object?
so you just give the new object all of the variables that the current one has?
this kinda defeats the purpose of my goal then, I can do it a better way…
thank you for helping me understand,
h3ckboy
What’s the problem doing it this way? Instead of
MyObject clone = (MyObject) clonedObject.clone();
you just have
MyObject clone = new MyObject(clonedObject);
It doesn’t seem like you’re inherently exposing anything other than the ability to create a copy of another object, which is what you want to accomplish with clone anyway. But yes, you could do it like this too:
public MyClass()
{
    id = getNextID();
    x = 4;
    y = 1000;
}
private MyClass(MyClass cloner)
{
    x = cloner.getX();
    y = cloner.getY();
    id = cloner.getID();
}
public Object clone()
{
    return new MyClass(this);
}
//or if you don't want to use clone()
public MyClass getClone()
{
    return new MyClass(this);
}
I don’t necessarily understand why that’s better - it leaves you with the same functionality in the end. The only difference is you’re not directly allocating a new object, you’re leaving that to the class’s method.
Doesn’t using the builder pattern concept seem a bit bloated if he just wants a clone? I guess if you’re moving towards what I defined above (different ways of cloning something with different parts and pieces) then the builder pattern could work well, but it seems extraneous just to clone.
And yeah, h3ckboy, you do it all manually, in the end. If you really wanted to you could externally create a new object and then use public mutators and accessors to assign all the data you needed - it’s all just a question of implementation.
The clone method is there for a reason: it can clone any subclass and gives you exactly the same type, whereas with constructor you’re creating some concrete type and not really a clone. The constructor approach is considered a bad practice.
Also clone method does shallow copy for you automatically, though you have to still do deep copy manually, but that’s leaved to programmer because it can’t be decided automatically which field should or shouldn’t be deep copied.
One probably overlooked feature in Java 5 and newer is that you can override clone method and change the method signature to return your type. Though you have to override it for every subclass to return the concrete type in every case.
Aha, that makes sense. Thanks for enlightening me on that. As I wrote that code above, I did indeed wonder if clone() has yet been adjusted to allow different return types (I suspected it probably had). Good to know you can do that now.
It didn’t need adjusting. Covariant return types are a general language feature, and any (non-final) virtual method supports them.
Didn’t Jezek say that being able to do that with clone() added with Java 5? That’s what I meant when I said “adjusted.”
He did, but that’s a narrow reading of what he said. It would be more precise to say that the ability to do that with any (non-final) virtual method was added in Java 5. clone() wasn’t singled out for special treatment.
Which is a logical thing to have added with templates - to be honest I didn’t know they had added that feature, though. It’s certainly a good one to know about.