上传文件大小导致失败

【问题场景】 在本地开发环境,上传超过2MB的数据都可以成功上传至服务器。但是,在测试环境出现上传超过200K就会失败的问题。 前端使用的是springMVC,中间件使用Nginx + Tomcat

【排查方法】 1、检查不同环境下的spring配置文件中,上传文件大小限制

<!-- 配置多文件上传 -->
<bean id="multipartResolver"  class="org.springframework.web.multipart.commons.CommonsMultipartResolver">
      <property name="defaultEncoding">
            <value>UTF-8</value>
      </property>
      <property name="maxUploadSize">
            <!-- 上传文件大小限制为31M,31*1024*1024 -->
            <value>32505856</value>
      </property>
      <property name="maxInMemorySize">
            <value>4096</value>
      </property>
</bean>

两个环节的代码都是同样的配置,没有问题,排查spring问题

2、检查两个环境的tomcat配置 conf/server.xml

<connector port="8080"    
   maxThreads="150" minSpareThreads="25" maxSpareThreads="75"    
   enableLookups="false" redirectPort="8443" acceptCount="100"    
   debug="0" connectionTimeout="20000"     
   disableUploadTimeout="true" URIEncoding="utf-8"    
   maxPostSize="0"></connector>    

当maxPostSize<=0时,POST方式上传的文件大小不会被限制。配置一致,排除tomcat配置问题, 注意:maxPostSize参数只有当request的Content-Type为“application/x-www-form-urlencoded”时起作用。 参考资料 https://blog.csdn.net/lafengwnagzi/article/details/72846195

3、检查两个环境的nginx配置

server  
{  
    listen   80;  
    server_name  chat.erp.360buy.com;  
    #access_log    /export/servers/nginx/logs/chat.erp.360buy.com;  
    location / {  
        proxy_pass     http://tomcat;  
        client_max_body_size    1000m;  
    }  
}  

最终发现,测试环境的nginx配置不一致导致,问题定位成功 参考资料 https://blog.csdn.net/bruce128/article/details/9665503

【总结】 需要问题不要着急,先自查,从自己的代码入手。如果代码确认没有问题,就外范围扩大,看一下外部环境是否存在问题,一般情况都可以准确定位。

spring getbean 方法分析

在最近的项目中,有个地方我们不得不实用getBean的方法,自己从Spring context中获取bean进行数据库操作。
方法一(效率低,极易出现bug,不推荐使用):
刚刚开始的时候,我们使用这中方式,但是在应用过程中发现此方式效率低下,而且极易出现bug。 在我们系统中会生成ehcache_auto_created_时间戳文件夹,

String[] xmlCfg = new String[] {"classpath:/spring/applicationContext-service.xml",
"classpath:/spring/applicationContext-util.xml",
"classpath:/spring/applicationContext.xml"};
ApplicationContext context = new FileSystemXmlApplicationContext(xmlCfg);
// 获取inspectionUtil bean
inspectionUtil = (InspectionUtil) context.getBean("inspectionUtil");

 

所以我google了一下,改用其他方法。
方法二(效率高,灵活性高,可复用,推荐使用): 创建一个工具类SpringContextsUtil ,通过实现Spring中的ApplicationContextAware接口,在applicationContext.xml中注入bean后Spring会自动调用setApplicationContext方法。此时我们就可以获取到Spring context。

public class SpringContextsUtil implements ApplicationContextAware {
    private static ApplicationContext applicationContext; //Spring应用上下文环境

    /**
    * 实现ApplicationContextAware接口的回调方法,设置上下文环境
    * @param applicationContext
    * @throws BeansException
    */
    public void setApplicationContext(ApplicationContext applicationContext)
        throws BeansException {
        SpringContextsUtil.applicationContext = applicationContext;
    }

    /**
    * @return ApplicationContext
    */
    public static ApplicationContext getApplicationContext() {
        return applicationContext;
    }

    /**
    * 获取对象
    * @param name
    * @return Object 一个以所给名字注册的bean的实例
    * @throws BeansException
    */
    public static Object getBean(String name) throws BeansException {
        return applicationContext.getBean(name);
    }

    /**
    * 获取类型为requiredType的对象
    * 如果bean不能被类型转换,相应的异常将会被抛出(BeanNotOfRequiredTypeException)
    * @param name bean注册名
    * @param requiredType 返回对象类型
    * @return Object 返回requiredType类型对象
    * @throws BeansException
    */
    public static Object getBean(String name, Class requiredType)
        throws BeansException {
        return applicationContext.getBean(name, requiredType);
    }

    /**
    * 如果BeanFactory包含一个与所给名称匹配的bean定义,则返回true
    * @param name
    * @return boolean
    */
    public static boolean containsBean(String name) {
        return applicationContext.containsBean(name);
    }

    /**
    * 判断以给定名字注册的bean定义是一个singleton还是一个prototype。
    * 如果与给定名字相应的bean定义没有被找到,将会抛出一个异常(NoSuchBeanDefinitionException)
    * @param name
    * @return boolean
    * @throws NoSuchBeanDefinitionException
    */
    public static boolean isSingleton(String name)
        throws NoSuchBeanDefinitionException {
        return applicationContext.isSingleton(name);
    }

    /**
    * @param name
    * @return Class 注册对象的类型
    * @throws NoSuchBeanDefinitionException
    */
    public static Class getType(String name)
        throws NoSuchBeanDefinitionException {
        return applicationContext.getType(name);
    }

    /**
    * 如果给定的bean名字在bean定义中有别名,则返回这些别名
    * @param name
    * @return
    * @throws NoSuchBeanDefinitionException
    */
    public static String[] getAliases(String name)
        throws NoSuchBeanDefinitionException {
        return applicationContext.getAliases(name);
    }
}

调用方法:

// 获取inspectionUtil bean
inspectionUtil = (InspectionUtil) SpringContextUtil.getBean("inspectionUtil");

注:
1、使用时会出现无法获取applicationContext,并抛出NullPointerException。

原因:使用此方法必须在spring applicationContext.xml中注入bean。否则spring无法自动调用setApplicationContext。如下

<!-- lang: xml -->
<bean id="springContextsUtil" class="com.sinosoft.sepmis.util.SpringContextsUtil" ></bean>

2、如果注入后仍然出现这个问题。 则修改中的default-lazy-init="false"。 或者是修改bean注入属性

<!-- lang: xml -->
<bean id="springContextsUtil" class="com.sinosoft.sepmis.util.SpringContextsUtil" lazy-init="false"></bean>