`
Rocky_rup
  • 浏览: 146126 次
  • 性别: Icon_minigender_1
  • 来自: 杭州
社区版块
存档分类
最新评论

对象的创建和销毁

 
阅读更多

考虑用静态工厂方法代替构造器
  • 静态工厂方法的名字使得调用者更容易明白构造的意图,
  • 静态工厂方法可以控制实例的创建数量,以提升性能.

代码示例:

public static Boolean valueOf(boolean b) {

return b ? Boolean.TRUE : Boolean.FALSE;

}
当构造参数很多的时候考虑使用建造者(模式)
  • 建造者使得构造的代码编写更为自然,同时保证了构造的一次性,
  • 建造者能够满足实例构造之后的不变性,这是setter方法无法保证的,
  • 一般而言,建造者适用于有四个或以上的可选参数的场景下使用.

代码示例:

// 原始

public class NutritionFacts {

	private final int servingSize; // (mL)  required

	private final int servings; // (per container) required

	private final int calories; //  optional

	private final int fat; // (g) optional

	private final int sodium; // (mg) optional

	private final int carbohydrate; // (g)   optional



	...



	public NutritionFacts(int servingSize, int servings,

	int calories, int fat, int sodium, int carbohydrate) {

		this.servingSize  = servingSize;

		this.servings     = servings;

		this.calories     = calories;

		this.fat          = fat;

		this.sodium       = sodium;

		this.carbohydrate = carbohydrate;

	}

}
// Builder Pattern

public class NutritionFacts {

	private final int servingSize;

	private final int servings;

	private final int calories;

	private final int fat;

	private final int sodium;

	private final int carbohydrate;

	

	public static class Builder {

		// Required parameters

		private final int servingSize;

		private final int servings;

		// Optional parameters - initialized to default values

		private int calories = 0;

		private int fat = 0;

		private int carbohydrate = 0;

		private int sodium = 0;

		

		public Builder(int servingSize, int servings) {

			this.servingSize = servingSize;

			this.servings    = servings;

		}



		public Builder calories(int val)

		{ calories = val; return this; }



		public Builder fat(int val)

		{ fat = val; return this; }



		public Builder carbohydrate(int val)

		{ carbohydrate = val; return this; }



		public Builder sodium(int val)

		{ sodium = val; return this; }



		public NutritionFacts build() {

			return new NutritionFacts(this);

		}

	}



	private NutritionFacts(Builder builder) {

		servingSize = builder.servingSize;

		servings = builder.servings;

		calories = builder.calories;

		fat = builder.fat;

		sodium = builder.sodium;

		carbohydrate = builder.carbohydrate;

	}

}
用枚举实现单例(模式)
  • 尽管通过private可以保证不能通过new来调用构造器,但通过反射机制可以绕过这一限制.

代码示例:

// Enum singleton - the preferred approach

public enum Elvis {

	INSTANCE;

	public void leaveTheBuilding() { ... }

}
避免创建不必要的对象
  • 可以重用对象的时候,不要创建它,
  • 考虑值不变的对象在static语句中创建,

代码示例:

public class Person {

	private final Date birthDate;

	// Other fields, methods, and constructor omitted

	

	// DON'T DO THIS!

	public boolean isBabyBoomer() {

		// Unnecessary allocation of expensive object

		Calendar gmtCal =

		Calendar.getInstance(TimeZone.getTimeZone("GMT"));

		gmtCal.set(1946, Calendar.JANUARY, 1, 0, 0, 0);

		Date boomStart = gmtCal.getTime();

		gmtCal.set(1965, Calendar.JANUARY, 1, 0, 0, 0);

		Date boomEnd = gmtCal.getTime();

		return birthDate.compareTo(boomStart) >= 0 &&

		birthDate.compareTo(boomEnd)   <  0;

	}

}
class Person {

	private final Date birthDate;

	// Other fields, methods, and constructor omitted

	/**

	* The starting and ending dates of the baby boom.

	*/

	private static final Date BOOM_START;

	private static final Date BOOM_END;

	

	static {

		Calendar gmtCal =

		Calendar.getInstance(TimeZone.getTimeZone("GMT"));

		gmtCal.set(1946, Calendar.JANUARY, 1, 0, 0, 0);

		BOOM_START = gmtCal.getTime();

		gmtCal.set(1965, Calendar.JANUARY, 1, 0, 0, 0);

		BOOM_END = gmtCal.getTime();

	}



	public boolean isBabyBoomer() {

		return birthDate.compareTo(BOOM_START) >= 0 &&

		birthDate.compareTo(BOOM_END)   <  0;

	}

}
  • 谨防自动包装(autoboxing),这会产生大量包装类的实例

代码示例:

// Hideously slow program! Can you spot the object creation?

public static void main(String[] args) {

	Long sum = 0L;

	for (long i = 0; i < Integer.MAX_VALUE; i++) {

		sum += i;

	}

	System.out.println(sum);

}
清除过时的对象引用
  • 即时清除过时对象的引用,能够促进对象被垃圾回收,从而避免内存泄漏,

代码示例:

// Can you spot the "memory leak"?

public class Stack {

	private Object[] elements;

	private int size = 0;

	private static final int DEFAULT_INITIAL_CAPACITY = 16;

	

	public Stack() {

		elements = new Object[DEFAULT_INITIAL_CAPACITY];

	}

	public void push(Object e) {

		ensureCapacity();

		elements[size++] = e;

	}

	public Object pop() {

		if (size == 0)

		throw new EmptyStackException();

		return elements[--size];

	}

	/**

	* Ensure space for at least one more element, roughly

	* doubling the capacity each time the array needs to grow.

	*/

	private void ensureCapacity() {

		if (elements.length == size)

		elements = Arrays.copyOf(elements, 2 * size + 1);

	}

}
public Object pop() {

	if (size == 0)

	throw new EmptyStackException();

	Object result = elements[--size];

	elements[size] = null; // Eliminate obsolete reference

	return result;

}
  • 内存泄漏的三大来源
    • 类自我管理内存
    • 缓存
    • 监听,回调中的引用
避免使用finalize()
  • finalize()是不可预知的,经常是危险的,通常是没必要的
分享到:
评论

相关推荐

    Qt6 QML Book/动态QML/创建和销毁对象示例源码

    总的来说,Qt6 QML的动态对象创建和销毁功能为开发者提供了极大的灵活性,使得我们可以根据应用程序的需求来高效地管理UI资源。通过学习和实践“load-component”和“create-object”中的示例代码,你将能够更好地...

    C#创建、销毁对象和继承

    在C#编程中,对象是程序的基本构建块,它们代表了...总结起来,C#中的对象创建、销毁和继承涉及了类的构造方法、初始化、内存管理、多态性以及资源的有效利用。理解并熟练运用这些概念是成为一名优秀的C#开发者的基础。

    监听器:监听三个对象的创建和销毁方法

    本次工程的目标是学习监听器,掌握监听三个对象的创建和销毁方法。 三个对象:rquest,session,ServletContext 实现的接口:ServletRequestListener,HttpSessionListener,ServletContextListener 工程的大致步骤...

    用new创建对象和直接定义的区别

    4. **性能**:栈内存分配和回收比堆内存更快,因此直接定义的对象创建和销毁速度较快。但堆内存可以提供更大的容量。 5. **异常处理**:`new`操作可能会抛出内存不足异常,而直接定义的对象不会面临这个问题(除非...

    ACE对象生命周期管理者:一种用于控制对象创建和销毁的补充模式

    目的对象生命周期管理者模式可以被用来控制对象的整个生命周期,从对象被首次使用前创建它们到应用程序中止前完全的销毁它们。此外通过在应用启动/中止时进行对象自动的预先创建/销毁,使这个模式能够用来替代静态...

    Java对象池技术的原理及其实现

    这种方式显著降低了对象创建和销毁所带来的性能开销,同时也减少了内存的使用量。 #### 实现一个对象池的关键类 实现一个完整的对象池需要以下几个关键类: 1. **对象池工厂 (ObjectPoolFactory) 类**: - 负责...

    C++对象池源码示例

    这种策略可以显著减少对象生命周期管理的开销,尤其是在对象创建和销毁过程中涉及到大量系统调用或内存分配的情况下。 在C++中,对象的创建(`new`操作符)和销毁(`delete`操作符)可能导致内存碎片和性能损失。...

    ART运行时CompactingGC为新创建对象分配内存的过程分析.docx

    在Android的ART运行时环境中,内存管理是一个至关重要的部分,特别是在处理大量对象创建和销毁的电子商务应用中。本文将深入探讨ART运行时引入的Compacting GC(垃圾收集器)如何为新创建的对象分配内存,以及它如何...

    C++面向对象程序设计课后题答案.pdf

    13. 统计对象的创建和销毁数量:暗示了在对象生命周期中跟踪对象创建和销毁的数量。 14. 函数递归调用:虽然没有直接的代码示例,但是通过上下文可以理解涉及到二分查找算法,其中可能有递归调用的实现。 15. ...

    TheadPool 线程池源码 自动管理线程的创建和销毁

    在Java、C++等编程语言中,线程池被广泛应用,用于管理线程的生命周期,避免频繁创建和销毁线程带来的开销。本文将基于提供的"ThreadPool.cpp"和"ThreadPool.h"文件,深入解析线程池的实现原理及其自动管理线程创建...

    对象池优化

    在IT行业中,优化是提升系统性能的关键步骤,尤其是在处理大量对象创建和销毁的场景下。对象池技术就是一种常用的优化手段,特别是在ActionScript 3(AS3)这种面向对象的编程语言中。本文将深入探讨“对象池优化”...

    一个ASP创建动态对象的工厂类(类似PHP的stdClass)

    `Class_Initialize`和`Class_Terminate`分别在对象创建和销毁时初始化和清理资源。`setClassName`用于设置对象的名称,`add`和`setValue`方法则负责添加和修改属性及其访问权限。 这个工厂类的核心在于`add`和`...

    第十一章 持有对象

    10. 性能考量:持有大量对象可能会影响程序性能,因此需要考虑内存占用、对象创建和销毁的时间开销。优化持有对象的方式,如对象池、延迟初始化等,可以提高程序效率。 综上所述,"第十一章 持有对象"可能涵盖的...

    unity对象池的案例Demo

    总的来说,Unity对象池是游戏性能优化的重要手段,通过合理的对象池设计,我们可以减少对象创建和销毁的开销,提高游戏运行的流畅度。"unity对象池的案例Demo"应该为我们提供了实践这些概念的实际代码和步骤,帮助...

    Flex性能,内存管理和对象缓存

    - **性能表现**:在每次“ENTER_FRAME”事件中创建和移除大量CircleRenderer对象时,使用对象缓存技术的应用程序表现出更好的性能,因为它减少了对象创建和销毁带来的开销。 - **稳定性**:对象缓存技术还能提高应用...

    C++源码:非模态对话框创建和销毁

    在`DlgModeless.cpp`和`DlgModeless.h`文件中,可能包含另一个非模态对话框类,它的创建和销毁过程与上述类似,只是类名和具体功能不同。 最后,`StdAfx.cpp`和`StdAfx.h`文件通常包含了预编译头文件,以提高编译...

    对象缓存池AS3实现

    在ActionScript 3(AS3)中,由于其基于ECMAScript的特性,对象创建和销毁可能会导致一定的性能影响,因此对象池模式的应用尤为必要。 接下来,我们将探讨如何用AS3实现对象缓存池。首先,我们需要定义一个基类或...

    轻量级的对象池

    - 优点:减少了对象创建和销毁的开销,提高了系统性能,降低了内存抖动。 - 缺点:增加了系统的复杂性,需要额外的内存来存储池中的对象,而且过度依赖对象池可能导致资源无法及时释放。 总结来说,轻量级对象池...

Global site tag (gtag.js) - Google Analytics