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

【Spring】 AOP面向切面编程

文章目录

  • AOP是什么?
  • 一、AOP术语名词介绍
  • 二、Spring AOP框架介绍和关系梳理
  • 三、Spring AOP基于注解方式实现和细节
    • 3.1 Spring AOP底层技术组成
    • 3.2 初步实现
    • 3.3 获取通知细节信息
    • 3.4 切点表达式语法
    • 3.5 重用(提取)切点表达式
    • 3.6 环绕通知
    • 3.7 切面优先级设置
    • 3.8 CGLib动态代理生效
    • 3.9 注解实现小结
  • 四、Spring AOP对获取Bean的影响理解
  • 总结
    • 代理介绍:
    • AOP名词理解:
    • AOP思维以及与代理的关系:


AOP是什么?

AOP:Aspect Oriented Programming面向切面编程,解决非核心代码的冗余。

  • AOP利用一种称为"横切"的技术,剖解开封装的对象内部,并将那些影响了多个类的公共行为封装到一个可重用模块,并将其命名为"Aspect",即切面。
  • 所谓"切面",简单说就是那些与业务无关,却为业务模块所共同调用的逻辑或责任封装起来,便于减少系统的重复代码,降低模块之间的耦合度,并有利于未来的可操作性和可维护性。
  • 使用AOP,可以在不修改原来代码的基础上添加新功能。

AOP思想主要的应用场景:

  1. 日志记录:在系统中记录日志是非常重要的,可以使用AOP来实现日志记录的功能,可以在方法执行前、执行后异常抛出时记录日志
  2. 事务处理:在数据库操作中使用事务可以保证数据的一致性,可以使用AOP来实现事务处理的功能,可以在方法开始前开启事务,在方法执行完毕后提交或回滚事务
  3. 安全控制:在系统中包含某些需要安全控制的操作,如登录、修改密码、授权等,可以使用AOP来实现安全控制的功能。可以在方法执行前进行权限判断,如果用户没有权限,则抛出异常或转向到错误页面,以防止未经授权的访问。
  4. 性能监控:在系统运行过程中,有时需要对某些方法的性能进行监控,以找到系统的瓶颈并进行优化。可以使用AOP来实现性能监控的功能,可以在方法执行前记录时间戳,在方法执行完毕后计算方法执行时间并输出到日志中。
  5. 异常处理:系统中可能出现各种异常情况,如空指针异常、数据库连接异常等,可以使用AOP来实现异常处理的功能,在方法执行过程中,如果出现异常,则进行异常处理(如记录日志、发送邮件等)。
  6. 缓存控制:在系统中有些数据可以缓存起来以提高访问速度,可以使用AOP来实现缓存控制的功能,可以在方法执行前查询缓存中是否有数据,如果有则返回,否则执行方法并将方法返回值存入缓存中。
  7. 动态代理:AOP的实现方式之一是通过动态代理,可以代理某个类的所有方法,用于实现各种功能。
  • 综上所述,AOP可以应用于各种场景,它的作用是将通用的横切关注点与业务逻辑分离,使得代码更加清晰、简洁、易于维护。

一、AOP术语名词介绍

  • 横切关注点
    • 从每个方法中抽取出来的同一类非核心业务
    • 在同一个项目中,我们可以使用多个横切关注点对相关方法进行多个不同方面的增强。
    • 这个概念不是语法层面天然存在的,而是根据附加功能的逻辑上的需要:有十个附加功能,就有十个横切关注点

·

  • AOP把软件系统分为两个部分:核心关注点横切关注点

    • 业务处理的主要流程是核心关注点,非核心代码是横切关注点。
    • 横切关注点的一个特点是,他们经常发生在核心关注点的多处,而各处基本相似,比如权限认证、日志、事务、异常等。
    • AOP的作用在于分离系统中的各种关注点,将核心关注点和横切关注点分离开来。(分离核心业务代码与非核心代码
  • 通知(增强)

    • 每一个横切关注点上要做的事情都需要写一个方法来实现,这样的方法就叫通知方法。
      • 前置通知:在被代理的目标方法执行
      • 返回通知:在被代理的目标方法成功结束后执行(寿终正寝
      • 异常通知:在被代理的目标方法异常结束后执行(死于非命
      • 后置通知:在被代理的目标方法最终结束后执行(盖棺定论
      • 环绕通知:使用try...catch...finally结构围绕整个被代理的目标方法,包括上面四种通知对应的所有位置

1

  • 切入点 pointcut
    定位连接点的方式,或者可以理解成被选中的连接点!
    是一个表达式,比如execution(* com.spring.service.impl..(…))。
    符合条件的每个方法都是一个具体的连接点。

  • 切面 aspect
    切入点和通知的结合。是一个类。
    1

  • 目标 target
    被代理的目标对象。

  • 代理 proxy
    向目标对象应用通知之后创建的代理对象。

  • 织入 weave
    指把通知应用到目标上,生成代理对象的过程。
    可以在编译期织入,也可以在运行期织入,Spring采用后者。

二、Spring AOP框架介绍和关系梳理

  • AOP一种区别于OOP编程思维,用来完善和解决OOP的非核心代码冗余和不方便统一维护问题!
  • 代理技术(动态代理|静态代理)是实现AOP思维编程的具体技术,但是自己使用动态代理实现代码比较繁琐!
  • Spring AOP框架基于AOP编程思维封装动态代理技术,简化动态代理技术实现的框架!
    • SpringAOP内部帮助我们实现动态代理,
    • 我们只需写少量的配置指定生效范围即可完成面向切面思维编程的实现!

三、Spring AOP基于注解方式实现和细节

3.1 Spring AOP底层技术组成

1

  • 动态代理(InvocationHandler):JDK原生的实现方式,需要被代理的目标类必须实现接口
    • 因为这个技术要求代理对象和目标对象实现同样的接口(兄弟两个拜把子模式)。
  • cglib:通过继承被代理的目标类(认干爹模式)实现代理,所以不需要目标类实现接口。
  • AspectJ:早期的AOP实现的框架,SpringAOP借用了AspectJ中的AOP注解。

3.2 初步实现

1
1

  1. 导入依赖
<!-- spring-aspects会帮我们传递过来aspectjweaver -->
<dependency><groupId>org.springframework</groupId><artifactId>spring-aop</artifactId><version>6.0.6</version>
</dependency><dependency><groupId>org.springframework</groupId><artifactId>spring-aspects</artifactId><version>6.0.6</version>
</dependency>
  1. 准备接口
public interface Calculator {int add(int i, int j);int sub(int i, int j);int mul(int i, int j);int div(int i, int j);}
  1. 纯净实现类
/*** @Description: 实现计算接口,单纯添加加减乘除 实现,掺杂其它功能*  AOP -》 只针对IOC容器对象 - 创建代理对象 -> 将代理对象存储到IOC容器*/
@Component
public class CalculatorImpl implements Calculator {@Overridepublic int add(int i, int j) {int result = i + j;return result;}@Overridepublic int sub(int i, int j) {int result = i - j;return result;}@Overridepublic int mul(int i, int j) {int result = i * j;return result;}@Overridepublic int div(int i, int j) {int result = i / j;return result;}
}
  1. 声明切面类
package com.doug.advice;import org.aspectj.lang.annotation.After;
import org.aspectj.lang.annotation.AfterThrowing;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.springframework.context.annotation.EnableAspectJAutoProxy;
import org.springframework.stereotype.Component;/*** @Description: 增强类 实现增强方法*/
@Component
@Aspect // @Aspect表示这个类是一个切面类
public class LogAspect {/** 1. 定义方法存储增强代码*       具体定义几个方法,根据插入的位置决定!* 2. 使用配置注解 指定插入目标方法的位置*   前置 @Before*   后置 @AfterReturning*   异常 @AfterThrowing*   最后 @After*   环绕 @Around**   try{*       前置*       目标方法执行*       后置*   }catch(){*       异常*   }finally{*       最后*   }** 3.配置切点表达式 [选择要插入的方法  切点* 4.注解补全*   加入IOC容器 @Component*   配置切面 @Aspect* 5.开启Aspect注解的支持* */@Before("execution(* com.doug.aop.impl.*.*(..))")public void start(){System.out.println("方法开始了");}@After("execution(* com.doug.aop.impl.*.*(..))")public void after(){System.out.println("方法结束了");}@AfterThrowing("execution(* com.doug.aop.impl.*.*(..))")public void error(){System.out.println("方法报错了");}
}
  1. 配置类
@Configuration
@ComponentScan("com.doug")
@EnableAspectJAutoProxy // 开启Aspectj的注解  xml: <aop:aspectj-autoproxy />
public class MyConfiguration {
}
  1. 测试
@SpringJUnitConfig(value = MyConfiguration.class)
public class TestAop {@Autowiredprivate Calculator calculator;@Testpublic void aopTest(){int add = calculator.add(1, 0);System.out.println(add);}
}

1

3.3 获取通知细节信息

  • JointPoint接口
    • JoinPoint 接口通过 getSignature() 方法获取目标方法的签名(方法声明时的完整信息)
  • 方法返回值
    • 在返回通知中,通过 @AfterReturning注解的returning属性获取目标方法的返回值!
  • 异常对象捕捉
    • 在异常通知中,通过**@AfterThrowing**注解的throwing属性获取目标方法抛出的异常对象
package com.doug.advice;import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.annotation.*;
import org.springframework.stereotype.Component;import java.lang.reflect.Modifier;/*** @Description: 定义四个增强方法,获取目标方法的信息 返回值 异常*  1. 定义方法 - 增强代码*  2. 使用注解指定对应的位置*  3. 配置切点表达式选中方法*  4. 切面和IOC的配置*  5. 开启aspectj注解的支持*/
@Component
@Aspect
public class MyAdvice {/**   增强方法中获取目标方法信息:**  1. 全部增强方法中,获取目标方法的信息(方法名,参数,访问修饰符,所属类的信息...)*   (JoinPoint joinPoint) import org.aspectj.lang.JoinPoint;**  2. 返回结果 - @AfterReturning*     (Object result) result 接受返回的结果*       @AfterReturning(value = "execution(* com..impl.*.*(..))",returning = "result")** 3.  异常信息 - @AfterThrowing*   (Throwing t) t接收异常信息*   @AfterThrowing(value = "execution(* com..impl.*.*(..))",throwing = "result")* */@Before("execution(* com..impl.*.*(..))")public void start(JoinPoint joinPoint){// 1. 获取方法属于类的信息String simpleName = joinPoint.getTarget().getClass().getSimpleName();// 2. 获取方法名称int modifiers = joinPoint.getSignature().getModifiers();String s = Modifier.toString(modifiers);String name = joinPoint.getSignature().getName();//获取方法名// 3. 获取参数列表Object[] args = joinPoint.getArgs();}@AfterReturning(value = "execution(* com..impl.*.*(..))",returning = "result")public void afterReturn(JoinPoint joinPoint,Object result){}@After("execution(* com..impl.*.*(..))")public void after(){}@AfterThrowing(value = "execution(* com..impl.*.*(..))",throwing = "result")public void afterThrowing(Throwable result){}
}

3.4 切点表达式语法

切点表达式作用
AOP切点表达式(Pointcut Expression)是一种用于指定切点的语言,它可以通过定义匹配规则,来选择需要被切入的目标对象。
1
切点表达式语法
1
语法细节:
1

  • 实战:
1.查询某包某类下,访问修饰符是公有,返回值是int的全部方法execution(public int com.doug.xClass.*(..))
2.查询某包下类中第一个参数是String的方法execution(* com.doug.xClass.*(String..))
3.查询全部包下,无参数的方法!execution(* *..*.*())
4.查询com包下,以int参数类型结尾的方法execution(* com..*.*(..int))
5.查询指定包下,Service开头类的私有返回值int的无参数方法execution(private int com.doug.Service*.*())

3.5 重用(提取)切点表达式

1
提取重复的切点表达式
1
切点统一管理:
将切点表达式统一存储到一个类中进行集中管理和维护!

3.6 环绕通知

三合一(四合一)
1

3.7 切面优先级设置

相同目标方法上同时存在多个切面时,切面的优先级控制切面的内外嵌套顺序。

  • 优先级高的切面:外面
  • 优先级低的切面:里面

使用 @Order 注解可以控制切面的优先级:

  • @Order(较小的数):优先级高
  • @Order(较大的数):优先级低
    1
    实际意义:
    实际开发时,如果有多个切面嵌套的情况,要慎重考虑。例如:如果事务切面优先级高,那么在缓存中命中数据的情况下,事务切面的操作都浪费了。
    1
    此时应该将缓存切面的优先级提高,在事务操作之前先检查缓存中是否存在目标数据。
    1

3.8 CGLib动态代理生效

在目标类没有实现任何接口的情况下,Spring会自动使用cglib技术实现代理。

总结:
a. 如果目标类有接口,选择使用jdk动态代理

b. 如果目标类没有接口,选择cglib动态代理

c. 如果有接口,接口接值

d. 如果没有接口,类进行接值

3.9 注解实现小结

1

四、Spring AOP对获取Bean的影响理解

对实现了接口的类应用切面
在这里插入图片描述
对没实现接口的类应用切面new
1
如果使用AOP技术,目标类有接口,必须使用接口类型接收IoC容器中代理组件!


总结

代理介绍:

1

AOP名词理解:

1

AOP思维以及与代理的关系:

1

相关文章:

【Spring】 AOP面向切面编程

文章目录 AOP是什么&#xff1f;一、AOP术语名词介绍二、Spring AOP框架介绍和关系梳理三、Spring AOP基于注解方式实现和细节3.1 Spring AOP底层技术组成3.2 初步实现3.3 获取通知细节信息3.4 切点表达式语法3.5 重用&#xff08;提取&#xff09;切点表达式3.6 环绕通知3.7 切…...

R语言入门笔记2.6

描述统计 分类数据与顺序数据的图表展示 为了下面代码便于看出颜色参数所对应的值&#xff0c;在这里先集中介绍&#xff0c; col1是黑色&#xff0c;2是粉红&#xff0c;3是绿色&#xff0c;4是天蓝&#xff0c;5是浅蓝&#xff0c;6是紫红&#xff0c;7是黄色&#xff0c;…...

PS人像处理磨皮插件

PS人像处理插件 Portraiture 人像照片进行自动磨皮和平滑处理Arcsoft Portrait 3 自动化人像磨皮软件 批量处理功能DR增强插件 含有磨皮滤镜Beautify Panel 高级质感磨皮插件PT Portrait 人像检测自动完成磨皮优化Retouch4me AI智能人能磨皮美容软件 1、Retouch4me_Heal…...

类型转换(C++)

一、C语言中的类型转换 在C语言中&#xff0c;如果赋值运算符左右两侧类型不同&#xff0c;或者形参与实参类型不匹配&#xff0c;或者返回值类型与 接收返回值类型不一致时&#xff0c;就需要发生类型转化&#xff0c;C语言中总共有两种形式的类型转换&#xff1a;隐式类型 …...

2.23数据结构

单向循环链表 创建单向循环链表&#xff0c;创建节点 &#xff0c;头插&#xff0c;按位置插入&#xff0c;输出&#xff0c;尾删&#xff0c;按位置删除功能 //main.c #include "loop_list.h" int main() {loop_p Hcreate_head();insert_head(H,12);insert_head(…...

c++笔记理解

1.封装 &#xff08;1&#xff09;构造函数不是必须在的 可以通过行为修改属性 &#xff08;2&#xff09;private和protected区别在于继承那里要学 &#xff08;3&#xff09;类默认是私有&#xff0c;struct是共有 私有的好处&#xff1a;控制数据的有效性&#xff0c;意…...

二进制部署k8s集群之cni网络插件

目录 k8s的三种网络模式 pod内容器之间的通信 同一个node节点中pod之间通信 不同的node节点的pod之间通信 flannel网络插件 flannel的三种工作方式 VxLAN host-GW UDP Flannel udp 模式 Flannel VXLAN 模式 flannel插件的三大模式的总结 calico网络插件 k8s 组网…...

二维矩阵子集的最大值

登录—专业IT笔试面试备考平台_牛客网 正好遇到了 对于一维,我们只需要贪一次 int ans -1E9; int suf -1E9; for (int i 0; i < n; i) {if (i && (a[i] - a[i - 1]) % 2 0) {suf 0;}suf std::max(suf, 0) a[i];ans std::max(ans, suf); } ans就是最大值…...

瑞_23种设计模式_装饰者模式

文章目录 1 装饰者模式&#xff08;Decorator Pattern&#xff09;1.1 介绍1.2 概述1.3 装饰者模式的结构 2 案例一2.1 需求2.2 代码实现 3 案例二3.1 需求3.2 代码实现 4 JDK源码解析5 总结5.1 装饰者模式的优缺点5.2 装饰者模式的使用场景5.3 装饰者模式 VS 代理模式 &#x…...

使用Python制作进度条有多少种方法?看这一篇文章就够了!

前言 偶然间刷到一个视频&#xff0c;说到&#xff1a;当程序正在运算时&#xff0c;会有一个较长时间的空白期&#xff0c;谁也不知道程序运行的进度如何&#xff0c;不如给他加个进度条。 于是我今个就搜寻一下&#xff0c;Python版的进度条都可以怎么写&#xff01; 送书…...

SpringBoot-2.7.6基于SLF4J日志门面的日志框架切换

SpringBoot 没有强制性的日志记录依赖项,但 Commons Logging API 除外,它通常由 Spring Framework 的模块提供。 要使用 Logback,您需要将其包含在类路径中。 推荐的方法是您只需要通过启动器,这都取决于 . 对于 Web 应用程序 ,因为它可传递地依赖于日志记录启动器。 如果…...

MongoDB聚合运算符:$binarySize

$binarySize聚合运算符返回给定字符串或二进制数据的字节数。 语法 { $binarySize: <string or binData> }使用 <string or bindData>可以是任何能够被解析为字符串和二进制数据的表达式&#xff1b;如果表达式解析为null&#xff0c;则$binarySize也返回null&a…...

Android的ViewModel

前言 在Compose的学习中&#xff0c;我们在可组合函数中使用rememberSaveable​​​​​​​保存应用数据&#xff0c;但这可能意味着将逻辑保留在可组合函数中或附近。随着应用体量不断变大&#xff0c;您应将数据和逻辑从可组合函数中移出。 而在之前的应用架构学习中&…...

Android 圆环带刻度条进度动画效果实现

效果图 需求是根据传感器做一个重力球效果&#xff0c;先实现了动画后续加上跟传感器联动. 又是摆烂的一天&#xff0c; 尚能呼吸&#xff0c;未来可期啊 View源码 package com.android.circlescalebar.view;import android.content.Context; import android.content.res.Typ…...

94. 二叉树的中序遍历

// 定义一个名为Solution的类&#xff0c;用于解决二叉树的中序遍历问题 class Solution { // 定义一个公共方法&#xff0c;输入是一个二叉树的根节点&#xff0c;返回一个包含中序遍历结果的整数列表 public List<Integer> inorderTraversal(TreeNode root) { // …...

汽车信息安全概述

随着智能网联汽车的迅猛发展&#xff0c;车辆不再是简单的交通工具&#xff0c;而是集数据收集、处理与通信于一体的移动智能终端。然而&#xff0c;这一变革也使得汽车成为黑客攻击的新目标。汽车信息安全问题日益凸显&#xff0c;成为行业关注的焦点。本文将深入探讨汽车信息…...

Linux——基础IO

&#x1f4d8;北尘_&#xff1a;个人主页 &#x1f30e;个人专栏:《Linux操作系统》《经典算法试题 》《C》 《数据结构与算法》 ☀️走在路上&#xff0c;不忘来时的初心 文章目录 一、C语言IO1、写文件2、读文件3、stdin & stdout & stderr 二、系统文件I/O1、写文件…...

数据结构-数组

一,数组基础及注意事项 1,用来储存一组相同的类型的数据. 2,在内存中,分配连续的空姐,数组创建时要指定容量(大小). 3,创建格式: 数据类型 []数组名 int[] arr new int[10] int[] arr2 {1,2,3,4}. 4,索引--访问数组时通过索引进行操作. (注意:一定要理解索引的含义,在数据结…...

【Java程序设计】【C00279】基于Springboot的智慧外贸平台(有论文)

基于Springboot的智慧外贸平台&#xff08;有论文&#xff09; 项目简介项目获取开发环境项目技术运行截图 项目简介 这是一个基于Springboot的智慧外贸平台 本系统分为系统功能模块、管理员功能模块、买家功能模块以及商家功能模块。 系统功能模块&#xff1a;在平台首页可以…...

C#,计算几何,计算机图形学(Computer Graphics)洪水填充算法(Flood Fill Algorithm)与源代码

1 泛洪填充算法(Flood Fill Algorithm) 泛洪填充算法(Flood Fill Algorithm) &#xff0c;又称洪水填充算法&#xff0c;是在很多图形绘制软件中常用的填充算法&#xff0c;最熟悉不过就是 windows 自带画图软件的油漆桶功能。 2 源程序 using System; using System.Collecti…...

AI-调查研究-01-正念冥想有用吗?对健康的影响及科学指南

点一下关注吧&#xff01;&#xff01;&#xff01;非常感谢&#xff01;&#xff01;持续更新&#xff01;&#xff01;&#xff01; &#x1f680; AI篇持续更新中&#xff01;&#xff08;长期更新&#xff09; 目前2025年06月05日更新到&#xff1a; AI炼丹日志-28 - Aud…...

7.4.分块查找

一.分块查找的算法思想&#xff1a; 1.实例&#xff1a; 以上述图片的顺序表为例&#xff0c; 该顺序表的数据元素从整体来看是乱序的&#xff0c;但如果把这些数据元素分成一块一块的小区间&#xff0c; 第一个区间[0,1]索引上的数据元素都是小于等于10的&#xff0c; 第二…...

linux之kylin系统nginx的安装

一、nginx的作用 1.可做高性能的web服务器 直接处理静态资源&#xff08;HTML/CSS/图片等&#xff09;&#xff0c;响应速度远超传统服务器类似apache支持高并发连接 2.反向代理服务器 隐藏后端服务器IP地址&#xff0c;提高安全性 3.负载均衡服务器 支持多种策略分发流量…...

C++初阶-list的底层

目录 1.std::list实现的所有代码 2.list的简单介绍 2.1实现list的类 2.2_list_iterator的实现 2.2.1_list_iterator实现的原因和好处 2.2.2_list_iterator实现 2.3_list_node的实现 2.3.1. 避免递归的模板依赖 2.3.2. 内存布局一致性 2.3.3. 类型安全的替代方案 2.3.…...

QMC5883L的驱动

简介 本篇文章的代码已经上传到了github上面&#xff0c;开源代码 作为一个电子罗盘模块&#xff0c;我们可以通过I2C从中获取偏航角yaw&#xff0c;相对于六轴陀螺仪的yaw&#xff0c;qmc5883l几乎不会零飘并且成本较低。 参考资料 QMC5883L磁场传感器驱动 QMC5883L磁力计…...

vscode(仍待补充)

写于2025 6.9 主包将加入vscode这个更权威的圈子 vscode的基本使用 侧边栏 vscode还能连接ssh&#xff1f; debug时使用的launch文件 1.task.json {"tasks": [{"type": "cppbuild","label": "C/C: gcc.exe 生成活动文件"…...

Golang dig框架与GraphQL的完美结合

将 Go 的 Dig 依赖注入框架与 GraphQL 结合使用&#xff0c;可以显著提升应用程序的可维护性、可测试性以及灵活性。 Dig 是一个强大的依赖注入容器&#xff0c;能够帮助开发者更好地管理复杂的依赖关系&#xff0c;而 GraphQL 则是一种用于 API 的查询语言&#xff0c;能够提…...

NFT模式:数字资产确权与链游经济系统构建

NFT模式&#xff1a;数字资产确权与链游经济系统构建 ——从技术架构到可持续生态的范式革命 一、确权技术革新&#xff1a;构建可信数字资产基石 1. 区块链底层架构的进化 跨链互操作协议&#xff1a;基于LayerZero协议实现以太坊、Solana等公链资产互通&#xff0c;通过零知…...

GruntJS-前端自动化任务运行器从入门到实战

Grunt 完全指南&#xff1a;从入门到实战 一、Grunt 是什么&#xff1f; Grunt是一个基于 Node.js 的前端自动化任务运行器&#xff0c;主要用于自动化执行项目开发中重复性高的任务&#xff0c;例如文件压缩、代码编译、语法检查、单元测试、文件合并等。通过配置简洁的任务…...

【Android】Android 开发 ADB 常用指令

查看当前连接的设备 adb devices 连接设备 adb connect 设备IP 断开已连接的设备 adb disconnect 设备IP 安装应用 adb install 安装包的路径 卸载应用 adb uninstall 应用包名 查看已安装的应用包名 adb shell pm list packages 查看已安装的第三方应用包名 adb shell pm list…...