CXF2.7整合spring發布webservice,返回值類型是Map和List類型


  

  在昨天研究了發布CXF發布webservice之后想着將以前的項目發布webservice接口,可是怎么也發布不起來,服務啟動失敗,原來是自己的接口有返回值類型是Map。

  研究了一番之后,發現:

    webService可以處理Java 數據類型、JavaBean、List等,但是卻不能處理Map數據類型,也不能返回List<Map>類型數據。於是做了兩個研究:

 

1.發布返回值類型是Map<String,Object>類型接口

  在發布的時候竟然神奇般的發布成功了,接口如下:

接口:

package cn.qlq.service;

import java.sql.SQLException;
import java.util.Map;

import javax.jws.WebMethod;
import javax.jws.WebService;
import javax.xml.bind.annotation.adapters.XmlJavaTypeAdapter;

import cn.qlq.adapter.MapAdapter;
import cn.qlq.domain.User;

@WebService
public interface UserService {
    /**
     * 保存用戶
     * 
     * @param user
     * @return
     * @throws SQLException
     */
    public int saveUser(User user) throws SQLException;

    /**
     * 根據userId獲取user
     * 
     * @param userId
     * @return
     */
    @WebMethod
    public User getUserById(int userId);
    
    /**
     * 根據userId獲取userMap
     * 
     * @param userId
     * @return
     */
    public Map<String,Object> getUserMapById(int id);
}

 

實現類:

package cn.qlq.service.impl;

import java.sql.SQLException;
import java.util.List;
import java.util.Map;

import javax.jws.WebService;

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;

import cn.qlq.dao.UserDao;
import cn.qlq.domain.User;
import cn.qlq.service.UserService;

@Service
@WebService(targetNamespace = "http://service.qlq.cn")
public class UserServiceImpl implements UserService {
    @Autowired
    private UserDao userDao;

    @Override
    public int saveUser(User user) throws SQLException {
        System.out.println("----------------保存user----------");
        return 0;
    }

    @Override
    public User getUserById(int userId) {
        System.out.println("----------------獲取user----------" + userId);
        return userDao.getUserById(userId);
    }

    @Override
    public Map<String, Object> getUserMapById(int id) {
        return userDao.getUserMapById(id);
    }

}

 

 

發布webservice還是上一篇的方式,啟動成功之后采用動態代理方式訪問:

import javax.xml.namespace.QName;

import org.apache.cxf.jaxws.endpoint.dynamic.JaxWsDynamicClientFactory;

public class TestWS {

    public static void main(String[] args) throws Exception {

        JaxWsDynamicClientFactory dcf = JaxWsDynamicClientFactory.newInstance();

        org.apache.cxf.endpoint.Client client = dcf.createClient("http://localhost/CXFTest/WS/userws?wsdl"); // url為調用webService的wsdl地址

        QName name = new QName("http://service.qlq.cn/", "getUserMapById");// namespace是命名空間,methodName是方法名

        Object[] objects;
        try {
            objects = client.invoke(name,5);// 第一個參數是上面的QName,第二個開始為參數,可變數組
            System.out.println(objects[0].toString());
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

}

 

結果是正常的,如下:(還是map類型)

  cn.qlq.service.GetUserMapByIdResponse$Return@273aa934

 

 

如果報錯,我們可以加一個轉換器,如下:

package cn.qlq.service;

import java.sql.SQLException;
import java.util.Map;

import javax.jws.WebMethod;
import javax.jws.WebService;
import javax.xml.bind.annotation.adapters.XmlJavaTypeAdapter;

import cn.qlq.adapter.MapAdapter;
import cn.qlq.domain.User;

@WebService
public interface UserService {
    /**
     * 保存用戶
     * 
     * @param user
     * @return
     * @throws SQLException
     */
    public int saveUser(User user) throws SQLException;

    /**
     * 根據userId獲取user
     * 
     * @param userId
     * @return
     */
    @WebMethod
    public User getUserById(int userId);
    
    /**
     * 根據userId獲取userMap
     * 
     * @param userId
     * @return
     */
    @XmlJavaTypeAdapter(MapAdapter.class)
    public Map<String,Object> getUserMapById(int id);
}

 

編寫轉換器:

  這里參數需要一個實現了XmlAdapter類的適配器類;

  這里的話XmlAdapter要加兩個參數,XmlAdapter<ValueType,BoundType> 

  ValueType是cxf能接收的類型,BoundType是cxf不能接受的類型;

package cn.qlq.adapter;

import java.util.HashMap;
import java.util.Map;

import javax.xml.bind.annotation.adapters.XmlAdapter;

import cn.qlq.domain.User;

public class MapAdapter extends XmlAdapter<User,Map<String, Object>> {

    @Override
    public Map<String, Object> unmarshal(User v) throws Exception {
        Map result = new HashMap();
        result.put("username", v.getUsername());
        result.put("password", v.getPassword());
        return result;
    }

    @Override
    public User marshal(Map<String, Object> v) throws Exception {
        User u = new User();
        u.setUsername((String) v.get("username"));
        u.setPassword((String) v.get("password"));
        return u;
    }
}

 

 

測試代碼還是上面代碼,結果:(被轉成User對象)

cn.qlq.service.User@530db33e

 

另外附一通用的map轉換器:

package cn.qlq.adapter;


import java.util.HashMap;
import java.util.Iterator;
import java.util.Map;
import java.util.Set;

import javax.xml.bind.annotation.adapters.XmlAdapter;


public class MapAdapter extends XmlAdapter<MapEntity[], Map> {
    
    @Override
    public MapEntity[] marshal(Map map) throws Exception {
        // TODO Auto-generated method stub
        MapEntity[] list = new MapEntity[map.size()];
        Set keyset = map.keySet();
        int index =0;
        for(Iterator itor=keyset.iterator();itor.hasNext();){
            MapEntity item = new MapEntity();
            item.key = itor.next();
            item.value = map.get(item.key);
            list[index++] = item;            
        }
        return list;
    }

    @Override
    public Map unmarshal(MapEntity[] list) throws Exception {
        // TODO Auto-generated method stub
        Map map = new HashMap();
        for(int i=0;i<list.length;i++){
            MapEntity item = list[i];
            map.put(item.key, item.value);
        }
        
        return map;
        
    }

}

class MapEntity{
    public Object key;
    public Object value;
}

 

 

2.發布返回值類型是List<Map<String,Object>>類型接口

在上面service增加一個接口:

package cn.qlq.service;

import java.sql.SQLException;
import java.util.List;
import java.util.Map;

import javax.jws.WebMethod;
import javax.jws.WebService;
import javax.xml.bind.annotation.adapters.XmlJavaTypeAdapter;

import cn.qlq.adapter.MapAdapter;
import cn.qlq.adapter.UserMapAdapter;
import cn.qlq.domain.User;

@WebService
public interface UserService {
    /**
     * 保存用戶
     * 
     * @param user
     * @return
     * @throws SQLException
     */
    public int saveUser(User user) throws SQLException;

    /**
     * 根據userId獲取user
     * 
     * @param userId
     * @return
     */
    @WebMethod
    public User getUserById(int userId);

    /**
     * 模擬獲取所有的用戶,一個map代表一個用戶
     * 
     * @return
     */public List<Map<String, Object>> getAllUsers();

    /**
     * 根據userId獲取userMap
     * 
     * @param userId
     * @return
     */
    @XmlJavaTypeAdapter(MapAdapter.class)
    public Map<String, Object> getUserMapById(int id);
}

 

實現類:

package cn.qlq.service.impl;

import java.sql.SQLException;
import java.util.List;
import java.util.Map;

import javax.jws.WebService;

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;

import cn.qlq.dao.UserDao;
import cn.qlq.domain.User;
import cn.qlq.service.UserService;

@Service
@WebService(targetNamespace = "http://service.qlq.cn")
public class UserServiceImpl implements UserService {
    @Autowired
    private UserDao userDao;

    @Override
    public int saveUser(User user) throws SQLException {
        System.out.println("----------------保存user----------");
        return 0;
    }

    @Override
    public User getUserById(int userId) {
        System.out.println("----------------獲取user----------" + userId);
        return userDao.getUserById(userId);
    }

 @Override public List<Map<String, Object>> getAllUsers() {
        return userDao.getAllUsers(); }

    @Override
    public Map<String, Object> getUserMapById(int id) {
        return userDao.getUserMapById(id);
    }

}

 

 

直接發布報錯如下:

嚴重: Exception sending context initialized event to listener instance of class org.springframework.web.context.ContextLoaderListener
org.springframework.beans.factory.BeanCreationException: Error creating bean with name 'userWS': Invocation of init method failed; nested exception is javax.xml.ws.WebServiceException: org.apache.cxf.service.factory.ServiceConstructionException
    at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.initializeBean(AbstractAutowireCapableBeanFactory.java:1422)
    at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.doCreateBean(AbstractAutowireCapableBeanFactory.java:518)
    at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.createBean(AbstractAutowireCapableBeanFactory.java:455)
    at org.springframework.beans.factory.support.AbstractBeanFactory$1.getObject(AbstractBeanFactory.java:293)
    at org.springframework.beans.factory.support.DefaultSingletonBeanRegistry.getSingleton(DefaultSingletonBeanRegistry.java:222)
    at org.springframework.beans.factory.support.AbstractBeanFactory.doGetBean(AbstractBeanFactory.java:290)
    at org.springframework.beans.factory.support.AbstractBeanFactory.getBean(AbstractBeanFactory.java:192)
    at org.springframework.beans.factory.support.DefaultListableBeanFactory.preInstantiateSingletons(DefaultListableBeanFactory.java:585)
    at org.springframework.context.support.AbstractApplicationContext.finishBeanFactoryInitialization(AbstractApplicationContext.java:895)
    at org.springframework.context.support.AbstractApplicationContext.refresh(AbstractApplicationContext.java:425)
    at org.springframework.web.context.ContextLoader.createWebApplicationContext(ContextLoader.java:282)
    at org.springframework.web.context.ContextLoader.initWebApplicationContext(ContextLoader.java:204)
    at org.springframework.web.context.ContextLoaderListener.contextInitialized(ContextLoaderListener.java:47)
    at org.apache.catalina.core.StandardContext.listenerStart(StandardContext.java:5118)
    at org.apache.catalina.core.StandardContext.startInternal(StandardContext.java:5634)
    at org.apache.catalina.util.LifecycleBase.start(LifecycleBase.java:145)
    at org.apache.catalina.core.ContainerBase$StartChild.call(ContainerBase.java:1571)
    at org.apache.catalina.core.ContainerBase$StartChild.call(ContainerBase.java:1561)
    at java.util.concurrent.FutureTask.run(FutureTask.java:262)
    at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1145)
    at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:615)
    at java.lang.Thread.run(Thread.java:745)
Caused by: javax.xml.ws.WebServiceException: org.apache.cxf.service.factory.ServiceConstructionException
    at org.apache.cxf.jaxws.EndpointImpl.doPublish(EndpointImpl.java:369)
    at org.apache.cxf.jaxws.EndpointImpl.publish(EndpointImpl.java:251)
    at org.apache.cxf.jaxws.EndpointImpl.publish(EndpointImpl.java:537)
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:57)
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
    at java.lang.reflect.Method.invoke(Method.java:606)
    at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.invokeCustomInitMethod(AbstractAutowireCapableBeanFactory.java:1546)
    at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.invokeInitMethods(AbstractAutowireCapableBeanFactory.java:1487)
    at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.initializeBean(AbstractAutowireCapableBeanFactory.java:1419)
    ... 21 more
Caused by: org.apache.cxf.service.factory.ServiceConstructionException
    at org.apache.cxf.jaxb.JAXBDataBinding.initialize(JAXBDataBinding.java:332)
    at org.apache.cxf.service.factory.AbstractServiceFactoryBean.initializeDataBindings(AbstractServiceFactoryBean.java:86)
    at org.apache.cxf.service.factory.ReflectionServiceFactoryBean.buildServiceFromClass(ReflectionServiceFactoryBean.java:478)
    at org.apache.cxf.jaxws.support.JaxWsServiceFactoryBean.buildServiceFromClass(JaxWsServiceFactoryBean.java:690)
    at org.apache.cxf.service.factory.ReflectionServiceFactoryBean.initializeServiceModel(ReflectionServiceFactoryBean.java:540)
    at org.apache.cxf.service.factory.ReflectionServiceFactoryBean.create(ReflectionServiceFactoryBean.java:252)
    at org.apache.cxf.jaxws.support.JaxWsServiceFactoryBean.create(JaxWsServiceFactoryBean.java:205)
    at org.apache.cxf.frontend.AbstractWSDLBasedEndpointFactory.createEndpoint(AbstractWSDLBasedEndpointFactory.java:102)
    at org.apache.cxf.frontend.ServerFactoryBean.create(ServerFactoryBean.java:159)
    at org.apache.cxf.jaxws.JaxWsServerFactoryBean.create(JaxWsServerFactoryBean.java:211)
    at org.apache.cxf.jaxws.EndpointImpl.getServer(EndpointImpl.java:454)
    at org.apache.cxf.jaxws.EndpointImpl.doPublish(EndpointImpl.java:334)
    ... 30 more
Caused by: com.sun.xml.internal.bind.v2.runtime.IllegalAnnotationsException: 1 counts of IllegalAnnotationExceptions
java.util.Map is an interface, and JAXB can't handle interfaces.
    this problem is related to the following location:
        at java.util.Map
        at private java.util.Map cn.qlq.service.jaxws_asm.GetAllUsersResponse._return
        at cn.qlq.service.jaxws_asm.GetAllUsersResponse

    at com.sun.xml.internal.bind.v2.runtime.IllegalAnnotationsException$Builder.check(IllegalAnnotationsException.java:91)
    at com.sun.xml.internal.bind.v2.runtime.JAXBContextImpl.getTypeInfoSet(JAXBContextImpl.java:442)
    at com.sun.xml.internal.bind.v2.runtime.JAXBContextImpl.<init>(JAXBContextImpl.java:274)
    at com.sun.xml.internal.bind.v2.runtime.JAXBContextImpl.<init>(JAXBContextImpl.java:125)
    at com.sun.xml.internal.bind.v2.runtime.JAXBContextImpl$JAXBContextBuilder.build(JAXBContextImpl.java:1127)
    at com.sun.xml.internal.bind.v2.ContextFactory.createContext(ContextFactory.java:130)
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:57)
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
    at java.lang.reflect.Method.invoke(Method.java:606)
    at javax.xml.bind.ContextFinder.newInstance(ContextFinder.java:248)
    at javax.xml.bind.ContextFinder.newInstance(ContextFinder.java:235)
    at javax.xml.bind.ContextFinder.find(ContextFinder.java:445)
    at javax.xml.bind.JAXBContext.newInstance(JAXBContext.java:637)
    at org.apache.cxf.common.jaxb.JAXBContextCache.createContext(JAXBContextCache.java:278)
    at org.apache.cxf.common.jaxb.JAXBContextCache.getCachedContextAndSchemas(JAXBContextCache.java:172)
    at org.apache.cxf.jaxb.JAXBDataBinding.createJAXBContextAndSchemas(JAXBDataBinding.java:464)
    at org.apache.cxf.jaxb.JAXBDataBinding.initialize(JAXBDataBinding.java:330)
    ... 41 more

 

 

 

解決辦法也是編寫轉換器並且在接口的抽象方法上聲明轉換器或者用一個bean接收Map中的entry

(1)第一種解決辦法:不知道什么原因我直接寫轉換器轉List<Map>老是報錯,最后沒辦法了就將List<Map>放進map中再進行轉換器轉換,如下:(需要轉換器)

dao層模擬從數據看查詢的直接是List<Map>

    @Override
    public List<Map<String, Object>> getAllUsers() {
        //模擬從數據庫取數據
        List<Map<String, Object>> result = new ArrayList<Map<String, Object>>();
        for(int i=0;i<3;i++){
            Map map = new HashMap();
            map.put("username", i);
            map.put("password", i);
            result.add((HashMap<String, Object>) map);
        }
        return result;
    }

 

 

service接口方法:

    @XmlJavaTypeAdapter(MapAdapter3.class)
    public Map<String,List<Map<String, Object>>> getAllUsers();

 

 

service實現類實現方法:(就是將List<Map>又裝到一個Map中了)

    @Override
    public Map<String, List<Map<String, Object>>> getAllUsers() {
        Map result = new HashMap();
        List<Map<String, Object>> allUsers = userDao.getAllUsers();
        result.put("allUsers", allUsers);
        return result;
    }

 

轉換器:

package cn.qlq.adapter;

import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import javax.xml.bind.annotation.adapters.XmlAdapter;
import cn.qlq.domain.User;

@SuppressWarnings("all")
public class MapAdapter3 extends XmlAdapter<User[], Map<String, List<Map<String, Object>>>> {

    @Override
    public Map<String, List<Map<String, Object>>> unmarshal(User[] v) throws Exception {
        Map result = new HashMap();
        List<Map> list = new ArrayList();
        for (User u : v) {
            Map map = new HashMap();
            map.put("username", u.getUsername());
            map.put("password", u.getPassword());
            list.add(map);
        }

        result.put("allUsers", list);
        return result;
    }

    @Override
    public User[] marshal(Map<String, List<Map<String, Object>>> v) throws Exception {
        int length = v.size();
        User u[] = new User[length];
        List<Map<String, Object>> userMap = v.get("allUsers");
        for (int i = 0; i < length; i++) {
            Map m = userMap.get(i);
            User user = new User();
            user.setPassword(m.get("username").toString());
            user.setPassword(m.get("password").toString());
            u[i] = user;
        }
        return u;
    }

}

 

 

測試代碼:

import javax.xml.namespace.QName;

import org.apache.cxf.jaxws.endpoint.dynamic.JaxWsDynamicClientFactory;

public class TestWS {

    public static void main(String[] args) throws Exception {

        JaxWsDynamicClientFactory dcf = JaxWsDynamicClientFactory.newInstance();

        org.apache.cxf.endpoint.Client client = dcf.createClient("http://localhost/CXFTest/WS/userws?wsdl"); // url為調用webService的wsdl地址

        QName name = new QName("http://service.qlq.cn/", "getAllUsers");// namespace是命名空間,methodName是方法名

        Object[] objects;
        try {
            objects = client.invoke(name);// 第一個參數是上面的QName,第二個開始為參數,可變數組
            System.out.println(objects[0].toString());
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

}

 

 

結果:

  cn.qlq.service.UserArray@7b17b8ad

 

(2)第二種方法我們可以編寫一個實體類接收Map,也就是map中的一個entry對應bean的一個屬性,這樣我們只用拿List<BeanName>替代List<Map>(不需要轉換器====推薦這種方式)

dao層還是上面的代碼:

    @Override
    public List<Map<String, Object>> getAllUsers() {
        //模擬從數據庫取數據
        List<Map<String, Object>> result = new ArrayList<Map<String, Object>>();
        for(int i=0;i<3;i++){
            Map map = new HashMap();
            map.put("username", i);
            map.put("password", i);
            result.add((HashMap<String, Object>) map);
        }
        return result;
    }

 

 

比如上面每個map都有username和password,我們可以寫一個實體:

package cn.qlq.domain;

public class User {

    private String username;
    private String password;

    public String getUsername() {
        return username;
    }

    public void setUsername(String username) {
        this.username = username;
    }

    public String getPassword() {
        return password;
    }

    public void setPassword(String password) {
        this.password = password;
    }

    @Override
    public String toString() {
        return "User [username=" + username + ", password=" + password + "]";
    }
}

 

 

service方法:

    /**
     * 模擬獲取所有的用戶,一個map代表一個用戶
     * 
     * @return
     */
    public List<User> getAllUsers();

 

 

serviceImpl實現方法

    @Override
    public List<User> getAllUsers() {
        List<Map<String, Object>> allUsers = userDao.getAllUsers();
        List<User> users = new ArrayList();
        for (Map userMap : allUsers) {
            User u = new User();
            u.setUsername(userMap.get("username").toString());
            u.setPassword(userMap.get("password").toString());
            users.add(u);
        }
        return users;
    }

 

 

測試代碼:

import javax.xml.namespace.QName;

import org.apache.cxf.jaxws.endpoint.dynamic.JaxWsDynamicClientFactory;

public class TestWS {

    public static void main(String[] args) throws Exception {

        JaxWsDynamicClientFactory dcf = JaxWsDynamicClientFactory.newInstance();

        org.apache.cxf.endpoint.Client client = dcf.createClient("http://localhost/CXFTest/WS/userws?wsdl"); // url為調用webService的wsdl地址

        QName name = new QName("http://service.qlq.cn/", "getAllUsers");// namespace是命名空間,methodName是方法名

        Object[] objects;
        try {
            objects = client.invoke(name);// 第一個參數是上面的QName,第二個開始為參數,可變數組
            System.out.println(objects[0].toString());
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

}

 

 

結果:

  [cn.qlq.service.User@1a89b61f, cn.qlq.service.User@4713bad5, cn.qlq.service.User@604f1a67]

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM