存档

文章标签 ‘Java’

SpringMVC获取所有映射URL

2014年8月30日 没有评论

  获取代码如下:

public static List<String> getAllRequestMapping(HttpServletRequest request){
    ServletContext sc = request.getSession().getServletContext();
    WebApplicationContext wac = WebApplicationContextUtils.getWebApplicationContext(sc,FrameworkServlet.class.getName() + ".CONTEXT.mvc");
    RequestMappingHandlerMapping mapping = wac.getBean(RequestMappingHandlerMapping.class);
 
    List<String> list = new ArrayList<String>();
    for(RequestMappingInfo info : mapping.getHandlerMethods().keySet()){
    	for(String url : info.getPatternsCondition().getPatterns()){
    		list.add(url);
    		break;
    	}
    }
 
    return list;
 
}
分类: 源码分享, 高级民工 标签:

JAVA序列化中的一些问题

2014年6月3日 没有评论

1.serialVersionUID的作用

  serialVersionUID可以认为是一个类的摘要,默认值的计算依赖于类的构造方法、成员变量和成员方法的定义规则,具体算法代码位于:java.io.ObjectStreamClass.computeDefaultSUID(Class<?>) 。对于implements Serializable的类而言可以显示或非显示的定义serialVersionUID,在通常情况下建议显示定义serialVersionUID,这样就算改变成员方法定义也不会影响到反序列化操作。

2.成员变量的增加或减少会影响反序列化吗?

  这个问题的前提是显示定义了serialVersionUID,如果非显示定义serialVersionUID则对 构造方法、成员变量和成员方法的任意修改都会导致反序列化失败,因为反序列化时会计算当前Class的serialVersionUID,如果序列化前后不一致则会导致反序列化失败。

  在显示定义了serialVersionUID的情况下如果保证serialVersionUID的值序列化前后不变,则对成员变量的增加或减少都不会导致反序列化失败,但如果是对原有成员变量的定义进行了修改,如:public int a;变成了public Srting a; 则会导致反序列化失败,具体代码位于:java.io.ObjectStreamClass.matchFields(ObjectStreamField[], ObjectStreamClass) 方法,这个方法会判断序列化对象和jvm加载对象的属性类型是否一样。

分类: 高级民工 标签:

Apache HttpClient无法解析错误头信息

2013年8月19日 没有评论

  在使用HttpClient 4.2时遇到服务器返回头信息错误的情况会导致一个异常发生。

Caused by: org.apache.http.ProtocolException: Invalid header: content-Type=text/html;charset=gbk
	at org.apache.http.impl.io.AbstractMessageParser.parseHeaders(AbstractMessageParser.java:226)
	at org.apache.http.impl.io.AbstractMessageParser.parse(AbstractMessageParser.java:261)
	at org.apache.http.impl.AbstractHttpClientConnection.receiveResponseHeader(AbstractHttpClientConnection.java:289)
	at org.apache.http.impl.conn.DefaultClientConnection.receiveResponseHeader(DefaultClientConnection.java:252)
	at org.apache.http.impl.conn.ManagedClientConnectionImpl.receiveResponseHeader(ManagedClientConnectionImpl.java:191)
	at org.apache.http.protocol.HttpRequestExecutor.doReceiveResponse(HttpRequestExecutor.java:300)
	at org.apache.http.protocol.HttpRequestExecutor.execute(HttpRequestExecutor.java:127)
	at org.apache.http.impl.client.DefaultRequestDirector.tryExecute(DefaultRequestDirector.java:712)
	at org.apache.http.impl.client.DefaultRequestDirector.execute(DefaultRequestDirector.java:517)
	at org.apache.http.impl.client.AbstractHttpClient.execute(AbstractHttpClient.java:906)

出现异常的原因是org.apache.http.message.BufferedHeader类无法解析,所以如果想要忽略header信息的要嘛修改HttpClient的源码,要嘛通过继承来重写方法实现。HttpClient的官方文档中已有说明 http://hc.apache.org/httpcomponents-client-ga/tutorial/html/advanced.html
  但上面的方法对于使用PoolingClientConnectionManager的情况是不适用的。在使用PoolingClientConnectionManager时由于其实现内部用到了连接池org.apache.http.impl.conn.HttpConnPool,在其中有个static类用于返回DefaultClientConnection,代码如下:

    static class InternalConnFactory implements ConnFactory<HttpRoute, OperatedClientConnection> {
 
        public OperatedClientConnection create(final HttpRoute route) throws IOException {
            return new DefaultClientConnection();
        }
 
    }

所以结合HttpClient文档中的方法则需要在这里对return new DefaultClientConnection();这句代码进行修改就能达到目的。当然为了不修改其源码一个实现方式是自己创建一个包路径为org.apache.http.impl.conn,在这个包下创建继承于PoolingClientConnectionManager的类来最终达到忽略异常header的目的。

分类: 高级民工 标签:

JAXB简介及教程

2013年6月3日 没有评论

JAXB(Java Architecture for XML Binding) 是一个业界的标准,是一项可以根据XML Schema产生Java类的技术。该过程中,JAXB也提供了将XML实例文档反向生成Java对象树的方法,并能将Java对象树的内容重新写到XML实例文档。从另一方面来讲,JAXB提供了快速而简便的方法将XML模式绑定到Java表示,从而使得Java开发者在Java应用程序中能方便地结合XML数据和处理函数。JAXB 2.0是JDK 1.6的组成部分。JAXB 2.2.3是JDK 1.7的组成部分。

jaxb为了应对各种情况的bean到java类的映射定义了一套注解机制,下面简单说明几个常用的注解及其作用:

@XmlAccessorType

控制字段或属性的序列化。XmlAccessType.FIELD表示JAXB将自动绑定Java类中的每个非静态的(static)、非瞬态的(由@XmlTransient标注)字段到XML。其他值还有XmlAccessType.PROPERTY和XmlAccessType.NONE。一般情况如果没有指定XmlAccessType注解可能会导致一个异常的产生:

Exception in thread "main" com.sun.xml.internal.bind.v2.runtime.IllegalAnnotationsException: 1 counts of IllegalAnnotationExceptions
Class has two properties of the same name "nodes"
	this problem is related to the following location:
		at public java.util.Map com.zznode.e2esqm.core.diagnosis.schema.DiagnosisSchema.getNodes()
		at com.zznode.e2esqm.core.diagnosis.schema.DiagnosisSchema
	this problem is related to the following location:
		at private java.util.Map com.zznode.e2esqm.core.diagnosis.schema.DiagnosisSchema.nodes
		at com.zznode.e2esqm.core.diagnosis.schema.DiagnosisSchema

此种情况只需在类上添加注解@XmlAccessorType(XmlAccessType.FIELD)即可解决

@XmlJavaTypeAdapter

使用定制的适配器(即扩展抽象类XmlAdapter并覆盖marshal()和unmarshal()方法),以序列化Java类为XML。需要注意的是XmlAdapter有泛型定义,其第一个泛型参数为jaxb可以直接识别的bean即由marshal()方法return的类型,第二个泛型参数为unmarshal()方法返回的类型.
@XmlElementWrapper

对于数组或集合(即包含多个元素的成员变量),生成一个包装该数组或集合的XML元素(称为包装器)。例如:
一个javabean 包含一个List属性,正常情况生成的xml片段如下,

<rs id="1"/>
<rs id="2"/>
<rs id="3"/>

如果加上本XmlElementWrapper注解则会变为如下形式

<list>
   <rs id="1"/>
   <rs id="2"/>
   <rs id="3"/>
</list>

@XmlRootElement
将Java类或枚举类型映射到XML元素。

@XmlElement
将Java类的一个属性映射到与属性同名的一个XML元素。

@XmlAttribute
将Java类的一个属性映射到与属性同名的一个XML属性。

另外默认情况jaxb下从xml解析到javabean的时候是必须要注解XmlRootElement的存在,下面提供一种方式让bean不使用XmlRootElement注解,代码片段如下:

JAXBContext context = JAXBContext.newInstance(clazz);
Unmarshaller unmarshaller = context.createUnmarshaller();
 
DocumentBuilderFactory dbf = DocumentBuilderFactory.newInstance();
dbf.setNamespaceAware(true);
DocumentBuilder db = dbf.newDocumentBuilder();
ByteArrayInputStream in = new ByteArrayInputStream(xml.getBytes());//xml为对应的需要转换为bean的xml字符串
Document doc = db.parse(in);
Node fooSubtree = doc.getFirstChild();
 
JAXBElement<?> el = unmarshaller.unmarshal( fooSubtree, clazz);
el.getValue();//这个就是bean的实例

当让以上方法只能应对简单bean和xml的互转,如果有复杂的还是需要通过注解来处理才能达到预期效果.

分类: 高级民工 标签:

在动态加载的JAR中使用Spring

2012年12月28日 没有评论

  目的是在动态加载的JAR中使用Spring的功能,即动态把JAR中的Class加载到Spring中管理,目前设计为使用Annation来管理配置Spring,动态加入的代码如下:

//加载jar
JARClassLoader loader = new JARClassLoader();
loader.loadJar("c:/x3.jar");
 
//设置当前线程的classLoader
Thread.currentThread().setContextClassLoader(loader);
 
//注册bean
ApplicationContext context = SystemCore.getApplicationContext();
ClassPathBeanDefinitionScanner scanner = new ClassPathBeanDefinitionScanner((DefaultListableBeanFactory)
context.getAutowireCapableBeanFactory());//使用Spring的注解自动扫描
//扫描jar中的包路径,使用通配符,另外在导出jar时必须选择add directory entries(即把目录也加入到jar中)
//否则spring扫描时将无法找到class
scanner.scan("com.test.moduleA.*");
 
//手动拿jar包里的bean的实例时因为存在ClassLoader的隔离虽然在开始设置了ContextClassLoader
//但是Spring默认getBean的时候并没有每次都去拿最新的ContextClassLoader使用,所以需要手动设置Bean的ClassLoader
//因为是手动设置的这里存在线程安全的问题...不知道有没有其他方法.
DefaultListableBeanFactory factory = (DefaultListableBeanFactory)context.getAutowireCapableBeanFactory();
		factory.setBeanClassLoader(loader);
 
System.out.println(context.getBean("questionDaoImpl"));
分类: 高级民工 标签: ,