`
xinklabi
  • 浏览: 1560189 次
  • 性别: Icon_minigender_1
  • 来自: 吉林
文章分类
社区版块
存档分类
最新评论

探索java类加载原理(1)

    博客分类:
  • Java
阅读更多

声明:转自java中文网 转载请注明!

原文请见:http://www.java-cn.com/club/html/96/n-496.html

 

 

类加载是java语言提供的最强大的机制之一。尽管类加载并不是讨论的热点话题,但所有的编程人员都应该了解其工作机制,明白如何做才能让其满足我们的需要。这能有效节省我们的编码时间,从不断调试ClassNotFoundException, ClassCastException的工作中解脱出来。 JAVA中文站社区门户?

这篇文章从基础讲起,比如代码与数据的不同之处是什么,他们是如何构成一个实例或对象的。然后深入探讨java虚拟机(JVM)是如何利用类加载器读取代码,以及java中类加载器的主要类型。接着用一个类加载的基本算法看一下类加载器如何加载一个内部类。本文的下一节演示一段代码来说明扩展和开发属于自己的类加载器的必要性。紧接着解释如何使用定制的类加载器来完成一个一般意义上的任务,使其可以加载任意远端客户的代码,在JVM中定义,实例化并执行它。本文包括了J2EE关于类加载的规范??事实上这已经成为了J2EE的标准之一。

  一个类代表要执行的代码,而数据则表示其相关状态。状态时常改变,而代码则不会。当我们将一个特定的状态与一个类相对应起来,也就意味着将一个类事例化。尽管相同的类对应的实例其状态千差万别,但其本质都对应着同一段代码。在JAVA中,一个类通常有着一个.class文件,但也有例外。在JAVA的运行时环境中(Java runtime),每一个类都有一个以第一类(first-class)Java对象所表现出现的代码,其是java.lang.Class的实例。我们编译一个JAVA文件,编译器都会嵌入一个public, static, final修饰的类型为java.lang.Class,名称为class的域变量在其字节码文件中。因为使用了public修饰,我们可以采用如下的形式对其访问:
java.lang.Class klass = Myclass.class;
一旦一个类被载入JVM中,同一个类就不会被再次载入了(切记,同一个类)。这里存在一个问题就是什么是同一个类?正如一个对象有一个具体的状态,即标识,一个对象始终和其代码()相关联。同理,载入JVM的类也有一个具体的标识,我们接下来看。JAVA中文站社区门户)n+y/y|0]

  Java中,一个类用其完全匹配类名(fully qualified class name)作为标识,这里指的完全匹配类名包括包名和类名。但在JVM中一个类用其全名和一个加载类ClassLoader的实例作为唯一标识。因此,如果一个名为Pg的包中,有一个名为Cl的类,被类加载器KlassLoader的一个实例kl1加载,Cl的实例,即C1.classJVM中表示为(Cl, Pg, kl1)。这意味着两个类加载器的实例(Cl, Pg, kl1)  (Cl, Pg, kl2)是不同的,被它们所加载的类也因此完全不同,互不兼容的。那么在JVM中到底有多少种类加载器的实例?下一节我们揭示答案。

  在JVM中,每一个类都被java.lang.ClassLoader的一些实例来加载.ClassLoader是在包中java.lang里,开发者可以自由地继承它并添加自己的功能来加载类。
  无论何时我们键入java MyMainClass来开始运行一个新的JVM引导类加载器(bootstrap class loader)”负责将一些关键的Java类,如java.lang.Object和其他一些运行时代码先加载进内存中。运行时的类在JRE\lib\rt.jar包文件中。因为这属于系统底层执行动作,我们无法在JAVA文档中找到引导类加载器的工作细节。基于同样的原因,引导类加载器的行为在各JVM之间也是大相径庭。
同理,如果我们按照如下方式:

log(java.lang.String.class.getClassLoader());

  接下来介绍java的扩展类加载器。扩展库提供比java运行代码更多的特性,我们可以把扩展库保存在由java.ext.dirs属性提供的路径中。
(编辑注:java.ext.dirs属性指的是系统属性下的一个key,所有的系统属性可以通过System.getProperties()方法获得。在编者的系统中,java.ext.dirsvalue” C:\Program Files\Java\jdk1.5.0_04\jre\lib\ext”。下面将要谈到的如java.class.path也同属系统属性的一个key) 类ExtClassLoader专门用来加载所有java.ext.dirs下的.jar文件。开发者可以通过把自己的.jar文件或库文件加入到扩展目录的classpath,使其可以被扩展类加载器读取。

  从开发者的角度,第三种同样也是最重要的一种类加载器是AppClassLoader。这种类加载器用来读取所有的对应在java.class.path系统属性的路径下的类。

  Sunjava指南中,文章理解扩展类加载Understanding Extension Class Loading)对以上三个类加载器路径有更详尽的解释,这是其他几个JDK中的类加载器
  ●java.net.URLClassLoader 
  ●java.security.SecureClassLoader JAVA中文站社区门户6CzAsS!c:z2u W
  ●java.rmi.server.RMIClassLoader JAVA中文站社区门户 iU5k_%EP!uE­\

  java.lang.Thread,包含了public ClassLoader getContextClassLoader()方法,这一方法返回针对一具体线程的上下文环境类加载器。此类加载器由线程的创建者提供,以供此线程中运行的代码在需要加载类或资源时使用。如果此加载器未被建立,缺省是其父线程的上下文类加载器。原始的类加载器一般由读取应用程序的类加载器建立。

  除了引导类加载器,所有的类加载器都有一个父类加载器,不仅如此,所有的类加载器也都是java.lang.ClassLoader类型。以上两种类加载器是不同的,而且对于开发者自订制的类加载器的正常运行也至关重要。最重要的方面是正确设置父类加载器。任何类加载器,其父类加载器是加载该类加载器的类加载器实例。(记住,类加载器本身也是一个类!)

  使用loadClass()方法可以从类加载器中获得该类。我们可以通过java.lang.ClassLoader的源代码来了解该方法工作的细节,如下:
protected synchronized Class loadClass (String name, boolean resolve) throws ClassNotFoundException{
 

// First check if the class is already loaded  

Class c = findLoadedClass(name);  

if (c == null)  {  

        try {   

            if (parent != null){

                     c = parent.loadClass(name, false);

           } else {

                    c = findBootstrapClass0(name);

              }catch (ClassNotFoundException e) {    

          // If still not found, then invoke // findClass to find the class.    

       c = findClass(name);  

       } 

    }if (resolve) {  

          resolveClass(c); 

    } 

  return c;

}
 我们可以使用ClassLoader的两种构造方法来设置父类加载器:
public class MyClassLoader extends ClassLoader{ public MyClassLoader() {  super(MyClassLoader.class.getClassLoader()); }}

public class MyClassLoader extends ClassLoader{
 public MyClassLoader() {  super(getClass().getClassLoader()); }}

 

  第一种方式较为常用,因为通常不建议在构造方法里调用getClass()方法,因为对象的初始化只是在构造方法的出口处才完全完成。因此,如果父类加载器被正确建立,当要示从一个类加载器的实例获得一个类时,如果它不能找到这个类,它应该首先去访问其父类。如果父类不能找到它(即其父类也不能找不这个类,等等),而且如果findBootstrapClass0()方法也失败了,则调用findClass()方法。findClass()方法的缺省实现会抛出ClassNotFoundException,当它们继承java.lang.ClassLoader来订制类加载器时开发者需要实现这个方法。findClass()的缺省实现方式如下:

protected Class findClass(String name) throws ClassNotFoundException { throw new ClassNotFoundException(name); }
  在findClass()方法内部,类加载器需要获取任意来源的字节码。来源可以是文件系统,URL,数据库,可以产生字节码的另一个应用程序,及其他类似的可以产生java规范的字节码的来源。你甚至可以使用BCEL (Byte Code Engineering Library:字节码工程库),它提供了运行时创建类的捷径。BCEL已经被成功地使用在以下方面:编译器,优化器,混淆器,代码产生器及其他分析工具。一旦字节码被检索,此方法就会调用defineClass()方法,此行为对不同的类加载实例是有差异的。因此,如果两个类加载实例从同一个来源定义一个类,所定义的结果是不同的。

  JAVA语言规范(Java language specification)详细解释了JAVA执行引擎中的类或接口的加载(loading),链接(linking)或初始化(initialization)过程。
  图一显示了一个主类称为MyMainClass的应用程序。依照之前的阐述,MyMainClass.class会被AppClassLoader加载。 MyMainClass创建了两个类加载器的实例:CustomClassLoader1  CustomClassLoader2,他们可以从某数据源(比如网络)获取名为Target的字节码。这表示类Target的类定义不在应用程序类路径或扩展类路径。在这种情况下,如果MyMainClass想要用自定义的类加载器加载Target类,CustomClassLoader1CustomClassLoader2会分别独立地加载并定义Target.class类。这在java中有重要的意义。如果Target类有一些静态的初始化代码,并且假设我们只希望这些代码在JVM中只执行一次,而这些代码在我们目前的步骤中会执行两次??分别被不同的CustomClassLoaders加载并执行。如果类Target被两个CustomClassLoaders加载并创建两个实例Target1Target2,如图一显示,它们不是类型兼容的。换句话说,在JVM中无法执行以下代码:

 

Target target3 = (Target) target2;

 

  以上代码会抛出一个ClassCastException。这是因为JVM把他们视为分别不同的类,因为他们被不同的类加载器所定义。这种情况当我们不是使用两个不同的类加载器CustomClassLoader1  CustomClassLoader2,而是使用同一个类加载器CustomClassLoader的不同实例时,也会出现同样的错误。这些会在本文后边用具体代码说明。
1. 在同一个JVM中多个类加载器加载同一个目标类

为什么我们需要我们自己的类加载器
  原因之一为开发者写自己的类加载器来控制JVM中的类加载行为,java中的类靠其包名和类名来标识,对于实现了java.io.Serializable接口的类,serialVersionUID扮演了一个标识类版本的重要角色。这个唯一标识是一个类名、接口名、成员方法及属性等组成的一个64位的哈希字段,而且也没有其他快捷的方式来标识一个类的版本。严格说来,如果以上的都匹配,那么则属于同一个类。

 

 

 

 

  但是让我们思考如下情况:我们需要开发一个通用的执行引擎。可以执行实现某一特定接口的任何任务。当任务被提交到这个引擎,首先需要加载这个任务的代码。假设不同的客户对此引擎提交了不同的任务,凑巧,这些所有的任务都有一个相同的类名和包名。现在面临的问题就是这个引擎是否可以针对不同的用户所提交的信息而做出不同的反应。这一情况在下文的参考一节有可供下载的代码样例,samepath  differentversions,这两个目录分别演示了这一概念。 显示了文件目录结构,有三个子目录samepath, differentversions,  differentversionspush,里边是例子:


 在samepath 中,类version.Version保存在v1v2两个子目录里,两个类具有同样的类名和包名,唯一不同的是下边这行:{
public void fx(){ log("this = " + this + "; Version.fx(1)."); }


  V1中,日志记录中有Version.fx(1),而在v2中则是Version.fx(2)。把这个两个存在细微不同的类放在一个classpath下,然后运行Test类:

 


set CLASSPATH=.;%CURRENT_ROOT%\v1;%CURRENT_ROOT%\v2%JAVA_HOME%\bin\java Test

 




 图3显示了控制台输出。我们可以看到对应着Version.fx(1)的代码被执行了,因为类加载器在classpath首先看到此版本的代码。

 3. 在类路径中samepath测试排在最前面的version 1

  再次运行,类路径做如下微小改动。 

set CLASSPATH=.;%CURRENT_ROOT%\v2;%CURRENT_ROOT%\v1%JAVA_HOME%\bin\java Test
  控制台的输出变为图4。对应着Version.fx(2)的代码被加载,因为类加载器在classpath中首先找到它的路径。
4. 在类路径中samepath测试排在最前面的version 2

 


  根据以上例子可以很明显地看出,类加载器加载在类路径中被首先找到的元素。如果我们在v1v2中删除了version.Version,做一个非version.Version形式的.jar文件,如myextension.jar,把它放到对应java.ext.dirs的路径下,再次执行后看到version.Version不再被AppClassLoader加载,而是被扩展类加载器加载。如图5所示。
 
5. AppClassLoaderExtClassLoader



  继续这个例子,文件夹differentversions包含了一个RMI执行引擎,客户端可以提供给执行引擎任何实现了common.TaskIntf接口的任务。子文件夹client1  client2包含了类client.TaskImpl有个细微不同的两个版本。两个类的区别在以下几行:


static{
 log("client.TaskImpl.class.getClassLoader (v1) : " + TaskImpl.class.getClassLoader());}public void execute(){ log("this = " + this + "; execute(1)"); }

  在client1client2里分别有getClassLoader(v1)  execute(1)getClassLoader(v2)  execute(2)的的log语句。并且,在开始执行引擎RMI服务器的代码中,我们随意地将client2的任务实现放在类路径的前面。
-W9qf6rXY6iCLASSPATH=%CURRENT_ROOT%\common;%CURRENT_ROOT%\server;%CURRENT_ROOT%\client2;%CURRENT_ROOT%\client1%JAVA_HOME%\bin\java server.Server

  如图678的屏幕截图,在客户端VM,各自的client.TaskImpl类被加载、实例化,并发送到服务端的VM来执行。从服务端的控制台,可以明显看到client.TaskImpl代码只被服务端的VM执行一次,这个单一的代码版本在服务端多次生成了许多实例,并执行任务。

  图6显示了服务端的控制台,加载并执行两个不同的客户端的请求,如图78所示。需要注意的是,代码只被加载了一次(从静态初始化块的日志中也可以明显看出),但对于客户端的调用这个方法被执行了两次。
 
7. 执行引擎客户端 1控制台 


  图7中,客户端VM加载了含有client.TaskImpl.class.getClassLoader(v1)的日志内容的类TaskImpl的代码,并提供给服务端的执行引擎。图8的客户端VM加载了另一个TaskImpl的代码,并发送给服务端。
8. 执行引擎客户端 2控制台 

  在客户端的VM中,类client.TaskImpl被分别加载,初始化,并发送到服务端执行。图6还揭示了client.TaskImpl的代码只在服务端的VM中加载了一次,但这唯一的一次却在服务端创造了许多实例并执行。或许客户端1该不高兴了因为并不是它的client.TaskImpl(v1)的方法调用被服务端执行了,而是其他的一些代码。如何解决这一问题?答案就是实现定制的类加载器。

定制类加载器
  要较好地控制类的加载,就要实现定制的类加载器。所有自定义的类加载器都应继承自java.lang.ClassLoader。而且在构造方法中,我们也应该设置父类加载器。然后重写findClass()方法。differentversionspush文件夹包含了一个叫做FileSystemClassLoader的自订制的类加载器。其结构如图9所示。
 
9. 定制类加载器关系
  以下是在common.FileSystemClassLoader实现的主方法:
public byte[] findClassBytes(String className){
 try {  String pathName = currentRoot + File.separatorChar + className. replace(’.’, File.separatorChar) + ".class";  FileInputStream inFile = new FileInputStream(pathName);   byte[] classBytes = new byte[inFile.available()];  inFile.read(classBytes);   return classBytes;  } catch (java.io.IOException ioEx) {  return null;  }}public Class findClass(String name)throws ClassNotFoundException{ byte[] classBytes = findClassBytes(name);  if (classBytes==null) {  throw new ClassNotFoundException(); } else{  return defineClass(name, classBytes, 0, classBytes.length); }}public Class findClass(String name, byte[] classBytes)throws ClassNotFoundException{ if (classBytes==null) {  throw new ClassNotFoundException( "(classBytes==null)");  } else{  return defineClass(name, classBytes, 0, classBytes.length); }}public void execute(String codeName, byte[] code){ Class klass = null; try {  klass = findClass(codeName, code);  TaskIntf task = (TaskIntf) klass.newInstance();  task.execute(); } catch(Exception exception){  exception.printStackTrace(); }}

  这个类供客户端把client.TaskImpl(v1)转换成字节数组,之后此字节数组被发送到RMI服务端。在服务端,一个同样的类用来把字节数组的内容转换回代码。客户端代码如下:

public class Client{
 public static void main (String[] args) {  try{ byte[] code = getClassDefinition ("client.TaskImpl");   serverIntf.execute("client.TaskImpl", code); } catch(RemoteException remoteException) {  remoteException.printStackTrace(); }}private static byte[] getClassDefinition (String codeName){ String userDir = System.getProperties(). getProperty("BytePath"); FileSystemClassLoader fscl1 = null;  try {  fscl1 = new FileSystemClassLoader (userDir); } catch(FileNotFoundException fileNotFoundException) {  fileNotFoundException.printStackTrace();  } return fscl1.findClassBytes(codeName);}}

  在执行引擎中,从客户端收到的代码被送到定制的类加载器中。定制的类加载器把其从字节数组定义成类,实例化并执行。需要指出的是,对每一个客户请求,我们用类FileSystemClassLoader的不同实例来定义客户端提交的client.TaskImpl。而且,client.TaskImpl并不在服务端的类路径中。这也就意味着当我们在FileSystemClassLoader调用findClass()方法时,findClass()调用内在的defineClass()方法。类client.TaskImpl被特定的类加载器实例所定义。因此,当FileSystemClassLoader的一个新的实例被使用,类又被重新定义为字节数组。因此,对每个客户端请求类client.TaskImpl被多次定义,我们就可以在相同执行引擎JVM中执行不同的client.TaskImpl的代码。
public void execute(String codeName, byte[] code)throws RemoteException{
 FileSystemClassLoader fileSystemClassLoader = null; try {  fileSystemClassLoader = new FileSystemClassLoader();  fileSystemClassLoader.execute(codeName, code); } catch(Exception exception) {  throw new RemoteException(exception.getMessage());  }}


  示例在differentversionspush文件夹下。服务端和客户端的控制台界面分别如图101112所示:

 
10. 定制类加载器执行引擎

  图10显示的是定制的类加载器控制台。我们可以看到client.TaskImpl的代码被多次加载。实际上针对每一个客户端,类都被加载并初始化。
 
11. 定制类加载器,客户端1

  图11中,含有client.TaskImpl.class.getClassLoader(v1)的日志记录的类TaskImpl的代码被客户端的VM加载,然后送到服务端。图12 另一个客户端把包含有client.TaskImpl.class.getClassLoader(v1)的类代码加载并送往服务端。


 
12. 定制类加载器,客户端1
  这段代码演示了我们如何利用不同的类加载器实例来在同一个VM上执行不同版本的代码。

  J2EE的类加载器

  J2EE的服务器倾向于以一定间隔频率,丢弃原有的类并重新载入新的类。在某些情况下会这样执行,而有些情况则不。同样,对于一个web服务器如果要丢弃一个servlet实例,可能是服务器管理员的手动操作,也可能是此实例长时间未相应。当一个JSP页面被首次请求,容器会把此JSP页面翻译成一个具有特定形式的servlet代码。一旦servlet代码被创建,容器就会把这个servlet翻译成class文件等待被使用。对于提交给容器的每次请求,容器都会首先检查这个JSP文件是否刚被修改过。是的话就重新翻译此文件,这可以确保每次的请求都是及时更新的。企业级的部署方案以.ear, .war, .rar等形式的文件,同样需要重复加载,可能是随意的也可能是依照某种配置方案定期执行。对所有的这些情况??类的加载、卸载、重新加载……全部都是建立在我们控制应用服务器的类加载机制的基础上的。实现这些需要扩展的类加载器,它可以执行由其自身所定义的类。Brett Peterson已经在他的文章 Understanding J2EE Application Server Class Loading Architectures给出了J2EE应用服务器的类加载方案的详细说明,详见网站TheServerSide.com

  本文探讨了类载入到虚拟机是如何进行唯一标识的,以及类如果存在同样的类名和包名时所产生的问题。因为没有一个直接可用的类版本管理机制,所以如果我们要按自己的意愿来加载类时,需要自己订制类加载器来扩展其行为。我们可以利用许多J2EE服务器所提供的热部署功能来重新加载一个新版本的类,而不改动服务器的VM。即使不涉及应用服务器,我们也可以利用定制类加载器来控制java应用程序载入类时的具体行为。Ted Neward的书Server-Based Java Programming中详细阐述java的类加载,J2EEAPI以及使用他们的最佳途径。

 

分享到:
评论

相关推荐

    实验-三、数据库安全性(目的、要求和模板).doc

    实验-三、数据库安全性(目的、要求和模板).doc

    基于Docker搭建K8s集群离线包

    基于Docker搭建K8s集群离线包,包含部署时所需的全部文件,可在内网环境中使用,K8s为1.23.0版本,docker为20.10.9-3版本

    基于springboot+vue实现的求职招聘类型网站源代码+数据库(优质毕设项目).zip

    基于springboot+vue实现的求职招聘类型网站源代码+数据库(优质毕设项目).zip个人经导师指导并认可通过的98分毕业设计项目,主要针对计算机相关专业的正在做毕设的学生和需要项目实战练习的学习者。也可作为课程设计、期末大作业。包含全部项目源码、该项目可以直接作为毕设使用。项目都经过严格调试,确保可以运行! 基于springboot+vue实现的求职招聘类型网站源代码+数据库(优质毕设项目).zip个人经导师指导并认可通过的98分毕业设计项目,主要针对计算机相关专业的正在做毕设的学生和需要项目实战练习的学习者。也可作为课程设计、期末大作业。包含全部项目源码、该项目可以直接作为毕设使用。项目都经过严格调试,确保可以运行! 基于springboot+vue实现的求职招聘类型网站源代码+数据库(优质毕设项目).zip个人经导师指导并认可通过的98分毕业设计项目,主要针对计算机相关专业的正在做毕设的学生和需要项目实战练习的学习者。也可作为课程设计、期末大作业。包含全部项目源码、该项目可以直接作为毕设使用。项目都经过严格调试,确保可以运行!基于springboot+vue实现的求

    基于Android系统Api封装常用工具类.zip

    基于Android系统Api封装常用工具类.zip

    基于PCA和SVM的人脸识别

    svm 基于PCA(主成分分析)和SVM(支持向量机)的人脸识别是一种常见的方法。这里是一个简要说明: PCA(主成分分析): PCA是一种降维技术,它通过线性变换将高维数据转换为低维数据,同时保留最大的数据方差。 在人脸识别中,PCA被用来提取人脸图像中的主要特征,从而减少数据的维度,并保留最重要的信息。 SVM(支持向量机): SVM是一种监督学习算法,用于分类和回归分析。 在人脸识别中,SVM被用来构建一个分类器,以将提取的人脸特征映射到相应的人脸身份标签。 基于PCA和SVM的人脸识别流程: 训练阶段: 收集训练数据集,包括多个人的人脸图像和相应的标签。 对每个人脸图像应用PCA,将其转换为低维特征向量。 使用这些特征向量训练一个SVM分类器,使其能够将人脸特征向量与相应的人脸标签关联起来。 测试阶段: 对待识别的人脸图像应用相同的PCA转换,将其转换为与训练数据相同的低维特征向量。 使用训练好的SVM分类器,将待识别的人脸特征向量与已知的人脸标签进行比较,从而确定其身份。 优点: PCA可以有效地降低数据的维度,减少计算复杂度,并提取最相关的特征。 SVM在处理

    天津科技大学-答辩通用PPT模板我给母校送模板作品.pptx

    PPT模板,答辩PPT模板,毕业答辩,学术汇报,母校模板,我给母校送模板作品,周会汇报,开题答辩,教育主题模板下载。PPT素材下载。

    Java SE Development Kit 11.0.23 macOS x64 DMG Installer

    Java SE Development Kit 11.0.23 macOS x64 DMG Installer

    课设&大作业-SSM毕业设计项目.zip

    【资源说明】【毕业设计】 1、该资源内项目代码都是经过测试运行成功,功能正常的情况下才上传的,请放心下载使用。 2、适用人群:主要针对计算机相关专业(如计科、信息安全、数据科学与大数据技术、人工智能、通信、物联网、数学、电子信息等)的同学或企业员工下载使用,具有较高的学习借鉴价值。 3、不仅适合小白学习实战练习,也可作为大作业、课程设计、毕设项目、初期项目立项演示等,欢迎下载,互相学习,共同进步!

    studyopencv2

    studyopencv2

    实验五-使用matlab实现卷积的运算.pdf

    实验五-使用matlab实现卷积的运算.pdf

    2024年中国纸杯蛋糕盒行业研究报告.docx

    2024年中国纸杯蛋糕盒行业研究报告

    总结作图常用的操作 Python数据分析及可视化-Matplotlib极简入门.zip

    matplotlib绘图 通过 Matplotlib,开发者可以仅需要几行代码,便可以生成绘图、直方图、功率谱、条形图、错误图、散点图等。 Matplotlib基础知识 1.Matplotlib中的基本图表包括的元素 x轴和y轴 水平和垂直的轴线 x轴和y轴刻度 刻度标示坐标轴的分隔,包括最小刻度和最大刻度 x轴和y轴刻度标签 表示特定坐标轴的值 绘图区域 实际绘图的区域 2.hold属性 hold属性默认为True,允许在一幅图中绘制多个曲线;将hold属性修改为False,每一个plot都会覆盖前面的plot。 但是不推荐去动hold这个属性,这种做法(会有警告)。因此使用默认设置即可。 3.网格线 grid方法 使用grid方法为图添加网格线 设置grid参数(参数与plot函数相同) .lw代表linewidth,线的粗细 .alpha表示线的明暗程度 4.axis方法 如果axis方法没有任何参数,则返回当前坐标轴的上下限 5.xlim方法和ylim方法 除了plt.axis方法,还可以通过xlim,ylim方法设置坐标轴范围

    java本科毕业设计基于RFID技术的国有资产管理系统源码后台项目.zip

    高分设计源码,详情请查看资源内容中使用说明 高分设计源码,详情请查看资源内容中使用说明高分设计源码,详情请查看资源内容中使用说明高分设计源码,详情请查看资源内容中使用说明高分设计源码,详情请查看资源内容中使用说明高分设计源码,详情请查看资源内容中使用说明高分设计源码,详情请查看资源内容中使用说明高分设计源码,详情请查看资源内容中使用说明高分设计源码,详情请查看资源内容中使用说明高分设计源码,详情请查看资源内容中使用说明高分设计源码,详情请查看资源内容中使用说明高分设计源码,详情请查看资源内容中使用说明高分设计源码,详情请查看资源内容中使用说明高分设计源码,详情请查看资源内容中使用说明高分设计源码,详情请查看资源内容中使用说明高分设计源码,详情请查看资源内容中使用说明高分设计源码,详情请查看资源内容中使用说明高分设计源码,详情请查看资源内容中使用说明高分设计源码,详情请查看资源内容中使用说明高分设计源码,详情请查看资源内容中使用说明高分设计源码,详情请查看资源内容中使用说明高分设计源码,详情请查看资源内容中使用说明

    springboot部署系统(自动化部署+缓存管理+业务降级+应用监控).zip

    springboot框架 一、Spring Boot基础应用 Spring Boot特征 概念: 约定优于配置,简单来说就是你所期待的配置与约定的配置一致,那么就可以不做任何配置,约定不符合期待时才需要对约定进行替换配置。 特征: 1. SpringBoot Starter:他将常用的依赖分组进行了整合,将其合并到一个依赖中,这样就可以一次性添加到项目的Maven或Gradle构建中。 2,使编码变得简单,SpringBoot采用 JavaConfig的方式对Spring进行配置,并且提供了大量的注解,极大的提高了工作效率,比如@Configuration和@bean注解结合,基于@Configuration完成类扫描,基于@bean注解把返回值注入IOC容器。 3.自动配置:SpringBoot的自动配置特性利用了Spring对条件化配置的支持,合理地推测应用所需的bean并自动化配置他们。 4.使部署变得简单,SpringBoot内置了三种Servlet容器,Tomcat,Jetty,undertow.我们只需要一个Java的运行环境就可以跑SpringBoot的项目了

    CUDA规约求和.cu

    CUDA规约求和.cu

    工程有限公司计算机信息系统数据信息安全管理办法.doc

    工程有限公司计算机信息系统数据信息安全管理办法.doc

    小学教育信息化工作计划.doc

    小学教育信息化工作计划.doc

    springboot停车位管理源码.rar

    springboot停车位管理源码.rarspringboot停车位管理源码.rarspringboot停车位管理源码.rar

    实验七-管道通信.doc

    实验七-管道通信.doc

    基于YOLOv5行人车辆跟踪检测识别计数系统源码.zip

    YOLO高分设计资源源码,详情请查看资源内容中使用说明 YOLO高分设计资源源码,详情请查看资源内容中使用说明 YOLO高分设计资源源码,详情请查看资源内容中使用说明 YOLO高分设计资源源码,详情请查看资源内容中使用说明YOLO高分设计资源源码,详情请查看资源内容中使用说明YOLO高分设计资源源码,详情请查看资源内容中使用说明YOLO高分设计资源源码,详情请查看资源内容中使用说明YOLO高分设计资源源码,详情请查看资源内容中使用说明YOLO高分设计资源源码,详情请查看资源内容中使用说明YOLO高分设计资源源码,详情请查看资源内容中使用说明YOLO高分设计资源源码,详情请查看资源内容中使用说明YOLO高分设计资源源码,详情请查看资源内容中使用说明YOLO高分设计资源源码,详情请查看资源内容中使用说明YOLO高分设计资源源码,详情请查看资源内容中使用说明YOLO高分设计资源源码,详情请查看资源内容中使用说明YOLO高分设计资源源码,详情请查看资源内容中使用说明YOLO高分设计资源源码,详情请查看资源内容中使用说明YOLO高分设计资源源码,详情请查看资源内容中使用说明YOLO高分设计资源源码,详情请查看资源内容中使用说明YOLO高分设计资源源码,详情请查看资源内容中使用说明YOLO高分设计资源源码,详情请查看资源内容中使用说明YOLO高分设计资源源码,详情请查看资源内容中使用说明YOLO高分设计资源源码,详情请查看资源内容中使用说明YOLO高分设计资源源码,详情请查看资源内容中使用说明YOLO高分设计资源源码,详情请查看资源内容中使用说明YOLO高分设计资源源码,详情请查看资源内容中使用说明

Global site tag (gtag.js) - Google Analytics