나는 웹 서비스 세계에 상대적으로 새롭고 내 연구는 나를 깨우는 것보다 더 혼란스러워하는 것 같습니다. 내 문제는 웹 서비스 기능으로 확장 해야하는 라이브러리 (jar)를 받았다는 것입니다.
이 라이브러리는 다른 개발자와 공유되며 jar의 클래스 중에는 웹 서비스를 호출하는 메서드가있는 클래스가 있습니다 (기본적으로 클래스의 속성을 설정하고, 객체를 db에 저장하는 것과 같은 일부 비즈니스 로직을 수행합니다. 등을 수정하여 개체를 다시 보냅니다). 이 서비스에 대한 호출을 가능한 한 간단하게 만들고 싶습니다. 클래스를 사용하는 개발자 만 수행하면됩니다.
Car c = new Car("Blue");
c.webmethod();
나는 서버에서 사용하기 위해 JAX-WS를 연구했지만 wsimport
서버 나 wsimport
클라이언트에서 둘 다 클래스가 있다는 것을 알고 있기 때문에 클래스 간 상호 작용이 필요하다는 것을 알기 때문에 서버 나 클라이언트 에서 만들 필요가 없다고 생각합니다. 서버와 클라이언트 모두에서 공유됩니다. 클래스에서 웹 서비스와 호출을 수행하는 것이 어떻게 합리적이라고 생각하십니까?
답변
귀하의 문제는 Java에서 SOAP (JAX-WS) 웹 서비스를 호출하고 반환하는 객체를 얻는 방법으로 귀결된다는 것을 이해합니다 . 이 경우 두 가지 가능한 접근 방식이 있습니다.
- 다음을 통해 Java 클래스 생성
wsimport
이를 하고 사용하십시오. 또는 - 다음과 같은 SOAP 클라이언트를 만듭니다.
- 서비스의 매개 변수를 XML로 직렬화합니다.
- HTTP 조작을 통해 웹 메서드를 호출합니다. 과
- 반환되는 XML 응답을 다시 객체로 구문 분석합니다.
첫 번째 접근 방법 (사용 wsimport
) :
귀하는 이미 서비스 (엔터티 또는 기타) 비즈니스 클래스를 보유하고 있습니다. wsimport
에서 완전히 새로운 클래스 집합 (이미 가지고있는 클래스의 중복)을 생성 .
하지만이 시나리오에서는 다음 중 하나만 수행 할 수 있습니다.
wsimport
생성 된 코드를 수정 (편집) 하여 비즈니스 클래스를 (이것은 어렵고 가치가 없습니다. WSDL이 변경 될 때마다 코드를 재생성하고 다시 수정해야합니다.) 또는wsimport
생성 된 클래스를 포기하고 사용하십시오 . (이 솔루션에서 비즈니스 코드는 생성 된 클래스를 다른 아키텍처 계층의 서비스로 “사용”할 수 있습니다.
두 번째 접근 방식 (사용자 지정 SOAP 클라이언트 생성) :
두 번째 접근 방식을 구현하려면 다음을 수행해야합니다.
- 전화 걸기 :
- SAAJ (SOAP with Attachments API for Java) 프레임 워크 (아래 참조, Java SE 1.6 이상과 함께 제공됨)를 사용하여 호출합니다. 또는
- 당신은 또한 그것을 통해 할 수 있습니다
java.net.HttpUrlconnection
(그리고 약간의java.io
처리).
- 객체를 XML로 또는 XML에서 되돌립니다.
- JAXB와 같은 OXM (Object to XML Mapping) 프레임 워크를 사용하여 객체에서 XML을 직렬화 / 역 직렬화합니다.
- 또는 필요한 경우 XML을 수동으로 생성 / 파싱합니다 (수신 된 객체가 전송 된 객체와 약간만 다를 경우 최상의 솔루션이 될 수 있음).
클래식을 사용하여 SOAP 클라이언트를 만드는 java.net.HttpUrlConnection
것은 그리 어렵지 않습니다 (하지만 그렇게 간단하지는 않습니다). 이 링크 에서 아주 좋은 시작 코드를 찾을 수 있습니다 .
SAAJ 프레임 워크를 사용하는 것이 좋습니다.
SAAJ (SOAP with Attachments API for Java) 는 주로 웹 서비스 API의 뒤에서 발생하는 SOAP 요청 / 응답 메시지를 직접 처리하는 데 사용됩니다. 개발자가 JAX-WS를 사용하는 대신 SOAP 메시지를 직접 보내고받을 수 있습니다.
SAAJ를 사용하는 SOAP 웹 서비스 호출의 작동 예제 (실행!)는 아래를 참조하십시오. 이 웹 서비스를 호출 합니다 .
import javax.xml.soap.*;
public class SOAPClientSAAJ {
// SAAJ - SOAP Client Testing
public static void main(String args[]) {
/*
The example below requests from the Web Service at:
https://www.w3schools.com/xml/tempconvert.asmx?op=CelsiusToFahrenheit
To call other WS, change the parameters below, which are:
- the SOAP Endpoint URL (that is, where the service is responding from)
- the SOAP Action
Also change the contents of the method createSoapEnvelope() in this class. It constructs
the inner part of the SOAP envelope that is actually sent.
*/
String soapEndpointUrl = "https://www.w3schools.com/xml/tempconvert.asmx";
String soapAction = "https://www.w3schools.com/xml/CelsiusToFahrenheit";
callSoapWebService(soapEndpointUrl, soapAction);
}
private static void createSoapEnvelope(SOAPMessage soapMessage) throws SOAPException {
SOAPPart soapPart = soapMessage.getSOAPPart();
String myNamespace = "myNamespace";
String myNamespaceURI = "https://www.w3schools.com/xml/";
// SOAP Envelope
SOAPEnvelope envelope = soapPart.getEnvelope();
envelope.addNamespaceDeclaration(myNamespace, myNamespaceURI);
/*
Constructed SOAP Request Message:
<SOAP-ENV:Envelope xmlns:SOAP-ENV="http://schemas.xmlsoap.org/soap/envelope/" xmlns:myNamespace="https://www.w3schools.com/xml/">
<SOAP-ENV:Header/>
<SOAP-ENV:Body>
<myNamespace:CelsiusToFahrenheit>
<myNamespace:Celsius>100</myNamespace:Celsius>
</myNamespace:CelsiusToFahrenheit>
</SOAP-ENV:Body>
</SOAP-ENV:Envelope>
*/
// SOAP Body
SOAPBody soapBody = envelope.getBody();
SOAPElement soapBodyElem = soapBody.addChildElement("CelsiusToFahrenheit", myNamespace);
SOAPElement soapBodyElem1 = soapBodyElem.addChildElement("Celsius", myNamespace);
soapBodyElem1.addTextNode("100");
}
private static void callSoapWebService(String soapEndpointUrl, String soapAction) {
try {
// Create SOAP Connection
SOAPConnectionFactory soapConnectionFactory = SOAPConnectionFactory.newInstance();
SOAPConnection soapConnection = soapConnectionFactory.createConnection();
// Send SOAP Message to SOAP Server
SOAPMessage soapResponse = soapConnection.call(createSOAPRequest(soapAction), soapEndpointUrl);
// Print the SOAP Response
System.out.println("Response SOAP Message:");
soapResponse.writeTo(System.out);
System.out.println();
soapConnection.close();
} catch (Exception e) {
System.err.println("\nError occurred while sending SOAP Request to Server!\nMake sure you have the correct endpoint URL and SOAPAction!\n");
e.printStackTrace();
}
}
private static SOAPMessage createSOAPRequest(String soapAction) throws Exception {
MessageFactory messageFactory = MessageFactory.newInstance();
SOAPMessage soapMessage = messageFactory.createMessage();
createSoapEnvelope(soapMessage);
MimeHeaders headers = soapMessage.getMimeHeaders();
headers.addHeader("SOAPAction", soapAction);
soapMessage.saveChanges();
/* Print the request message, just for debugging purposes */
System.out.println("Request SOAP Message:");
soapMessage.writeTo(System.out);
System.out.println("\n");
return soapMessage;
}
}
직렬화 / 역 직렬화를 위해 JAXB를 사용하는 방법에 대한 정보를 찾는 것은 매우 쉽습니다. http://www.mkyong.com/java/jaxb-hello-world-example/에서 시작할 수 있습니다 .
답변
또는 Apache CXF의 wsdl2java 를 사용하여 사용할 수있는 개체를 생성하십시오.
웹 사이트에서 다운로드 할 수있는 바이너리 패키지에 포함되어 있습니다. 다음과 같은 명령을 간단히 실행할 수 있습니다.
$ ./wsdl2java -p com.mynamespace.for.the.api.objects -autoNameResolution http://www.someurl.com/DefaultWebService?wsdl
그것은 당신처럼 사용할 수있는 객체를 생성하기 위해 WSDL을 사용하여 이 (당신은 다른의 약간 수 있도록 개체 이름은 또한 WSDL에서 잡고있다)
DefaultWebService defaultWebService = new DefaultWebService();
String res = defaultWebService.getDefaultWebServiceHttpSoap11Endpoint().login("webservice","dadsadasdasd");
System.out.println(res);
소스를 생성하는 Maven 플러그인도 있습니다 : https://cxf.apache.org/docs/maven-cxf-codegen-plugin-wsdl-to-java.html
참고 : CXF 및 IDEA를 사용하여 소스를 생성하는 경우 https://stackoverflow.com/a/46812593/840315 를 살펴볼 수 있습니다.
답변
비누 메시지를 생성하는 훨씬 더 간단한 대안을 찾았습니다. Person 객체가 주어지면 :
import com.fasterxml.jackson.annotation.JsonInclude;
@JsonInclude(JsonInclude.Include.NON_NULL)
public class Person {
private String name;
private int age;
private String address; //setter and getters below
}
다음은 간단한 Soap 메시지 생성기입니다.
import com.fasterxml.jackson.databind.DeserializationFeature;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.databind.SerializationFeature;
import com.fasterxml.jackson.datatype.jsr310.JavaTimeModule;
import lombok.extern.slf4j.Slf4j;
import org.apache.commons.lang3.StringUtils;
import com.fasterxml.jackson.dataformat.xml.XmlMapper;
@Slf4j
public class SoapGenerator {
protected static final ObjectMapper XML_MAPPER = new XmlMapper()
.enable(DeserializationFeature.READ_UNKNOWN_ENUM_VALUES_AS_NULL)
.configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false)
.configure(SerializationFeature.WRITE_DATES_AS_TIMESTAMPS, false)
.registerModule(new JavaTimeModule());
private static final String SOAP_BODY_OPEN = "<soap:Body>";
private static final String SOAP_BODY_CLOSE = "</soap:Body>";
private static final String SOAP_ENVELOPE_OPEN = "<soap:Envelope xmlns:soap=\"http://schemas.xmlsoap.org/soap/envelope/\">";
private static final String SOAP_ENVELOPE_CLOSE = "</soap:Envelope>";
public static String soapWrap(String xml) {
return SOAP_ENVELOPE_OPEN + SOAP_BODY_OPEN + xml + SOAP_BODY_CLOSE + SOAP_ENVELOPE_CLOSE;
}
public static String soapUnwrap(String xml) {
return StringUtils.substringBetween(xml, SOAP_BODY_OPEN, SOAP_BODY_CLOSE);
}
}
다음과 같이 사용할 수 있습니다.
public static void main(String[] args) throws Exception{
Person p = new Person();
p.setName("Test");
p.setAge(12);
String xml = SoapGenerator.soapWrap(XML_MAPPER.writeValueAsString(p));
log.info("Generated String");
log.info(xml);
}