一个程序可以在内存里面存在多个运行实例比如你可以打开多个微软的Word程序但是有些时候我们需要控制程序运行的实例只有一个也就是说该程序同一时刻在内存里面运行的只有一个实例这样当这个程序在内存中已经存在一个运行实例而用户又再次运行了该程序的时候有两种结果第一种结果是结束目前的运行实例打开新运行的实例第二种就是让新运行的实例退出原有的运行实例继续运行
原理
因为任何时候只有一个实例所以在实现这种功能的时候必须借助只能被独享的资源如果我们的程序是基于某个平台的那么就可以借助操作系统的内核对象来完成比如Windows操作系统就提供了CreateMutex这个API来创建一个独享的内核对象但是因为要考虑平台无关Java程序的实例控制不应该使用系统的内核对象来完成那么我们就必须找到其它的可以独享的资源实际上一台机器无论是在什么操作系统上网络端口都是独享的也就是说基于网络端口这个独享的原理我们可以很方便地让我们的Java程序实现在内存里面只有一个运行实例这个功能而且这个功能的实现是与平台无关的
实现
我们先来看看第一种情况是如何实现的也就是说如果系统中已经存在运行实例的话那么结束原有的运行实例让新实例运行这个实现实例控制的Java类也是一个线程具体的实现如下
import *;
public class InstanceControl extends Thread {
public void run() {
try{
Socket sock = new Socket();
//创建socket连接端口
}
catch (Exception e)
{}
try{
ServerSocket server = new ServerSocket();//创建socket在端口监听
serveraccept(); //等待连接
serverclose(); //有连接到来也就是说有新的实例
Systemexit(); //这个实例退出
}catch (Exception e)
{
eprintStackTrace();
}
}
}
下面这个Java程序的程序入口是没有实例控制功能的
public class ProgramMain {
public static void main(String argv[])
{
mainFrame frame = new mainFrame();
}
}
现在想加入实例控制只需要添加两行代码添加后代码如下所示
public class ProgramMain {
public static void main(String argv[])
{
InstanceControl ic = new InstanceControl();
icstart();
mainFrame frame = new mainFrame();
}
}
在这个基础上要实现第二种情况也就是已经有实例运行的情况下新的实例退出保持原有的运行实例就只需要一点小的改动了具体的实现如下
import *;
public class InstanceControl extends Thread {
public void run() {
try{
Socket sock = new Socket( );//创建socket连接端口
Systemexit(); //连接成功说明有实例存在则退出
}catch (Exception e)
{}
try{
ServerSocket server = new ServerSocket();//创建socket连接端口
while (true)
{
serveraccept(); //接受连接请求
}
}catch (Exception e)
{
eprintStackTrace();
}
}
}
这个类的使用方法和第一种情况的那个类是一样的只需要在原有的代码上加入两行代码即可
InstanceControl ic = new InstanceControl();
icstart();
扩展
上面的程序也许有一个小bug就是如果程序在开始运行时ServerSocket监听的端口已经被其它程序占用那么程序的运行就会受到影响所以程序的端口应该尽量取得大一些在这种情况下其它程序占用这个程序使用的端口的概率是可以忽略不计的同时还可以做两种扩展第一种是把端口写在配置文件中可通过读配置文件得到端口这样就能够在其它程序占用目前端口的情况下改变这个程序使用的端口还有一种是在运行的时候用两个InstanceControl类分别在两个端口监听只要有一个InstanceControl类得到连接就做出响应这样两个端口都被其它程序占用的概率就更加的微乎其微了