各位大虾你们好,我做了个软件,不过这个软件的规则是不允许把这个软件同时开启两个或两个以上。问题的重点是每次开启一个程序,都会启动一个javaw,我怎么能让第二个程序知道这个软件已经启动了并在运行中? 

解决方案 »

  1.   

    在类中用一个static类的对象保存单实例例如:
    单实例类public class ConnectionPool {
    //only one instance is needed
    private static ConnectionPool instance;
    private ConnectionPool() {
    System.out.println(Createing the single instance of connection pool);}public static ConnectionPool getInstance() {
    if(instance != null) {
    System.out.println(Getting existing instance of connection pool );
    return instance;
    } else {
    instance = new ConnectionPool();
    return instance;
    }}
    public static void main(String args[]) {
    System.out.println(Get connection pool first time);
    ConnectionPool cPool = ConnectionPool.getInstance();
    System.out.println(Get Connection pool second time );
    ConnectionPool annotherPool = ConnectionPool.getInstance();}
    }不过此方法只使用于一个jvm上不能有多个实例
    如果在多个jvm上要实现单实例,此种方法不适合
      

  2.   

    可以用文件注册在启动以后往固定的文件写入注册信息,如果再有一个实例启动的时候就可以通过文件内容来判断了
    也可以通过socket通信来判断,方法还是很多的
      

  3.   

    congliu你好:
    謝謝你的回復,但是單類模式之針對在一個程序裏面(只啓動一個java或者javaw),但是在第二個程序啓動的時候,他會再次啓動一個javaw,這樣的話,單類模式是不能起到任何作用的。wangbo118你好:
    謝謝你的回復,但是你想沒想過一個問題,就是如果計算機不正常從新啓動,或是突然閒停電,你的程序將無法把文件内容改回到初始值,那麽,等你再次開機的時候,你的程序將會無法啓動。
      

  4.   

    goldfinger你好:
    谢谢你的答复,不过我不是很明白你所说的意思,有没有什么电子书籍,或是什么网站给介绍一下,谢谢!
      

  5.   

    最Cheap的方法是用一个文件记录启动信息,如何启动时记1,退出时将一改为0,启动时若此文件显示1就不给启动.
      

  6.   

    along1你好:你好像没有看其他人的帖子和我回复的帖子吧,你这个方法是行不通的。
    为什么行不通,你看看帖子就明白了。谢谢
      

  7.   

    http://www.yesky.com/42/1720542_1.shtml 这里说的很详细了,不过是VC的,需要做一个JNI来调用。除了这个,还有进程管道间的通讯,也可以用来控制程序的唯一性启动,这个可以直接用java实现的。^_^
      

  8.   

    一起贴给你吧,管道间的java通讯,注意,要用线程来实现。*******************************************************
    Java技巧:使用管道数据流传送数据 作者: BUILDER.COMThursday, January 2 2003 3:08 PM Java I/O系统是建立在数据流概念之上的,在UNIX操作系统中有一个类似的概念很流行,那就是管道,它具有将一个程序的输出当作另一个程序的输入的能力。Java为这种管道概念提供了PipedInputStream和PipedOutputStream类。将这两者结合在一起,它们允许一个Java组件输出数据到输出流,而另一个组件将这个输出流当作输入流来读取。举个例子可以说明,比如一个用来记录应用程序日志信息的组件和一个用来显示流数据动态报表的组件。通过给日志记录组件提供PipedOutputStream,给报表组件提供相应的PipedInputStream,这两个组件就不用知道对方的情况而可以相互通信。最简单的情况下,代码段如下所示:importjava.io.*;public class Foo {static public void main(String[] args) throwsIOException {PipedOutputStream pout = newPipedOutputStream();PipedInputStream= new PipedInputStream(pout);for(int i=0; i < 100; i++) {pout.write((byte)i );}pout.close();int j=0;while( (j = pin.read()) != -1){System.err.println(j);}pin.close();}}上面的代码中,一定要记得调用close()以关闭输出流"pout",否则第二轮循环永远也不会结束。这个粗泛的例子并不是正常的可用的代码。随着第一轮循环数量越来越大,PipedOutputStream中的缓冲将会用尽,从而出现问题,比如它在等待某些程序从流里面删除数据时会导致线程死锁。这就是为什么管道流的Javadoc中声明了PipedInputStream和PipedOutputStream必须用在多线程环境里面的原因。
    原作者: BUILDER.COM 
    原出处: BUILDER.COM
      

  9.   

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

  10.   

    Socket 端口监听,每次启动先检测 程序目录下 myBrotherIsRunning.pid 文件里面有个端口号,如果这个端口号非空,就尝试连接一下,如果能连接就是正在运行,如果有端口号却不能连接那可能是上次以外停电了/死机了,直接修改文件并正常运行。我看 Linux /Unix 上很多程序都是这么做的, 创建一个 pid 文件上面是 进程 ID 或者 Socket 端口号,它存在并且可用那就是正在运行,如果不存在就是未运行, 如果存在却不可用那就是上次灾难发生了(停电,死机)。
      

  11.   

    socket 的方式比文件好
      

  12.   

    谢谢各位,问题已经解决,方法使用的是ServerSocket。