android 浅复制和深复制-Java Generic Deep Copy 篇

简介:

关于Java Generic Deep Copy 在java中的应用和注意事项,请参考:http://blog.csdn.net/yang_hui1986527/article/details/7039425

而关于在android程序中通过clone方法来进行浅复制和深复制,请参考:http://blog.csdn.net/yang_hui1986527/article/details/7036818


众所周知,android上层使用的是java语言,因此理论上于Java Generic Deep Copy 方法,同样适用于android程序中对象的深度拷贝。但是,上次在一个android的demo中使用试图通过该方法来进行深度拷贝,一直未成功。出错提示如下:

W/ActivityManager(  154): Launch timeout has expired, giving up wake lock!
W/ActivityManager(  154): Activity idle timeout for HistoryRecord{40540f08 com.clone/.CloneActivity}
E/dalvikvm( 3080): Could not find class 'javax.print.attribute.standard.PrinterStateReason', referenced from method com.ajexperience.utils.DeepCopyUtil.a
W/dalvikvm( 3080): VFY: unable to resolve instanceof 97 (Ljavax/print/attribute/standard/PrinterStateReason;) in Lcom/ajexperience/utils/DeepCopyUtil;
D/dalvikvm( 3080): VFY: replacing opcode 0x20 at 0x011d
D/dalvikvm( 3080): VFY: dead code 0x011f-0131 in Lcom/ajexperience/utils/DeepCopyUtil;.a (Ljava/lang/Object;Ljava/lang/Object;Lcom/ajexperience/utils/DeepCopyUtil$a;)Ljava/lang/Object;
W/System.err( 3080): java.lang.RuntimeException: com.ajexperience.utils.DeepCopyException: java.lang.NoSuchFieldException: modifiers
W/System.err( 3080): 	at com.clone.CloneUtil.getDeepCloneUtil(CloneUtil.java:14)
W/System.err( 3080): 	at com.clone.CloneUtil.<init>(CloneUtil.java:8)
W/System.err( 3080): 	at com.clone.CloneView.<init>(CloneView.java:24)
W/System.err( 3080): 	at com.clone.CloneActivity.onCreate(CloneActivity.java:12)
W/System.err( 3080): 	at android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1047)
W/System.err( 3080): 	at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:1722)
W/System.err( 3080): 	at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:1784)
W/System.err( 3080): 	at android.app.ActivityThread.access$1500(ActivityThread.java:123)
W/System.err( 3080): 	at android.app.ActivityThread$H.handleMessage(ActivityThread.java:939)
W/System.err( 3080): 	at android.os.Handler.dispatchMessage(Handler.java:99)
W/System.err( 3080): 	at android.os.Looper.loop(Looper.java:130)
W/System.err( 3080): 	at android.app.ActivityThread.main(ActivityThread.java:3835)
W/System.err( 3080): 	at java.lang.reflect.Method.invokeNative(Native Method)
W/System.err( 3080): 	at java.lang.reflect.Method.invoke(Method.java:507)
W/System.err( 3080): 	at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:841)
W/System.err( 3080): 	at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:599)
W/System.err( 3080): 	at dalvik.system.NativeStart.main(Native Method)
W/System.err( 3080): Caused by: com.ajexperience.utils.DeepCopyException: java.lang.NoSuchFieldException: modifiers
W/System.err( 3080): 	at com.ajexperience.utils.DeepCopyUtil.<init>(SourceFile:99)
W/System.err( 3080): 	at com.clone.CloneUtil.getDeepCloneUtil(CloneUtil.java:12)
W/System.err( 3080): 	... 16 more
W/System.err( 3080): Caused by: java.lang.NoSuchFieldException: modifiers
W/System.err( 3080): 	at java.lang.ClassCache.findFieldByName(ClassCache.java:446)
W/System.err( 3080): 	at java.lang.Class.getDeclaredField(Class.java:666)
W/System.err( 3080): 	at com.ajexperience.utils.DeepCopyUtil.<init>(SourceFile:95)
W/System.err( 3080): 	... 17 more
D/dalvikvm(  424): GC_EXPLICIT freed 5K, 51% free 3666K/7367K, external 0K/0K, paused 95ms
I/ActivityManager(  154): Displayed com.clone/.CloneActivity: +20s115ms

于是,通过邮件向作者咨询该问题。今天,终于又收到作者的邮件,通知Java Generic Deep Copy 有更新,让我再试试。

下载最新版DeepCopyUtil-1.5,替换以前的库,运行成功。


主要思路:
1、创建TextBox,然后根据TextBox复制出一个对象TextBox2(点击屏幕,进行复制);
2、改变TextBox2的位置,方便观察比较差异。
3、改变TextBox2内部文本的位置。如果是浅复制,那么TextBox2内部文本改变位置的同时,TextBox内部文本的位置会随着改变。否则,如果是深度复制,TextBox内部文本的位置不会改变。
效果如下:
注:请注意文本“test”在文本框内的位置。
深复制:
                                           

源程序:
CloneActivity.java
package com.clone;

import android.app.Activity;
import android.os.Bundle;

public class CloneActivity extends Activity {
    /** Called when the activity is first created. */
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        //setContentView(R.layout.main);
        CloneView mCloneView = new CloneView(this);
        setContentView(mCloneView);
    }
}

CloneView.java
package com.clone;

import android.content.Context;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.Paint;
import android.graphics.PointF;
import android.util.AttributeSet;
import android.view.MotionEvent;
import android.view.View;

public class CloneView extends View {
	private Document mDocument = null;
	public Paint mPaint= null;
	public TextBox mTextBox = null;
	public TextBox mTextBox2 = null;
	private CloneUtil mCloneUtil = null;
	
	public CloneView(Context context) {
		super(context);
		mDocument = new Document();
		mPaint = new Paint();
		try {
			mCloneUtil = new CloneUtil();
		} catch (Exception e) {
			e.printStackTrace();
		}
		initData();
	}

	public CloneView(Context context, AttributeSet attrs) {
		super(context, attrs);
		mDocument = new Document();
		mPaint = new Paint();
		try {
			mCloneUtil = new CloneUtil();
		} catch (Exception e) {
			e.printStackTrace();
		}	
		initData();
	}

	public CloneView(Context context, AttributeSet attrs, int defStyle) {
		super(context, attrs, defStyle);
		mDocument = new Document();
	}

	public void initData()
	{
		PointF textpos = new PointF(30,30);
		String text = new String("test");

		mTextBox = new TextBox(50, 50, 100, 100, textpos, text);

		mDocument.add(mTextBox);
	}

	@Override
	protected void onDraw(Canvas canvas) {
		super.onDraw(canvas);
		canvas.drawColor(Color.WHITE);
		mDocument.draw(canvas, mPaint);
	}

	@Override
	public boolean onTouchEvent(MotionEvent event) {
		switch (event.getAction()) {
		case MotionEvent.ACTION_DOWN:
		{
			if (mCloneUtil != null) {
				mTextBox2 = (TextBox) mCloneUtil.clone(mTextBox); //深克隆  
			}

			if(mTextBox2 != null)
			{
				mTextBox2.moveBound();
				mTextBox2.changeData();
				mDocument.add(mTextBox2);
			}
		}
		invalidate();
		break;
		case MotionEvent.ACTION_MOVE:
			break;	
		case MotionEvent.ACTION_UP:
			break;
		default:
			break;
		}
		return super.onTouchEvent(event);
	}


}

Document.java
package com.clone;

import java.util.Vector;

import android.graphics.Canvas;
import android.graphics.Paint;

public class Document {

	private Vector<Object> objs = null;

	public Document() {
		objs = new Vector<Object>();
	}

	public void add(Object obj){
		objs.add(obj);
	}

	void draw(Canvas canvas, Paint paint) {
		canvas.save();
		
		for (int i = 0; i < objs.size(); i++) {
			Object obj = objs.get(i);
			obj.draw(canvas, paint);
		}
		
		canvas.restore();
	}


}

Object.java
package com.clone;

import android.graphics.Canvas;
import android.graphics.Paint;



public abstract class Object{
	protected float posx = 0;	
	protected float posy =0;
	protected float width = 0;
	protected float height = 0;
	
	public Object() {
		posx = 0;
		posy =0;
		width = 0;
		height = 0;
	}

	public Object(float posx,float posy,float width,float height){
		this.posx = posx;
		this.posy = posy;
		this.width = width;
		this.height = height;		
	}
	
	public void moveBound() {
		this.posx += 100;
		this.posy += 100;
	}
	
	abstract  void draw(Canvas canvas,Paint paint);
	abstract  void changeData();
}

TextBox.java
package com.clone;

import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.Paint;
import android.graphics.Paint.Style;
import android.graphics.PointF;

public class TextBox extends Object{
	private PointF mtextpos = null;
	private String mtextString = null;
	
	public TextBox() {
		posx = 0;
		posy =0;
		width = 0;
		height = 0;
		mtextpos = new PointF();
		mtextString = new String();
	}

	public TextBox(float posx, float posy, float width, float height,PointF textpos,String text) {
		this.posx = posx;
		this.posy = posy;
		this.width = width;
		this.height = height;	
		mtextpos = new PointF(textpos.x,textpos.y);
		mtextString = new String(text);
	}

	@Override
	void draw(Canvas canvas, Paint paint) {
		canvas.save();
		paint.setColor(Color.BLUE);
		paint.setStyle(Style.STROKE);
		canvas.drawRect(posx, posy, posx+width, posy+height, paint);
		
		canvas.translate(posx, posy);
		paint.setColor(Color.BLACK);
		paint.setTextSize(20);
		canvas.drawText(mtextString, mtextpos.x, mtextpos.y, paint);
		canvas.restore();
		
	}

	@Override
	void changeData() {
        if(mtextpos != null)
        	mtextpos.offset(20, 50);
	} 
}

CloneUtil.java
package com.clone;

import com.ajexperience.utils.DeepCopyException;
import com.ajexperience.utils.DeepCopyUtil;

public class CloneUtil {

	private DeepCopyUtil deepCopyUtil = getDeepCloneUtil();

	private DeepCopyUtil getDeepCloneUtil() {
		try {
			return new DeepCopyUtil();
		} catch (DeepCopyException e) {
			throw new RuntimeException(e);
		}
	}

	public Object clone(Object object) {
		try {
			return deepCopyUtil.deepCopy(object);
		} catch (DeepCopyException e) {
			throw new RuntimeException(e);
		}
	}

}



相关文章
|
28天前
|
移动开发 Java Android开发
构建高效Android应用:探究Kotlin与Java的性能差异
【4月更文挑战第3天】在移动开发领域,性能优化一直是开发者关注的焦点。随着Kotlin的兴起,其在Android开发中的地位逐渐上升,但关于其与Java在性能方面的对比,尚无明确共识。本文通过深入分析并结合实际测试数据,探讨了Kotlin与Java在Android平台上的性能表现,揭示了在不同场景下两者的差异及其对应用性能的潜在影响,为开发者在选择编程语言时提供参考依据。
|
2月前
|
Java 编译器 Android开发
构建高效Android应用:探究Kotlin与Java的性能差异
【2月更文挑战第30天】 随着Kotlin成为开发Android应用的首选语言,开发者社区对于其性能表现持续关注。本文通过深入分析与基准测试,探讨Kotlin与Java在Android平台上的性能差异,揭示两种语言在编译效率、运行时性能和内存消耗方面的具体表现,并提供优化建议。我们的目标是为Android开发者提供科学依据,帮助他们在项目实践中做出明智的编程语言选择。
|
2月前
|
Java 编译器 Android开发
构建高效Android应用:探究Kotlin与Java的性能差异
在开发高性能的Android应用时,选择合适的编程语言至关重要。近年来,Kotlin因其简洁性和功能性受到开发者的青睐,但其性能是否与传统的Java相比有所不足?本文通过对比分析Kotlin与Java在Android平台上的运行效率,揭示二者在编译速度、运行时性能及资源消耗方面的具体差异,并探讨在实际项目中如何做出最佳选择。
18 4
|
1天前
|
Java 编译器 Android开发
构建高效Android应用:探究Kotlin与Java的性能差异
【4月更文挑战第30天】在Android开发领域,Kotlin作为一种现代化的编程语言,因其简洁性和功能性受到了开发者的广泛欢迎。尽管与传统的Java相比,Kotlin提供了诸多便利,但关于其性能表现的讨论始终未息。本文将深入分析Kotlin和Java在Android平台上的性能差异,通过实际测试数据揭示两种语言在编译效率、运行速度以及内存占用方面的具体表现,并探讨如何利用Kotlin的优势来提升Android应用的整体性能。
|
5天前
|
Java 测试技术 Android开发
构建高效Android应用:探究Kotlin与Java的性能对比
【4月更文挑战第26天】 在移动开发领域,性能优化一直是开发者追求的重要目标。随着Kotlin的兴起,其在Android平台上的应用逐渐增多,但关于Kotlin与Java在性能方面的对比,社区中仍存在诸多讨论。本文通过实际的性能测试,分析比较了使用Kotlin和Java编写的Android应用在多个维度上的运行效率,旨在为开发者提供一个明确的性能参考,帮助他们在选择编程语言时做出更加明智的决策。
|
6天前
|
Java 编译器 Android开发
构建高效Android应用:Kotlin与Java的性能比较
【4月更文挑战第25天】 在移动开发领域,性能优化始终是开发者关注的焦点。随着Kotlin的普及,许多Android开发者开始考虑是否应该采用这种新的编程语言来替代传统的Java。本文通过对Kotlin和Java进行深入的性能比较,揭示了两者在Android开发中的效率差异,并提供了选择合适语言以提升应用性能的策略。通过实际案例分析和基准测试,我们探讨了两种语言在编译速度、运行时性能以及内存消耗等方面的表现,旨在为开发者提供一个关于如何在Android项目中做出明智语言选择的指南。
14 0
|
7天前
|
移动开发 Java Android开发
构建高效Android应用:Kotlin与Java的性能比较
【4月更文挑战第24天】在移动开发领域,性能优化始终是关键议题之一。随着Kotlin的普及,许多Android开发者开始考虑将其作为首选语言。本文深入分析了Kotlin与Java在Android平台上的性能差异,并通过实际案例演示了Kotlin的优化策略。我们将探讨如何通过智能语言特性和现代编程实践来提升应用效率,同时保持代码的可读性和可维护性。
|
15天前
|
数据采集 小程序 数据可视化
Java Android原生智慧校园管理系统源码
对班牌的考试模式、班牌模式上课模式进行设置及管理,设置成功后,班牌端将同步应用。
23 0
|
16天前
|
传感器 小程序 Java
Java+saas模式 智慧校园系统源码Java Android +MySQL+ IDEA 多校运营数字化校园云平台源码
Java+saas模式 智慧校园系统源码Java Android +MySQL+ IDEA 多校运营数字化校园云平台源码 智慧校园即智慧化的校园,也指按智慧化标准进行的校园建设,按标准《智慧校园总体框架》中对智慧校园的标准定义是:物理空间和信息空间的有机衔接,使任何人、任何时间、任何地点都能便捷的获取资源和服务。
16 1
|
2月前
|
Java Android开发
android 开发常见错误:TransformException: java.lang.IllegalStateException:
android 开发常见错误:TransformException: java.lang.IllegalStateException:
9 0