SOAP WS - 使@WebParam可选

2022-09-03 06:51:42

我有一个非常简单的方法,我在JAX-WS注解的WS API中使用:

@WebMethod
public MyResponse sendSingle2(
    @WebParam(name="username") String username,
    @WebParam(name="password") String password,
    @WebParam(name="newParam") String newParam) {
        // the code
    }

现在我希望newParam是可选的。我的意思是,我希望该方法不仅在传递的xml中的参数为空时仍然有效:

<ws:sendSingle2>
    <username>user</username>
    <password>pass</password>
    <newParam></newParam>
</ws:sendSingle2>

但是当它不存在时也是如此:

<ws:sendSingle2>
    <username>user</username>
    <password>pass</password>
</ws:sendSingle2>

我需要它来不破坏现有的API,它在没有新参数的情况下工作。


答案 1

@WebParam消息部件映射到参数,并且部件不能是可选的。请参见 WSDL 中的可选消息部分。因此,简短的答案是,你所问的恰恰是做不到的。但是,如果可以重构此方法,则可以使用下面描述的方法之一。

通常,参数的可选性是通过架构设置的。此外,您可以使用多个参数,而不是在架构中定义一个 Request 参数,并将其定义为 .可选性现在封装在参数中,并且对于具有或不具有可选参数的调用,将调用相同的方法。minOccurs=0WebMethod

我更喜欢先定义合约,而不是依赖自动生成的文件。一旦你弄清楚了 XSD、SOAP 和 WSDL 是如何协同工作的,你几乎不想再使用基于注释/代码优先的定义,因为反过来你更灵活了。

代码示例:

<xs:schema
    targetNamespace="http://your.namespace.com"
    xmlns:tns="http://your.namespace.com"
    xmlns:xs="http://www.w3.org/2001/XMLSchema"
    elementFromDefault="qualified"
    attributeFromDefault="qualified">

...

<xs:element name="MyRequest" type="tns:MyRequestType" />
<xs:element name="MyResponse" type="tns:MyResponseType" />

<xs:complexType name"MyRequestType">
    <xs:sequence>
        <xs:element name="username" type="xs:string" minOccurs="1" maxOccurs="1" />
        <xs:element name="password" type="xs:string" minOccurs="1" maxOccurs="1" />
        <xs:element name="newParam" type="xs:string" minOccurs="0" maxOccurs="1" />
    </xs:sequence>
</xs:complexType>

...

</xs:schema>

在 WSDL 文件中,您可以像这样定义消息:

<wsdl:definitions
    xmlns:soap="http://schemas.xmlsoap.org/wsdl/soap/"
    xmlns:msg="http://your.namespace.com"
    xmlns:xs="http://www.w3.org/2001/XMLSchema"
    xmlns:http="http://schemas.xmlsoap.org/wsdl/http/"
    xmlns:wsdl="http://schemas.xmlsoap.org/wsdl/"
    xmlns:soapenc="http://schemas.xmlsoap.org/soap/encoding/"
    targetNamespace="http://your.namespace.com">

    <wsdl:types>
        <xs:schema>
            <!-- either import the externalized schema -->
            <xs:import namespace="http://your.namespace.com"
                       schemaLocation="someDir/yourMessageSchema.xsd" />
        </xs:schema>
        <!-- or define the schema within the WSDL - just copy the schema here -->
        <xs:schema
            targetNamespace="http://your.namespace.com"
            xmlns:tns="http://your.namespace.com"
            xmlns:xs="http://www.w3.org/2001/XMLSchema"
            elementFromDefault="qualified"
            attributeFromDefault="qualified">
                ...
        </xs:schema>
    </wsdl:types>

    ...

    <wsdl:message name="sendSingle2Request">
        <wsdl:part name="in" element="msg:MyRequest" />
    </wsdl:message>

    <wsdl:message name="sendSingle2Response">
        <wsdl:part name="out" element="msg:MyResponse" />
    </wsdl:message>

    ...

    <wsdl:portType name="YourServiceEndpoint">
        <wsdl:operation name="sendSingle2">
            <wsdl:input message="tns:sendSingle2Request" />
            <wsdl:output message="tns:sendSingle2Response" />
        </wsdl:operation>
        ...
    </wsdl:portType>

    <wsdl:binding name="YourServiceBinding" type="YourServiceEndpoint">
        <soap:binding style="document" transport="http://schemas.xmlsoap.org/soap/http" />
        <wsdl:operation name=""sendSingle2">
            <soap:operation soapAction="http://your.namespace.com/SendSingle2" style="document" />
            <wsdl:input>
                <soap:body parts="in" use="literal" />
            </wsdl:input>
            <wsdl:output>
                <soap:body parts="out" use="literal" />
            </wsdl:output>
        </wsdl:operation>
        ...
    </wsdl:binding>

    <wsdl:service name="YourService">
        <wsdl:port name="YourServicePort binding="tns:YourServiceBinding">
            <soap:address location="http://your.server:port/path/to/the/service" />
        </wsdl:port>
    </wsdl:service>
</wsdl:definitions>

这里的 WSDL 协定定义了使用样式:在模式的帮助下,实际的 SOAP 消息将更加符合 WS-I。document/literaldocument/literal wrapped

因此,您的方法将更改为 public,现在封装 和 。如果没有与SOAP请求一起发送,它只是会返回,所以最好在使用它之前先检查一下。MyResponse sendSinge2(MyRequest request)requestusernamepassowrdnewParamnewParamnull

如果坚持使用代码优先方法,则需要首先定义用作请求参数的类,而不是这 2 或 3 个值。MyRequest

@XmlAccessorType(XmlAccessType.FIELD)
@XmlRootElement(name = "MyRequest", namespace="http://your.namespace.com")
public class MyRequest implements Serializable
{
   @XmlElement(name = "username", required = true)
   protected String username;
   @XmlElement(name = "password", required = true)
   protected String password;
   @XmlElement(name = "newParam", required = false)
   protected String newParam;
   ...
}

如果您还没有这样做,也应该这样做。Web 方法现在可能如下所示:MyResult

@WebMethod(operationName = "sendSingle2")
@WebResult(name = "sendSingle2Response", targetNamespace = "http://your.namespace.com")
public MyResult sendSingle2(@WebParam(name = "sendSingle2Request") MyRequest request)
{
   ...
}

同样,封装 3 个参数,您应首先检查可选参数是否为 null。request

呵呵


答案 2

这完全取决于您使用这些参数的实现类。在端点接口上,只需将此参数添加为 webparam。

确保在实现类中 如果在任何地方使用此参数,请添加替代代码(Else 部分)以在没有此参数的情况下执行操作或执行。

我认为所有参数都被视为可选参数,除非您验证它们或按照建议在实现类中使用它们。


推荐