欢迎大家关注我的公众号【老周聊架构】,Java后端主流技术栈的原理、源码分析、架构以及各种互联网高并发、高性能、高可用的解决方案。

一、代理

(1)、什么是代理?

大道理上讲代理是一种软件设计模式,目的地希望能做到代码重用。具体上讲,代理这种设计模式是通过不直接访问被代理对象的方式,而访问被代理对象的方法。这个就好比 商户---->明星经纪人(代理)---->明星这种模式。我们可以不通过直接与明星对话的情况下,而通过明星经纪人(代理)与其产生间接对话。

(2)、什么情况下使用代理?

a.设计模式中有一个设计原则是开闭原则,是说对修改关闭对扩展开放,我们在工作中有时会接手很多前人的代码,里面代码逻辑让人摸不着头脑(sometimes the code is really like shit),这时就很难去下手修改代码,那么这时我们就可以通过代理对类进行增强。

b.我们在使用RPC框架的时候,框架本身并不能提前知道各个业务方要调用哪些接口的哪些方法 。那么这个时候,就可用通过动态代理的方式来建立一个中间人给客户端使用,也方便框架进行搭建逻辑,某种程度上也是客户端代码和框架松耦合的一种表现。

c.Spring的AOP机制就是采用动态代理的机制来实现切面编程。

(3)静态代理和动态代理

我们根据加载被代理类的时机不同,将代理分为静态代理和动态代理。如果我们在代码编译时就确定了被代理的类是哪一个,那么就可以直接使用静态代理;如果不能确定,那么可以使用类的动态加载机制,在代码运行期间加载被代理的类这就是动态代理,比如RPC框架和Spring AOP机制。

二、代理模式

代理模式是面向对象编程中比较常见的设计模式。

这是常见代理模式常见的 UML 示意图。

需要注意的有下面几点:

  • 用户只关心接口功能,而不在乎谁提供了功能。上图中接口是 Subject。
  • 接口真正实现者是上图的 RealSubject,但是它不与用户直接接触,而是通过代理。
  • 代理就是上图中的 Proxy,由于它实现了 Subject 接口,所以它能够直接与用户接触。
  • 用户调用 Proxy 的时候,Proxy 内部调用了 RealSubject。所以,Proxy 是中介者,它可以增强 RealSubject 操作。

如果难于理解的话,我用事例说明好了。值得注意的是,代理可以分为静态代理和动态代理两种。先从静态代理讲起。

三、静态代理

我们平常去电影院看电影的时候,在电影开始的阶段是不是经常会放广告呢?

电影是电影公司委托给影院进行播放的,但是影院可以在播放电影的时候,产生一些自己的经济收益,比如卖爆米花、可乐等,然后在影片开始结束时播放一些广告。

现在用代码来进行模拟。

首先得有一个接口,通用的接口是代理模式实现的基础。这个接口我们命名为 Movie,代表电影播放的能力。

package com.proxy;public interface Movie {void play();
}

然后,我们要有一个真正的实现这个 Movie 接口的类,和一个只是实现接口的代理类。

package com.proxy;public class RealMovie implements Movie {@Overridepublic void play() {System.out.println("您正在观看电影 《盗梦空间》");}}

这个表示真正的影片。它实现了 Movie 接口,play() 方法调用时,影片就开始播放。那么 Proxy 代理呢?

package com.proxy;public class Cinema implements Movie {RealMovie movie;public Cinema(RealMovie movie) {super();this.movie = movie;}@Overridepublic void play() {guanggao(true);movie.play();guanggao(false);}public void guanggao(boolean isStart) {if (isStart) {System.out.println("电影马上开始了,爆米花、可乐、口香糖9.8折,快来买啊!");} else {System.out.println("电影马上结束了,爆米花、可乐、口香糖8.8折,买回家吃吧!");}}}

Cinema 就是 Proxy 代理对象,它有一个 play() 方法。不过调用 play() 方法时,它进行了一些相关利益的处理,那就是广告。现在,我们编写测试代码。

package com.proxy;public class ProxyTest {public static void main(String[] args) {RealMovie realMovie = new RealMovie();Movie movie = new Cinema(realMovie);movie.play();}
}

输出结果:

电影马上开始了,爆米花、可乐、口香糖9.8折,快来买啊!
您正在观看电影 《盗梦空间》
电影马上结束了,爆米花、可乐、口香糖8.8折,买回家吃吧!

现在可以看到,代理模式可以在不修改被代理对象的基础上,通过扩展代理类,进行一些功能的附加与增强。值得注意的是,代理类和被代理类应该共同实现一个接口,或者是共同继承某个类。

上面介绍的是静态代理的内容,为什么叫做静态呢?因为它的类型是事先预定好的,比如上面代码中的 Cinema 这个类。下面要介绍的内容就是动态代理。

四、动态代理

既然是代理,那么它与静态代理的功能与目的是没有区别的,唯一有区别的就是动态与静态的差别。

那么在动态代理的中这个动态体现在什么地方?

上一节代码中 Cinema 类是代理,我们需要手动编写代码让 Cinema 实现 Movie 接口,而在动态代理中,我们可以让程序在运行的时候自动在内存中创建一个实现 Movie 接口的代理,而不需要去定义 Cinema 这个类。这就是它被称为动态的原因。

也许概念比较抽象。现在实例说明一下情况。

假设有一个大商场,商场有很多的柜台,有一个柜台卖茅台酒。我们进行代码的模拟。

package com.proxy;public interface SellWine {void mainJiu();
}

SellWine 是一个接口,你可以理解它为卖酒的许可证。

package com.proxy;public class MaotaiJiu implements SellWine {@Overridepublic void mainJiu() {System.out.println("我卖得是茅台酒。");}}

然后创建一个类 MaotaiJiu,对的,就是茅台酒的意思。

我们还需要一个柜台来卖酒:

package com.proxy;import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;public class GuitaiA implements InvocationHandler {Object brand;public GuitaiA(Object brand) {this.brand = brand;}@Overridepublic Object invoke(Object proxy, Method method, Object[] args) throws Throwable {System.out.println("销售开始  柜台是: "+this.getClass().getSimpleName());method.invoke(brand, args);System.out.println("销售结束");return null;}}

GuitaiA 实现了 InvocationHandler 这个类,这个类是什么意思呢?大家不要慌张,待会我会解释。

然后,我们就可以卖酒了。

package com.proxy;import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Proxy;public class SellWineTest {public static void main(String[] args) {MaotaiJiu maotaiJiu = new MaotaiJiu();InvocationHandler riemann = new GuitaiA(maotaiJiu);SellWine dynamicProxy = (SellWine) Proxy.newProxyInstance(MaotaiJiu.class.getClassLoader(),MaotaiJiu.class.getInterfaces(), riemann);dynamicProxy.mainJiu();}}

输出结果:

销售开始  柜台是: GuitaiA
我卖得是茅台酒。
销售结束

看到没有,我并没有像静态代理那样为 SellWine 接口实现一个代理类,但最终它仍然实现了相同的功能,这其中的差别,就是之前讨论的动态代理所谓“动态”的原因。

五、动态代理语法

动态代码涉及了一个非常重要的类 Proxy。正是通过 Proxy 的静态方法 newProxyInstance 才会动态创建代理。

Proxy

public static Object newProxyInstance(ClassLoader loader,Class<?>[] interfaces,InvocationHandler h)

下面讲解它的 3 个参数意义。

  • loader 自然是类加载器
  • interfaces 代码要用来代理的接口
  • h 一个 InvocationHandler 对象

初学者应该对于 InvocationHandler 很陌生,我马上就讲到这一块。

InvocationHandler

InvocationHandler 是一个接口,官方文档解释说,每个代理的实例都有一个与之关联的 InvocationHandler 实现类,如果代理的方法被调用,那么代理便会通知和转发给内部的 InvocationHandler 实现类,由它决定处理。

public interface InvocationHandler {public Object invoke(Object proxy, Method method, Object[] args)throws Throwable;}

InvocationHandler 内部只是一个 invoke() 方法,正是这个方法决定了怎么样处理代理传递过来的方法调用。

  • proxy 代理对象
  • method 代理对象调用的方法
  • args 调用的方法中的参数

因为,Proxy 动态产生的代理会调用 InvocationHandler 实现类,所以 InvocationHandler 是实际执行者。

public class GuitaiA implements InvocationHandler {private Object pingpai;public GuitaiA(Object pingpai) {this.pingpai = pingpai;}@Overridepublic Object invoke(Object proxy, Method method, Object[] args)throws Throwable {// TODO Auto-generated method stubSystem.out.println("销售开始  柜台是: "+this.getClass().getSimpleName());method.invoke(pingpai, args);System.out.println("销售结束");return null;}}

GuitaiA 就是实际上卖酒的地方。

现在,我们加大难度,我们不仅要卖茅台酒,还想卖五粮液。

package com.proxy;public class Wuliangye implements SellWine {@Overridepublic void mainJiu() {System.out.println("我卖得是五粮液。");}}

Wuliangye 这个类也实现了 SellWine 这个接口,说明它也拥有卖酒的许可证,同样把它放到 GuitaiA 上售卖。

package com.proxy;import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Proxy;public class SellWineTest {public static void main(String[] args) {MaotaiJiu maotaiJiu = new MaotaiJiu();Wuliangye wu = new Wuliangye();InvocationHandler riemann = new GuitaiA(maotaiJiu);InvocationHandler riemann2 = new GuitaiA(wu);SellWine dynamicProxy = (SellWine) Proxy.newProxyInstance(MaotaiJiu.class.getClassLoader(),MaotaiJiu.class.getInterfaces(), riemann);SellWine dynamicProxy2 = (SellWine) Proxy.newProxyInstance(Wuliangye.class.getClassLoader(),Wuliangye.class.getInterfaces(),riemann2);dynamicProxy.mainJiu();dynamicProxy2.mainJiu();}}

输出结果:

销售开始  柜台是: GuitaiA
我卖得是茅台酒。
销售结束
销售开始  柜台是: GuitaiA
我卖得是五粮液。
销售结束

有人会问,dynamicProxy 和 dynamicProxy12有什么区没有?他们都是动态产生的代理,都是售货员,都拥有卖酒的技术证书。

我现在扩大商场的经营,除了卖酒之外,还要卖烟。

首先,同样要创建一个接口,作为卖烟的许可证。

package com.proxy;public interface SellCigarette {void sell();
}

然后,卖什么烟呢?就芙蓉王好了。

public class Furongwang implements SellCigarette {@Overridepublic void sell() {// TODO Auto-generated method stubSystem.out.println("售卖的是正宗的芙蓉王,可以扫描条形码查证。");}}

然后再次测试验证:

package com.proxy;import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Proxy;public class SellWineTest {public static void main(String[] args) {MaotaiJiu maotaiJiu = new MaotaiJiu();Wuliangye wu = new Wuliangye();Furongwang fu = new Furongwang();InvocationHandler riemann = new GuitaiA(maotaiJiu);InvocationHandler riemann2 = new GuitaiA(wu);InvocationHandler riemann3 = new GuitaiA(fu);SellWine dynamicProxy = (SellWine) Proxy.newProxyInstance(MaotaiJiu.class.getClassLoader(),MaotaiJiu.class.getInterfaces(), riemann);SellWine dynamicProxy2 = (SellWine) Proxy.newProxyInstance(Wuliangye.class.getClassLoader(),Wuliangye.class.getInterfaces(),riemann2);SellCigarette dynamicProxy3 = (SellCigarette) Proxy.newProxyInstance(Furongwang.class.getClassLoader(),Furongwang.class.getInterfaces(), jingxiao3);dynamicProxy.mainJiu();dynamicProxy2.mainJiu();dynamicProxy3.sell();}}

输出结果:

销售开始  柜台是: GuitaiA
我卖得是茅台酒。
销售结束
销售开始  柜台是: GuitaiA
我卖得是五粮液。
销售结束
销售开始  柜台是: GuitaiA
售卖的是正宗的芙蓉王,可以扫描条形码查证。
销售结束

结果符合预期。大家仔细观察一下代码,同样是通过 Proxy.newProxyInstance() 方法,却产生了 SellWine 和 SellCigarette 两种接口的实现类代理,这就是动态代理的魔力。

动态代理的秘密

一定有同学对于为什么 Proxy 能够动态产生不同接口类型的代理感兴趣,我的猜测是肯定通过传入进去的接口然后通过反射动态生成了一个接口实例。
比如 SellWine 是一个接口,那么 Proxy.newProxyInstance() 内部肯定会有
new SellWine();

这样相同作用的代码,不过它是通过反射机制创建的。那么事实是不是这样子呢?直接查看它们的源码好了。需要说明的是,我当前查看的源码是 1.8 版本。

public static Object newProxyInstance(ClassLoader loader,Class<?>[] interfaces,InvocationHandler h)throws IllegalArgumentException{Objects.requireNonNull(h);final Class<?>[] intfs = interfaces.clone();/** Look up or generate the designated proxy class.*/Class<?> cl = getProxyClass0(loader, intfs);/** Invoke its constructor with the designated invocation handler.*/try {final Constructor<?> cons = cl.getConstructor(constructorParams);final InvocationHandler ih = h;if (!Modifier.isPublic(cl.getModifiers())) {AccessController.doPrivileged(new PrivilegedAction<Void>() {public Void run() {cons.setAccessible(true);return null;}});}return cons.newInstance(new Object[]{h});} catch (IllegalAccessException|InstantiationException e) {throw new InternalError(e.toString(), e);} catch (InvocationTargetException e) {Throwable t = e.getCause();if (t instanceof RuntimeException) {throw (RuntimeException) t;} else {throw new InternalError(t.toString(), t);}} catch (NoSuchMethodException e) {throw new InternalError(e.toString(), e);}}

newProxyInstance 的确创建了一个实例,它是通过 cl 这个 Class 文件的构造方法反射生成。cl 由 getProxyClass0() 方法获取。

private static Class<?> getProxyClass0(ClassLoader loader,Class<?>... interfaces) {if (interfaces.length > 65535) {throw new IllegalArgumentException("interface limit exceeded");}// If the proxy class defined by the given loader implementing// the given interfaces exists, this will simply return the cached copy;// otherwise, it will create the proxy class via the ProxyClassFactoryreturn proxyClassCache.get(loader, interfaces);
}

直接通过缓存获取,如果获取不到,注释说会通过 ProxyClassFactory 生成。

/*** A factory function that generates, defines and returns the proxy class given* the ClassLoader and array of interfaces.*/private static final class ProxyClassFactoryimplements BiFunction<ClassLoader, Class<?>[], Class<?>>{// Proxy class 的前缀是 “$Proxy”,private static final String proxyClassNamePrefix = "$Proxy";// next number to use for generation of unique proxy class namesprivate static final AtomicLong nextUniqueNumber = new AtomicLong();@Overridepublic Class<?> apply(ClassLoader loader, Class<?>[] interfaces) {Map<Class<?>, Boolean> interfaceSet = new IdentityHashMap<>(interfaces.length);for (Class<?> intf : interfaces) {/** Verify that the class loader resolves the name of this* interface to the same Class object.*/Class<?> interfaceClass = null;try {interfaceClass = Class.forName(intf.getName(), false, loader);} catch (ClassNotFoundException e) {}if (interfaceClass != intf) {throw new IllegalArgumentException(intf + " is not visible from class loader");}/** Verify that the Class object actually represents an* interface.*/if (!interfaceClass.isInterface()) {throw new IllegalArgumentException(interfaceClass.getName() + " is not an interface");}/** Verify that this interface is not a duplicate.*/if (interfaceSet.put(interfaceClass, Boolean.TRUE) != null) {throw new IllegalArgumentException("repeated interface: " + interfaceClass.getName());}}String proxyPkg = null;     // package to define proxy class inint accessFlags = Modifier.PUBLIC | Modifier.FINAL;/** Record the package of a non-public proxy interface so that the* proxy class will be defined in the same package.  Verify that* all non-public proxy interfaces are in the same package.*/for (Class<?> intf : interfaces) {int flags = intf.getModifiers();if (!Modifier.isPublic(flags)) {accessFlags = Modifier.FINAL;String name = intf.getName();int n = name.lastIndexOf('.');String pkg = ((n == -1) ? "" : name.substring(0, n + 1));if (proxyPkg == null) {proxyPkg = pkg;} else if (!pkg.equals(proxyPkg)) {throw new IllegalArgumentException("non-public interfaces from different packages");}}}if (proxyPkg == null) {// if no non-public proxy interfaces, use com.sun.proxy packageproxyPkg = ReflectUtil.PROXY_PACKAGE + ".";}/** Choose a name for the proxy class to generate.*/long num = nextUniqueNumber.getAndIncrement();String proxyName = proxyPkg + proxyClassNamePrefix + num;/** Generate the specified proxy class.*/byte[] proxyClassFile = ProxyGenerator.generateProxyClass(proxyName, interfaces, accessFlags);try {return defineClass0(loader, proxyName,proxyClassFile, 0, proxyClassFile.length);} catch (ClassFormatError e) {/** A ClassFormatError here means that (barring bugs in the* proxy class generation code) there was some other* invalid aspect of the arguments supplied to the proxy* class creation (such as virtual machine limitations* exceeded).*/throw new IllegalArgumentException(e.toString());}}}

这个类的注释说,通过指定的 ClassLoader 和 接口数组 用工厂方法生成 proxy class。 然后这个 proxy class 的名字是:

// Proxy class 的前缀是 “$Proxy”,
private static final String proxyClassNamePrefix = "$Proxy";long num = nextUniqueNumber.getAndIncrement();String proxyName = proxyPkg + proxyClassNamePrefix + num;

所以,动态生成的代理类名称是包名+$Proxy+id序号。

生成的过程,核心代码如下:

byte[] proxyClassFile = ProxyGenerator.generateProxyClass(proxyName, interfaces, accessFlags);return defineClass0(loader, proxyName,proxyClassFile, 0, proxyClassFile.length);

这两个方法,我没有继续追踪下去,defineClass0() 甚至是一个 native 方法。我们只要知道,动态创建代理这回事就好了。

现在我们还需要做一些验证,我要检测一下动态生成的代理类的名字是不是包名+$Proxy+id序号。

package com.proxy;import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Proxy;public class SellWineTest {public static void main(String[] args) {MaotaiJiu maotaiJiu = new MaotaiJiu();Wuliangye wu = new Wuliangye();Furongwang fu = new Furongwang();InvocationHandler riemann = new GuitaiA(maotaiJiu);InvocationHandler riemann2 = new GuitaiA(wu);InvocationHandler riemann3 = new GuitaiA(fu);SellWine dynamicProxy = (SellWine) Proxy.newProxyInstance(MaotaiJiu.class.getClassLoader(),MaotaiJiu.class.getInterfaces(), riemann);SellWine dynamicProxy2 = (SellWine) Proxy.newProxyInstance(Wuliangye.class.getClassLoader(),Wuliangye.class.getInterfaces(),riemann2);SellCigarette dynamicProxy3 = (SellCigarette) Proxy.newProxyInstance(Furongwang.class.getClassLoader(),Furongwang.class.getInterfaces(), jingxiao3);dynamicProxy.mainJiu();dynamicProxy2.mainJiu();dynamicProxy3.sell();System.out.println("dynamicProxy class name:"+dynamicProxy.getClass().getName());System.out.println("dynamicProxy1 class name:"+dynamicProxy2.getClass().getName());System.out.println("dynamicProxy3 class name:"+dynamicProxy3.getClass().getName());}}

输出结果:

销售开始  柜台是: GuitaiA
我卖得是茅台酒。
销售结束
销售开始  柜台是: GuitaiA
我卖得是五粮液。
销售结束
销售开始  柜台是: GuitaiA
售卖的是正宗的芙蓉王,可以扫描条形码查证。
销售结束dynamicProxy class name:com.sun.proxy.$Proxy0
dynamicProxy2 class name:com.sun.proxy.$Proxy0
dynamicProxy3 class name:com.sun.proxy.$Proxy1

SellWine 接口的代理类名是:com.sun.proxy.Proxy0SellCigarette接口的代理类名是:com.sun.proxy.Proxy0 SellCigarette 接口的代理类名是:com.sun.proxy.Proxy0SellCigarette接口的代理类名是:com.sun.proxy.Proxy1

这说明动态生成的 proxy class 与 Proxy 这个类同一个包。

下面用一张图让大家记住动态代理涉及到的角色。

红框中 $Proxy0就是通过 Proxy 动态生成的。
$Proxy0实现了要代理的接口。
$Proxy0通过调用 InvocationHandler来执行任务。

六、总结

  • 代理分为静态代理和动态代理两种。
  • 静态代理,代理类需要自己编写代码写成。
  • 动态代理,代理类通过 Proxy.newInstance() 方法生成。
  • 不管是静态代理还是动态代理,代理与被代理者都要实现两样接口,它们的实质是面向接口编程。
  • 静态代理和动态代理的区别是在于要不要开发者自己定义 Proxy 类。
  • 动态代理通过 Proxy 动态生成 proxy class,但是它也指定了一个 InvocationHandler 的实现类。
  • 代理模式本质上的目的是为了增强现有代码的功能。

java动态代理实现与原理详细分析相关推荐

  1. java动态代理实现与原理详细分析(代码层面解释了AOP的实现)

    关于Java中的动态代理,我们首先需要了解的是一种常用的设计模式--代理模式,而对于代理,根据创建代理类的时间点,又可以分为静态代理和动态代理. 一.代理模式    代理模式是常用的java设计模式, ...

  2. java动态代理实现与原理详细分析(【转载】By--- Gonjan )

    [转载]By---    Gonjan  关于Java中的动态代理,我们首先需要了解的是一种常用的设计模式--代理模式,而对于代理,根据创建代理类的时间点,又可以分为静态代理和动态代理. 一.代理模式 ...

  3. (转)java动态代理实现与原理详细分析

    https://www.cnblogs.com/gonjan-blog/p/6685611.html 1.静态代理 静态代理:由程序员创建或特定工具自动生成源代码,也就是在编译时就已经将接口,被代理类 ...

  4. 【java】java动态代理实现与原理详细分析

    文章目录 一.代理模式 二.静态代理 1.静态代理 2.静态代理简单实现 三.动态代理 1.动态代理 2.动态代理简单实现 四.动态代理原理分析 五.总结 关于Java中的动态代理,我们首先需要了解的 ...

  5. class转java_java动态代理实现与原理详细分析

    关于Java中的动态代理,我们首先需要了解的是一种常用的设计模式--代理模式,而对于代理,根据创建代理类的时间点,又可以分为静态代理和动态代理. 一.代理模式    代理模式是常用的java设计模式, ...

  6. java动态代理实现与原理

    java动态代理实现与原理详细分析 关于Java中的动态代理,我们首先需要了解的是一种常用的设计模式–代理模式,而对于代理,根据创建代理类的时间点,又可以分为静态代理和动态代理. 一.代理模式 代理模 ...

  7. 高级JAVA - 动态代理的实现原理和源码分析

    在之前的一篇文章中 , 我们简单了解了一下代理模式(JAVA设计模式 - 代理模式) , 本篇我们来学习一下动态代理的实现原理 , 以及源码是怎样的 . JDK动态代理的主要实现步骤如下 : 1 . ...

  8. Java动态代理的实现原理

    概述 AOP用到了两种动态代理来实现织入功能: jdk动态代理 cglib动态代理 比较: jdk动态代理是由java内部的反射机制来实现的,cglib动态代理底层则是借助asm来实现的.反射机制在生 ...

  9. 02.Java动态代理实现与原理分析之静态代理

    一.代理模式 代理模式是常用的java设计模式,他的特征是代理类与委托类有同样的接口,代理类主要负责为委托类预处理消息.过滤消息.把消息转发给委托类,以及事后处理消息等.代理类与委托类之间通常会存在关 ...

最新文章

  1. NetBeans 时事通讯(刊号 # 4 - Apr 22, 2008)
  2. 优秀学生是如何高效利用时间的?
  3. Jenkins中安装Credentials Binding插件实现凭证管理与安装Git插件和拉取代码构建项目
  4. eclipse 在 Linux中常用命令,持续更新....
  5. mysql用sql语句怎么做个脚本备份_mysql备份脚本
  6. Comet OJ(Contest #8)-D菜菜种菜【树状数组,指针】
  7. linux tune2fs命令详解
  8. android 中断处理流程,Android P的native crash处理流程
  9. 杭电oj首字母变大写
  10. 百科知识 已知三角形三条边长,如何求解三角形的面积
  11. Linux系统重要日志文件
  12. 计算器c语言源代码全,C语言的计算器源代码
  13. pil库修改图片大小_利用pillow库(PIL)批量修改图片尺寸
  14. Android——打电话(选择指定手机卡)、发短信
  15. 阿里云6·18新玩法上线:邀好友送天猫超市卡 更有机会赢得iPhone 12 Pro!
  16. 视频号|常见违规限流情况及解除方法
  17. 芙拉机器人_【诉说实情】芙拉2020新品女士小号斜挎包心形印花怎么样?别花冤枉钱,内幕大揭秘 | 智能扫地机器人评测...
  18. 2022年618活动4000价位笔记本推荐
  19. useful eclipse plugins
  20. winform 三层(BLL.DAL.MODEL)

热门文章

  1. 三言两语 (不定时更)
  2. jitter单位_Jitter知识
  3. webrtc jitterbuffer 模块分析
  4. 服务注册eureka上显示ip地址出现的问题
  5. 守护云原生安全,青藤让浙江移动“心里更有底”
  6. C++设计模式——桥接模式(高屋建瓴)
  7. 三天2亿游玩人次,近期爆火的「人生重开模拟器」,好玩在哪?
  8. Android设计模式-迭代器模式
  9. 纽约时报称刘强东卷入悉尼嫩模性侵案 京东回应:深表同情
  10. matlab程序 地震 相干噪声_基于 matlab 的背景噪声计算程序的设计与应用