遇到的問題如題,是一個比較常見的需求吧。其實我要實現的功能是在自己寫的filter中注入一個mapper對象,然後在過filter時候校驗ak,防止多端登錄,在剛開始天真的認爲直接@Autowire就可以了,後來被NullPointException整到爆炸。
0x01 錯誤的嘗試
大概的邏輯就像下邊的filter中寫的一樣,後來才發現愛你是……走錯片場了,後來才發現是mapper空指針。
filter.java
public class Patient implements Filter{
@Autowired
private PatientMapper patientMapper;
@Override
public void init(FilterConfig filterConfig) throws ServletException {
}
@Override
public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain)
throws IOException, ServletException {
String ak = "";
HttpServletRequest httpServletRequest = (HttpServletRequest)request;
ak = httpServletrequest.getCookies()[1].getValues();
Users user = patientMapper.findUserByAk(ak);
chain.doFilter(request, response);
}
@Override
public void destroy() {
}
}
0x02 近乎成功的嘗試
網上有一部分人提出用org.springframework.web.filter.DelegatingFilterProxy
把filter加入到Spring容器的管理中去,我嘗試了,但是注入時候仍然是空指針。
web.xml中filter部分的代碼如下:
<!-- 患者filter-->
<filter>
<filter-name>PatientFilter</filter-name>
<filter-class>com.yy.filter.PatientFilter</filter-class>
<init-param>
<param-name>charset</param-name>
<param-value>UTF-8</param-value>
</init-param>
<init-param>
<param-name>contentType</param-name>
<param-value>text/html;charset=UTF-8</param-value>
</init-param>
</filter>
<filter-mapping>
<filter-name>PatientFilter</filter-name>
<!-- 要攔截的哪一級目錄 -->
<url-pattern>/patient/*</url-pattern>
</filter-mapping>
<!--將filter加入到Spring容器的管理-->
<filter>
<filter-name>patientFilter</filter-name>
<filter-class>org.springframework.web.filter.DelegatingFilterProxy</filter-class>
<init-param>
<param-name>targetFilterLifecycle</param-name>
<param-value>true</param-value>
</init-param>
</filter>
<filter-mapping>
<filter-name>patientFilter</filter-name>
<url-pattern>/patient/*</url-pattern>
</filter-mapping>
Spring的applicationContext.xml中和filter相關的內容如下:
<bean id="patientFilter" class="com.yy.filter.PatientFilter" />
0x02 Bug解決!
當我某一次檢查web.xml的時候,突然想到我有兩個filter,而請求經過filter的順序是根據web.xml中filter聲明的順序進行的。bingo!因爲web.xml中第一個聲明的是PatientFilter,第二個纔是patientFilter(暫且忽略我拙劣的起名方式),而且兩個filter的類還都是同一個,PatientFilter.java!!當然在請求剛來的時候經過PatientFilter.java,這個時候第二個用DelegatingFilterProxy
聲明的filter還沒有起作用,當然會報出空指針異常的錯誤,所以只要把對於同一個filter的配置合成一個或把兩個filter聲明的順序修改即可。我選擇的是合併,因爲我的兩個filter的聲明都是對於同一個filter,合併了比較合理
最終的web.xml中關於filter的配置:
<!-- 患者filter-->
<!--將filter加入到Spring容器的管理-->
<filter>
<filter-name>patientFilter</filter-name>
<filter-class>org.springframework.web.filter.DelegatingFilterProxy</filter-class>
<init-param>
<param-name>targetFilterLifecycle</param-name>
<param-value>true</param-value>
</init-param>
<init-param>
<param-name>charset</param-name>
<param-value>UTF-8</param-value>
</init-param>
<init-param>
<param-name>contentType</param-name>
<param-value>text/html;charset=UTF-8</param-value>
</init-param>
</filter>
<filter-mapping>
<filter-name>patientFilter</filter-name>
<url-pattern>/patient/*</url-pattern>
</filter-mapping>