当前位置: 首页 > news >正文

中国建设银行官网站诚聘英才网络营销专业技能

中国建设银行官网站诚聘英才,网络营销专业技能,高明专业网站建设报价,c2c的电子商务平台有哪些文章目录零、本讲学习目标一、Spring AOP(一)AOP基本含义(二)AOP基本作用(三)AOP与OOP对比(四)AOP使用方式(五)AOP基本概念二、提出游吟诗人唱赞歌任务&#…

文章目录

  • 零、本讲学习目标
  • 一、Spring AOP
    • (一)AOP基本含义
    • (二)AOP基本作用
    • (三)AOP与OOP对比
    • (四)AOP使用方式
    • (五)AOP基本概念
  • 二、提出游吟诗人唱赞歌任务
    • (一)采用传统方式实现
    • (二)采用传统方式实现的缺点
  • 三、采用配置方式使用AOP
    • (一)创建本讲所需子包
    • (二)创建杀龙任务类
    • (三)创建勇敢骑士类
    • (四)创建游吟诗人类
    • (五)创建Spring配置文件
      • 1、切点
      • 2、切点表达式
      • 3、切点函数
    • (六)在pom文件里添加AOP相关依赖
    • (七)创建测试类 - TestKnight
    • (八)运行测试方法testBraveKnight(),查看结果
    • (九)课堂练习
      • 1、增加救美任务类与救美骑士类
      • 2、在测试类里增加测试方法 - testDamselRescuingKnight()
      • 3、运行testDamselRescuingKnight()方法,查看结果
  • 四、采用注解方式使用AOP
    • (一)创建本讲所需子包
    • (二)创建杀龙任务类
    • (三)创建勇敢骑士类
    • (四)创建游吟诗人切面
    • (五)创建Spring配置类
    • (六)创建骑士测试类
    • (七)运行测试方法testBraveKnight(),查看效果
    • (八)课堂练习
      • 1、增加救美任务类与救美骑士类
      • 2、在测试类里增救美骑士测试方法 - testDamselRescuingKnight()
      • 3、运行testDamselRescuingKnight()方法,查看结果
  • 四、实现注解式拦截
    • (一)拦截的含义
    • (二)创建注解接口
      • 1、@Target({ElementType.TYPE}) 注解
      • 2、 @Retention({RetentionPolicy.Runtime}) 注解
      • 3、@Documented注解
    • (三)修改勇敢骑士类
    • (四)修改游吟诗人切面
    • (五)运行测试方法testBraveKnight(),查看效果
    • (五)修改救美骑士类,给embarkOnQuest()添加自定义注解Action
    • (六)运行测试方法testDamselRescuingKnight(),查看效果
    • (七)课堂练习
      • 任务:输出骑士完成任务的耗时

零、本讲学习目标

  1. 理解AOP与OOP的关系
  2. 掌握采用配置方式使用AOP
  3. 掌握采用注解方式使用AOP

一、Spring AOP

(一)AOP基本含义

  • AOP(Aspect-OrientedProgramming,面向切面编程),可以说是OOP(Object-Oriented Programing,面向对象编程)的补充和完善。OOP引入封装、继承和多态性等概念来建立一种对象层次结构,用以模拟公共行为的一个集合。当我们需要为分散的对象引入公共行为的时候,OOP则显得无能为力。
  • OOP允许你定义从上到下的关系,但并不适合定义从左到右的关系。例如日志功能。日志代码往往水平地散布在所有对象层次中,而与它所散布到的对象的核心功能毫无关系。对于其他类型的代码,如安全性、异常处理和透明的持续性也是如此。这种散布在各处的无关的代码被称为横切(cross-cutting)代码,在OOP设计中,它导致了大量代码的重复,而不利于各个模块的重用。
  • AOP技术则恰恰相反,它利用一种称为“横切”的技术,剖解开封装的对象内部,并将那些影响了多个类的公共行为封装到一个可重用模块,并将其名为“Aspect”,即方面。所谓“方面”,简单地说,就是将那些与业务无关,却为业务模块所共同调用的逻辑或责任封装起来,便于减少系统的重复代码,降低模块间的耦合度,并有利于未来的可操作性和可维护性。
  • AOP代表的是一个横向的关系,如果说“对象”是一个空心的圆柱体,其中封装的是对象的属性和行为;那么面向方面编程的方法,就仿佛一把利刃,将这些空心圆柱体剖开,以获得其内部的消息。而剖开的切面,也就是所谓的“方面”了。然后它又以巧夺天功的妙手将这些剖开的切面复原,不留痕迹。

(二)AOP基本作用

  • 软件开发原则:高内聚,低耦合
  • Spring的AOP作用在于解耦。AOP让一组类共享相同的行为(比如事务管理、日志管理、安全管理)。
  • OOP(Object-Oriented Programming)只能通过继承类或实现接口来增加代码的耦合度,而且类继承是单根继承(不允许一子多父),阻碍了将更多的行为添加到一组类上,此时AOP可以弥补OOP的不足。

(三)AOP与OOP对比

  • AOP(Aspect-Oriented Programming)—— 横向的关系
  • OOP(Object-Oriented Programming)—— 纵向的关系
    在这里插入图片描述

(四)AOP使用方式

  1. Spring采用配置方式使用AOP
  2. Spring采用注解方式使用AOP

(五)AOP基本概念

  1. Aspect(切面):通常是一个类,里面可以定义切入点和通知
  2. JointPoint(连接点):程序执行过程中明确的点,一般是方法的调用
  3. Advice(通知):AOP在特定的切入点上执行的增强处理,有before、after、after-returning、after-throwing、around
  4. Pointcut(切入点):就是带有通知的连接点,在程序中主要体现为书写切入点表达式
  5. AOP代理:AOP框架创建的对象,代理就是目标对象的加强。Spring中的AOP代理可以使JDK动态代理,也可以是CGLIB代理,前者基于接口,后者基于子类

二、提出游吟诗人唱赞歌任务

  • 骑士执行任务前和执行任务后,游吟诗人唱赞歌

(一)采用传统方式实现

  • 修改day04子包的勇敢骑士类
    在这里插入图片描述

  • 修改day04子包里的救美骑士类
    在这里插入图片描述

  • 执行测试类 - TestKnight
    在这里插入图片描述

(二)采用传统方式实现的缺点

  • 每个骑士类的embarkOnQuest()方法都要修改,耦合度太高,当骑士类数量很大时,这个任务完成起来就十分枯燥繁琐。

下面我们采用AOP方式来实现同样的功能,大家通过对比可以更好地体会采用AOP方式的优越性。

三、采用配置方式使用AOP

(一)创建本讲所需子包

  • 在net.hw.spring包里创建lesson05.aop_xml子包
    在这里插入图片描述

(二)创建杀龙任务类

  • 在aop_xml子包里创建杀龙任务类 - SlayDragonQuest
    在这里插入图片描述
package net.hw.spring.lesson05.aop_xml;import org.springframework.stereotype.Component;/*** 功能:杀龙任务类* 作者:华卫* 日期:2021年03月24日*/
@Component
public class SlayDragonQuest {public void embark() {System.out.println("执行杀龙任务。");}
}

(三)创建勇敢骑士类

  • 在aop_xml子包里创建勇敢骑士类 - BraveKnight
    在这里插入图片描述
package net.hw.spring.lesson05.aop_xml;/*** 功能:勇敢骑士类* 作者:华卫* 日期:2021年03月24日*/import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;@Component("Mike")
public class BraveKnight {@Autowiredprivate SlayDragonQuest slayDragonQuest;public void embarkOnQuest() {slayDragonQuest.embark();}
}

(四)创建游吟诗人类

  • 在aop_xml子包里创建游吟诗人类 - Minstrel
    在这里插入图片描述
package net.hw.spring.lesson05.aop_xml;import org.springframework.stereotype.Component;/*** 功能:游吟诗人类* 作者:华卫* 日期:2021年03月24日*/
@Component
public class Minstrel {public void singBeforeQuest() {System.out.println("啦啦啦,骑士出发了!");}public void singAfterQuest() {System.out.println("真棒啊!骑士完成了任务!");}
}

(五)创建Spring配置文件

  • 在resources里创建aop_xml目录,在里面创建spring-config.xml配置文件
    在这里插入图片描述
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xmlns:context="http://www.springframework.org/schema/context"xmlns:aop="http://www.springframework.org/schema/aop"xsi:schemaLocation="http://www.springframework.org/schema/beanshttp://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context https://www.springframework.org/schema/context/spring-context.xsd http://www.springframework.org/schema/aop https://www.springframework.org/schema/aop/spring-aop.xsd"><!--组件扫描--><context:component-scan base-package="net.hw.spring.lesson05.aop_xml"/><!--AOP配置--><aop:config><!--定义切面--><aop:aspect ref="minstrel"><!--定义切点--><aop:pointcut id="embark" expression="execution(* net.hw.spring.lesson05..*.embarkOnQuest(..))"/><!--声明前置通知--><aop:before method="singBeforeQuest" pointcut-ref="embark"/><!--声明后置通知--><aop:after method="singAfterQuest" pointcut-ref="embark"/></aop:aspect></aop:config>
</beans>

在这里插入图片描述

1、切点

在使用Spring框架配置AOP时,不管是通过XML配置文件还是注解方式,都需要定义pointcut(切点)。

2、切点表达式

  • 拦截类里的指定方法:"execution(* net.hw.spring.lesson05..*.embarkOnQuest(..))"
  • 拦截类里的所有方法:"execution(* net.hw.spring.lesson05..*.*(..))"

3、切点函数

execution()是最常用的切点函数,整个表达式可以分为五个部分。

  • execution():表达式主体。
  • 第一个*号:表示返回类型,*号表示所有的类型。
  • 包名:表示需要拦截的包名,后面的两个句点表示当前包和当前包的所有子包,net.hw.spring.lesson05包、子孙包下所有类的方法。
  • 第二个*号:表示类名,*号表示所有的类。
  • *(..):最后这个星号表示方法名,*号表示所有的方法,后面括弧里面表示方法的参数,两个句点表示任何参数。

(六)在pom文件里添加AOP相关依赖

在这里插入图片描述

<!--Spring AOP-->                                         
<dependency>                                              <groupId>org.springframework</groupId>                <artifactId>spring-aop</artifactId>                   <version>${spring.version}</version>                  
</dependency>                                             
<!--AspectJ支持-->                                          
<dependency>                                              <groupId>aspectj</groupId>                            <artifactId>aspectjrt</artifactId>                    <version>1.5.4</version>                              
</dependency>                                             
<dependency>                                              <groupId>org.aspectj</groupId>                        <artifactId>aspectjweaver</artifactId>                <version>1.9.6</version>                              <scope>runtime</scope>                                
</dependency>                                             
  • 注意:添加新的依赖之后,记得更新
    在这里插入图片描述

(七)创建测试类 - TestKnight

  • 在test/java里创建net.hw.spring.lesson05.aop_xml包,在包里创建TestKnight
    在这里插入图片描述
package net.hw.spring.lesson05.aop_xml;import org.junit.After;
import org.junit.Before;
import org.junit.Test;
import org.springframework.context.support.ClassPathXmlApplicationContext;/*** 功能:测试骑士类* 作者:华卫* 日期:2021年03月24日*/
public class TestKnight {private ClassPathXmlApplicationContext context; // 基于类路径XML配置文件的应用容器@Beforepublic void init() {// 基于Spring配置文件创建应用容器context = new ClassPathXmlApplicationContext("aop_xml/spring-config.xml");}@Testpublic void testBraveKnight() {// 根据名称从应用容器中获取勇敢骑士对象BraveKnight braveKnight = (BraveKnight) context.getBean("Mike");// 勇敢骑士执行任务braveKnight.embarkOnQuest();}@Afterpublic void destroy() {// 关闭应用容器context.close();}
}

(八)运行测试方法testBraveKnight(),查看结果

在这里插入图片描述

(九)课堂练习

1、增加救美任务类与救美骑士类

在这里插入图片描述

2、在测试类里增加测试方法 - testDamselRescuingKnight()

在这里插入图片描述

3、运行testDamselRescuingKnight()方法,查看结果

在这里插入图片描述

四、采用注解方式使用AOP

(一)创建本讲所需子包

  • 在net.hw.spring包里创建lesson05.aop_annotation子包
    在这里插入图片描述

(二)创建杀龙任务类

  • 在aop_annotation子包里创建杀龙任务类 - SlayDragonQuest
    在这里插入图片描述
package net.hw.spring.lesson05.aop_annotation;import org.springframework.stereotype.Component;/*** 功能:杀龙任务类* 作者:华卫* 日期:2021年03月29日*/
@Component
public class SlayDragonQuest {public void embark() {System.out.println("执行杀龙任务。");}
}

(三)创建勇敢骑士类

  • 在aop_annotation子包里创建勇敢骑士类 - BraveKnight
  • 在这里插入图片描述
package net.hw.spring.lesson05.aop_annotation;/*** 功能:勇敢骑士类* 作者:华卫* 日期:2021年03月29日*/import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;@Component("Mike")
public class BraveKnight {@Autowiredprivate SlayDragonQuest slayDragonQuest;public void embarkOnQuest() {slayDragonQuest.embark();}
}

(四)创建游吟诗人切面

  • 在aop_annotation子包里创建游吟诗人切面 - MinstrelAspect
    在这里插入图片描述

在这里插入图片描述

package net.hw.spring.lesson05.aop_annotation;import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.annotation.After;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.aspectj.lang.annotation.Pointcut;
import org.springframework.stereotype.Component;/*** 功能:游吟诗人切面* 作者:华卫* 日期:2021年03月29日*/
@Aspect // 声明为切面
@Component // 交给Spring容器管理
public class MinstrelAspect {// 注解声明切点@Pointcut("execution(* net.hw.spring.lesson05..*.embarkOnQuest(..))")public void embark() {}// 注解声明前置通知@Before("embark()")public void singBeforeQuest(JoinPoint joinPoint) {System.out.println("啦啦啦,骑士出发了!");}// 注解声明后置通知@After("embark()")public void singAfterQuest(JoinPoint joinPoint) {System.out.println("真棒啊!骑士完成了任务!");}
}

(五)创建Spring配置类

  • 在aop_annotation子包里创建Spring配置类 - AopConfig
    在这里插入图片描述
package net.hw.spring.lesson05.aop_annotation;import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.EnableAspectJAutoProxy;/*** 功能:AOP配置类* 作者:华卫* 日期:2021年03月29日*/
@Configuration // 标明Spring配置类
@ComponentScan("net.hw.spring.lesson05.aop_annotation") // 组件扫描
@EnableAspectJAutoProxy // 开启Spring对AspectJ的支持
public class AopConfig {
}

(六)创建骑士测试类

  • 在test/java/aop_annotation子包里创建测试类 - TestKnight
    在这里插入图片描述
package net.hw.spring.lesson05.aop_annotation;import org.junit.After;
import org.junit.Before;
import org.junit.Test;
import org.springframework.context.annotation.AnnotationConfigApplicationContext;/*** 功能:测试骑士类* 作者:华卫* 日期:2021年03月29日*/
public class TestKnight {private AnnotationConfigApplicationContext context; // 基于注解配置类的应用容器@Beforepublic void init() {// 基于注解配置类创建应用容器context = new AnnotationConfigApplicationContext(AopConfig.class);}@Testpublic void testBraveKnight() {// 根据名称从应用容器里获取勇敢骑士对象BraveKnight knight = (BraveKnight) context.getBean("Mike");// 勇敢骑士执行任务knight.embarkOnQuest();}@Afterpublic void destroy() {// 关闭应用容器context.close();}
}

(七)运行测试方法testBraveKnight(),查看效果

在这里插入图片描述

(八)课堂练习

1、增加救美任务类与救美骑士类

在这里插入图片描述

2、在测试类里增救美骑士测试方法 - testDamselRescuingKnight()

在这里插入图片描述

3、运行testDamselRescuingKnight()方法,查看结果

在这里插入图片描述

四、实现注解式拦截

(一)拦截的含义

在某个方法被访问之前进行拦截,然后在方法执行之前或之后加入某些操作,其实就是AOP的一种实现策略。Spring提供拦截器(Interceptor),它通过动态拦截Action调用的对象,允许开发者定义在一个action执行的前后执行的代码,也可以在一个action执行前阻止其执行。同时也是提供了一种可以提取action中可重用的部分的方式。本案例,我们不用拦截器,学习注解式拦截。

(二)创建注解接口

  • 在aop_annotation子包里创建注解接口 - Action
    在这里插入图片描述
    在这里插入图片描述
package net.hw.spring.lesson05.aop_annotation;import java.lang.annotation.*;/*** 功能:动作注解接口* 作者:华卫* 日期:2021年03月29日*/
@Target(ElementType.METHOD) // 拦截目标 - 方法
@Retention(RetentionPolicy.RUNTIME) // 保留策略 - 运行时
@Documented // 注解文档化
public @interface Action {String name();
}

1、@Target({ElementType.TYPE}) 注解

  • ElementType 这个枚举类型的常量提供了一个简单的分类:注解可能出现在Java程序中的语法位置(这些常量与元注解类型(@Target)一起指定在何处写入注解的合法位置)

2、 @Retention({RetentionPolicy.Runtime}) 注解

  • RetentionPolicy这个枚举类型的常量描述保留注解的各种策略,它们与元注解(@Retention)一起指定注释要保留多长时间

3、@Documented注解

  • Documented注解表明这个注解是由 javadoc记录的,在默认情况下也有类似的记录工具。 如果一个类型声明被注解了文档化,它的注解成为公共API的一部分。

(三)修改勇敢骑士类

  • 给embarkOnQuest()添加自定义注解Action,并设置其name属性
    在这里插入图片描述
package net.hw.spring.lesson05.aop_annotation;/*** 功能:勇敢骑士类* 作者:华卫* 日期:2021年03月29日*/import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;@Component("Mike")
public class BraveKnight {@Autowiredprivate SlayDragonQuest slayDragonQuest;@Action(name = "动作拦截器")public void embarkOnQuest() {slayDragonQuest.embark();}
}

(四)修改游吟诗人切面

在这里插入图片描述

package net.hw.spring.lesson05.aop_annotation;import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.annotation.After;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.aspectj.lang.annotation.Pointcut;
import org.aspectj.lang.reflect.MethodSignature;
import org.springframework.stereotype.Component;import java.lang.reflect.Method;/*** 功能:游吟诗人切面* 作者:华卫* 日期:2021年03月29日*/
@Aspect // 声明为切面
@Component // 交给Spring容器管理
public class MinstrelAspect {// 注解声明切点@Pointcut("@annotation(net.hw.spring.lesson05.aop_annotation.Action)")public void embark() {}// 注解声明前置通知@Before("embark()")public void singBeforeQuest(JoinPoint joinPoint) {// 获取方法签名MethodSignature signature = (MethodSignature) joinPoint.getSignature();// 获取被拦截的方法Method method = signature.getMethod();// 获取注解式拦截Action action = method.getAnnotation(Action.class);// 提示用户被拦截了System.out.println("[" + action.name() + "]拦截了[" + method.getName() + "]:拦截前!");System.out.println("啦啦啦,骑士出发了!");}// 注解声明后置通知@After("embark()")public void singAfterQuest(JoinPoint joinPoint) {// 获取方法签名MethodSignature signature = (MethodSignature) joinPoint.getSignature();// 获取被拦截的方法Method method = signature.getMethod();// 获取注解式拦截Action action = method.getAnnotation(Action.class);// 提示用户被拦截了System.out.println("[" + action.name() + "]拦截了[" + method.getName() + "]:拦截后!");System.out.println("真棒啊!骑士完成了任务!");}
}

(五)运行测试方法testBraveKnight(),查看效果

在这里插入图片描述

(五)修改救美骑士类,给embarkOnQuest()添加自定义注解Action

在这里插入图片描述

(六)运行测试方法testDamselRescuingKnight(),查看效果

在这里插入图片描述

(七)课堂练习

任务:输出骑士完成任务的耗时

  • 创建耗时切面类 - ElapseAspect
    在这里插入图片描述

  • 运行测试方法testBraveKnight(),查看效果
    在这里插入图片描述

  • 运行测试方法testDamselRescuingKnight(),查看效果
    在这里插入图片描述

  • 只让耗时切面起作用,查看运行结果
    在这里插入图片描述


文章转载自:
http://goonie.c7622.cn
http://xanthophore.c7622.cn
http://athens.c7622.cn
http://loudspeaker.c7622.cn
http://talma.c7622.cn
http://sunbow.c7622.cn
http://resh.c7622.cn
http://bartend.c7622.cn
http://vertiginous.c7622.cn
http://verbally.c7622.cn
http://aquosity.c7622.cn
http://amassment.c7622.cn
http://caesarism.c7622.cn
http://sociogenous.c7622.cn
http://recreate.c7622.cn
http://initiatress.c7622.cn
http://uncomplaining.c7622.cn
http://hageman.c7622.cn
http://vanward.c7622.cn
http://agamogenesis.c7622.cn
http://portage.c7622.cn
http://monoblastic.c7622.cn
http://orthocephalous.c7622.cn
http://vaporise.c7622.cn
http://purebred.c7622.cn
http://bioinstrumentation.c7622.cn
http://sodomize.c7622.cn
http://monosynaptic.c7622.cn
http://bullionist.c7622.cn
http://possy.c7622.cn
http://betaine.c7622.cn
http://kulun.c7622.cn
http://finalist.c7622.cn
http://subdiscipline.c7622.cn
http://evocator.c7622.cn
http://fermentation.c7622.cn
http://floozie.c7622.cn
http://player.c7622.cn
http://strawy.c7622.cn
http://nonlinear.c7622.cn
http://durability.c7622.cn
http://rhathymia.c7622.cn
http://solus.c7622.cn
http://inhabitancy.c7622.cn
http://lump.c7622.cn
http://noncancelability.c7622.cn
http://comtesse.c7622.cn
http://unwatered.c7622.cn
http://crate.c7622.cn
http://danny.c7622.cn
http://pantoum.c7622.cn
http://removed.c7622.cn
http://rheophilic.c7622.cn
http://proliferous.c7622.cn
http://catachrestically.c7622.cn
http://breastsummer.c7622.cn
http://citified.c7622.cn
http://montessorian.c7622.cn
http://exhalent.c7622.cn
http://candlefish.c7622.cn
http://carpophore.c7622.cn
http://calculated.c7622.cn
http://roselike.c7622.cn
http://tripersonal.c7622.cn
http://washroom.c7622.cn
http://chitling.c7622.cn
http://droning.c7622.cn
http://yodel.c7622.cn
http://anagenesis.c7622.cn
http://trainee.c7622.cn
http://pythonic.c7622.cn
http://wheezily.c7622.cn
http://farmstead.c7622.cn
http://alonso.c7622.cn
http://nuke.c7622.cn
http://fascismo.c7622.cn
http://kingfisher.c7622.cn
http://tortuosity.c7622.cn
http://compulsorily.c7622.cn
http://nautili.c7622.cn
http://bernie.c7622.cn
http://unconsumed.c7622.cn
http://assignable.c7622.cn
http://globefish.c7622.cn
http://assiut.c7622.cn
http://elves.c7622.cn
http://undeliverable.c7622.cn
http://counterpropaganda.c7622.cn
http://riproarious.c7622.cn
http://migronaut.c7622.cn
http://bony.c7622.cn
http://menstrual.c7622.cn
http://receptaculum.c7622.cn
http://reprobative.c7622.cn
http://photoabsorption.c7622.cn
http://scrapbasket.c7622.cn
http://slenderize.c7622.cn
http://recoin.c7622.cn
http://oopm.c7622.cn
http://disillusionary.c7622.cn
http://www.zhongyajixie.com/news/96087.html

相关文章:

  • 中山网站建设文化搜索排名广告营销
  • 建站公司上海百度一下你就知道官方网站
  • 湖州 网站建设公司bt搜索引擎下载
  • 网站解析怎么做郑州有没有厉害的seo
  • 温州网站建设公司哪个好郑州网站排名推广
  • 网站建设深营销网站建设都是专业技术人员
  • 国际网站怎么做优化直链平台
  • 网站托管解决方案微信推广怎么弄
  • 成都软件开发工资一般多少在seo优化中
  • 网页程序开发学什么语言杭州seo网站排名
  • 石门网站建设网络营销的推广手段
  • 怎么做qq业务网站软件推广是什么工作
  • 做网站毕业论文的参考文献吉林seo管理平台
  • 佛山专业建设网站平台seo公司北京
  • 湖北网站建设费用百度商店应用市场
  • 哔哩哔哩网站4 3比例怎么做今日新闻快讯
  • 国外的设计网站app国家高新技术企业名单
  • 青岛哪里有做网站的什么是seo优化推广
  • html5官网首页网站seo方案撰写
  • 网站首页一般做多大太原网站优化
  • 公司使用威联通nas做网站存储12345浏览器网址大全
  • 网站目录怎么做的seo教育
  • 做logo什么网站什么是白帽seo
  • 服务器安装多个wordpress关键seo排名点击软件
  • 企业网站项目流程app营销十大成功案例
  • 免费域名申请网站空间seo网站优化方案案例
  • wordpress当前菜单湛江百度seo公司
  • 建筑网片产品资料seo是如何优化
  • 网站建设liluokj腰肌劳损的自我治疗和恢复的方法有什么?
  • 哪些公司做网站首页