单例模式

   5.3.7  单例模式的优缺点 1、时间和空间 

比较上面两种写法:懒汉式是典型的时间换空间,也就是每次获取实例都会进行判断,看是否需要创建实例,浪费判断的时间。当然,如果一直没有人使用的话,那就不会创建实例,则节约内存空间。 
饿汉式是典型的空间换时间,当类装载的时候就会创建类实例,不管你用不用,先创建出来,然后每次调用的时候,就不需要再判断了,节省了运行时间。 
2、线程安全 
(1)从线程安全性上讲,不加同步的懒汉式是线程不安全的,比如,有两个线程,一个是线程A,一个是线程B,它们同时调用getInstance方法,那就可能导致并发问题。如下示例: 
public static  Singleton getInstance(){  

if(instance == null)

{   
         instance = new Singleton();    

}   
     return instance;   

 }  
程序继续运行,两个线程都向前走了一步,如下: 
1public static  Singleton getInstance(){  

 if(instane == null){  

 instance = new Singleton(); 

  }   
return instance;   


 }  
可能有些朋友会觉得文字描述还是不够直观,再来画个图说明一下,如图5.4所示。 
  
(点击查看大图)图5.4  懒汉式单例的线程问题示意图 
通过图5.4的分解描述,明显地看出,当A、B线程并发的情况下,会创建出两个实例来,也就是单例的控制在并发情况下失效了。 
(2)饿汉式是线程安全的,因为虚拟机保证只会装载一次,在装载类的时候是不会发生并发的。 
(3)如何实现懒汉式的线程安全呢? 
当然懒汉式也是可以实现线程安全的,只要加上synchronized即可,如下:  
1. public static synchronized Singleton getInstance(){}  
但是这样一来,会降低整个访问的速度,而且每次都要判断。那么有没有更好的方式来实现呢? 






  




(4)双重检查加锁 
可以使用"双重检查加锁"的方式来实现,就可以既实现线程安全,又能够使性能不受到很大的影响。那么什么是"双重检查加锁"机制呢? 
所谓双重检查加锁机制,指的是:并不是每次进入getInstance方法都需要同步,而是先不同步,进入方法过后,先检查实例是否存在,如果不存在才进入下面的同步块,这是第一重检查。进入同步块过后,再次检查实例是否存在,如果不存在,就在同步的情况下创建一个实例,这是第二重检查。这样一来,就只需要同步一次了,从而减少了多次在同步情况下进行判断所浪费的时间。 
双重检查加锁机制的实现会使用一个关键字volatile,它的意思是:被volatile修饰的变量的值,将不会被本地线程缓存,所有对该变量的读写都是直接操作共享内存,从而确保多个线程能正确的处理该变量。 
看看代码可能会更加清楚些。示例代码如下: 
 public class Singleton {   

 /**   
   * 对保存实例的变量添加volatile的修饰   

*/   
private volatile static Singleton instance = null;   

  private Singleton(){   }   
 public static  Singleton getInstance(){   
     //先检查实例是否存在,如果不存在才进入下面的同步块   

 if(instance == null){   
//同步块,线程安全地创建实例   

synchronized(Singleton.class){   
//再次检查实例是否存在,如果不存在才真正地创建实例

if(instance == null){   
              instance = new Singleton();     

}  

 }  

 }   
return instance;   

  }  

 }  
这种实现方式可以实现既线程安全地创建实例,而又不会对性能造成太大的影响。它只是在第一次创建实例的时候同步,以后就不需要同步了,从而加快了运行速度。


饿汉式:
        public class Singleton{
            private static Singleton singleton = new Singleton ();
            private Singleton (){}
            public Singleton getInstance(){return singletion;}
       } 

     懒汉式:
       public class Singleton{
            private static Singleton singleton = null;
            public static synchronized synchronized getInstance(){
                 if(singleton==null){
                     singleton = new Singleton();
                 }
                return singleton;
            }

       } 

  

public class SocketClient
{
private volatile static Socket socket = null;
private volatile static SocketClient client = null;
private final static String address = "10.161.12.190"; // "10.230.3.201";
private final static int port = 9500;
private Log log;
private PrintWriter pw = null;
private InputStream br = null;


private SocketClient() throws UnknownHostException, IOException
{
socket = new Socket(address, port);
socket.setKeepAlive(true);
log = LogFactory.getLog(SocketClient.class);
pw = new PrintWriter(new OutputStreamWriter(socket.getOutputStream()));
br = socket.getInputStream();
}


public static SocketClient getInstance () throws UnknownHostException,IOException
{
if (client == null)
{
synchronized (SocketClient.class)
{
if (client == null)
{
client = new SocketClient();
}
}
}
return client;
}

public synchronized void closeSocket()
{
try
{
if (pw != null)
pw.close();
if (br != null)
br.close();
if(socket!=null)
socket.close();

catch (Exception e)

{
log.error("function closeSocket throws: "+e.getMessage());
}
}


public synchronized String sendMessage ( String msg )
{
String result = null;
try
{
if (socket == null || socket.isClosed())
{
socket = new Socket(address, port);
socket.setKeepAlive(true);
pw = new PrintWriter(new OutputStreamWriter(socket.getOutputStream()));
br = socket.getInputStream();
}
pw.println(msg);
pw.flush();
byte[] reads = null;
long count = (new Date()).getTime();
while (reads == null || reads.length == 0)
{
reads = new byte[br.available()];
br.read(reads);
long current = (new Date()).getTime();
if (current - count > 1 * 10 * 1000)
{
break;
}
}
StringBuffer sb = new StringBuffer();
for (int i = 0; i < reads.length; i++)
{
byte current = reads[i];
sb.append((char) current);


}
result = sb.toString();
} catch (UnknownHostException ue)
{
result = "error:socketerror";
} catch (IOException ie)
{
result = "error:ioerror";
} catch (Exception ex)
{
result = "error:" + ex.getMessage();
}
return result;
}
}






  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值