如何注册java 程序为windows 服务

最近想找个软件来控制电脑的关机时间,在网上找了几个,都是可视化界面的可以设置具体的关机时间的软件。由于我想编写的关机程序是运行在别人机器上,只能让该机器在晚上17 点到23 点25 分才上网,到了23 点25 分就可以自动关机。为了让别人感觉不到这个软件的“存在”(免得使用者自己关闭定时关机软件),所以我想把关机软件注册为服务的方式,运行在后台。这里介绍下如何利用javaService 软件把java 程序注册为windows 服务。

一、  利用javaService 注册java 程序为windows 服务

[1] 下载javaService

访问网址http://javaservice.objectweb.org/ 下载windows 版本的javaService 文件,我下载的是JavaService-2.0.10.rar ,目前最新的版本就是“2.0.10 ”。

[2] 安装javaService

解压我们下载下来的javaServices 到一个目录,我是解压到目录“D:/software/JavaService-2.0.10 ”下(解压到任何目录都可以,最好别解压到中文目录,省的出现问题

[3] 编写定时关机代码,见第二章的定时关机代码

1)   具体代码参加第二章,类的名字为:

com.test.timer.TimerShutDownWindows

2)   把编写后的java 文件导出为class 的形式,把导出的类放到目录“D:/software/JavaService-2.0.10/classes/com/test/timer ”下。也就是把导出的com 包放到

“D:/software/JavaService-2.0.10/classes” 目录下。

[4] 注册java 程序为windows 服务

进入“D:/software/JavaService-2.0.10 “目录,执行如下命令:

JavaService.exe -install MyShutDownService "%JAVA_HOME%"/jre/bin/server/jvm.dll -Djava.class.path="%JAVA_HOME%"/lib/tools.jar;D:/software/JavaService-2.0.10/classes -start com.test.timer.TimerShutDownWindows

其中“-install “后面的参数为服务的名称,“-start ”参数后边是要启动的类名,“Djava.class.path ”后面参数中的

“D:/software/JavaService-2.0.10/classe ”地址是我的“TimerShutDownWindows ”类存放的路径,实际应用中修改为自己的classPath 即可。

  这里需要注意几点:

1)   “%JAVA_HOME% ”jdk 目录,如果没有配置jdk 目录,则替换为jdk 的实际绝对地址。

2)   -Djava.class.path 是必须的,因为服务启动的时候无法访问系统的CLASSPATH 变量,所以必须在这里声明;如果jar 比较多,为避免写的命令过长,我们可以使用“-Djava.ext.dirs=jars 所在目录”参数。

3)   服务添加之后,可以在命令行中敲入“services.msc ”命令来查看所有服务,并可以对服务的启动类型(自动启动还是手动启动等)进行修改。

[5] 测试

1)  启动服务

当我们注册完服务后,我们可以通过命令“net start MyShutDownService ”来启动该服务,服务启动后会在D 盘根目录生成my_shutdown.log 日志文件。

2)  关闭服务

如果我们要关闭服务,可以通过命令“net stop MyShutDownService ”来关闭该服务。

3)  删除服务

当我们想删除该服务时,可以使用命令“sc delete MyShutDownService ”来删除该服务。

二、  定时关机代码

package com.test.timer;

 

import java.io.File;

import java.io.FileWriter;

import java.io.IOException;

import java.io.PrintWriter;

import java.text.SimpleDateFormat;

import java.util.Calendar;

import java.util.Date;

 

public class TimerShutDownWindows {

 

    /* 检测是否需要关机的时间间隔 */

    private static long m_nDetectInterval = 5000;

 

    /* 记录上次检测的时间,以毫秒为单位 */

    private static long m_lLastMilliSeconds = 0;

 

    /* 可以使用电脑的最小小时 */

    private static int m_nUsePCMinHour = 17;

 

    /* 可以使用电脑的最大小时 */

    private static int m_nUseComputerMaxHour = 23;

 

    /* 如果分钟超过这个时间,则关机计算机 */

    private static int m_nMinutes = 25;

 

    /* 记录日志的文件的保存位置 */

    private static String m_sLogFile = "D:" + File.separator

           + "my_shutdown.log";

 

    /* 记录当前系统是否已经启动自动关闭程序 */

    private static boolean bHasShutDownPC = false;

 

    /**

      * @param args

      */

    public static void main(String[] args) {

 

       // 1. 单独开启一个线程去检测

       Thread aThread = new Thread(new TimerDetector());

       aThread.start();

    }

 

    /**

      * 定义内部类

      *

      * @author Administrator

      *

      */

    static class TimerDetector implements Runnable {

       /*

         * (non-Javadoc)

         *

         * @see java.lang.Runnable#run()

         */

       public void run() {

 

           // 1. 获取日志文件

           PrintWriter out = null;

           SimpleDateFormat df = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");

           try {

              out = new PrintWriter(new FileWriter(m_sLogFile, true), true);

           } catch (IOException e1) {

              out = null;

              e1.printStackTrace();

           }

 

           // 2. 记录服务启动时间

           appendLog(out, " 服务启动时间:" + df.format(new Date()));

 

           while (true) {

              // 1. 判断当前系统时间是否被修改过

              boolean bShoudShutDownPC = validateShoudShutDownPC(out);

 

              if (bShoudShutDownPC) {

                  // 验证没通过,强制关机

                  exectueShutDown(out);

              } else {

                  bHasShutDownPC = false;

              }

 

              // 2. 当前线程休眠下

              try {

                  Thread.sleep(m_nDetectInterval);

              } catch (InterruptedException e) {

                  appendLog(out, e.getMessage());

              }

           }

 

       }

 

       /**

         * 验证当前时间是否是需要关机的时间

         *

         * @return

         */

       private boolean validateShoudShutDownPC(PrintWriter _out) {

 

           // 1. 判断是否修改了系统时间

           boolean bHasModifySystemTime = detectModifySytemTime(_out);

           appendLog(_out, "bHasModifySystemTime :" + bHasModifySystemTime);

           if (bHasModifySystemTime) {

              return bHasModifySystemTime;

           }

 

           // 2. 没有修改系统时间,则判断当前时间是否超过了指定的时间

           boolean bShoudSleep = nowIsSleepTime();

           appendLog(_out, "bShoudSleep :" + bShoudSleep);

           if (bShoudSleep) {

              return bShoudSleep;

           }

 

           return false;

       }

 

       /**

         * 判断当前时间是否应该休息的时间

         *

         * @return

         */

       private boolean nowIsSleepTime() {

           // 1. 获取当前小时和分钟

           Calendar aCalendar = Calendar.getInstance();

           int nHour = aCalendar.get(Calendar.HOUR_OF_DAY);

           int nMinute = aCalendar.get(Calendar.MINUTE);

 

           // 2. 判断当前小时是否在可以使用PC 的时间内, 最大小时为23

           if (nHour < m_nUsePCMinHour) {

              return true;

           }

 

           // 23 点需要单独判断,超过23 点30 就应该休息

           if ((nHour >= m_nUseComputerMaxHour) && (nMinute >= m_nMinutes)) {

              return true;

           }

 

           // 3. 非休息时间

           return false;

       }

 

       /**

         * 判断是否有人修改了系统时间,如果有人修改了系统时间返回true ,<BR>

         * 否则返回false

         *

         * @return

         */

       private boolean detectModifySytemTime(PrintWriter _out) {

           // 1. 第一次检测系统时间

           if (m_lLastMilliSeconds == 0) {

              m_lLastMilliSeconds = System.currentTimeMillis();

              return false;

           }

 

           // 2. 检测两次时间的差值

           long lInteral = System.currentTimeMillis() - m_lLastMilliSeconds;

           lInteral = Math.abs(lInteral);

 

           // 3. 判断两次的时间间隔, 两次结果不一定完全等于 m_nDetectInterval ,允许误差为1 分钟

           long lMaxInterval = m_nDetectInterval + 60 * 1000;

           appendLog(_out, "lInteral :::" + lInteral);

           appendLog(_out, "lMaxInterval :::" + lMaxInterval);

 

           if (lInteral > lMaxInterval) {

              // 有人修改了系统时间,强制关机

              return true;

           }

 

           // 4. 只有没人修改时间才记录上次检测时间

           m_lLastMilliSeconds = System.currentTimeMillis();

           return false;

       }

 

       /**

         * 在指定的流中写入日志信息

         *

         * @param _outWriter

         * @param _sAppendContent

         */

       private void appendLog(PrintWriter _outWriter, String _sAppendContent) {

           if (_outWriter == null) {

              return;

           }

 

           _outWriter.println(_sAppendContent);

 

       }

 

       /**

         * 执行关机命令

         */

       private void exectueShutDown(PrintWriter _out) {

           if (bHasShutDownPC) {

              SimpleDateFormat df = new SimpleDateFormat(

                     "yyyy-MM-dd HH:mm:ss");

              appendLog(_out, " 系统即将关闭, 当前时间:" + df.format(new Date()));

              return;

           }

           appendLog(_out, " 有人修改了系统时间,系统强制关机!");

 

           // 关机

           try {

              Runtime.getRuntime().exec(

                     "shutdown -s -t 120 -c /" 很晚了,该睡觉了,2 分钟后关闭计算机。/"");

           } catch (IOException e) {

              appendLog(_out, e.getMessage());

           }

           bHasShutDownPC = true;

       }

    }

 

}

 

你可能感兴趣的:(java,thread,windows,String,calendar,import)