`
schy_hqh
  • 浏览: 555807 次
  • 性别: Icon_minigender_1
社区版块
存档分类
最新评论

单例设计模式

 
阅读更多

设计模式

即解决某类问题的最优方案,强调解决问题的思想。

某种问题有各种解决的办法,从中选取最好的一种办法,最为以后处理问题的统一处理方式,该办法就成为一种模式!

 

 

单例设计模式

解决的问题:

确保一个类在内存中只有唯一的对象;

步骤:

构造方法私有化,保证外界无法创建对象

本类创建自己的对象,并对外提供获取该唯一对象的“接口”

外界通过这个接口获取返回的对象的引用

 

饿汉式(推荐开发中使用)

类一加载就在堆内存创建对象,在外界获取之前就创建好。

package com.gc.design.single;


public class SingleHungry {
	
	//保证成员属性可控,私有化
	private static SingleHungry instance = new SingleHungry();
	
	//禁止外界创建对象
	private SingleHungry() {
		
	}
	
	//由于外界没有对象来调用本类方法,所以方法静态化
	//返回内存中唯一的对象给调用者
	public static SingleHungry getInstance(String code) {
		//通过参数来判断是否返回实例给调用者
		if(code.equals("xxx"))
			return instance;
		return null;
	}
	
}

 

懒汉式

真正获取对象实例的时候,才创建对象

特别注意:为了保证多线程并发时对象的唯一性,需要加synchronized修饰方法。

 

package com.gc.design.single;

public class SingleLazy {
	//初始时不创建对象
	private static SingleLazy instance = null;
	
	//禁止外界创建对象
	private SingleLazy() {
		
	}
	
	//synchronized---多线程并发时,保证对象的唯一性
	public static synchronized SingleLazy getInstance() {
		//真正需要的时候才创建对象
		if(instance == null)
			instance = new SingleLazy();
		return instance;
	}
	
}

  

改进懒汉式,改变加锁的位置

package com.gc.design.single;

public class SingleLazy2 {
	//初始时不创建对象
	private static SingleLazy2 instance = null;
	
	//禁止外界创建对象
	private SingleLazy2() {
		
	}
	
	//synchronized---锁加在方法上,效率低,每次进入方法都需要判断锁
	public static SingleLazy2 getInstance() {
		if(instance==null) {
			//静态函数中,没有this锁,只能使用类的class对象
			synchronized(SingleLazy2.class) {
				//真正需要的时候才创建对象
				if(instance == null)
					instance = new SingleLazy2();
			}
		}
		return instance;
	}
	
}

 

 

 

========================================================================

 

单例模式的实际应用

---读取配置文件

资源文件,如189.properties被放在 xxx/WEB-INF/classes目录下

本示例程序涉及以下几方面的知识:

1.单例模式的应用

2.属性文件的加载

3.文件和流的操作

4.多线程下同步问题的考虑

 

注意:

未针对发生修改的文件进行单独加载,而是一旦有文件发生修改,就重新加载所有的配置,以后改进。程序健壮性有待提高,考虑的点不全,需要改进。没有时间了,暂时先这样。

代码如下:
package com.gc.sms.util;

import java.io.File;
import java.io.FilenameFilter;
import java.io.IOException;
import java.io.InputStream;
import java.util.Enumeration;
import java.util.HashMap;
import java.util.Map;
import java.util.Properties;

public class ConfigManager {
	
	//是否动态加载properties文件
	//动态加载虽然好,但是影响效率,每次都要比较文件修改时间
	private boolean autoReload = false;
	
	private final String BASE_PATH = "/";
	
	private Properties props = null;
	
	private Map<String,Long> lastModifiedMap = new HashMap<String, Long>();

	private static ConfigManager instance = new ConfigManager();
	
	private ConfigManager() {
		props = new Properties();
		init();
	}
	
	public static ConfigManager getInstance() {
		return instance;
	}


	private void init() {
		InputStream[] ins = getAllPropertiesInputStream();
		for (int i = 0; i < ins.length; i++) {
			try {
				props.load(ins[i]);
			} catch (IOException e) {
				throw new RuntimeException("加载配置文件出现异常",e);
			} finally {
				try {
					ins[i].close();
				} catch (IOException e) {
					e.printStackTrace();
				}
			}
		}
	}
	
	private void init(File[] files) {
		props.clear();
		InputStream[] ins = getInputStreamsFromFiles(files);
		for (int i = 0; i < ins.length; i++) {
			try {
				props.load(ins[i]);
			} catch (IOException e) {
				throw new RuntimeException("加载配置文件出现异常",e);
			} finally {
				try {
					ins[i].close();
				} catch (IOException e) {
					e.printStackTrace();
				}
			}
		}
	}

	private InputStream[] getAllPropertiesInputStream() {
		File[] files = getAllPropertiesFiles();
		InputStream[] ins = getInputStreamsFromFiles(files);
		return ins;
	}

	private InputStream[] getInputStreamsFromFiles(File[] files) {
		InputStream[] ins = new InputStream[files.length];
		for (int i = 0; i < files.length; i++) {
			File file = files[i];
			String fileName = file.getName();
			//第1次加载和之后重新加载,这里都对文件的最后修改时间进行了保存。
			if(autoReload)
				lastModifiedMap.put(fileName, Long.valueOf(file.lastModified()));
			ins[i] = ClassLoader.getSystemResourceAsStream(fileName);
		}
		return ins;
	}

	private File[] getAllPropertiesFiles() {
		String parentPath = Class.class.getResource(BASE_PATH).getPath();
		File parentFile = new File(parentPath);
		File[] files = parentFile.listFiles(new FilenameFilter() {
			@Override
			public boolean accept(File dir, String name) {
				return name.endsWith(".properties");
			}
		});
		return files;
	}

	public String getConfigItem(String key) {
		if(autoReload) {
			accordingModifiedTimeDecideReload();
		}
		return props.getProperty(key);
	}

	
	public Map<String,String> getAllConfigItem() {
		if(autoReload) {
			accordingModifiedTimeDecideReload();
		}
		Enumeration<?> enums = props.propertyNames();
		Map<String,String> confs = new HashMap<String,String>();
		while(enums.hasMoreElements()) {
			String key = (String)enums.nextElement();
			confs.put(key, (String) props.get(key));
		}
		return confs;
	}
	
	//多线程并发时,如果有一个文件发生修改,则重新加载所有的配置,需要同步
	private synchronized void accordingModifiedTimeDecideReload() {
		File[] files = getAllPropertiesFiles();
		for (File file : files) {
			if(lastModifiedMap.get(file.getName())!=null &&
					file.lastModified() > lastModifiedMap.get(file.getName())) {
				init(files);
				break;
			}
		}
	}
	

}
 
main()创建多线程环境进行测试
	//测试,线程睡眠的时候,修改配置文件中的数据,线程醒过来后应该读取到新的配置
	//前提是autoReload = true
	public static void main(String[] args) {

		for(int i=0;i<100;i++) {
			new Thread(new Runnable() {
				@Override
				public void run() {
					try {
						int time = 1000+new Random().nextInt(100000);
						System.out.println(Thread.currentThread().getName()+":sleep\t"+time/1000);
						Thread.currentThread().sleep(time);
					} catch (InterruptedException e) {e.printStackTrace();}
					ConfigManager instance = ConfigManager.getInstance();
					String value = instance.getConfigItem("ip");
					System.out.println(value);
				}
			}).start();
		}
	}
 
分享到:
评论
发表评论

文章已被作者锁定,不允许评论。

相关推荐

    单例设计模式的优缺点和设计思想

    单例设计模式是一种在软件工程中广泛使用的创建型设计模式,其核心思想是确保一个类仅有一个实例,并提供一个全局访问点。这种模式在多种场景下具有显著的优势,同时也存在一定的局限性和潜在的问题。 ### 单例设计...

    单例设计模式源码和案例解析

    单例设计模式是一种常用的设计模式,它保证一个类只有一个实例,并提供一个全局访问点。在Java编程中,单例模式的应用广泛,例如控制资源的共享、管理配置信息等。本篇将深入解析单例设计模式的四种实现方式,并通过...

    Java的单例设计模式

    单例设计模式是软件设计模式中的经典模式之一,它的核心思想是确保一个类只有一个实例,并提供全局访问点。这种模式在Java中广泛应用于系统配置、线程池、缓存管理等场景,因为这些场景通常只需要一个共享的全局对象...

    单例设计模式.pdf

    单例设计模式是一种在软件设计中广泛使用的设计模式,它属于创建型模式,其核心目标是确保类只有一个实例,并提供一个全局访问点。这个模式在Java中尤其常见,因为Java的垃圾回收机制允许对象长时间驻留在内存中,...

    java单例设计模式的好处

    单例设计模式是软件开发中一种重要的设计模式,它的核心思想是确保一个类只有一个实例,并提供全局访问点。在Java中,单例模式通常用于控制特定类的实例化过程,以达到节省系统资源、控制并发访问和实现数据共享等...

    C++单例设计模式,单例模式

    C++单例设计模式,单例模式 C++单例设计模式,单例模式

    单例设计模式_单例设计模式_

    单例设计模式是一种在软件工程中广泛使用的设计模式,它的主要目标是确保一个类只有一个实例,并提供一个全局访问点。这种模式在系统中需要频繁创建和销毁对象,且对象需要跨模块共享时特别有用,例如数据库连接、...

    C++单例设计模式

    单例设计模式是一种重要的软件设计模式,用于确保一个类只有一个实例,并提供全局访问点。这种模式在C++中尤其常见,因为它允许多个组件共享同一对象,从而提高效率和一致性。以下是对C++单例模式的详细说明: 1. *...

    Qt单例设计模式(1-8)

    在Qt框架中,单例设计模式是一种经常被用到的设计模式,它确保一个类只有一个实例,并提供全局访问点。这个模式在管理共享资源、配置文件或者数据库连接等方面特别有用。本系列教程将深入探讨Qt中的单例设计模式,...

    设计模式:单例设计模式(全部实现方式)工厂设计模式,抽象工厂模式

    单例设计模式、工厂设计模式和抽象工厂模式是常见的软件开发设计模式。这些设计模式提供了一些有用的思想和实现方式,可以帮助开发人员在设计和实现复杂的软件系统时,更加灵活和高效地进行编程。 单例设计模式是一...

     单例设计模式Singleton1

    单例设计模式Singleton1 单例设计模式Singleton1是Java设计模式中的一种创建型模式,指某个类采用Singleton模式,则在这个类被创建后,只可能产生一个实例供外部访问,并且提供一个全局的访问点。这个模式的核心...

    java单例设计模式

    java单例设计模式

    单例设计模式五种例子

    单例设计模式是一种在软件设计中广泛使用的设计模式,它保证了一个类只有一个实例,并提供一个全局访问点。这种模式在需要频繁创建和销毁对象,或者对象的创建代价高昂时特别有用,例如数据库连接、线程池等。在C#中...

    Java中懒汉单例设计模式线程安全测试

    Java中懒汉单例设计模式线程安全测试,单例设计模式的测试

    单例模式(单例设计模式)详解1

    单例模式(单例设计模式)详解1

    java 单例设计模式

    Java中的单例设计模式是一种创建型设计模式,用于确保一个类只有一个实例,并提供一个全局访问点。这种模式常用于管理共享资源,如数据库连接、线程池或配置对象。单例模式的关键在于限制类的实例化,防止多个实例的...

    PHP单例设计模式,连接多数据库源码

    PHP单例设计模式,连接多数据库源码 单例模式的三个特点 1: 一个类只能有一个实例 2: 它保修自行创建这个实例 3: 必须自行向整个系统提供这个实例. 单例模式中主要的角色 Singleton定义一个Instance操作,允许客户...

    设计模式-单例设计模式

    ### 设计模式—单例设计模式详解 #### 定义 单例模式是一种常见的创建型设计模式,其核心思想在于确保一个类仅有一个实例存在,并且该实例由该类自行创建,随后向整个系统提供这一唯一实例。 #### 类图解析 在设计...

Global site tag (gtag.js) - Google Analytics