【Apache CXF】Interceptor与Phase

server端和client端都可以使用拦截器做一些记录、转换、验证、错误处理什么的。
比如处理传输的对象较大时使用GZIPIn/OutInterceptor。



interceptor以phase方式组织起来,相似功能的interceptor属一个phase。
phase实现Comparator接口,以priority属性进行排序,phaseManager将其依次添加至SortedSet。
当需要doInterceptor()时则按顺序依次执行。



CXF提供的Interceptor有in和out的区别。
比如当服务被调用时拦截器链会被创建并调用,此时对于client端是out interceptor,而对于server端则是in interceptor。
反之,当server端作响应时,对于server端是out interceptor,而对于client端则是in interceptor。



打开看org.apache.cxf.interceptor.Interceptor<T extendsMessage>的类型层级关系发现数量好多。
wKiom1Nt2qzSirJnAAG_2waTPOQ559.jpg



CXF中的很多组件都继承org.apache.cxf.interceptor.InterceptorProvider,以自由地添加/移除过滤器。
wKiom1Nt4CLCUquyAACt_qwBRts979.jpg


定义一个拦截器只需要继承org.apache.cxf.phase.AbstractPhaseInterceptor或其子类。
如果需要使用到Message接口中的一些方法的话还是直接继承AbstractPhaseInterceptor比较好;
如果想获取一些特定的信息,则需要继承那些特定的拦截器。



原先我是想直接实现phaseInterceptor或者继承AbstractPhaseInterceptor,但我以annotation方式为服务设置interceptor时必须在constructor中指定phase,而且此时interceptor无法定义一个无参constructor。
于是我继承了SoapActionInInterceptor输出点信息,继承AbstractPhaseInterceptor的子类则是同一个phase:

package pac.king.webservice.interceptor;
import org.apache.cxf.binding.soap.SoapMessage;
import org.apache.cxf.binding.soap.interceptor.SoapActionInInterceptor;
import org.apache.cxf.interceptor.Fault;
public class MyInterceptor extends SoapActionInInterceptor {
    @Override
    public void handleMessage(SoapMessage m) throws Fault {
        System.out.println(m.get(m.PROTOCOL_HEADERS));
        System.out.println(m.get(m.REQUEST_URL));
    }
}


有三种方式设置interceptor。


・调用方法
比如像这样以编程方式启动服务时顺便添加interceptor:

JaxWsServerFactoryBean factory = new JaxWsServerFactoryBean();
factory.setServiceClass(MyCxfServiceImpl.class);
factory.setAddress("http://localhost:8080/runtrain/services/MyCxfService");
factory.create().getEndpoint().getInInterceptors().add(new MyInterceptor(Phase.USER_PROTOCOL));



・annotation
没什么特别的,interceptor注解支持两种参数:

/**
 * Specifies a list of classes that are added to the inbound interceptor
 * chain. This annotation effects SEI classes and service implementation
 * classes.
 */
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
public @interface InInterceptors {
                                                                                                                                                                                                                                         
    String[] interceptors() default { };
                                                                                                                                                                                                                                         
    Class<? extends Interceptor<? extends Message>>[] classes() default { };
}


于是我可以直接写到service类上面:

package pac.king.webservice;
import java.util.HashMap;
import javax.jws.WebMethod;
import javax.jws.WebParam;
import javax.jws.WebResult;
import javax.jws.WebService;
import javax.xml.bind.annotation.adapters.XmlJavaTypeAdapter;
import org.apache.cxf.interceptor.InInterceptors;
import pac.king.pojo.User;
import pac.king.webservice.interceptor.MyInterceptor;
import pac.king.webservice.utils.MyMapAdapter;
@InInterceptors(classes={MyInterceptor.class})
@WebService
public interface MyCxfService {
    @WebMethod
    @XmlJavaTypeAdapter(MyMapAdapter.class)
    public @WebResult HashMap<String,String> convertUserInfoToMap(@WebParam User user);
}



・XML configuration

<bean id="myInterceptor" class="pac.king.webservice.interceptor.MyInterceptor" />
<cxf:bus>
    <cxf:inInterceptors>
        <ref bean="myInterceptor"/>
    </cxf:inInterceptors>
</cxf:bus>


client端设置interceptor的方法几乎相同。
这里记录一下编程方式设置Interceptor的方式,虽然代码中的service是从spring context中拿的,但是配置interceptor则是通过在代码中add:

package pac.king.webservice;
import java.util.Map;
import org.apache.cxf.frontend.ClientProxy;
import org.apache.cxf.phase.Phase;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
import pac.king.pojo.User;
import pac.king.webservice.interceptor.MyInterceptor;
public class TestCase {
    public static void main(String[] args) {
        ApplicationContext context = new ClassPathXmlApplicationContext("classpath*:applicationContext*.xml");
        MyCxfService client = (MyCxfService) context.getBean("MyCxfClient");
        ClientProxy.getClient(client).getInInterceptors().add(new MyInterceptor(Phase.INVOKE));
        Map<String,String> map = client.convertUserInfoToMap(new User("100001","King.","t;stmdtkg"));
        System.out.println(map.get("id"));
        System.out.println(map.get("name"));
        System.out.println(map.get("password"));
                                                   
    }
}



你可能感兴趣的:(webservice,CXF,Interceptor,phase)