|
马上注册,结交更多好友,享用更多功能,让你轻松玩转社区。
您需要 登录 才可以下载或查看,没有帐号?立即注册
x
对于一个大型项目,如果用java来作,可能需要9个月,并且可能需要翻阅10本以上的书,但如果用ruby来作,3个月,3本书就足够了,而.net也不过3,4本书足以,这就是区别。servlet|web|web服务Servlet在webservices中起到很年夜的感化,卖力剖析soap文件(动静和附件封套),利用wsdl来校验soap动静等。作为基础的编程是能够用servlet来剖析soap动静,特别是关于带附件的soap动静,如许做很通明,可是servlet不克不及作为Web服务,固然拓展的JAXM是能够做到的。如今那些撑持webservices的项目都是经由过程底层上完成了servlet的功效,来完成HTTP+SOAP的通讯,就不必要程序员思索基于HTTP的SOAP动静的通讯历程(不要有如许的曲解,在java平台上HTTP的通讯的服务程序是servlet/jsp,而如今你不必servlet和jsp来通讯,岂不是冲突?现实不冲突,记着是底层已完成了servlet功效,详细通讯有底曾本人办理),间接上升到营业逻辑的服务编程。固然偶然也能够编写剖析SOAP动静的servlet,那就仅仅只是SOAP通讯了(servlet没法作为服务来形貌)。上面依据设置文件,来讲明一些底层与webservices的干系:
回忆一下servlet的映照形式。我们晓得,servlet是从javax.servlet.http.HttpServlet承继的,在服务器端被载进JVM实行,然后向客户端输入html流。
servlet的web.xml文件(位于webapps/foo/WEB-INF目次):
<?xmlversion="1.0"encoding="UTF-8"?>
<!DOCTYPEweb-appPUBLIC"-//SunMicrosystems,Inc.//DTDWebApplication2.2//EN"
"http://java.sun.com/j2ee/dtds/web-app_2.2.dtd">
<web-app>
<servlet-mapping>
<servlet-name>invoker</servlet-name>
<url-pattern>/servlet/*</url-pattern>
</servlet-mapping>
</web-app>
invokerservlet实际上是:org.apache.catalina.servlets.InvokerServlet
按类名供应小服务程序。比方,假如您挪用foo/servlet/HelloServlet,
invokerservlet将装进该HelloServlet(假如它在其类路径中的话)并实行。
初看下面的web.xml,仿佛只给出了一个servlet映照,而没有界说invokerservlet。
实在,invokerservlet是在tomcat的conf目次中的web.xml中界说的::
<servlet>
<servlet-name>invoker</servlet-name>
<servlet-class>
org.apache.catalina.servlets.InvokerServlet
</servlet-class>
<init-param>
<param-name>debug</param-name>
<param-value>0</param-value>
</init-param>
<load-on-startup>2</load-on-startup>
</servlet>
以是,假如抛开Tomcat_HOME/conf/web.xml,我们如许界说一个web.xml,仿佛更能分明的申明成绩:
<?xmlversion="1.0"encoding="UTF-8"?>
<!DOCTYPEweb-appPUBLIC"-//SunMicrosystems,Inc.//DTDWebApplication2.2//EN"
"http://java.sun.com/j2ee/dtds/web-app_2.2.dtd">
<web-app>
<servlet-name>MyInvoker</servlet-name>
<servlet-class>
org.apache.catalina.servlets.InvokerServlet
</servlet-class>
<init-param>
<param-name>debug</param-name>
<param-value>0</param-value>
</init-param>
<load-on-startup>2</load-on-startup>
</servlet>
<servlet-mapping>
<servlet-name>MyInvoker</servlet-name>
<url-pattern>/servlet/*</url-pattern>
</servlet-mapping>
</web-app>
即一切/servlet/*形式的url,城市交给org.apache.catalina.servlets.InvokerServlet来处置。
大概说,一切/servlet/*形式的url,实在都是挪用InvokerServlet这个类,而InvokerServlet自己也是
一个servlet,它也是从HttpServlet承继而来的。
如许,我们本人的servlet就可以够经由过程特定的url实行,即/servlet/OurServlet。
固然,假如你乐意,能够界说任何的urlpattern,而纷歧定是/servlet/*,这一点,正如我们前面
看到的Axis处置Soap动静的办法。
进一步,假如不想让InvokerServlet在两头“捣乱”,我们固然能够间接界说本人的servlet:
<?xmlversion="1.0"encoding="UTF-8"?>
<!DOCTYPEweb-appPUBLIC"-//SunMicrosystems,Inc.//DTDWebApplication2.2//EN"
"http://java.sun.com/j2ee/dtds/web-app_2.2.dtd">
<web-app>
<servlet-name>MyInvoker2</servlet-name>
<servlet-class>
com.foo.MyServlet
</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>MyInvoker2</servlet-name>
<url-pattern>/AnyName/*</url-pattern>
</servlet-mapping>
</web-app>
JSP也是一样的事理,有了下面的剖析,
看看Tomcat_HOME/conf/web.xml中的以下语句就能够JSP的处置办法了,这里就不再空话了:
....
<servlet>
<servlet-name>jsp</servlet-name>
<servlet-class>org.apache.jasper.servlet.JspServlet</servlet-class>
<init-param>
<param-name>logVerbosityLevel</param-name>
<param-value>WARNING</param-value>
</init-param>
<load-on-startup>3</load-on-startup>
</servlet>
<servlet-mapping>
<servlet-name>jsp</servlet-name>
<url-pattern>*.jsp</url-pattern>
</servlet-mapping>
....
进进正题。
我们先来看部署WebService的web.xml:
<?xmlversion="1.0"encoding="ISO-8859-1"?>
<!DOCTYPEweb-appPUBLIC"-//SunMicrosystems,Inc.//DTDWebApplication2.2//EN""http://java.sun.com/j2ee/dtds/web-app_2_2.dtd">
<web-app>
<servlet>
<servlet-name>Axis</servlet-name>
<!--实践servlet程序,这里是AxisServlet-->
<servlet-class>org.apache.axis.transport.http.AxisServlet</servlet-class>
</servlet>
<!--###界说servlet和url的对应干系-->
<servlet-mapping>
<servlet-name>Axis</servlet-name>
<url-pattern>/services/*</url-pattern>
</servlet-mapping>
</web-app>
一切/services/*形式的url城市交给org.apache.axis.transport.http.AxisServlet处置,
AxisServlet固然也是从HttpServlet承继而来的。这就是为何我们部署的Web服务在挪用时都要在
服务称号前加上services/了。
能够说,AxisServlet是一切Web服务挪用的出口。
那末AxisServlet在接办Web服务挪用后都做了哪些事情呢?
客户端用call.invoke()挪用web服务用的是POST,以是出口是AxisServlet.doPost...
而不是AxisServlet.doGet...
先来看看AxisServlet的doPost函数,这里只给出了关头语句及正文:
/**
*ProcessaPOSTtotheservletbyhandingitofftotheAxisEngine.
*HereiswhereSOAPmessagesarereceived
*@paramreqpostedrequest
*@paramresrespose
*@throwsServletExceptiontrouble
*@throwsIOExceptiondifferenttrouble
*/
publicvoiddoPost(HttpServletRequestreq,HttpServletResponseres)
throwsServletException,IOException
{
msgContext=createMessageContext(engine,req,res);//猎取客户哀求信息
engine.invoke(msgContext);//挪用客户端哀求的服务
responseMsg=msgContext.getResponseMessage();//失掉挪用的前往了局
sendResponse(getProtocolVersion(req),contentType,res,responseMsg);//将了局送至客户端
}
如许一来,Web服务挪用的前因后果就大抵分明了。。。
为了高分明后面我们的三个url
http://192.168.0.1/test/services
http://192.168.0.1/test/services/sayHelloService?wsdl
http://192.168.0.1/test/services/sayHelloService?method=sayHelloTo&aname=everybody
是如何取得输入了局的,再来看看AxisServlet的doGet函数,这里只给出了流程框架及正文:
**
*ProcessGETrequests.BecauseAxisdoesnotsupporttheGET-style
*pseudoexecutionofSOAPmethods,thishandlerdealswithqueries
*ofvariouskinds,notrealSOAPactions.
*
*@todoforsecureinstallations,dontstacktraceonfaults
*@paramrequestrequestin
*@paramresponserequestout
*@throwsServletException
*@throwsIOException
*/
publicvoiddoGet(HttpServletRequestreq,HttpServletResponseres)
throwsServletException,IOException
{
//假如路径为空,好比:http://localhost/wstk/services或http://localhost/wstk/services/*
if((pathInfo==null||pathInfo.equals(""))&&!realpath.endsWith(".jws"))
{
//从server-config.wsdd文件中读取一切部署的服务信息,向向客户端列出一切部署的服务,
//包含每一个服务可挪用的办法。
}else
//假如路径不为空,好比:http://localhost/wstk/services/sayHelloService
if(realpath!=null)
{
//假如哀求wsdl,好比:http://localhost/wstk/services/sayHelloService?wsdl
if(wsdlRequested)
{
//创立sayHelloService的WSDL文件并传送至客户端
}else
//这里是使用url挪用Web服务的出口,好比http://192.168.0.1/test/services/sayHelloService?method=sayHelloTo&aname=everybody
if(req.getParameterNames().hasMoreElements())
{
//假如客户端挪用的办法准确,则Axis会挪用响应的JavaBean,并把JavaBean的前往了局
//封装为Soap动静流前往给客户端。
}
}
}
而Axis如何找到我们所哀求的JavaBean呢?谜底是server-config.wsdd文件。
server-config.wsdd
<?xmlversion="1.0"encoding="UTF-8"?>
<deploymentxmlns:java="http://xml.apache.org/axis/wsdd/providers/java"xmlns="http://xml.apache.org/axis/wsdd/">
<servicename="sayHelloService"provider="java:RPC">
<parametername="className"value="sayHello"/>
<parametername="allowedMethods"value="sayHelloTo"/>
</service>
<handlertype="java:org.apache.axis.handlers.http.URLMapper"name="URLMapper"/>
<transportname="http">
<requestFlow>
<handlertype="URLMapper"/>
</requestFlow>
</transport>
</deployment>
WSDD是webservicedeploymentdescriptor的缩写。
最表面的<deployment>元素唆使这是WSDD,并界说了java的名字空间。
接着的<service>元素界说了service。一个service是一个方针链,包含哀求request、内容供应者provider、呼应response。
在这个例子中,我们指出service名字是sayHelloService,provider是"java:RPC",它是axis的标志,唆使这是一个java的RPCservice,
而处置它的真实的class是org.apache.axis.providers.java.RPCProvider。
接着我们要在<parameter>中告知RPCProvider,它怎样实例化并挪用准确的class(如:com.foo.MyService)。
<parameter>元素的className唆使class名,allowedMethods告知引擎那些共用的办法要经由过程soap来挪用。
"*"暗示一切的大众办法,我们也列出办法名字列表,能够空格或逗号支解它们。
大型的应用一般不会用这些框架(因为性能考虑);开发人员根据需要选择用一些框架,也可以不选用框架;不用框架并不代表要自己写框架;修改框架的可能性更小。 |
|