Java编写服务的准备
1、作为本地化的实现,实现NT服务的Java程序当然不是100%纯Java,单靠标准类库是无法实现我们的编写NT服务的目的,所以MS提供了一套SDK for Java(本文采用的是Microsoft SDK for Java 4.0),提到了如何利用MS提供的扩展类库和相应的工具,实现符合Windows平台需要的程序。其中包括了实现NT服务的所需要的类库API框架以及将Java编译的class文件组装成标准的NT服务程序的工具。SDK的下载路径可以从www.microsoft.com/java/查找到。
2、安装完SDK后可以看到在安装目录下有jntsvc目录,此目录就包含了service.zip文件,它实际上是一个NT services的类库框架,封装了一些NT服务实现细节,使得我们可以按照框架舒服实现我们关心的细节。将service.zip展开至开发机器的系统安装Service库到Java扩展库\Winnt\java\TrustLib下,如果在其他操作系统下进行开发,参照此系统目录进行安装文件。
3、在该目录下还有一个jntsvc.exe文件,也就是Java NT Service的意思啦。她可以帮助您实现将按照SDK提供的框架实现的编译后的class文件组装成一个标准的NT服务可执行文件。JntSvc帮助我们在已经编译好的.class文件基础上设置了所有NT服务程序必须的特征,是很重要的工具,得到NT服务取决于如何有效利用她。为了我们能够方便从任何其他目录的控制台窗口调用她,我们将JntSvc.exe所在的目录全路径加入path环境变量。这可以通过设置系统属性的高级属性页当中进行环境变量的设定。
4、按照要求,我们写好各项代码,然后编译编写Java程序,得到class文件。我们当然不会在Vj Studio中启动她,因为它目前还没有可执行文件的入口,系统无法启动她。为了得到NT服务程序,我们需要在class文件所在目录的控制台窗口执行一个命令:X:jntsvc *.class /OUT:ECHOSvc.exe /SVCMAIN:EchoSvc????????? "/SERVICENAME:ECHOSvc"。具体的Jntsvc的参数我们可以看一看jntsvc -?得到,这里的意思大概是:将当前目录下的所有class文件组装成一个NT服务进程exe文件,文件名为EchoSvc.exe,服务的启动入口在echosvc.class中,在注册表中相应的服务名称为/Servicename参数指定的EchoSvc。如果有多个多个NT服务需要组装在一个Exe文件中,还可以在 /Out参数后指定每一个服务展示名称。/SVCMAIN参数指定服务的入口,所谓入口是指服务启动之初是从哪一个类的实例开始的。"/SERVICENAME:"参数指定了该服务将以什么名称出现。这些参数都是jntsvc.exe实用工具需要组装服务所必须的信息,根据这些信息将编译后的.class文件按照win32格式要求得到一个可执行文件。
需要注意的是,这个exe文件的运行必须要有JVM存在,她实际上是通过解释.class来实现服务提供的。如果需要另外的扩展包,可以通过在/Classpath参数指定另外的扩展包的位置。所以在安装Java编写得到的NT服务的机器上必须存在JVM。如果是拥有IE5.x那么不用操心这个问题,IE核心组件已经包括了JVM;但是如果是IE6版本,则需要到MS的网站上下载JVM。如果您讲SDK for Java安装在服务器上就更方便了。
5、如果没有什么错误,您将得到一个可执行文件echosvc.exe。像大多数服务可执行文件一样,它可以将自己安装到系统中: echosvc.exe ?install,这一个过程将会往系统注册表添加一些项目,特别是关于服务的项目,SCM也可以列出这个服务了。我们可以在控制台下采用DOS NT服务控制命令Net start/stop来测试服务是否真像普通服务一样可以按照标准方式来控制,当然在服务管理器当中启停该服务更不会有问题。
==
Echo服务的样例
当系统载入服务进程时,入口是在EchoSvc的构造函数中,我们可以看到此构造函数带有同一般程序的入口main()类似的参数。
import com.ms.service.* ;
?public class EchoSvc extends Service
{???? static Thread mainSvc=null ; //定义服务主线程
?????? public? EchoSvc (String[] args) //构造此服务
?????? {
???????? CheckPoint(1000);???
????????????? setRunning(ACCEPT_SHUTDOWN | ACCEPT_PAUSE_CONTINUE |ACCEPT_STOP); // 该项服务接受的关于服务控制的命令
????????????? mainSvc = new Thread((Runnable) new? MainSvcThread());
??????? mainSvc.start();????
?????? System.out.println( "The Echo Service Was Started Successfully!");//纪录事件,可以通过事件察看器看到
?????? }
}
CheckPoint是 Service的同步方法,指示系统正改变服务的状态,需要让系统等待1秒。这里我们启动的是一个线程,实际上相当于一个进程,她是服务进程的主线程。在这个线程中我们响应SCM对此服务的控制。大致的表达为:
public class MainSvcThread implements Runnable //实现线程控制
{????
public static boolean STOP = false;? //由系统来控制的内部变量,决定着服务进程(线程)的启动、暂停等
????? public static boolean PAUSE = false;
?
???????? public void run()
???????? {
??????? while (!STOP)
???????????????? {?
??????????????????????? while (!PAUSE && !STOP)
??????????????????????? {
????????????????????????????? 。。。//此处为服务控制逻辑,下面会充实此处
????????????????????? }
??????????????? try
??????????????? {Thread.sleep(5000);//休眠5秒后实现暂停或者停止}
??????????????? catch (InterruptedException e)
{ }
????????????? }
??????????? try
???????????????????? {Thread.sleep(1000);}
???????????????????? catch (InterruptedException ie)
???????????????????? {}
???????? }
???????? }? //Run结束?
}
在服务逻辑控制当中,我们会具体实现Echo服务。我们的Echo服务监听2002端口,接收客户端任何一行输入,然后加上“Echo:”后返回。如果客户端输入一个quit词组那么服务认为这是客户关闭此套接字的命令,会自动关闭当前的套接字连接,停止对当前连接的服务。具体的实现(EchoThread.java的代码):
public void run()
?????? {
??????? String line;
??????? DataInputStream in;
??????? PrintWriter out;
??????? boolean exitflag=false;
?
?????? try
?????? {
??????? in=new DataInputStream(so.getInputStream()) ;//获取套接字的输入流
??????? out=new PrintWriter(new DataOutputStream(so.getOutputStream())) ;
????? out.println("You have connected to EchoSvc!");? //发送问候
??????? out.flush();
??????? while((line=in.readLine())!=null) //读取
???????? {
?????????????? line=line.trim();
?????????????? if (line.equalsIgnoreCase("quit") )
?????????????? {
?????????? out.println("ECHO:" + line );
???????????????? out.flush();
return;
}
?????????????? else
?????????????? {
?????????????? out.println("ECHO:" + line );
?????????????? out.flush();
?????????????? }
???????? }
??????? in.close();
??????? out.close();
}
catch(IOException ioe)
????????????? {}
}
Echo服务主要就是将客户发送的字符回显给客户,并加上Echo:的前缀,以表明是从服务器返回的内容。如果客户输入“quit”那么表示这是要求服务器停止服务的表现。
如何调试NT服务进程工程。如果直接将此函数调用来提供客户端的ECHO套接字服务,逻辑上是没有什么错误,但是就是无法支持多个用户同时访问。为了能够提供多服务,允许同时又多个用户连接此服务器(这种情况在很多网络服务都不可少),我们可以将此逻辑在由MainSvcTread创建的线程中实现,而且可以允许多个用户同时访问此服务。具体的表达在MainSvcTread的run函数中实现:
while(ListenThreadCount
{
????????? server=li.accept(); //监听
????????? EchoThread p=new EchoThread(server,this);//创建实现该服务的具体逻辑对象,是一个支持线程的类
????????? Thread t=new Thread(g,(Runnable)p) ; //将当前线程并入线成组
????????? t.start(); //启动服务线程
????????? ListenThreadCount++; //修改当前线程数量
?????? }
参照上面提到的工具Jntsvc.exe可以帮助你讲编译好的.class文件组装成exe文件,运行此文并加上-install参数可以自动帮助您讲些好的服务添加到注册表中,可以通过服务管理器或者相当的实用程序来如同其他服务一样来进行控制了。撤除服务采用-uninstall参数。
本例程采用套接字、多线程实现技术来解释实现Java编写NT服务,实际上类似这样的很多网络方面的服务都可以按照此规范实现,譬如POP3服务、FTP服务,甚至WWW服务等。我们也接触过像Tomcat、Jrun等Java应用服务器在NT平台的启动往往采用NT服务形式,那么通过此例你也可以尝试编写自己的Java服务应用。
??? 最后,如果需要调试NT服务的逻辑,可以采用一个变通的办法。我们在EchoSvc.java中添加一个Main静态方法,然后产生一个EchoSvc的实例,这样就是一个标准的VJ产生的Exe文件,利用Vj的调试功能我们可以排除隐藏的错误。一旦调通后,我们注释掉main静态方法,编译后就可以得到一个调试好的NT服务。
==
比较VC等“原装”NT服务开发方式而言,Java开发模式可以更快捷,因为几乎所有得服务框架通过扩展Services类就可以达到,省下不少复杂的细节处理。Java语言提供了丰富的类库,可以为自己使用,可以提高效率,而且编写的程序结构清晰容易理解,方便以后维护。
Java提供的异常处理模式,可以让我们写好结构良好,更加安全的代码。试想如果编写的服务进程由于采用VC编写却忘记对某块内存的释放,那么服务启动后一段时间由于内存泄漏造成服务性能下降,甚至系统崩溃;但是Java本身的语言特性可以使我们不用时刻提防内存管理,可以更加关注服务逻辑本身,是的实现起来更加有效率。
采用VC如果编写多线程服务进程,虽然可以实现,但是会相当麻烦。而服务进程多线程几乎是每一个性能良好的服务必备特征,Java语言本身可以提供这方面良好的支持,同时Java自身对网络的天然良好支持,使各种网络套接字编程容易。
最后,如果不采用其他扩展库,我们很容易将此服务逻辑实现在其他操作系统上。一个编写好的NT服务程序,可以在去掉对Ms的相关本地化扩展实现的类引用后,方便移植到其他例如Linux等平台上,尽可能向Java的“一次编写、到处可运行”的理想境界靠拢。
=======================
可考虑用修改注册表的方法,但java没有尝试过。
将HKEY_LOCAL_MACHINE\SYSTEM\ControlSet001\Services里面,随便找个顺眼一点的主键,导出,修改reg文件,把里面的display name改称你要的service name, 把imagepath改称你的C:\jdk\java.exe -cp ...... 这个样子
注:本人没有尝试过java,但是其他exe的服务,我用这个方法做过(比如在另一个Win下安装的MySQL的服务),一切OK。
=======================
利用工具:
一个叫做“JavaExe”的小工具,可以注册service,非常方便的