如何使用Java编写NT服务

 

 

如何使用Java编写NT服务

杨山河

内容简介:本文通过例子讲解了如何利用Java的特性快速编写安全可靠的NT服务,并展示了Java的多线程如何实施,以及如何应用套接字实现网络服务。

关键词:Java  JntSvc.exe NT服务 多线程 套接字编程

 

一、NT服务介绍

所谓NT服务,实际上是一类特殊的应用程序所谓NT服务,实际上就是一个可以在系统启动时自动在一定身份下启动的伴随系统长时间存在的进程。象FTP serverHTTP server、脱机打印等都是采用NT服务的形式提供的。这实际上类似Unixroot daemon进程。NT服务归纳起来,NT服务又以下几个特征:

1、可以自启动,不需要交互启动。这对于服务器来说是一个重要的特征。当然,你可以决定服务是否自启动,甚至可以屏蔽某个服务。
    2
NT服务没有用户界面,基本上类似一个DOS 程序,因为NT服务必须长时间运行,所以不想普通win32进程一样有自己的界面。但是NT服务可以同用户有界面交互,这是一类特殊的服务进程。可以通过NT的任务管理器来看到服务进程。
    3
NT服务通过SCMServices Control Manager)接口来管理,安装、启动、停止、撤除等都需要SCM的接口功能来进行。控制面板的服务控制器就是利用SCM接口来管理系统中的所有服务的。实际上,还有一些可以控制服务的程序或者命令,有net.exe 、服务器管理器等 SCM.exe等。
    4
、这些进程都以一定的身份运行,以方便进行服务器资源的存取。一般情况下使用域中的LocalSystem账号运行,此账号对本机上的大多数资源(除非特别禁止)有完全的存取权限,这样可以保证服务程序的“强大”。但是,也有些服务采用特别的账号运行,你也可以特别设定一个服务的帐号。
    5
、由系统自动以线程方式运行,一般情况下不过多占用系统资源,这同普通的进程有所区别,如果不采用线程方式,一般进程往往消耗整个CPU资源。一般需要时时存在,又不能过多消耗资源的任务以服务来实现最合适。

 

二、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创建的线程中实现,而且可以允许多个用户同时访问此服务。具体的表达在MainSvcTreadrun函数中实现:

while(ListenThreadCount<maxSocket) //如果当前启动的线程数在系统允许的范围内

{

          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服务等。我们也接触过像TomcatJrunJava应用服务器在NT平台的启动往往采用NT服务形式,那么通过此例你也可以尝试编写自己的Java服务应用。

    最后,如果需要调试NT服务的逻辑,可以采用一个变通的办法。我们在EchoSvc.java中添加一个Main静态方法,然后产生一个EchoSvc的实例,这样就是一个标准的VJ产生的Exe文件,利用Vj的调试功能我们可以排除隐藏的错误。一旦调通后,我们注释掉main静态方法,编译后就可以得到一个调试好的NT服务。

 

四、为什么要采用Java编写NT服务

比较VC等“原装”NT服务开发方式而言,Java开发模式可以更快捷,因为几乎所有得服务框架通过扩展Services类就可以达到,省下不少复杂的细节处理。Java语言提供了丰富的类库,可以为自己使用,可以提高效率,而且编写的程序结构清晰容易理解,方便以后维护。

Java提供的异常处理模式,可以让我们写好结构良好,更加安全的代码。试想如果编写的服务进程由于采用VC编写却忘记对某块内存的释放,那么服务启动后一段时间由于内存泄漏造成服务性能下降,甚至系统崩溃;但是Java本身的语言特性可以使我们不用时刻提防内存管理,可以更加关注服务逻辑本身,是的实现起来更加有效率。

采用VC如果编写多线程服务进程,虽然可以实现,但是会相当麻烦。而服务进程多线程几乎是每一个性能良好的服务必备特征,Java语言本身可以提供这方面良好的支持,同时Java自身对网络的天然良好支持,使各种网络套接字编程容易。

最后,如果不采用其他扩展库,我们很容易将此服务逻辑实现在其他操作系统上。一个编写好的NT服务程序,可以在去掉对Ms的相关本地化扩展实现的类引用后,方便移植到其他例如Linux等平台上,尽可能向Java的“一次编写、到处可运行”的理想境界靠拢。

 

五、源码

/*所附的ZIP文件报含示例的全部工程文件,还有编译后的NT服务的可执行文件,您可以直接测试此服务exe文件的安装、服务启停*/

/*EchoSvc.java*/

import com.ms.service.* ;

public class EchoSvc extends Service

{

       static Thread mainSvc=null ; //定义主线程

 

       public  EchoSvc (String[] args) //构造服务

       {

                CheckPoint(1000);    //服务是系统的一部分,作为Log纪录,可以帮助用户理解系统故障

                     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!");

       }

}

/*-------------- EchoSvc.java源码结束-------------------*/

 

/*MainSvcThread.java*/

import java.io.*;

import java.net.*;

public class MainSvcThread implements Runnable //实现线程控制多线程接口

{     /将启动一组线程来监听多个服务请求

         public static boolean STOP = false;  //由系统来控制的内部变量,决定着服务进程(线程)的启动、暂停等

      public static boolean PAUSE = false;

         public int ListenThreadCount=0;  //本服务支持的当前线程数量

         int maxSocket=10;  //最大支持的同时连结数

         int SvcPort=2002;  //服务监听的端口

        

         public void run()

         {

        try

               {

           while (!STOP)

                 { 

                        while (!PAUSE && !STOP)

                        {

                               {//创建监听服务器

                             Socket server; 

                             ServerSocket  li=new ServerSocket(SvcPort);  //创建服务器端套接字

                             ThreadGroup g=new ThreadGroup("EchoThreads"); //创建一组线程

                    System.out.println("Echo service starting...");  //记录在Log

                             while(ListenThreadCount<maxSocket)

                                      {

                                             server=li.accept();  //监听

                                             EchoThread p=new EchoThread(server,this); //创建服务单线程

                                             Thread t=new Thread(g,(Runnable)p) ; //创建新线程

                                             t.start(); //启动服务线程

                                             ListenThreadCount++; //当前线程的数量

                                      }

                      }

                   try

                   {

                     Thread.sleep(5000);//暂停5

                    }

                    catch (InterruptedException e)

{   }

              }

               try

                         {

                           Thread.sleep(1000);

                         }

                        catch (InterruptedException ie)

                       {  }

          }

              }

              catch (IOException ioe)

              {}

         }  //Run结束 

}

/*-------------- MainSvcThread.java源码结束-------------------*/

 

/*EchoThread.java*/

import java.io.*;

import java.net.*;

/*实现每一个客户连接到此NT服务时的服务器端的线程单元逻辑*/

public class EchoThread implements Runnable   //实现线程接口

{

    Socket so=null;//套接字

       MainSvcThread p;  //一个指向父线程的指针,EchoThread的线程是服务线程的创建的子线程

       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 && ! exitflag)

                {

                      line=line.trim();

                      if (line.equalsIgnoreCase("quit") )

                      {//如果是退出命令,则关闭当前套接字上的输入输出流

                            in.close();

                out.flush();

                            out.close();

p.ListenThreadCount --; //主线程的服务线程单元数量控制

                            return;   //退出当前的服务逻辑线程单元

                      }

                      else

                      {

                      out.println("ECHO:" + line );

                      out.flush();

                      }

                 }

          in.close();

                out.close();

                p.ListenThreadCount --;

              }

              catch(IOException ioe)

              {}

       }

      

  EchoThread(Socket s,MainSvcThread parent)

       {

              so=s;

              p= parent;

       }

}

/*-------------- EchoThread.java源码结束-------------------*/

阅读终点,创作起航,您可以撰写心得或摘录文章要点写篇博文。去创作
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 1
    评论
### 回答1: nt98562 lint out 设计是指在软件开发过程中利用代码静态分析工具对代码进行审查和优化的一种方法。该方法主要用于检查代码中的潜在错误和缺陷,确保代码的可读性、可维护性和可靠性。在进行代码审查时,通常使用的工具是针对特定编程语言的语法检查器或静态分析工具,例如 C++ 程序中的 Lint 工具或者 Java 程序中的 FindBugs 工具等。 nt98562 lint out 设计的优势在于提高软件的质量和稳定性,减少系统崩溃和漏洞的风险。通过使用这种方法,开发人员可以快速找到代码中的问题并及时解决,减少调试和修复的时间和成本。此外,nt98562 lint out 设计也帮助开发人员更好地理解和掌握代码,促进开发团队之间的协作和交流。 值得注意的是,nt98562 lint out 设计并不能完全替代人工代码审查和设计评审。开发人员需要在编写代码的过程中遵循良好的编程规范和设计原则,并配合代码审查工具进行操作。只有这样才能实现高质量的软件开发和维护,为用户提供优质的产品和服务。 ### 回答2: nt98562 lint out设计是一项针对软件程序开发的静态代码分析工具。它的主要目标是帮助开发者发现潜在的缺陷或者错误,以提高软件品质。该工具能够自动地扫描源代码,检查是否有编码错误、规则违反、潜在的内存泄漏等方面的问题,并给出相关的警告或者建议。 nt98562 lint out设计的优势不仅仅在于其准确性和高效性,更重要的是它可以大幅度地降低软件开发过程中的错误率。在软件开发过程中,经常会出现一些低级或者微小的错误,但这些错误可能会在后期导致严重的问题。nt98562 lint out设计能够在开发者提交代码之前及时发现这些问题,并帮助开发者及时修复。 使用nt98562 lint out设计有助于提高软件的可靠性、可维护性和稳定性。它不仅能够帮助开发者发现问题,还能根据项目的规模和复杂度进行定制化的配置,以适应不同的需求。总之,nt98562 lint out设计是一项非常有价值的工具,可以帮助开发者提高软件质量,减少成本并提高效率。 ### 回答3: NT98562 Lint Out 设计是一种对于电子元器件进行检测的设计方法。Lint Out 是指在进行编程时,对于代码进行静态分析,进而检查其中的错误和不规范的编程风格,以及有潜在的漏洞或者安全隐患的地方,在这个设计中,NT98562是一种常用的处理器芯片,可以用来执行代码的分析和检测。 在NT98562 Lint Out 设计中,通过在编译过程中进行静态分析,可以提前检测一些代码中的问题,并给出相应的提示信息,有利于开发者进行快速的调试和修改。这种设计可以提高代码的可读性和可维护性,减少因为编程风格不规范而导致的代码可读性差、难以维护的问题。同时,NT98562作为处理器芯片的选择,可以保证代码分析的效率和准确性,为开发者提供更多的便利。 总的来说,NT98562 Lint Out 设计是一种有益的编码规范检查方法,对于提高代码质量、优化开发效率有重要作用,被广泛应用于各种领域的开发中。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

shanhe

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值