国产成人精品亚洲777人妖,欧美日韩精品一区视频,最新亚洲国产,国产乱码精品一区二区亚洲

您的位置:首頁(yè)技術(shù)文章
文章詳情頁(yè)

Spring中異步注解@Async的使用、原理及使用時(shí)可能導(dǎo)致的問題及解決方法

瀏覽:3日期:2023-08-25 11:07:23

前言

其實(shí)最近都在研究事務(wù)相關(guān)的內(nèi)容,之所以寫這么一篇文章是因?yàn)榍懊鎸懥艘黄P(guān)于循環(huán)依賴的文章:

《Spring循環(huán)依賴的解決辦法,你真的懂了嗎》

然后,很多同學(xué)碰到了下面這個(gè)問題,添加了Spring提供的一個(gè)異步注解@Async循環(huán)依賴無法被解決了,下面是一些讀者的留言跟群里同學(xué)碰到的問題:

Spring中異步注解@Async的使用、原理及使用時(shí)可能導(dǎo)致的問題及解決方法

Spring中異步注解@Async的使用、原理及使用時(shí)可能導(dǎo)致的問題及解決方法

本著講一個(gè)知識(shí)點(diǎn)就要講明白、講透徹的原則,我決定單獨(dú)寫一篇這樣的文章對(duì)@Async這個(gè)注解做一下詳細(xì)的介紹,這個(gè)注解帶來的問題遠(yuǎn)遠(yuǎn)不止循環(huán)依賴這么簡(jiǎn)單,如果對(duì)它不夠熟悉的話建議慎用。

文章要點(diǎn)

Spring中異步注解@Async的使用、原理及使用時(shí)可能導(dǎo)致的問題及解決方法

@Async的基本使用

這個(gè)注解的作用在于可以讓被標(biāo)注的方法異步執(zhí)行,但是有兩個(gè)前提條件

配置類上添加@EnableAsync注解需要異步執(zhí)行的方法的所在類由Spring管理需要異步執(zhí)行的方法上添加了@Async注解

我們通過一個(gè)Demo體會(huì)下這個(gè)注解的作用吧

第一步,配置類上開啟異步:

@EnableAsync@Configuration@ComponentScan('com.dmz.spring.async')public class Config {}

第二步,

[code]@Component // 這個(gè)類本身要被Spring管理public class DmzAsyncService { @Async // 添加注解表示這

@Component // 這個(gè)類本身要被Spring管理public class DmzAsyncService { @Async // 添加注解表示這個(gè)方法要異步執(zhí)行public void testAsync(){try {TimeUnit.SECONDS.sleep(1);} catch (InterruptedException e) {e.printStackTrace();}System.out.println('testAsync invoked');}}

第三步,測(cè)試異步執(zhí)行

public class Main {public static void main(String[] args) {AnnotationConfigApplicationContext ac = new AnnotationConfigApplicationContext(Config.class);DmzAsyncService bean = ac.getBean(DmzAsyncService.class);bean.testAsync();System.out.println('main函數(shù)執(zhí)行完成');}}// 程序執(zhí)行結(jié)果如下:// main函數(shù)執(zhí)行完成// testAsync invoked

通過上面的例子我們可以發(fā)現(xiàn),DmzAsyncService中的testAsync方法是異步執(zhí)行的,那么這背后的原理是什么呢?我們接著分析

原理分析

我們?cè)诜治瞿骋粋€(gè)技術(shù)的時(shí)候,最重要的事情是,一定一定要找到代碼的入口,像Spring這種都很明顯,入口必定是在@EnableAsync這個(gè)注解上面,我們來看看這個(gè)注解干了啥事(本文基于5.2.x版本)

@Target(ElementType.TYPE)@Retention(RetentionPolicy.RUNTIME)@Documented// 這里是重點(diǎn),導(dǎo)入了一個(gè)ImportSelector@Import(AsyncConfigurationSelector.class)public @interface EnableAsync { // 這個(gè)配置可以讓程序員配置需要被檢查的注解,默認(rèn)情況下檢查的就是@Async注解Class<? extends Annotation> annotation() default Annotation.class; // 默認(rèn)使用jdk代理boolean proxyTargetClass() default false; // 默認(rèn)使用Spring AOPAdviceMode mode() default AdviceMode.PROXY; // 在后續(xù)分析我們會(huì)發(fā)現(xiàn),這個(gè)注解實(shí)際往容器中添加了一個(gè) // AsyncAnnotationBeanPostProcessor,這個(gè)后置處理器實(shí)現(xiàn)了Ordered接口 // 這個(gè)配置主要代表了AsyncAnnotationBeanPostProcessor執(zhí)行的順序int order() default Ordered.LOWEST_PRECEDENCE;}

上面這個(gè)注解做的最重要的事情就是導(dǎo)入了一個(gè)AsyncConfigurationSelector,這個(gè)類的源碼如下:

public class AsyncConfigurationSelector extends AdviceModeImportSelector<EnableAsync> {private static final String ASYNC_EXECUTION_ASPECT_CONFIGURATION_CLASS_NAME ='org.springframework.scheduling.aspectj.AspectJAsyncConfiguration';@Override@Nullablepublic String[] selectImports(AdviceMode adviceMode) {switch (adviceMode) { // 默認(rèn)會(huì)使用SpringAOP進(jìn)行代理case PROXY:return new String[] {ProxyAsyncConfiguration.class.getName()};case ASPECTJ:return new String[] {ASYNC_EXECUTION_ASPECT_CONFIGURATION_CLASS_NAME};default:return null;}}}

這個(gè)類的作用是像容器中注冊(cè)了一個(gè)ProxyAsyncConfiguration,這個(gè)類的繼承關(guān)系如下:

Spring中異步注解@Async的使用、原理及使用時(shí)可能導(dǎo)致的問題及解決方法

我們先看下它的父類AbstractAsyncConfiguration,其源碼如下:

@Configurationpublic abstract class AbstractAsyncConfiguration implements ImportAware {@Nullableprotected AnnotationAttributes enableAsync;@Nullableprotected Supplier<Executor> executor;@Nullableprotected Supplier<AsyncUncaughtExceptionHandler> exceptionHandler; // 這里主要就是檢查將其導(dǎo)入的類上是否有EnableAsync注解 // 如果沒有的話就報(bào)錯(cuò)@Overridepublic void setImportMetadata(AnnotationMetadata importMetadata) {this.enableAsync = AnnotationAttributes.fromMap(importMetadata.getAnnotationAttributes(EnableAsync.class.getName(), false));if (this.enableAsync == null) {throw new IllegalArgumentException('@EnableAsync is not present on importing class ' + importMetadata.getClassName());}} // 將容器中配置的AsyncConfigurer注入 // 異步執(zhí)行嘛,所以我們可以配置使用的線程池 // 另外也可以配置異常處理器@Autowired(required = false)void setConfigurers(Collection<AsyncConfigurer> configurers) {if (CollectionUtils.isEmpty(configurers)) {return;}if (configurers.size() > 1) {throw new IllegalStateException('Only one AsyncConfigurer may exist');}AsyncConfigurer configurer = configurers.iterator().next();this.executor = configurer::getAsyncExecutor;this.exceptionHandler = configurer::getAsyncUncaughtExceptionHandler;}}

再來看看ProxyAsyncConfiguration這個(gè)類的源碼

@Configuration@Role(BeanDefinition.ROLE_INFRASTRUCTURE)public class ProxyAsyncConfiguration extends AbstractAsyncConfiguration {@Bean(name = TaskManagementConfigUtils.ASYNC_ANNOTATION_PROCESSOR_BEAN_NAME)@Role(BeanDefinition.ROLE_INFRASTRUCTURE)public AsyncAnnotationBeanPostProcessor asyncAdvisor() {AsyncAnnotationBeanPostProcessor bpp = new AsyncAnnotationBeanPostProcessor(); // 將通過AsyncConfigurer配置好的線程池跟異常處理器設(shè)置到這個(gè)后置處理器中 bpp.configure(this.executor, this.exceptionHandler);Class<? extends Annotation> customAsyncAnnotation = this.enableAsync.getClass('annotation');if (customAsyncAnnotation != AnnotationUtils.getDefaultValue(EnableAsync.class, 'annotation')) {bpp.setAsyncAnnotationType(customAsyncAnnotation);}bpp.setProxyTargetClass(this.enableAsync.getBoolean('proxyTargetClass'));bpp.setOrder(this.enableAsync.<Integer>getNumber('order'));return bpp;}}

這個(gè)類本身是一個(gè)配置類,它的作用是向容器中添加一個(gè)AsyncAnnotationBeanPostProcessor。到這一步我們基本上就可以明白了,@Async注解的就是通過AsyncAnnotationBeanPostProcessor這個(gè)后置處理器生成一個(gè)代理對(duì)象來實(shí)現(xiàn)異步的,接下來我們就具體看看AsyncAnnotationBeanPostProcessor是如何生成代理對(duì)象的,我們主要關(guān)注一下幾點(diǎn)即可:

是在生命周期的哪一步完成的代理? 切點(diǎn)的邏輯是怎么樣的?它會(huì)對(duì)什么樣的類進(jìn)行攔截? 通知的邏輯是怎么樣的?是如何實(shí)現(xiàn)異步的?

基于上面幾個(gè)問題,我們進(jìn)行逐一分析

是在生命周期的哪一步完成的代理?

我們抓住重點(diǎn),AsyncAnnotationBeanPostProcessor是一個(gè)后置處理器器,按照我們對(duì)Spring的了解,大概率是在這個(gè)后置處理器的postProcessAfterInitialization方法中完成了代理,直接定位到這個(gè)方法,這個(gè)方法位于父類AbstractAdvisingBeanPostProcessor中,具體代碼如下:

public Object postProcessAfterInitialization(Object bean, String beanName) { // 沒有通知,或者是AOP的基礎(chǔ)設(shè)施類,那么不進(jìn)行代理 if (this.advisor == null || bean instanceof AopInfrastructureBean) { return bean; } // 對(duì)已經(jīng)被代理的類,不再生成代理,只是將通知添加到代理類的邏輯中 // 這里通過beforeExistingAdvisors決定是將通知添加到所有通知之前還是添加到所有通知之后 // 在使用@Async注解的時(shí)候,beforeExistingAdvisors被設(shè)置成了true // 意味著整個(gè)方法及其攔截邏輯都會(huì)異步執(zhí)行 if (bean instanceof Advised) { Advised advised = (Advised) bean; if (!advised.isFrozen() && isEligible(AopUtils.getTargetClass(bean))) { if (this.beforeExistingAdvisors) { advised.addAdvisor(0, this.advisor); } else { advised.addAdvisor(this.advisor); } return bean; } } // 判斷需要對(duì)哪些Bean進(jìn)行來代理 if (isEligible(bean, beanName)) { ProxyFactory proxyFactory = prepareProxyFactory(bean, beanName); if (!proxyFactory.isProxyTargetClass()) { evaluateProxyInterfaces(bean.getClass(), proxyFactory); } proxyFactory.addAdvisor(this.advisor); customizeProxyFactory(proxyFactory); return proxyFactory.getProxy(getProxyClassLoader()); } return bean;}

果不其然,確實(shí)是在這個(gè)方法中完成的代理。接著我們就要思考,切點(diǎn)的過濾規(guī)則是什么呢?

切點(diǎn)的邏輯是怎么樣的?

其實(shí)也不難猜到肯定就是類上添加了@Async注解或者類中含有被@Async注解修飾的方法。基于此,我們看看這個(gè)isEligible這個(gè)方法的實(shí)現(xiàn)邏輯,這個(gè)方位位于AbstractBeanFactoryAwareAdvisingPostProcessor中,也是AsyncAnnotationBeanPostProcessor的父類,對(duì)應(yīng)代碼如下:

// AbstractBeanFactoryAwareAdvisingPostProcessor的isEligible方法// 調(diào)用了父類protected boolean isEligible(Object bean, String beanName) { return (!AutoProxyUtils.isOriginalInstance(beanName, bean.getClass()) && super.isEligible(bean, beanName));}protected boolean isEligible(Object bean, String beanName) { return isEligible(bean.getClass());}protected boolean isEligible(Class<?> targetClass) { Boolean eligible = this.eligibleBeans.get(targetClass); if (eligible != null) { return eligible; } if (this.advisor == null) { return false; } // 這里完成的判斷 eligible = AopUtils.canApply(this.advisor, targetClass); this.eligibleBeans.put(targetClass, eligible); return eligible;}

實(shí)際上最后就是根據(jù)advisor來確定是否要進(jìn)行代理,在Spring中基于xml的AOP的詳細(xì)步驟這篇文章中我們提到過,advisor實(shí)際就是一個(gè)綁定了切點(diǎn)的通知,那么AsyncAnnotationBeanPostProcessor這個(gè)advisor是什么時(shí)候被初始化的呢?我們直接定位到AsyncAnnotationBeanPostProcessor的setBeanFactory方法,其源碼如下:

public void setBeanFactory(BeanFactory beanFactory) { super.setBeanFactory(beanFactory); // 在這里new了一個(gè)AsyncAnnotationAdvisor AsyncAnnotationAdvisor advisor = new AsyncAnnotationAdvisor(this.executor, this.exceptionHandler); if (this.asyncAnnotationType != null) { advisor.setAsyncAnnotationType(this.asyncAnnotationType); } advisor.setBeanFactory(beanFactory); // 完成了初始化 this.advisor = advisor;}

我們來看看AsyncAnnotationAdvisor中的切點(diǎn)匹配規(guī)程是怎么樣的,直接定位到這個(gè)類的buildPointcut方法中,其源碼如下:

protected Pointcut buildPointcut(Set<Class<? extends Annotation>> asyncAnnotationTypes) { ComposablePointcut result = null; for (Class<? extends Annotation> asyncAnnotationType : asyncAnnotationTypes) { // 就是根據(jù)這兩個(gè)匹配器進(jìn)行匹配的 Pointcut cpc = new AnnotationMatchingPointcut(asyncAnnotationType, true); Pointcut mpc = new AnnotationMatchingPointcut(null, asyncAnnotationType, true); if (result == null) { result = new ComposablePointcut(cpc); } else { result.union(cpc); } result = result.union(mpc); } return (result != null ? result : Pointcut.TRUE);}

代碼很簡(jiǎn)單,就是根據(jù)cpc跟mpc兩個(gè)匹配器來進(jìn)行匹配的,第一個(gè)是檢查類上是否有@Async注解,第二個(gè)是檢查方法是是否有@Async注解。

那么,到現(xiàn)在為止,我們已經(jīng)知道了它在何時(shí)創(chuàng)建代理,會(huì)為什么對(duì)象創(chuàng)建代理,最后我們還需要解決一個(gè)問題,代理的邏輯是怎么樣的,異步到底是如何實(shí)現(xiàn)的?

通知的邏輯是怎么樣的?是如何實(shí)現(xiàn)異步的?

前面也提到了advisor是一個(gè)綁定了切點(diǎn)的通知,前面分析了它的切點(diǎn),那么現(xiàn)在我們就來看看它的通知邏輯,直接定位到AsyncAnnotationAdvisor中的buildAdvice方法,源碼如下:

protected Advice buildAdvice( @Nullable Supplier<Executor> executor, @Nullable Supplier<AsyncUncaughtExceptionHandler> exceptionHandler) { AnnotationAsyncExecutionInterceptor interceptor = new AnnotationAsyncExecutionInterceptor(null); interceptor.configure(executor, exceptionHandler); return interceptor;}

簡(jiǎn)單吧,加了一個(gè)攔截器而已,對(duì)于interceptor類型的對(duì)象,我們關(guān)注它的核心方法invoke就行了,代碼如下:

public Object invoke(final MethodInvocation invocation) throws Throwable { Class<?> targetClass = (invocation.getThis() != null ? AopUtils.getTargetClass(invocation.getThis()) : null); Method specificMethod = ClassUtils.getMostSpecificMethod(invocation.getMethod(), targetClass); final Method userDeclaredMethod = BridgeMethodResolver.findBridgedMethod(specificMethod); // 異步執(zhí)行嘛,先獲取到一個(gè)線程池 AsyncTaskExecutor executor = determineAsyncExecutor(userDeclaredMethod); if (executor == null) { throw new IllegalStateException( 'No executor specified and no default executor set on AsyncExecutionInterceptor either'); } // 然后將這個(gè)方法封裝成一個(gè) Callable對(duì)象傳入到線程池中執(zhí)行 Callable<Object> task = () -> { try { Object result = invocation.proceed(); if (result instanceof Future) { return ((Future<?>) result).get(); } } catch (ExecutionException ex) { handleError(ex.getCause(), userDeclaredMethod, invocation.getArguments()); } catch (Throwable ex) { handleError(ex, userDeclaredMethod, invocation.getArguments()); } return null; };// 將任務(wù)提交到線程池 return doSubmit(task, executor, invocation.getMethod().getReturnType());}

導(dǎo)致的問題及解決方案

問題1:循環(huán)依賴報(bào)錯(cuò)

就像在這張圖里這個(gè)讀者問的問題,

Spring中異步注解@Async的使用、原理及使用時(shí)可能導(dǎo)致的問題及解決方法

分為兩點(diǎn)回答:

第一:循環(huán)依賴為什么不能被解決?

這個(gè)問題其實(shí)很簡(jiǎn)單,在《講一講Spring中的循環(huán)依賴》這篇文章中我從兩個(gè)方面分析了循環(huán)依賴的處理流程

簡(jiǎn)單對(duì)象間的循環(huán)依賴處理AOP對(duì)象間的循環(huán)依賴處理

按照這種思路,@Async注解導(dǎo)致的循環(huán)依賴應(yīng)該屬于AOP對(duì)象間的循環(huán)依賴,也應(yīng)該能被處理。但是,重點(diǎn)來了,解決AOP對(duì)象間循環(huán)依賴的核心方法是三級(jí)緩存,如下:

Spring中異步注解@Async的使用、原理及使用時(shí)可能導(dǎo)致的問題及解決方法

在三級(jí)緩存緩存了一個(gè)工廠對(duì)象,這個(gè)工廠對(duì)象會(huì)調(diào)用getEarlyBeanReference方法來獲取一個(gè)早期的代理對(duì)象的引用,其源碼如下:

protected Object getEarlyBeanReference(String beanName, RootBeanDefinition mbd, Object bean) { Object exposedObject = bean; if (!mbd.isSynthetic() && hasInstantiationAwareBeanPostProcessors()) { for (BeanPostProcessor bp : getBeanPostProcessors()) { // 看到這個(gè)判斷了嗎,通過@EnableAsync導(dǎo)入的后置處理器 // AsyncAnnotationBeanPostProcessor根本就不是一個(gè)SmartInstantiationAwareBeanPostProcessor // 這就意味著即使我們通過AsyncAnnotationBeanPostProcessor創(chuàng)建了一個(gè)代理對(duì)象 // 但是早期暴露出去的用于給別的Bean進(jìn)行注入的那個(gè)對(duì)象還是原始對(duì)象 if (bp instanceof SmartInstantiationAwareBeanPostProcessor) { SmartInstantiationAwareBeanPostProcessor ibp = (SmartInstantiationAwareBeanPostProcessor) bp; exposedObject = ibp.getEarlyBeanReference(exposedObject, beanName); } } } return exposedObject;}

看完上面的代碼循環(huán)依賴的問題就很明顯了,因?yàn)樵缙诒┞兜膶?duì)象跟最終放入容器中的對(duì)象不是同一個(gè),所以報(bào)錯(cuò)了。報(bào)錯(cuò)的具體位置我在談?wù)勎覍?duì)Spring Bean 生命周期的理解 文章末尾已經(jīng)分析過了,本文不再贅述

Spring中異步注解@Async的使用、原理及使用時(shí)可能導(dǎo)致的問題及解決方法

解決方案

就以上面讀者給出的Demo為例,只需要在為B注入A時(shí)添加一個(gè)@Lazy注解即可

@Componentpublic class B implements BService { @Autowired@Lazyprivate A a;public void doSomething() {}}

這個(gè)注解的作用在于,當(dāng)為B注入A時(shí),會(huì)為A生成一個(gè)代理對(duì)象注入到B中,當(dāng)真正調(diào)用代理對(duì)象的方法時(shí),底層會(huì)調(diào)用getBean(a)去創(chuàng)建A對(duì)象,然后調(diào)用方法,這個(gè)注解的處理時(shí)機(jī)是在org.springframework.beans.factory.support.DefaultListableBeanFactory#resolveDependency方法中,處理這個(gè)注解的代碼位于org.springframework.context.annotation.ContextAnnotationAutowireCandidateResolver#buildLazyResolutionProxy,這些代碼其實(shí)都在我之前的文章中分析過了

《Spring雜談 | Spring中的AutowireCandidateResolver》

《談?wù)凷pring中的對(duì)象跟Bean,你知道Spring怎么創(chuàng)建對(duì)象的嗎?》

所以本文不再做詳細(xì)分析

問題2:默認(rèn)線程池不會(huì)復(fù)用線程

我覺得這是這個(gè)注解最坑的地方,沒有之一!我們來看看它默認(rèn)使用的線程池是哪個(gè),在前文的源碼分析中,我們可以看到?jīng)Q定要使用線程池的方法是org.springframework.aop.interceptor.AsyncExecutionAspectSupport#determineAsyncExecutor。其源碼如下:

protected AsyncTaskExecutor determineAsyncExecutor(Method method) { AsyncTaskExecutor executor = this.executors.get(method); if (executor == null) { Executor targetExecutor; // 可以在@Async注解中配置線程池的名字 String qualifier = getExecutorQualifier(method); if (StringUtils.hasLength(qualifier)) { targetExecutor = findQualifiedExecutor(this.beanFactory, qualifier); } else { // 獲取默認(rèn)的線程池 targetExecutor = this.defaultExecutor.get(); } if (targetExecutor == null) { return null; } executor = (targetExecutor instanceof AsyncListenableTaskExecutor ? (AsyncListenableTaskExecutor) targetExecutor : new TaskExecutorAdapter(targetExecutor)); this.executors.put(method, executor); } return executor;}

最終會(huì)調(diào)用到org.springframework.aop.interceptor.AsyncExecutionInterceptor#getDefaultExecutor這個(gè)方法中

protected Executor getDefaultExecutor(@Nullable BeanFactory beanFactory) { Executor defaultExecutor = super.getDefaultExecutor(beanFactory); return (defaultExecutor != null ? defaultExecutor : new SimpleAsyncTaskExecutor());}

可以看到,它默認(rèn)使用的線程池是SimpleAsyncTaskExecutor。我們不看這個(gè)類的源碼,只看它上面的文檔注釋,如下:

Spring中異步注解@Async的使用、原理及使用時(shí)可能導(dǎo)致的問題及解決方法

主要說了三點(diǎn)

為每個(gè)任務(wù)新起一個(gè)線程 默認(rèn)線程數(shù)不做限制 不復(fù)用線程

就這三點(diǎn),你還敢用嗎?只要你的任務(wù)耗時(shí)長(zhǎng)一點(diǎn),說不定服務(wù)器就給你來個(gè)OOM。

解決方案

最好的辦法就是使用自定義的線程池,主要有這么幾種配置方法

在之前的源碼分析中,我們可以知道,可以通過AsyncConfigurer來配置使用的線程池

如下:

public class DmzAsyncConfigurer implements AsyncConfigurer { @Override public Executor getAsyncExecutor() { // 創(chuàng)建自定義的線程池 }}

直接在@Async注解中配置要使用的線程池的名稱

如下:

public class A implements AService {private B b;@Autowiredpublic void setB(B b) {System.out.println(b);this.b = b;}@Async('dmzExecutor')public void doSomething() {}}

@EnableAsync@Configuration@ComponentScan('com.dmz.spring.async')@Aspectpublic class Config {@Bean('dmzExecutor')public Executor executor(){// 創(chuàng)建自定義的線程池return executor;}}

總結(jié)

本文主要介紹了Spring中異步注解的使用、原理及可能碰到的問題,針對(duì)每個(gè)問題文中也給出了方案。希望通過這篇文章能幫助你徹底掌握@Async注解的使用,知其然并知其所以然!

到此這篇關(guān)于Spring中異步注解@Async的使用、原理及使用時(shí)可能導(dǎo)致的問題及解決方法的文章就介紹到這了,更多相關(guān)Spring 異步注解@Async使用原理內(nèi)容請(qǐng)搜索好吧啦網(wǎng)以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持好吧啦網(wǎng)!

標(biāo)簽: Spring
相關(guān)文章:
主站蜘蛛池模板: 临泉县| 丹江口市| 丰原市| 鄂伦春自治旗| 黄龙县| 扎囊县| 宁津县| 德化县| 浙江省| 错那县| 莱阳市| 自贡市| 右玉县| 连云港市| 贵德县| 伽师县| 铜陵市| 广州市| 英超| 石嘴山市| 枣强县| 巍山| 绥德县| 台江县| 漳平市| 马山县| 河东区| 玛纳斯县| 台山市| 平遥县| 永新县| 定襄县| 卢湾区| 疏勒县| 玉龙| 阿图什市| 大连市| 黔西县| 锦州市| 囊谦县| 黎平县|