spring的bean创建流程源码解析

devtools/2024/10/24 10:17:13/

文章目录

    • IOC 和 DI
    • BeanFactory
    • ApplicationContext
      • 实现的接口
        • 1、BeanFactory接口
        • 2、MessageSource 国际化接口
        • 3、ResourcePatternResolver,资源解析接口
        • 4、EnvironmentCapable接口,用于获取环境变量,配置信息
        • 5、ApplicationEventPublisher 事件发布接口
      • Application 的继承体系
    • Bean 的创建执行流程
    • Spring 给开发者留下的扩展点
    • 结合代码描述
    • Bean 的创建流程总结:
    • 面试中说的三级缓存是什么?

本文基于 Spring 5.0.2.RELEASE

spring 是 Java Web 开发中,非常常用的框架。它的基础作用是为我们生成对象。这里涉及到 2 个概念。IOC 和 DI。

IOC 和 DI

IOC:Inversion of Control,控制反转。强调的是原来在程序中创建Bean的权利反转给第三方,这里的第三方,其实就是 spring 框架。

DI:依赖注入。比如 UserService类,依赖 UserDao,但是 UserDao 是由 BeanFactory 注入给 UserService 的。

BeanFactory

1、BeanFactory是Spring的早期接口,称为Spring的Bean工厂.它提供了一系列的方法,用于获取 Bean
在这里插入图片描述
BeanFactory 的默认实现类是 DefaultListableBeanFactory,DefaultListableBeanFactory主要是管理 BeanDefinition 对象,并根据BeanDefinition 对象创建 Bean 对象。
在这里插入图片描述
在DefaultListableBeanFactory中,实现了BeanDefinitionRegistry接口,这个接口定义了关于 BeanDefinition 的注册、移除、查询等一系列的操作。

DefaultListableBeanFactory还继承了DefaultSingletonBeanRegistry,这个类中,实现了 三级缓存,创建单例 Bean 的操作。????BeanDefinition对象,怎么传给DefaultSingletonBeanRegistry对象的????

DefaultListableBeanFactory总结起来,就是可以增删改查BeanDefinition定义对象,并通过父类方法,生成 Bean 对象。但是它并没有把 xml 文件,或者注解,解析成BeanDefinition对象的功能。这个功能是在ApplicationContext实现的。

ApplicationContext

实现的接口

1、BeanFactory接口

ApplicationContext 继承了很多接口,每增加一个接口,都增加了对应的功能。
在这里插入图片描述
但是,ApplicationContext 中,并不是继承了 DefaultListableBeanFactory,而是实现了 BeanFactory接口,并引入了 DefaultListableBeanFactory 作为属性,由 DefaultListableBeanFactory 实现 BeanFactory 相关的功能。(组合优于继承)

在这里插入图片描述

注意看,AnnotationConfigApplicationContext中包含 beanFactory 属性,在它的父类 AbstractApplicationContext中,实现了 getBean 等 方法,都是调用 beanFactory 来实现的。

@Override
public Object getBean(String name) throws BeansException {assertBeanFactoryActive();return getBeanFactory().getBean(name);
}

ApplicationContext 相比 BeanFactory,还多继承了四个接口,功能如下:

2、MessageSource 国际化接口
public static void main(String[] args) {ConfigurableApplicationContext context = SpringApplication.run(Application.class, args);System.out.println(context.getMessage("hi", null, Locale.CHINA));System.out.println(context.getMessage("hi", null, Locale.ENGLISH));
}

在 resources 目录下创建三个配置文件,注意设置 idea 的 properties 编码格式是 utf-8
messages.properties 空内容
messages_en.properties 填写内容 hi=hello
messages_zh.properties 填写内容 hi=你好

运行代码,控制台打印如下,说明国际化生效

你好
hello
3、ResourcePatternResolver,资源解析接口

可以用于读取配置文件

public static void main(String[] args) throws IOException {ConfigurableApplicationContext context = SpringApplication.run(Application.class, args);//读取类类路径下文件Resource[] resources1 = context.getResources("classpath:application.properties");//读取jar 包中的类路径文件,注意带个 * 号Resource[] resources2 = context.getResources("classpath*:META-INF/ spring.factories");//读取文件目录下的文件Resource[] resources3 = context.getResources("file:/tmp/test.log");
}
4、EnvironmentCapable接口,用于获取环境变量,配置信息
  public static void main(String[] args) throws IOException {ConfigurableApplicationContext context = SpringApplication.run(Application.class, args);//读取application.properties 中的内容System.out.println(context.getEnvironment().getProperty("server.port"));//读取系统环境变量System.out.println(context.getEnvironment().getProperty("JAVA_HOME"));}
5、ApplicationEventPublisher 事件发布接口
// 自定义事件
public class UserLoginEvent extends ApplicationEvent {public UserLoginEvent(Object source) {super(source);}
}
//发布事件public static void main(String[] args) throws IOException {ConfigurableApplicationContext context = SpringApplication.run(Application.class, args);context.publishEvent(new UserLoginEvent("username"));}
//接收事件
@Component
public class UserMessageService {@EventListenerpublic void sendMessage(UserLoginEvent event){Object source = event.getSource();System.out.println("欢迎用户:"+source);}
}

当调用publishEvent发布事件时,会调用到 sendMessage 方法。(观察者模式)

总结:
1、BeanFactory 类,用于生成 Bean。
2、ApplicationContext,继承了 BeanFactory 接口,引入了 DefaultListableBeanFactory对象,体现了组合优于继承的原则。
3、ApplicationContext,实现了其他接口,增加了国际化,事件发布,资源解析,读取配置等功能。

Application 的继承体系

在这里插入图片描述

常用的 ApplicationContext 有 3 个,分别是
AnnotationConfigApplicationContext,用于加载注解配置类的ApplicationContext
FileSystemXmlApplicationContext,用于加载磁盘路径下的xml配置的ApplicationContext
ClassPathXmlApplicationContext,用于加载类路径下的xml配置的ApplicationContext
如果 spring 中加入了 Web 相关的组件,那么使用的 ApplicationContext 是
XmlWebApplicationContext,用于web环境下,加载类路径下的xml配置的ApplicationContext
AnnotationConfigWebApplicationContext,用于web环境下,加载磁盘路径下的xml配置的ApplicationContext

Bean 的创建执行流程

以解析 xml 文件为例,使用ClassPathXmlApplicationContext,流程如下:

1、加载xml配置文件,解析获取配置中的每个的信息,封装成一个个的BeanDefinition对象;
2、将BeanDefinition存储在一个名为beanDefinitionMap的Map<String,BeanDefinition>中;
3、ApplicationContext底层遍历beanDefinitionMap,创建Bean实例对象;
4、创建好的Bean实例对象,被存储到一个名为singletonObjects的Map<String,Object>中;
5、当执行applicationContext.getBean(beanName)时,从singletonObjects去匹配Bean实例返回。

在这里插入图片描述

Spring 给开发者留下的扩展点

主要有两个地方,
1 是对 beanDefinitionMap 进行添加,修改,只要向 beanDefinitionMap 中添加了 BeanDefinition,就会被 Spring 生成 Bean。
2 是当 Bean 创建完成后,可以对 Bean 对象进行修改,比如 AOP,注解,都是生成代理 Bean,替换原有的 Bean。

流程如下:

结合代码描述

代码入口:

ApplicationContext context = new ClassPathXmlApplicationContext("classpath*:application-single.xml");UserService userService = context.getBean("userServiceBeanId",UserService.class);System.out.println(userService.getName());

进入ClassPathXmlApplicationContext

	public ClassPathXmlApplicationContext(String[] configLocations, boolean refresh, @Nullable ApplicationContext parent)throws BeansException {//1、返回一个classloader//2、返回一个解析器super(parent);  // ===> 1// 1、获取环境(系统环境、jvm环境)// 2、设置Placeholder占位符解析器,${xxx}.xml解析为完整文件名// 2、将xml的路径解析完存储到数组setConfigLocations(configLocations); //  ===>//默认为trueif (refresh) {//核心方法,入口refresh();  // ===>  magic happens here!}}

进入 Refresh 方法

	@Overridepublic void refresh() throws BeansException, IllegalStateException {// synchronized块锁(monitorenter --monitorexit)// 不然 refresh() 还没结束,又来个启动或销毁容器的操作//	 startupShutdownMonitor就是个空对象,锁synchronized (this.startupShutdownMonitor) {//1、【准备刷新】,设置了几个变量,也是准备工作prepareRefresh();   //  ===>// 2、【获得新的bean工厂】关键步骤,重点!//2.1、关闭旧的 BeanFactory//2.2、创建新的 BeanFactory(DefaluListbaleBeanFactory)//2.3、解析xml/加载 Bean 定义、注册 Bean定义到beanFactory(不初始化)//2.4、返回全新的工厂ConfigurableListableBeanFactory beanFactory = obtainFreshBeanFactory();//3、【bean工厂前置操作 】为BeanFactory配置容器特性// 例如类加载器、表达式解析器、注册默认环境bean、后置管理器prepareBeanFactory(beanFactory);// ===>try {// 4、【bean工厂后置操作】此处为空方法,如果子类需要,自己去实现postProcessBeanFactory(beanFactory);  // ===> 空的!//5、【调用bean工厂后置处理器】,开始调用我们自己实现的接口//调用顺序一:先执行BeanDefinitionRegistryPostProcessor接口的方法,它也继承了BeanFactoryPostProcessor//调用顺序二:后执行BeanFactoryPostProcessor接口的方法,用于修改 beanDefinitioninvokeBeanFactoryPostProcessors(beanFactory);  // ===>  重头戏//6、【注册bean后置处理器】只是注册,但是还不会调用//逻辑:找出所有实现BeanPostProcessor接口的类,分类、排序、注册registerBeanPostProcessors(beanFactory);  // ===>  关键点// Initialize message source for this context.//7、【初始化消息源】国际化问题i18n,参照https://nacos.io/initMessageSource(); // ===> 就是往factory加了个single bean// Initialize event multicaster for this context.//8、【初始化事件广播器】初始化自定义的事件监听多路广播器// 如果需要发布事件,就调它的multicastEvent方法// 把事件广播给listeners,其实就是起一个线程来处理,把Event扔给listener处理// (可以通过 SimpleApplicationEventMulticaster的代码来验证)initApplicationEventMulticaster(); // ===> 同样,加了个bean// 9、【刷新】这是个protected空方法,交给具体的子类来实现//  可以在这里初始化一些特殊的 Bean// (在初始化 singleton beans 之前)onRefresh();  // ===> 空的!一般没人管它//10、【注册监听器】,监听器需要实现 ApplicationListener 接口// 也就是扫描这些实现了接口的类,给他放进广播器的列表中// 其实就是个观察者模式,广播器接到事件的调用时,去循环listeners列表,// 挨个调它们的onApplicationEvent方法,把event扔给它们。registerListeners();  // ===> 观察者模式//11、 【结束bean工厂初始化操作】//1、初始化所有的 singleton beans,反射生成对象/填充//2、 调用Bean的前置处理器和后置处理器// 关键点:getBean方法里完成finishBeanFactoryInitialization(beanFactory);  // ===>  关键点!// 12、结束refresh操作// 发布事件与清除上下文环境finishRefresh();} catch (BeansException ex) {} finally {}}}

Bean 的创建流程总结:

在这里插入图片描述

面试中说的 三级缓存是什么?

三级缓存是为了解决循环依赖,也是为了存储单例对象

Spring提供了 三级缓存存储 完整Bean实例 和 半成品Bean实例 ,用于解决循环引用问题

在DefaultListableBeanFactory的上四级父类DefaultSingletonBeanRegistry中提供如下三个Map:

public class DefaultSingletonBeanRegistry ... { //1、最终存储单例Bean成品的容器,即实例化和初始化都完成的Bean,称之为"一级缓存"Map<String, Object> singletonObjects = new ConcurrentHashMap(256); //2、早期Bean单例池,缓存半成品对象,且当前对象已经被其他对象引用了,称之为"二级缓存" Map<String, Object> earlySingletonObjects = new ConcurrentHashMap(16); //3、单例Bean的工厂池,缓存半成品对象,对象未被引用,使用时在通过工厂创建Bean,称之为" 三级缓存" Map<String, ObjectFactory<?>> singletonFactories = new HashMap(16);
}

三级缓存怎么解决循环依赖问题?核心就是先把半成品的 Bean 放入一个缓存中

假设 Bean1 和 Bean2 循环依赖,那么创建过程如下:
1、Bean1 实例化对象,但尚未初始化,将 Bean1 存储到 三级缓存;
2、Bean1 属性注入,需要 Bean2,从缓存中获取,没有 Bean2;
3、Bean2 实例化对象,但尚未初始化,将 Bean2 存储到到 三级缓存;
4、Bean2 属性注入,需要 Bean1,从 三级缓存获取 Bean1,Bean1 从 三级缓存移入二级缓存;
5、Bean2 执行其他生命周期过程,最终成为一个完成Bean,存储到一级缓存,删除二 三级缓存;
6、Bean1 注入 Bean2;
7、Bean1 执行其他生命周期过程,最终成为一个完整 Bean,存储到一级缓存,删除二 三级缓存。

示例如下:

public static void main(String[] args) {GenericApplicationContext context = new GenericApplicationContext();context.registerBean(ConfigurationClassPostProcessor.class);context.registerBean(Config.class);context.refresh();context.close();
}@Configuration
static class Config {@Bean // 解析 @Aspect、产生代理public AnnotationAwareAspectJAutoProxyCreator annotationAwareAspectJAutoProxyCreator() {return new AnnotationAwareAspectJAutoProxyCreator();}@Bean // 解析 @Autowiredpublic AutowiredAnnotationBeanPostProcessor autowiredAnnotationBeanPostProcessor() {return new AutowiredAnnotationBeanPostProcessor();}@Bean // 解析 @PostConstructpublic CommonAnnotationBeanPostProcessor commonAnnotationBeanPostProcessor() {return new CommonAnnotationBeanPostProcessor();}@Beanpublic Advisor advisor(MethodInterceptor advice) {AspectJExpressionPointcut pointcut = new AspectJExpressionPointcut();pointcut.setExpression("execution(* foo())");return new DefaultPointcutAdvisor(pointcut, advice);}@Beanpublic MethodInterceptor advice() {return (MethodInvocation invocation) -> {System.out.println("before...");return invocation.proceed();};}@Beanpublic Bean1 bean1() {return new Bean1();}@Beanpublic Bean2 bean2() {return new Bean2();}
}static class Bean1 {public Bean1() {System.out.println("Bean1()");}@Autowired public void setBean2(Bean2 bean2) {System.out.println("Bean1 setBean2(bean2) class is: " + bean2.getClass());}@PostConstruct public void init() {System.out.println("Bean1 init()");}
}static class Bean2 {public Bean2() {System.out.println("Bean2()");}@Autowired public void setBean1(Bean1 bean1) {System.out.println("Bean2 setBean1(bean1) class is: " + bean1.getClass());}@PostConstruct public void init() {System.out.println("Bean2 init()");}
}

执行 main 方法后,打印如下:

Bean1()
Bean2()
Bean2 setBean1(bean1) class is: class com. spring.test.A17_1$Bean1
Bean2 init()
Bean1 setBean2(bean2) class is: class com. spring.test.A17_1$Bean2
Bean1 init()

为什么是 三级缓存而不是二级缓存?

如果只是为了解决循环依赖问题,那么二级缓存就够了。为什么要设计 三级缓存?为了解决循环依赖时,某个 Bean 需要被 AOP 增强,生成代理对象。

在Bean 的创建流程总结中,注意看图,Bean 的初始化方法执行完成后,才执行 BeanPostProcess的后处理方法,如果这个 Bean 被 AOP 增强了,会生成代理对象。

org. springframework.beans.factory.support.AbstractAutowireCapableBeanFactory#doCreateBean方法

// 方法一:实例化! Bean的构造函数在这里被调用!
instanceWrapper = createBeanInstance(beanName, mbd, args); 
//方法二:注入相关的属性!【关键点】
populateBean(beanName, mbd, instanceWrapper);
//方法三:初始化方法,这里会调用 BeanPostProcess
exposedObject = initializeBean(beanName, exposedObject, mbd);

org. springframework.beans.factory.support.AbstractAutowireCapableBeanFactory#initializeBean( java.lang.String, java.lang.Object, org. springframework.beans.factory.support.RootBeanDefinition) 方法

//调用Bean的前置处理器!我们自己实现的bean的processer在这里!
wrappedBean = applyBeanPostProcessorsBeforeInitialization(wrappedBean, beanName);// 1、调用初始化方法, 处理 bean 中定义的 init-method,
//2、如果 bean 实现了 InitializingBean 接口,调用 afterPropertiesSet() 方法
invokeInitMethods(beanName, wrappedBean, mbd); // 调用Bean的后置管理器!用于生成代理aop也在这里偷偷做了手脚……注意aop调试时,看这句后wrappedBean的变化!
wrappedBean = applyBeanPostProcessorsAfterInitialization(wrappedBean, beanName); 

通过上述源码,可以看到,代理对象 生成,是在初始化方法执行后,才生成代理对象。

这个功能,在循环依赖场景下,就会存在问题。

刚才的 Bean1 和 Bean2 循环依赖时,打印如下:

1、Bean1()
2、Bean2()
3、Bean2 setBean1(bean1) class is: class com. spring.test.A17_1$Bean1
4、Bean2 init()
5、Bean1 setBean2(bean2) class is: class com. spring.test.A17_1$Bean2
6、Bean1 init()

如果是在初始化后,才对 Bean1 进行增强生成代理对象,那么是在第 6 步完成后,才生成代理对象。这样Bean2 setBean1(bean1) 注入的就是原始对象,不是代理对象。不符合预期。

所以 Spring 打了一个补丁,提供一个 三级缓存, 三级缓存中存储的不是未完成的对象,而是 Bean 的创建工厂ObjectFactory。当需要Bean1 对象时,执行的是 ObjectFactory 的 getObject 方法,这个方法中,也会进行是否 AOP 的判断,如果是,就生成代理对象。

源码如下:

org. springframework.beans.factory.support.AbstractAutowireCapableBeanFactory#doCreateBean

// () -> getEarlyBeanReference(beanName, mbd, bean) 是 ObjectFactory 匿名实现类 的简写
addSingletonFactory(beanName, () -> getEarlyBeanReference(beanName, mbd, bean));

getEarlyBeanReference 方法

	protected Object getEarlyBeanReference(String beanName, RootBeanDefinition mbd, Object bean) {Object exposedObject = bean;//是否存在实现InstantiationAwareBeanPostProcessors接口的类if (!mbd.isSynthetic() && hasInstantiationAwareBeanPostProcessors()) {//循环所有Bean后置处理器for (BeanPostProcessor bp : getBeanPostProcessors()) {if (bp instanceof SmartInstantiationAwareBeanPostProcessor) {SmartInstantiationAwareBeanPostProcessor ibp = (SmartInstantiationAwareBeanPostProcessor) bp;//重点:开始创建AOP代理,exposedObject = ibp.getEarlyBeanReference(exposedObject, beanName);}}}//总结//1、如果不调用后置,返回的bean和 三级缓存一样//2、如果调用AOP后置,返回的就是代理对象//3、这就是 三级缓存设计的巧妙之处!!!!Map<String, ObjectFactory<?>>// 结论:虽然二级缓存能解决循环依赖,但是使用不了aop了,也就是扩展点没有了return exposedObject;}

示例如下:修改 Bean1 代码增加一个 foo 方法,让 Bean1 被 AOP 增强

    static class Bean1 {public void foo() {}public Bean1() {System.out.println("Bean1()");}@Autowired public void setBean2(Bean2 bean2) {System.out.println("Bean1 setBean2(bean2) class is: " + bean2.getClass());}@PostConstruct public void init() {System.out.println("Bean1 init()");}}

再次执行,观察打印效果:

Bean1()
Bean2()
Bean2 setBean1(bean1) class is: class com. spring.test.A17_1$Bean1$$EnhancerBySpringCGLIB$$28831f11
Bean2 init()
Bean1 setBean2(bean2) class is: class com. spring.test.A17_1$Bean2
Bean1 init()

注意看,Bean1 对象,很早就变成了代理对象,说明是 三级缓存生效了。注意:虽然很早就变成代理对象了,但是初始化方法还没执行。

总结下:

AOP 代理对象生成的时机?

1、如果存在循环依赖,是在 三级缓存中生成代理对象

2、如果不存在循环依赖,是在 Bean 初始化方法执行后生成。

为什么需要 三级缓存?

为了打补丁,让循环依赖的 AOP 的 Bean,能正常注入到其他 Bean 中。

1,2,3 三级缓存的作用是什么?

一级缓存存储已经完全创建好的单例对象

二级缓存存储未完成的,还被其他 Bean 引用的对象

三级缓存,用于存储未完成的对象工厂,且没被其他对象引用过。


http://www.ppmy.cn/devtools/14201.html

相关文章

应用在防蓝光显示器中的LED防蓝光灯珠

应用在防蓝光显示器中的LED防蓝光灯珠

相比抗蓝光眼镜、防蓝光覆膜、软体降低蓝光强度这些“软”净蓝手段&#xff0c;通过对LED的发光磷粉进行LED背光进行技术革新&#xff0c;可实现硬件“净蓝”。其能够将90%以上的有害蓝光转换为450nm以上的长波低能光线&#xff0c;从硬件的角度解决了蓝光危害眼睛的问题&#…
阅读更多...
25计算机考研院校数据分析 | 四川大学

25计算机考研院校数据分析 | 四川大学

四川大学(Sichuan University)简称“川大”&#xff0c;由中华人民共和国教育部直属&#xff0c;中央直管副部级建制&#xff0c;是世界一流大学建设高校、985工程”、"211工程"重点建设的高水平综合性全国重点大学&#xff0c;入选”2011计划"、"珠峰计划…
阅读更多...
真实世界的密码学(一)

真实世界的密码学(一)

原文&#xff1a;annas-archive.org/md5/655c944001312f47533514408a1a919a 译者&#xff1a;飞龙 协议&#xff1a;CC BY-NC-SA 4.0 前言 序言 当你拿起这本书时&#xff0c;你可能会想&#xff0c;为什么又一本关于密码学的书&#xff1f;甚至&#xff0c;为什么我要读这本…
阅读更多...
《从零开始的Java世界》10File类与IO流

《从零开始的Java世界》10File类与IO流

《从零开始的Java世界》系列主要讲解Javase部分&#xff0c;从最简单的程序设计到面向对象编程&#xff0c;再到异常处理、常用API的使用&#xff0c;最后到注解、反射&#xff0c;涵盖Java基础所需的所有知识点。学习者应该从学会如何使用&#xff0c;到知道其实现原理全方位式…
阅读更多...
Hadoop - 安装

Hadoop - 安装

文章目录 关于 Hadoop架构变迁 1.0 --> 2.0 --> 3.0 安装配置安装配置环境变量配置core-site.xmlhdfs-site.xmlmapped-site.xmlyarn-site.xml配置 hadoop-env 启动/停止 Hadoop 服务查看 hdfs report 关于 Hadoop The Apache™ Hadoop project develops open-source soft…
阅读更多...
数字信号处理操作教程_音频解码:3-8 G711A音频解码实验

数字信号处理操作教程_音频解码:3-8 G711A音频解码实验

一、实验目的 学习G711音频的格式和G711A音频解码的原理&#xff0c;并实现将BIT格式解码为PCM格式。 二、实验原理 G711 G711是国际电信联盟订定出来的一套语音压缩标准&#xff0c;主要用于电话。它主要用脉冲编码调制对音频采样&#xff0c;采样率为8k每秒。它利用一个 …
阅读更多...
python根据一定概率判断是否选择

python根据一定概率判断是否选择

&#x1f47d;发现宝藏 前些天发现了一个巨牛的人工智能学习网站&#xff0c;通俗易懂&#xff0c;风趣幽默&#xff0c;忍不住分享一下给大家。【点击进入巨牛的人工智能学习网站】。 Python中基于概率进行选择的方法 在编程中&#xff0c;我们经常会遇到需要根据一定的概率来…
阅读更多...
PHP定期给自己网站目录做个特征镜像供快速对比

PHP定期给自己网站目录做个特征镜像供快速对比

效果图 上代码&#xff1a; <style> h1{font-size:24px;line-height:180%;font-weight:600;margin:1px 2px;color:#0180cf;} h2{font-size:20px;line-height:140%;font-weight:600;margin:2px 4px;color:green;} h3{font-size:16px;line-height:140%;font-weight:600;m…
阅读更多...
最新文章

玻璃钢生产厂家玻璃钢菩萨雕塑濮阳大型玻璃钢仿铜雕塑厂家玻璃钢雕塑胶模驻马店玻璃钢卡通雕塑厂家地址大理市玻璃钢雕塑报价商洛玻璃钢人物雕塑艺术商场美陈制造徐汇区玻璃钢雕塑销售价格柳州玻璃钢泡沫雕塑价格范围户内玻璃钢雕塑价格福建动物玻璃钢雕塑市场临沧市玻璃钢雕塑生产厂家浙江季节性商场美陈批发商场美陈市场调研报告梧州玻璃钢雕塑厂家宝山专业玻璃钢雕塑雕塑玻璃钢动物玻璃钢玫瑰花雕塑佛山玻璃钢园林雕塑毕节玻璃钢座椅雕塑价格澳门玻璃钢瓜果雕塑装饰商场美陈哪里买惠州市义乌江玻璃钢雕塑厂商场美陈布搭配地贴猴子玻璃钢雕塑玻璃钢金属雕塑浙江大型商场美陈生产公司淮安美陈商场布置望城玻璃钢雕塑设计漳州玻璃钢卡通雕塑香港通过《维护国家安全条例》两大学生合买彩票中奖一人不认账让美丽中国“从细节出发”19岁小伙救下5人后溺亡 多方发声单亲妈妈陷入热恋 14岁儿子报警汪小菲曝离婚始末遭遇山火的松茸之乡雅江山火三名扑火人员牺牲系谣言何赛飞追着代拍打萧美琴窜访捷克 外交部回应卫健委通报少年有偿捐血浆16次猝死手机成瘾是影响睡眠质量重要因素高校汽车撞人致3死16伤 司机系学生315晚会后胖东来又人满为患了小米汽车超级工厂正式揭幕中国拥有亿元资产的家庭达13.3万户周杰伦一审败诉网易男孩8年未见母亲被告知被遗忘许家印被限制高消费饲养员用铁锨驱打大熊猫被辞退男子被猫抓伤后确诊“猫抓病”特朗普无法缴纳4.54亿美元罚金倪萍分享减重40斤方法联合利华开始重组张家界的山上“长”满了韩国人?张立群任西安交通大学校长杨倩无缘巴黎奥运“重生之我在北大当嫡校长”黑马情侣提车了专访95后高颜值猪保姆考生莫言也上北大硕士复试名单了网友洛杉矶偶遇贾玲专家建议不必谈骨泥色变沉迷短剧的人就像掉进了杀猪盘奥巴马现身唐宁街 黑色着装引猜测七年后宇文玥被薅头发捞上岸事业单位女子向同事水杯投不明物质凯特王妃现身!外出购物视频曝光河南驻马店通报西平中学跳楼事件王树国卸任西安交大校长 师生送别恒大被罚41.75亿到底怎么缴男子被流浪猫绊倒 投喂者赔24万房客欠租失踪 房东直发愁西双版纳热带植物园回应蜉蝣大爆发钱人豪晒法院裁定实锤抄袭外国人感慨凌晨的中国很安全胖东来员工每周单休无小长假白宫:哈马斯三号人物被杀测试车高速逃费 小米:已补缴老人退休金被冒领16年 金额超20万

玻璃钢生产厂家 XML地图 TXT地图 虚拟主机 SEO 网站制作 网站优化