SPI 原理
文章目录
- SPI
- 使用介绍
- 源码分析
- ServiceLoader.load(Car.class)
- ServiceLoader.iterator()
- LazyIterator.hasNext()
- LazyIterator.next()
- JDBC驱动加载
- 加载MySQL驱动类
- 驱动类注册到DriverManager
- MySQL驱动连接数据库
JDK版本为1.8
SPI
SPI全称Service Provider Interface,是Java提供的一套用来被第三方实现或者扩展的API,它可以用来启用框架扩展和替换组件。
Java SPI 实际上是“基于接口的编程+策略模式+配置文件”组合实现的动态加载机制。
系统设计的各个抽象,往往有很多不同的实现方案,在面向的对象的设计里,一般推荐模块之间基于接口编程,模块之间不对实现类进行硬编码。一旦代码里涉及具体的实现类,就违反了可拔插的原则,如果需要替换一种实现,就需要修改代码。为了实现在模块装配的时候能不在程序里动态指明,这就需要一种服务发现机制。
Java SPI就是提供这样的一个机制:为某个接口寻找服务实现的机制。有点类似IOC的思想,就是将装配的控制权移到程序之外,在模块化设计中这个机制尤其重要。所以SPI的核心思想就是解耦。
使用介绍
要使用Java SPI,需要遵循如下约定:
1、当服务提供者提供了接口的一种具体实现后,在jar包的META-INF/services
目录下创建一个以“接口全限定名”为命名的文件,内容为实现类的全限定名;
2、接口实现类所在的jar包放在主程序的classpath中;
3、主程序通过java.util.ServiceLoder
动态装载实现模块,它通过扫描META-INF/services
目录下的配置文件找到实现类的全限定名,把类加载到JVM;
4、SPI的实现类必须携带一个不带参数的构造方法;
项目结构如下:
src/main
--------javalzc.spidemoapiCar.javaimplBCCar.javaBMCar.javaTest.java
--------resourceMETA-INF.serviceslzc.spidemo.api.Car
Car.java
public interface Car {public String getCarName();
}
BCCar.java
public class BCCar implements Car {@Overridepublic String getCarName() {return "奔驰车";}
}
BMCar.java
public class BMCar implements Car {@Overridepublic String getCarName() {return "宝马车";}
}
META-INF/services/lzc.spidemo.api.Car
在META-INF/services/
下新建一个文件,名字为lzc.spidemo.api.Car
lzc.spidemo.impl.BCCar
lzc.spidemo.impl.BMCar
Test.java
public class Test {public static void main(String[] args) {// ServiceLoader.load(Car.class)ServiceLoader<Car> carList = ServiceLoader.load(Car.class);// ServiceLoader.iterator()Iterator<Car> carIterator = carList.iterator();while (carIterator.hasNext()) { // LazyIterator.hasNext()Car car = carIterator.next(); // LazyIterator.next()System.out.println(car.getCarName());}}
}
根据运行结果可以发现,ServiceLoader.load(Car.class)
可以帮我们找到Car
的实现类BCCar
和BMCar
,接下来通过原来来分析一下它是如何找到Car
的实现类的。
源码分析
Java的SPI机制实现跟ServiceLoader
这个类有关
public final class ServiceLoader<S> implements Iterable<S> {// 读取配置文件的前缀路径 META-INF/services/private static final String PREFIX = "META-INF/services/";// 需要被加载的服务接口或者服务类private final Class<S> service;// 类加载器private final ClassLoader loader;// 创建ServiceLoader时采用的访问控制上下文,默认情况下为 nullprivate final AccessControlContext acc;// 缓存SPI的实现,key是完整类名private LinkedHashMap<String,S> providers = new LinkedHashMap<>();// 当前的迭代器,默认初始化为: new LazyIterator(service, loader)// 这里是懒加载的,只有使用的时候才去迭代,加载// LazyIterator 是 ServiceLoader 的内部类private LazyIterator lookupIterator;
}
可以看到,ServiceLoader
实现了Iterable
接口,覆写其iterator
方法能产生一个迭代器;同时ServiceLoader
有一个内部类LazyIterator
,而LazyIterator
又实现了Iterator
接口,说明LazyIterator
是一个迭代器。
ServiceLoader.load(Car.class)
从ServiceLoader.load(Car.class)
开始分析
// java.util.ServiceLoader
public static <S> ServiceLoader<S> load(Class<S> service) {// 获取当前线程上下文类加载器 AppClassLoaderClassLoader cl = Thread.currentThread().getContextClassLoader();// 把刚才取出的线程上下文类加载器作为参数传入,用于后面去加载classpath中的外部厂商提供的驱动类// 将service接口类和线程上下文类加载器作为参数传入,继续调用load方法return ServiceLoader.load(service, cl);
}
查看ServiceLoader.load(service, cl)
// java.util.ServiceLoader
public static <S> ServiceLoader<S> load(Class<S> service, ClassLoader loader) {// 将service接口类和线程上下文类加载器作为构造参数,创建一个ServiceLoader对象return new ServiceLoader<>(service, loader);
}
查看new ServiceLoader<>(service, loader)
// java.util.ServiceLoader
private ServiceLoader(Class<S> svc, ClassLoader cl) {service = Objects.requireNonNull(svc, "Service interface cannot be null");loader = (cl == null) ? ClassLoader.getSystemClassLoader() : cl;acc = (System.getSecurityManager() != null) ? AccessController.getContext() : null;reload();
}
这里主要是对类属性做赋值操作,然后调用了reload()
方法
查看reload()
方法
// java.util.ServiceLoader
public void reload() {// 清除缓存providers.clear();// 新建 LazyIteratorlookupIterator = new LazyIterator(service, loader);
}
在reload
方法中又新建了一个LazyIterator
对象,然后赋值给lookupIterator
。
// java.util.ServiceLoader
// LazyIterator 是 ServiceLoader的内部类
private class LazyIterator implements Iterator<S> {Class<S> service;ClassLoader loader;Enumeration<URL> configs = null;Iterator<String> pending = null;String nextName = null;private LazyIterator(Class<S> service, ClassLoader loader) {this.service = service;this.loader = loader;}
}
在构建LazyIterator
对象时,只是给其service
和loader
属性赋值,并没有去加载接口的实现类。
ServiceLoader.iterator()
查看ServiceLoader.iterator()
// java.util.ServiceLoader
public Iterator<S> iterator() {// 这里返回的是一个匿名的迭代器对象return new Iterator<S>() {// 将 providers 缓存的数据转换成迭代器 Iterator 并赋值给 knownProvidersIterator<Map.Entry<String,S>> knownProviders= providers.entrySet().iterator();public boolean hasNext() {// 如果缓存中有数据就直接返回trueif (knownProviders.hasNext())return true;// 缓存中没有数据,调用 LazyIterator.hasNext()return lookupIterator.hasNext();}public S next() {// 如果缓存中有数据就直接从缓存中返回if (knownProviders.hasNext())return knownProviders.next().getValue();// 缓存中没有数据,调用 LazyIterator.next()return lookupIterator.next();}public void remove() {throw new UnsupportedOperationException();}};
}
LazyIterator.hasNext()
查看LazyIterator.hasNext()
// java.util.ServiceLoader.LazyIterator
// LazyIterator 是 ServiceLoader 的内部类
public boolean hasNext() {// 默认情况下 acc = nullif (acc == null) {return hasNextService();} else {PrivilegedAction<Boolean> action = new PrivilegedAction<Boolean>() {public Boolean run() { return hasNextService(); }};return AccessController.doPrivileged(action, acc);}
}
查看hasNextService()
方法
// java.util.ServiceLoader.LazyIterator
// LazyIterator 是 ServiceLoader 的内部类
// 这里会涉及到一个名词 "全限定名"
// 这里说的 "全限定名" = "包路径"."类名名称或者是接口名名称"
private boolean hasNextService() {// 如果 nextName 不为空,则直接返回 trueif (nextName != null) {return true;}if (configs == null) {try {// PREFIX = "META-INF/services/"// service.getName() 获取接口的全限定名// 在构建LazyIterator对象时已经给其成员属性service赋值String fullName = PREFIX + service.getName();// 在构建LazyIterator对象时已经给其成员属性loader赋值// 加载 "META-INF/services/接口的全限定名" 文件if (loader == null)configs = ClassLoader.getSystemResources(fullName);elseconfigs = loader.getResources(fullName);} catch (IOException x) {fail(service, "Error locating configuration files", x);}}while ((pending == null) || !pending.hasNext()) {if (!configs.hasMoreElements()) {return false;}// 解析 "META-INF/services/接口的全限定名" 文件内容// 返回 "META-INF/services/接口的全限定名" 文件内容中的服务提供者的全限定名并赋值给pending属性pending = parse(service, configs.nextElement());}// 然后取出一个服务提供者全限定名赋值给LazyIterator的成员变量nextNamenextName = pending.next();return true;
}
这里用到了懒加载的思想,当需要时才去加载配置文件。
该方法的主要功能是:去META-INF/services/
目录下加载接口文件的内容,解析文件内容赋值给LazyIterator
的成员变量pending
,pending
是一个Iterator
,然后取出第一个值赋值给LazyIterator
的成员变量nextName
。
比如META-INF/services/lzc.spidemo.api.Car
的文件内容为。
lzc.spidemo.impl.BCCar
lzc.spidemo.impl.BMCar
那么LazyIterator
的pending
属性就保存了lzc.spidemo.impl.BCCar
和lzc.spidemo.impl.BMCar
这两个值,然后取出lzc.spidemo.impl.BCCar
赋值给LazyIterator
的成员变量nextName
。
执行完LazyIterator
的hasNext
方法后,会继续执行LazyIterator
的next
方法
LazyIterator.next()
查看LazyIterator.next()
方法
// java.util.ServiceLoader.LazyIterator
// LazyIterator 是 ServiceLoader 的内部类
public S next() {// 默认情况下 acc = nullif (acc == null) {return nextService();} else {PrivilegedAction<S> action = new PrivilegedAction<S>() {public S run() { return nextService(); }};return AccessController.doPrivileged(action, acc);}
}
查看LazyIterator.nextService()
方法
// java.util.ServiceLoader.LazyIterator
// LazyIterator 是 ServiceLoader 的内部类
private S nextService() {if (!hasNextService())throw new NoSuchElementException();// hasNextService()方法中为 nextNam e赋值过服务提供者实现类的全限定名String cn = nextName;// 将 nextName 设置为空// 下次调用 hasNextService() 时会从 pending 获取值并赋值给 nextNamenextName = null;Class<?> c = null;try {// 传入类加载器和服务提供者实现类的 全限定名 去加载服务提供者实现类c = Class.forName(cn, false, loader);} catch (ClassNotFoundException x) {fail(service,"Provider " + cn + " not found");}if (!service.isAssignableFrom(c)) {fail(service,"Provider " + cn + " not a subtype");}try {// 实例化加载的服务提供者实现类,并进行转换S p = service.cast(c.newInstance());// 将实例化后的服务提供者实现类放进providers集合进行缓存providers.put(cn, p);// 返回实例return p;} catch (Throwable x) {fail(service,"Provider " + cn + " could not be instantiated",x);}throw new Error(); // This cannot happen
}
LazyIterator.nextService()
利用Class.forName(String name, boolean initialize,ClassLoader loader)
加载服务提供者实现类,然后利用c.newInstance()
实例化,把实例化的类存储到providers
中缓存起来。
JDBC驱动加载
JDBC驱动加载是利用了Java的SPI机制。JDBC提供了一组接口规范,不同的数据库厂商只要编写符合这套JDBC接口规范的驱动代码,那么就可以用Java语言来连接数据库。
以MySQL为例分析JDBC驱动加载源码。
首先需要引入mysql-connector-java
依赖包,版本为8.0.20。mysql-connector-java.jar
包目录下的META-INF/services/
下有一个java.sql.Driver
文件,文件内容如下所示:
com.mysql.cj.jdbc.Driver
加载MySQL驱动类
public class DBUtils {private static String dirverClassName = "com.mysql.cj.jdbc.Driver";private static String url = "jdbc:mysql://127.0.0.1:3306/lzc?characterEncoding=utf8";private static String user = "root";private static String password = "root";public static Connection getDBConnection() {Connection conn = null;
// try {
// // 在JDBC 4.0 规范中可以省略这一步了
// // Class.forName 用来加载类信息并执行类的静态块
// Class.forName(dirverClassName);
// } catch (ClassNotFoundException e) {
// e.printStackTrace();
// }try {// 通过 DriverManager 获取 Connectionconn = DriverManager.getConnection(url, user, password);} catch (SQLException e) {e.printStackTrace();}return conn;}
}
从上面的代码可以发现,通过DriverManager.getConnection(String url,String user, String password)
可以获取Connection
连接信息,此时肯定会执行DriverManager
的静态块代码。
查看java.sql.DriverManager
静态代码:
// java.sql.DriverManager
public class DriverManager {// 用来保存JDBC驱动列表// 就是实现了 java.sql.Driver 接口的类在类加载时会主动注册到 registeredDriversprivate final static CopyOnWriteArrayList<DriverInfo> registeredDrivers = new CopyOnWriteArrayList<>();static {loadInitialDrivers();println("JDBC DriverManager initialized");}
}
查看loadInitialDrivers()
方法
// java.sql.DriverManager
private static void loadInitialDrivers() {String drivers;try {drivers = AccessController.doPrivileged(new PrivilegedAction<String>() {public String run() {return System.getProperty("jdbc.drivers");}});} catch (Exception ex) {drivers = null;}// 重点看这里AccessController.doPrivileged(new PrivilegedAction<Void>() {public Void run() {// 前面已经分析过这里的原理了ServiceLoader<Driver> loadedDrivers = ServiceLoader.load(Driver.class);// 调用 ServiceLoader 的 iterator 方法Iterator<Driver> driversIterator = loadedDrivers.iterator();try{// 在迭代的时候会去加载META-INF/services/java.sql.Driver文件,// 文件内容为 com.mysql.cj.jdbc.Driver// 然后实例化 com.mysql.cj.jdbc.Driverwhile(driversIterator.hasNext()) {driversIterator.next();}} catch(Throwable t) {// Do nothing}return null;}});println("DriverManager.initialize: jdbc.drivers = " + drivers);if (drivers == null || drivers.equals("")) {return;}String[] driversList = drivers.split(":");println("number of Drivers:" + driversList.length);for (String aDriver : driversList) {try {println("DriverManager.Initialize: loading " + aDriver);Class.forName(aDriver, true,ClassLoader.getSystemClassLoader());} catch (Exception ex) {println("DriverManager.Initialize: load failed: " + ex);}}
}
这里主要是利用了Java的SPI机制去实例化MySQL驱动类,下面查看MySQL驱动类是如何注册到DriverManager
类的registeredDrivers
集合中
驱动类注册到DriverManager
查看com.mysql.cj.jdbc.Driver
的静态块代码
public class Driver extends NonRegisteringDriver implements java.sql.Driver {//// Register ourselves with the DriverManager//static {try {// 这里会将 com.mysql.cj.jdbc.Driver 实例注册到// DriverManager类的registeredDrivers集合中java.sql.DriverManager.registerDriver(new Driver());} catch (SQLException E) {throw new RuntimeException("Can't register driver!");}}
}
看到这里可以知道,com.mysql.cj.jdbc.Driver
在实例化的时候会执行它的静态块代码,此时会将自己注册到DriverManager
类的registeredDrivers
集合中。
MySQL驱动连接数据库
com.mysql.cj.jdbc.Driver
已经注册到DriverManager
类的registeredDrivers
集合中,下面查看它是何时被调用的。
通过前面的例子可以知道,通过DriverManager.getConnection(String url,String user, String password)
可以获取Connection
连接信息,查看这个方法的代码:
// java.sql.DriverManager
public static Connection getConnection(String url, String user, String password) throws SQLException {java.util.Properties info = new java.util.Properties();if (user != null) {info.put("user", user);}if (password != null) {info.put("password", password);}return (getConnection(url, info, Reflection.getCallerClass()));
}
继续往下看
// java.sql.DriverManager
private static Connection getConnection(String url, java.util.Properties info, Class<?> caller) throws SQLException {ClassLoader callerCL = caller != null ? caller.getClassLoader() : null;synchronized(DriverManager.class) {// synchronize loading of the correct classloader.if (callerCL == null) {callerCL = Thread.currentThread().getContextClassLoader();}}if(url == null) {throw new SQLException("The url cannot be null", "08001");}println("DriverManager.getConnection(\"" + url + "\")");// Walk through the loaded registeredDrivers attempting to make a connection.// Remember the first exception that gets raised so we can reraise it.SQLException reason = null;// 主要查看这里// 遍历 registeredDrivers 集合for(DriverInfo aDriver : registeredDrivers) {// If the caller does not have permission to load the driver then// skip it.if(isDriverAllowed(aDriver.driver, callerCL)) {try {println(" trying " + aDriver.driver.getClass().getName());// 利用驱动类来连接数据库Connection con = aDriver.driver.connect(url, info);// 只要连接上就直接放回 Connection,如果有多个驱动类,其余的就会被忽略if (con != null) {// Success!println("getConnection returning " + aDriver.driver.getClass().getName());return (con);}} catch (SQLException ex) {if (reason == null) {reason = ex;}}} else {println(" skipping: " + aDriver.getClass().getName());}}// if we got here nobody could connect.if (reason != null) {println("getConnection failed: " + reason);throw reason;}println("getConnection: no suitable driver found for "+ url);throw new SQLException("No suitable driver found for "+ url, "08001");
}
如若内容造成侵权/违法违规/事实不符,请联系编程学习网邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
相关文章
- 【JZ2440笔记】裸机实验点亮LED
一、前言最近在学韦东山的JZ2440开发板,于是记录下学习过程中的笔记。一般学程序写的第一个例子是打印“Hello World”,而学单片机的第一个例子一般都是点亮LED,学ARM的话如果从裸机开始学,也跟玩单片机差不多,从点亮LED开始。二、实验步骤1、目标点亮开发板上的3个LED灯。…...
2024/5/8 15:30:26 - LeetCode 673.最长递增子序列
这道卡了我一天的题目。我自己想的思路是对的,但是没能从纷杂的if else中得到正确的结果,最后还是向官方题解妥协了。 题目 给定一个未排序的整数数组,找到最长递增子序列的个数。 不要求子序列是连续的。 思路 最开始我设想用一维dp数组,dp[i]代表从0-i这一段最长子序列的…...
2024/4/24 12:36:18 - linux编程-文件操作
在Linux系统中,一切皆是文件,当我们初期学习Linux的系统编程的时候,要学会如何用代码来对文件进行操作,实现文件的创建、打开、编辑等Linux系统提供了一系列对于文件操作的API。 文件操作步骤 创建文件/打开文件 注:一定要打开文件成功后才能进行下面以下操作 读取文件/写…...
2024/4/24 12:36:17 - Python基础02:基本数据类型
```python # 1、整型 int age = 18 print(type(age)) # <class int> # 2、浮点型 float salary = 9.9 # <class float> print(type(salary))# 3、字符串类型 str # 定义:用引号(,"", ,""" """")包含的一串字符 …...
2024/4/24 12:36:18 - C语言--函数
使用函数来写连续运算,getch()表示得到一个字符...
2024/5/8 16:48:48 - java 的 锁(附测试源码)
*还是要使用才能暴露问题呀!并发与锁的知识整理目录*还是要使用才能暴露问题呀!并发与锁的知识整理什么是重量级锁?什么是轻量级锁轻量级锁的问题偏向锁与匿名偏向锁什么是偏向锁?为什么会出现偏向锁?啥是匿名偏向锁?偏向锁什么时候升级?关于偏向锁的应用场景1.偏向锁一…...
2024/4/26 22:49:50 - RabbitMq系列(二):最简单的例子
系列文章RabbitMq系列(一):服务器搭建RabbitMq系列(二):最简单的例子目录前言添加依赖创建生产者创建消费者测试其他前言前面已经把RabbitMq服务器搭起来,并且添加了一个基本用户,这里基于服务器构建最基本的RabbitMq程序(一个消费者和一个生产者)。添加依赖这里使用最新(当…...
2024/4/27 5:40:49 - Day07——白盒测试技术
白盒测试基于软件的源代码,已知产品的内部工作过程,主要是对程序内部结构展开测试,关注程序实现的细节。 白盒测试关注的对象包括: (1)源代码,即直接查看源代码,查看代码的规范性,并对照函数功能查找代码的逻辑缺陷、内存管理缺陷、数据定义和使用缺陷等。 (2)程序结…...
2024/5/8 1:59:00 - Kafka - 基本概述&操作
消息队列两种模式点对点(一对一,消费者消费后立刻清除消息)发布、订阅模式(一对多,消费者消费数据之后不会清除消息)基础架构Producer:消息生产者,想kafka broker发送消息的客户端 Consumer:消息消费者,想kafka broker取消息的客户端 Consumer Group(CG):消费者组,多…...
2024/5/8 2:54:42 - Cesium官方英文论坛
Cesium官方刚刚完成了将Google Groups论坛转移到Discourse的工作,Discourse是一个面向在线社区的现代开源平台。这是Cesium社区论坛成立以来最大的一次更新。论坛汇集了近十年的相关GIS的智慧结晶,从基本的故障排除到讨论世界可视化的最佳方式,共有30000多篇文章。 大多数人…...
2024/4/15 4:29:26 - 常见的搜索算法原理及其时间复杂度
常见的搜索算法原理及其时间复杂度 1.顺序查找:最基础的查找方法,对比每一个元素进行查找。在数据量很大的时候效率相当的慢。 数据结构:有序或者无需的队列 时间复杂度:O(n)2.二分查找:二分查找首先要求数组有序 每次查找从中间开始查找,比较查找对象和中间值,如果比中…...
2024/5/7 17:15:43 - Python基础03:垃圾回收机制简述
# 垃圾回收机制# 1、引用计数 x = 10 y=x # 2、标记清除 # 每次全量扫描栈区引用计数,清除计数为0的 # 3、分代回收 # 新生代 青春代 老年代...
2024/4/20 1:39:13 - rocketmq结合webservice的实例分析
1.websocket在接口中被调用。@Controller @RequestMapping("/checkcenter") public class WebSocketController {//推送数据接口@ResponseBody@RequestMapping("/socket/push/{cid}")public String pushToWeb(@PathVariable String cid,String message) {t…...
2024/5/8 8:18:58 - 2007 合成孔径雷达成像算法与实现 各figure的matlab实现(第三章 - 图3.8 )
说明: (1)图3.8大部分程序与图3.6一样,一点不同:增添了tc,将基带信号转化为非基带信号 matlab程序如下: % initial matlab workspace % figure 3.8; page 56 clc clear close all% 设置时间宽度T 和 TBP T = 7.2e-6; TBP = 42;% 计算B和K B = TBP/T; K = B/T;% 设置tc…...
2024/5/8 7:58:03 - 从零整合springMVC(eclipse)
springMVC:表现层框架;从请求中获取参数,将结果返回给前台。请求过程:用户发送请求至前端控制器DispatcherServletDispatcherServlet收到请求调用HandlerMapping处理器映射器。处理器映射器根据请求url找到具体的处理器,生成处理器对象及处理器拦截器(如果有则生成)一并返…...
2024/5/7 18:14:22 - PHP——MySQL笔记(8)之正则表达式
什么是正则表达式? 正则表达式,就是一些规则。通过这些规则,可以从大量的数据中,匹配出自己所需要的数据。 比如说,下面有一个图片链接,我想匹配出这张图片的名称2020060919564022.png: https://img-blog.csdnimg.cn/2020060919564022.png?x-oss-process=image/waterma…...
2024/5/8 2:00:30 - 基于Servlet+EasyUI+Mysql的学生信息系统开源
知识点:jsp servlet,MySQL数据库的基本操作,前端easyui框架。适合人群:Java初学者、在校学生,(已经学过Java基础语法,对html有简单的了解,熟悉js、jquery语法)。用到的工具:eclipse、MySQL首先看下系统的截图dao层的代码如下 所示:package com.ischoolbar.programme…...
2024/4/24 12:36:11 - Message(codeforces)
Message 问题:给字符串A,字符串B。有三种操作:1,在字符串的前端或后端加上一个字符2,删掉前端或后端一个字符,改变字符串中间一个字符。求将一个A的子串变成B最少的操作数。 我们要求A一个子串,和B的子串有着最大匹配 说一下最大匹配:这个可能是和B的子串相同,也有可能…...
2024/4/24 12:36:10 - 如何实现刷脸登录
如何实现刷脸登录 本文主要采用百度AI平台的人脸识别技术实现,原理比较简单,注册面部到人脸库,之后拍照获取照片,在人脸库中查找相似的图片,如果相似度超过设定的阈值,则认为存在这个用户,这里主要介绍下如何获取access_token,并建立自己的人脸库。并且对自己的人脸库进…...
2024/5/8 10:35:47 - JavaScript:11-元素偏移量 offset 系列、元素可视区 client 系列、元素滚动 scroll 系列、动画函数封装
day05 - Web APIs学习目标: 能够说出常见 offset 系列属性的作用 能够说出常见 client 系列属性的作用 能够说出常见 scroll 系列属性的作用 能够封装简单动画函数1.1.元素偏移量 offset 系列 1.1.1 offset 概述 offset 翻译过来就是偏移量, 我们使用 offset系列相关属性可以动…...
2024/4/24 12:36:17
最新文章
- 卡尔曼滤波实战
入门内容 假如有个超声波,它传回的数据是这样的,这样的数据是用不了的 我们想要的是稳定的数据 此时我们引入滤波,把里面的噪声去掉,使得数据更平滑 适用系统: 符合两个特质:线性和高斯。 也叫线性高斯…...
2024/5/8 17:29:58 - 梯度消失和梯度爆炸的一些处理方法
在这里是记录一下梯度消失或梯度爆炸的一些处理技巧。全当学习总结了如有错误还请留言,在此感激不尽。 权重和梯度的更新公式如下: w w − η ⋅ ∇ w w w - \eta \cdot \nabla w ww−η⋅∇w 个人通俗的理解梯度消失就是网络模型在反向求导的时候出…...
2024/5/7 10:36:02 - 利用Sentinel解决雪崩问题(一)
1、解决雪崩问题的常见方式有四种: 超时处理:设定超时时间,请求超过一定时间没有响应就返回错误信息,不会无休止等待;舱壁模式:限定每个业务能使用的线程数,避免耗尽整个tomcat的资源,因此也叫线程隔离;熔断降级:由断路器统计业务…...
2024/5/4 23:53:05 - 自我介绍的HTML 页面(入门)
一.前情提要 1.主要是代码示例,具体内容需自己填充 2.代码后是详解 二.代码实例和解析 代码 <!DOCTYPE html> <html lang"zh-CN"> <head> <meta charset"UTF-8"> <title>自我介绍页面</title>…...
2024/5/5 7:22:10 - 【外汇早评】美通胀数据走低,美元调整
原标题:【外汇早评】美通胀数据走低,美元调整昨日美国方面公布了新一期的核心PCE物价指数数据,同比增长1.6%,低于前值和预期值的1.7%,距离美联储的通胀目标2%继续走低,通胀压力较低,且此前美国一季度GDP初值中的消费部分下滑明显,因此市场对美联储后续更可能降息的政策…...
2024/5/8 6:01:22 - 【原油贵金属周评】原油多头拥挤,价格调整
原标题:【原油贵金属周评】原油多头拥挤,价格调整本周国际劳动节,我们喜迎四天假期,但是整个金融市场确实流动性充沛,大事频发,各个商品波动剧烈。美国方面,在本周四凌晨公布5月份的利率决议和新闻发布会,维持联邦基金利率在2.25%-2.50%不变,符合市场预期。同时美联储…...
2024/5/7 9:45:25 - 【外汇周评】靓丽非农不及疲软通胀影响
原标题:【外汇周评】靓丽非农不及疲软通胀影响在刚结束的周五,美国方面公布了新一期的非农就业数据,大幅好于前值和预期,新增就业重新回到20万以上。具体数据: 美国4月非农就业人口变动 26.3万人,预期 19万人,前值 19.6万人。 美国4月失业率 3.6%,预期 3.8%,前值 3…...
2024/5/4 23:54:56 - 【原油贵金属早评】库存继续增加,油价收跌
原标题:【原油贵金属早评】库存继续增加,油价收跌周三清晨公布美国当周API原油库存数据,上周原油库存增加281万桶至4.692亿桶,增幅超过预期的74.4万桶。且有消息人士称,沙特阿美据悉将于6月向亚洲炼油厂额外出售更多原油,印度炼油商预计将每日获得至多20万桶的额外原油供…...
2024/5/7 14:25:14 - 【外汇早评】日本央行会议纪要不改日元强势
原标题:【外汇早评】日本央行会议纪要不改日元强势近两日日元大幅走强与近期市场风险情绪上升,避险资金回流日元有关,也与前一段时间的美日贸易谈判给日本缓冲期,日本方面对汇率问题也避免继续贬值有关。虽然今日早间日本央行公布的利率会议纪要仍然是支持宽松政策,但这符…...
2024/5/4 23:54:56 - 【原油贵金属早评】欧佩克稳定市场,填补伊朗问题的影响
原标题:【原油贵金属早评】欧佩克稳定市场,填补伊朗问题的影响近日伊朗局势升温,导致市场担忧影响原油供给,油价试图反弹。此时OPEC表态稳定市场。据消息人士透露,沙特6月石油出口料将低于700万桶/日,沙特已经收到石油消费国提出的6月份扩大出口的“适度要求”,沙特将满…...
2024/5/4 23:55:05 - 【外汇早评】美欲与伊朗重谈协议
原标题:【外汇早评】美欲与伊朗重谈协议美国对伊朗的制裁遭到伊朗的抗议,昨日伊朗方面提出将部分退出伊核协议。而此行为又遭到欧洲方面对伊朗的谴责和警告,伊朗外长昨日回应称,欧洲国家履行它们的义务,伊核协议就能保证存续。据传闻伊朗的导弹已经对准了以色列和美国的航…...
2024/5/4 23:54:56 - 【原油贵金属早评】波动率飙升,市场情绪动荡
原标题:【原油贵金属早评】波动率飙升,市场情绪动荡因中美贸易谈判不安情绪影响,金融市场各资产品种出现明显的波动。随着美国与中方开启第十一轮谈判之际,美国按照既定计划向中国2000亿商品征收25%的关税,市场情绪有所平复,已经开始接受这一事实。虽然波动率-恐慌指数VI…...
2024/5/7 11:36:39 - 【原油贵金属周评】伊朗局势升温,黄金多头跃跃欲试
原标题:【原油贵金属周评】伊朗局势升温,黄金多头跃跃欲试美国和伊朗的局势继续升温,市场风险情绪上升,避险黄金有向上突破阻力的迹象。原油方面稍显平稳,近期美国和OPEC加大供给及市场需求回落的影响,伊朗局势并未推升油价走强。近期中美贸易谈判摩擦再度升级,美国对中…...
2024/5/4 23:54:56 - 【原油贵金属早评】市场情绪继续恶化,黄金上破
原标题:【原油贵金属早评】市场情绪继续恶化,黄金上破周初中国针对于美国加征关税的进行的反制措施引发市场情绪的大幅波动,人民币汇率出现大幅的贬值动能,金融市场受到非常明显的冲击。尤其是波动率起来之后,对于股市的表现尤其不安。隔夜美国股市出现明显的下行走势,这…...
2024/5/6 1:40:42 - 【外汇早评】美伊僵持,风险情绪继续升温
原标题:【外汇早评】美伊僵持,风险情绪继续升温昨日沙特两艘油轮再次发生爆炸事件,导致波斯湾局势进一步恶化,市场担忧美伊可能会出现摩擦生火,避险品种获得支撑,黄金和日元大幅走强。美指受中美贸易问题影响而在低位震荡。继5月12日,四艘商船在阿联酋领海附近的阿曼湾、…...
2024/5/4 23:54:56 - 【原油贵金属早评】贸易冲突导致需求低迷,油价弱势
原标题:【原油贵金属早评】贸易冲突导致需求低迷,油价弱势近日虽然伊朗局势升温,中东地区几起油船被袭击事件影响,但油价并未走高,而是出于调整结构中。由于市场预期局势失控的可能性较低,而中美贸易问题导致的全球经济衰退风险更大,需求会持续低迷,因此油价调整压力较…...
2024/5/4 23:55:17 - 氧生福地 玩美北湖(上)——为时光守候两千年
原标题:氧生福地 玩美北湖(上)——为时光守候两千年一次说走就走的旅行,只有一张高铁票的距离~ 所以,湖南郴州,我来了~ 从广州南站出发,一个半小时就到达郴州西站了。在动车上,同时改票的南风兄和我居然被分到了一个车厢,所以一路非常愉快地聊了过来。 挺好,最起…...
2024/5/7 9:26:26 - 氧生福地 玩美北湖(中)——永春梯田里的美与鲜
原标题:氧生福地 玩美北湖(中)——永春梯田里的美与鲜一觉醒来,因为大家太爱“美”照,在柳毅山庄去寻找龙女而错过了早餐时间。近十点,向导坏坏还是带着饥肠辘辘的我们去吃郴州最富有盛名的“鱼头粉”。说这是“十二分推荐”,到郴州必吃的美食之一。 哇塞!那个味美香甜…...
2024/5/4 23:54:56 - 氧生福地 玩美北湖(下)——奔跑吧骚年!
原标题:氧生福地 玩美北湖(下)——奔跑吧骚年!让我们红尘做伴 活得潇潇洒洒 策马奔腾共享人世繁华 对酒当歌唱出心中喜悦 轰轰烈烈把握青春年华 让我们红尘做伴 活得潇潇洒洒 策马奔腾共享人世繁华 对酒当歌唱出心中喜悦 轰轰烈烈把握青春年华 啊……啊……啊 两…...
2024/5/4 23:55:06 - 扒开伪装医用面膜,翻六倍价格宰客,小姐姐注意了!
原标题:扒开伪装医用面膜,翻六倍价格宰客,小姐姐注意了!扒开伪装医用面膜,翻六倍价格宰客!当行业里的某一品项火爆了,就会有很多商家蹭热度,装逼忽悠,最近火爆朋友圈的医用面膜,被沾上了污点,到底怎么回事呢? “比普通面膜安全、效果好!痘痘、痘印、敏感肌都能用…...
2024/5/5 8:13:33 - 「发现」铁皮石斛仙草之神奇功效用于医用面膜
原标题:「发现」铁皮石斛仙草之神奇功效用于医用面膜丽彦妆铁皮石斛医用面膜|石斛多糖无菌修护补水贴19大优势: 1、铁皮石斛:自唐宋以来,一直被列为皇室贡品,铁皮石斛生于海拔1600米的悬崖峭壁之上,繁殖力差,产量极低,所以古代仅供皇室、贵族享用 2、铁皮石斛自古民间…...
2024/5/4 23:55:16 - 丽彦妆\医用面膜\冷敷贴轻奢医学护肤引导者
原标题:丽彦妆\医用面膜\冷敷贴轻奢医学护肤引导者【公司简介】 广州华彬企业隶属香港华彬集团有限公司,专注美业21年,其旗下品牌: 「圣茵美」私密荷尔蒙抗衰,产后修复 「圣仪轩」私密荷尔蒙抗衰,产后修复 「花茵莳」私密荷尔蒙抗衰,产后修复 「丽彦妆」专注医学护…...
2024/5/4 23:54:58 - 广州械字号面膜生产厂家OEM/ODM4项须知!
原标题:广州械字号面膜生产厂家OEM/ODM4项须知!广州械字号面膜生产厂家OEM/ODM流程及注意事项解读: 械字号医用面膜,其实在我国并没有严格的定义,通常我们说的医美面膜指的应该是一种「医用敷料」,也就是说,医用面膜其实算作「医疗器械」的一种,又称「医用冷敷贴」。 …...
2024/5/6 21:42:42 - 械字号医用眼膜缓解用眼过度到底有无作用?
原标题:械字号医用眼膜缓解用眼过度到底有无作用?医用眼膜/械字号眼膜/医用冷敷眼贴 凝胶层为亲水高分子材料,含70%以上的水分。体表皮肤温度传导到本产品的凝胶层,热量被凝胶内水分子吸收,通过水分的蒸发带走大量的热量,可迅速地降低体表皮肤局部温度,减轻局部皮肤的灼…...
2024/5/4 23:54:56 - 配置失败还原请勿关闭计算机,电脑开机屏幕上面显示,配置失败还原更改 请勿关闭计算机 开不了机 这个问题怎么办...
解析如下:1、长按电脑电源键直至关机,然后再按一次电源健重启电脑,按F8健进入安全模式2、安全模式下进入Windows系统桌面后,按住“winR”打开运行窗口,输入“services.msc”打开服务设置3、在服务界面,选中…...
2022/11/19 21:17:18 - 错误使用 reshape要执行 RESHAPE,请勿更改元素数目。
%读入6幅图像(每一幅图像的大小是564*564) f1 imread(WashingtonDC_Band1_564.tif); subplot(3,2,1),imshow(f1); f2 imread(WashingtonDC_Band2_564.tif); subplot(3,2,2),imshow(f2); f3 imread(WashingtonDC_Band3_564.tif); subplot(3,2,3),imsho…...
2022/11/19 21:17:16 - 配置 已完成 请勿关闭计算机,win7系统关机提示“配置Windows Update已完成30%请勿关闭计算机...
win7系统关机提示“配置Windows Update已完成30%请勿关闭计算机”问题的解决方法在win7系统关机时如果有升级系统的或者其他需要会直接进入一个 等待界面,在等待界面中我们需要等待操作结束才能关机,虽然这比较麻烦,但是对系统进行配置和升级…...
2022/11/19 21:17:15 - 台式电脑显示配置100%请勿关闭计算机,“准备配置windows 请勿关闭计算机”的解决方法...
有不少用户在重装Win7系统或更新系统后会遇到“准备配置windows,请勿关闭计算机”的提示,要过很久才能进入系统,有的用户甚至几个小时也无法进入,下面就教大家这个问题的解决方法。第一种方法:我们首先在左下角的“开始…...
2022/11/19 21:17:14 - win7 正在配置 请勿关闭计算机,怎么办Win7开机显示正在配置Windows Update请勿关机...
置信有很多用户都跟小编一样遇到过这样的问题,电脑时发现开机屏幕显现“正在配置Windows Update,请勿关机”(如下图所示),而且还需求等大约5分钟才干进入系统。这是怎样回事呢?一切都是正常操作的,为什么开时机呈现“正…...
2022/11/19 21:17:13 - 准备配置windows 请勿关闭计算机 蓝屏,Win7开机总是出现提示“配置Windows请勿关机”...
Win7系统开机启动时总是出现“配置Windows请勿关机”的提示,没过几秒后电脑自动重启,每次开机都这样无法进入系统,此时碰到这种现象的用户就可以使用以下5种方法解决问题。方法一:开机按下F8,在出现的Windows高级启动选…...
2022/11/19 21:17:12 - 准备windows请勿关闭计算机要多久,windows10系统提示正在准备windows请勿关闭计算机怎么办...
有不少windows10系统用户反映说碰到这样一个情况,就是电脑提示正在准备windows请勿关闭计算机,碰到这样的问题该怎么解决呢,现在小编就给大家分享一下windows10系统提示正在准备windows请勿关闭计算机的具体第一种方法:1、2、依次…...
2022/11/19 21:17:11 - 配置 已完成 请勿关闭计算机,win7系统关机提示“配置Windows Update已完成30%请勿关闭计算机”的解决方法...
今天和大家分享一下win7系统重装了Win7旗舰版系统后,每次关机的时候桌面上都会显示一个“配置Windows Update的界面,提示请勿关闭计算机”,每次停留好几分钟才能正常关机,导致什么情况引起的呢?出现配置Windows Update…...
2022/11/19 21:17:10 - 电脑桌面一直是清理请关闭计算机,windows7一直卡在清理 请勿关闭计算机-win7清理请勿关机,win7配置更新35%不动...
只能是等着,别无他法。说是卡着如果你看硬盘灯应该在读写。如果从 Win 10 无法正常回滚,只能是考虑备份数据后重装系统了。解决来方案一:管理员运行cmd:net stop WuAuServcd %windir%ren SoftwareDistribution SDoldnet start WuA…...
2022/11/19 21:17:09 - 计算机配置更新不起,电脑提示“配置Windows Update请勿关闭计算机”怎么办?
原标题:电脑提示“配置Windows Update请勿关闭计算机”怎么办?win7系统中在开机与关闭的时候总是显示“配置windows update请勿关闭计算机”相信有不少朋友都曾遇到过一次两次还能忍但经常遇到就叫人感到心烦了遇到这种问题怎么办呢?一般的方…...
2022/11/19 21:17:08 - 计算机正在配置无法关机,关机提示 windows7 正在配置windows 请勿关闭计算机 ,然后等了一晚上也没有关掉。现在电脑无法正常关机...
关机提示 windows7 正在配置windows 请勿关闭计算机 ,然后等了一晚上也没有关掉。现在电脑无法正常关机以下文字资料是由(历史新知网www.lishixinzhi.com)小编为大家搜集整理后发布的内容,让我们赶快一起来看一下吧!关机提示 windows7 正在配…...
2022/11/19 21:17:05 - 钉钉提示请勿通过开发者调试模式_钉钉请勿通过开发者调试模式是真的吗好不好用...
钉钉请勿通过开发者调试模式是真的吗好不好用 更新时间:2020-04-20 22:24:19 浏览次数:729次 区域: 南阳 > 卧龙 列举网提醒您:为保障您的权益,请不要提前支付任何费用! 虚拟位置外设器!!轨迹模拟&虚拟位置外设神器 专业用于:钉钉,外勤365,红圈通,企业微信和…...
2022/11/19 21:17:05 - 配置失败还原请勿关闭计算机怎么办,win7系统出现“配置windows update失败 还原更改 请勿关闭计算机”,长时间没反应,无法进入系统的解决方案...
前几天班里有位学生电脑(windows 7系统)出问题了,具体表现是开机时一直停留在“配置windows update失败 还原更改 请勿关闭计算机”这个界面,长时间没反应,无法进入系统。这个问题原来帮其他同学也解决过,网上搜了不少资料&#x…...
2022/11/19 21:17:04 - 一个电脑无法关闭计算机你应该怎么办,电脑显示“清理请勿关闭计算机”怎么办?...
本文为你提供了3个有效解决电脑显示“清理请勿关闭计算机”问题的方法,并在最后教给你1种保护系统安全的好方法,一起来看看!电脑出现“清理请勿关闭计算机”在Windows 7(SP1)和Windows Server 2008 R2 SP1中,添加了1个新功能在“磁…...
2022/11/19 21:17:03 - 请勿关闭计算机还原更改要多久,电脑显示:配置windows更新失败,正在还原更改,请勿关闭计算机怎么办...
许多用户在长期不使用电脑的时候,开启电脑发现电脑显示:配置windows更新失败,正在还原更改,请勿关闭计算机。。.这要怎么办呢?下面小编就带着大家一起看看吧!如果能够正常进入系统,建议您暂时移…...
2022/11/19 21:17:02 - 还原更改请勿关闭计算机 要多久,配置windows update失败 还原更改 请勿关闭计算机,电脑开机后一直显示以...
配置windows update失败 还原更改 请勿关闭计算机,电脑开机后一直显示以以下文字资料是由(历史新知网www.lishixinzhi.com)小编为大家搜集整理后发布的内容,让我们赶快一起来看一下吧!配置windows update失败 还原更改 请勿关闭计算机&#x…...
2022/11/19 21:17:01 - 电脑配置中请勿关闭计算机怎么办,准备配置windows请勿关闭计算机一直显示怎么办【图解】...
不知道大家有没有遇到过这样的一个问题,就是我们的win7系统在关机的时候,总是喜欢显示“准备配置windows,请勿关机”这样的一个页面,没有什么大碍,但是如果一直等着的话就要两个小时甚至更久都关不了机,非常…...
2022/11/19 21:17:00 - 正在准备配置请勿关闭计算机,正在准备配置windows请勿关闭计算机时间长了解决教程...
当电脑出现正在准备配置windows请勿关闭计算机时,一般是您正对windows进行升级,但是这个要是长时间没有反应,我们不能再傻等下去了。可能是电脑出了别的问题了,来看看教程的说法。正在准备配置windows请勿关闭计算机时间长了方法一…...
2022/11/19 21:16:59 - 配置失败还原请勿关闭计算机,配置Windows Update失败,还原更改请勿关闭计算机...
我们使用电脑的过程中有时会遇到这种情况,当我们打开电脑之后,发现一直停留在一个界面:“配置Windows Update失败,还原更改请勿关闭计算机”,等了许久还是无法进入系统。如果我们遇到此类问题应该如何解决呢࿰…...
2022/11/19 21:16:58 - 如何在iPhone上关闭“请勿打扰”
Apple’s “Do Not Disturb While Driving” is a potentially lifesaving iPhone feature, but it doesn’t always turn on automatically at the appropriate time. For example, you might be a passenger in a moving car, but your iPhone may think you’re the one dri…...
2022/11/19 21:16:57