我的编程空间,编程开发者的网络收藏夹
学习永远不晚

JAVA classloader怎么理解

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

JAVA classloader怎么理解

这篇文章给大家介绍JAVA classloader怎么理解,内容非常详细,感兴趣的小伙伴们可以参考借鉴,希望对大家能有所帮助。

什么是 ClassLoader?

在流行的商业化编程语言中,Java 语言由于在 Java 虚拟机 (JVM) 上运行而显得与众不同。这意味着已编译的程序是一种非凡的、独立于平台的格式,并非依靠于它们所运行的机器。在很大程度上,这种格式不同于传统的可执行程序格式。

与 C 或 C++ 编写的程序不同,Java 程序并不是一个可执行文件,而是由许多独立的类文件组成,每一个文件对应于一个 Java 类。
此外,这些类文件并非立即全部都装入内存,而是根据程序需要装入内存。ClassLoader 是 JVM 中将类装入内存的那部分。
而且,Java ClassLoader 就是用 Java 语言编写的。这意味着创建您自己的 ClassLoader 非常轻易,不必了解 JVM 的微小细节。


为什么编写 ClassLoader?


假如 JVM 已经有一个 ClassLoader,那么为什么还要编写另一个呢?问得好。缺省的 ClassLoader 只知道如何从本地文件系统装入类文件。不过这只适合于常规情况,即已全部编译完 Java 程序,并且计算机处于等待状态。

但 Java 语言最具新意的事就是 JVM 可以非常轻易地从那些非本地硬盘或从网络上获取类。例如,浏览者可以使用定制的 ClassLoader 从 Web 站点装入可执行内容。

有许多其它方式可以获取类文件。除了简单地从本地或网络装入文件以外,可以使用定制的 ClassLoader 完成以下任务:
在执行非置信代码之前,自动验证数字签名
使用用户提供的密码透明地解密代码
动态地创建符合用户特定需要的定制化构建类
任何您认为可以生成 Java 字节码的内容都可以集成到应用程序中。

[@more@]

定制 ClassLoader 示例


假如使用过 JDK 或任何基于 Java 浏览器中的 Applet 查看器,那么您差不多肯定使用过定制的 ClassLoader。

Sun 最初发布 Java 语言时,其中最令人兴奋的一件事是观看这项新技术是如何执行在运行时从远程的 Web 服务器装入的代码。(此外,还有更令人兴奋的事 -- Java 技术提供了一种便于编写代码的强大语言。)更一些令人激动的是它可以执行从远程 Web 服务器通过 HTTP 连接发送过来的字节码。

此项功能归功于 Java 语言可以安装定制 ClassLoader。Applet 查看器包含一个 ClassLoader,它不在本地文件系统中寻找类,而是访问远程服务器上的 Web 站点,经过 HTTP 装入原始的字节码文件,并把它们转换成 JVM 内的类。

浏览器和 Applet 查看器中的 ClassLoaders 还可以做其它事情:它们支持安全性以及使不同的 Applet 在不同的页面上运行而互不干扰。

Luke Gorrie 编写的 Echidna 是一个开放源码包,它可以使您在单个虚拟机上运行多个 Java 应用程序。它使用定制的 ClassLoader,通过向每个应用程序提供该类文件的自身副本,以防止应用程序互相干扰。


我们的 ClassLoader 示例


了解了 ClassLoader 如何工作以及如何编写 ClassLoader 之后,我们将创建称作 CompilingClassLoader (CCL) 的 Classloader。CCL 为我们编译 Java 代码,而无需要我们干涉这个过程。它基本上就类似于直接构建到运行时系统中的 "make" 程序。
注:进一步了解之前,应注重在 JDK 版本 1.2 中已改进了 ClassLoader 系统的某些方面(即 Java 2 平台)。本教程是按 JDK 版本 1.0 和 1.1 写的,但也可以在以后的版本中运行。

Java 2 中 ClassLoader 的变动描述了 Java 版本 1.2 中的变动,并提供了一些具体信息,以便修改 ClassLoader 来利用这些变动。

ClassLoader 的基本目标是对类的请求提供服务。当 JVM 需要使用类时,它根据名称向 ClassLoader 请求这个类,然后 ClassLoader 试图返回一个表示这个类的 Class 对象。 通过覆盖对应于这个过程不同阶段的方法,可以创建定制的 ClassLoader。

在本文的其余部分,您会学习 Java ClassLoader 的要害方法。您将了解每一个方法的作用以及它是如何适合装入类文件这个过程的。您也会知道,创建自己的 ClassLoader 时,需要编写什么代码。

在下文中,您将会利用这些知识来使用我们的 ClassLoader 示例 -- CompilingClassLoader。


方法 loadClass

ClassLoader.loadClass() 是 ClassLoader 的入口点。其特征如下:
Class loadClass( String name, boolean resolve );
name 参数指定了 JVM 需要的类的名称,该名称以包表示法表示,如 Foo 或 java.lang.Object。 resolve 参数告诉方法是否需要解析类。在预备执行类之前,应考虑类解析。并不总是需要解析。假如 JVM 只需要知道该类是否存在或找出该类的超类,那么就不需要解析。
在 Java 版本 1.1 和以前的版本中,loadClass 方法是创建定制的 ClassLoader 时唯一需要覆盖的方法。(Java 2 中 ClassLoader 的变动提供了关于 Java 1.2 中 findClass() 方法的信息。)


方法 defineClass


defineClass 方法是 ClassLoader 的主要诀窍。该方法接受由原始字节组成的数组并把它转换成 Class 对象。原始数组包含如从文件系统或网络装入的数据。

defineClass 治理 JVM 的许多复杂、神秘和倚赖于实现的方面 -- 它把字节码分析成运行时数据结构、校验有效性等等。不必担心,您无需亲自编写它。事实上,即使您想要这么做也不能覆盖它,因为它已被标记成最终的。


方法 findSystemClass


findSystemClass 方法从本地文件系统装入文件。它在本地文件系统中寻找类文件,假如存在,就使用 defineClass 将原始字节转换成 Class 对象,以将该文件转换成类。当运行 Java 应用程序时,这是 JVM 正常装入类的缺省机制。(Java 2 中 ClassLoader 的变动提供了关于 Java 版本 1.2 这个过程变动的具体信息。)

对于定制的 ClassLoader,只有在尝试其它方法装入类之后,再使用 findSystemClass。原因很简单:ClassLoader 是负责执行装入类的非凡步骤,不是负责所有类。例如,即使 ClassLoader 从远程的 Web 站点装入了某些类,仍然需要在本地机器上装入大量的基本 Java 库。而这些类不是我们所关心的,所以要 JVM 以缺省方式装入它们:从本地文件系统。这就是 findSystemClass 的用途。

其工作流程如下:
请求定制的 ClassLoader 装入类。
检查远程 Web 站点,查看是否有所需要的类。
假如有,那么好;抓取这个类,完成任务。
假如没有,假定这个类是在基本 Java 库中,那么调用 findSystemClass,使它从文件系统装入该类。


在大多数定制 ClassLoaders 中,首先调用 findSystemClass 以节省在本地就可以装入的许多 Java 库类而要在远程 Web 站点上查找所花的时间。然而,正如,在下一章节所看到的,直到确信能自动编译我们的应用程序代码时,才让 JVM 从本地文件系统装入类。


方法 resolveClass
正如前面所提到的,可以不完全地(不带解析)装入类,也可以完全地(带解析)装入类。当编写我们自己的 loadClass 时,可以调用 resolveClass,这取决于 loadClass 的 resolve 参数的值。


方法 findLoadedClass
findLoadedClass 充当一个缓存:当请求 loadClass 装入类时,它调用该方法来查看 ClassLoader 是否已装入这个类,这样可以避免重新装入已存在类所造成的麻烦。应首先调用该方法。


组装
让我们看一下如何组装所有方法。
我们的 loadClass 实现示例执行以下步骤。(这里,我们没有指定生成类文件是采用了哪种技术 -- 它可以是从 Net 上装入、或者从归档文件中提取、或者实时编译。无论是哪一种,那是种非凡的神奇方式,使我们获得了原始类文件字节。)


CCL 揭密
我们的 ClassLoader (CCL) 的任务是确保代码被编译和更新。
下面描述了它的工作方式:
当请求一个类时,先查看它是否在磁盘的当前目录或相应的子目录。
假如该类不存在,但源码中有,那么调用 Java 编译器来生成类文件。
假如该类已存在,检查它是否比源码旧。假如是,调用 Java 编译器来重新生成类文件。
假如编译失败,或者由于其它原因不能从现有的源码中生成类文件,返回 ClassNotFoundException。
假如仍然没有该类,也许它在其它库中,所以调用 findSystemClass 来寻找该类。
假如还是没有,则返回 ClassNotFoundException。
否则,返回该类。
调用 findLoadedClass 来查看是否存在已装入的类。
假如没有,那么采用那种非凡的神奇方式来获取原始字节。
假如已有原始字节,调用 defineClass 将它们转换成 Class 对象。
假如没有原始字节,然后调用 findSystemClass 查看是否从本地文件系统获取类。
假如 resolve 参数是 true,那么调用 resolveClass 解析 Class 对象。
假如还没有类,返回 ClassNotFoundException。
否则,将类返回给调用程序。


Java 编译的工作方式
在深入讨论之前,应该先退一步,讨论 Java 编译。通常,Java 编译器不只是编译您要求它编译的类。它还会编译其它类,假如这些类是您要求编译的类所需要的类。
CCL 逐个编译应用程序中的需要编译的每一个类。但一般来说,在编译器编译完第一个类后,CCL 会查找所有需要编译的类,然后编译它。为什么?Java 编译器类似于我们正在使用的规则:假如类不存在,或者与它的源码相比,它比较旧,那么它需要编译。其实,Java 编译器在 CCL 之前的一个步骤,它会做大部分的工作。

当 CCL 编译它们时,会报告它正在编译哪个应用程序上的类。在大多数的情况下,CCL 会在程序中的主类上调用编译器,它会做完所有要做的 -- 编译器的单一调用已足够了。

然而,有一种情形,在第一步时不会编译某些类。假如使用 Class.forName 方法,通过名称来装入类,Java 编译器会不知道这个类时所需要的。在这种情况下,您会看到 CCL 再次运行 Java 编译器来编译这个类。在源代码中演示了这个过程。


使用 CompilationClassLoader
要使用 CCL,必须以非凡方式调用程序。不能直接运行该程序,如: % java Foo arg1 arg2
应以下列方式运行它:
% java CCLRun Foo arg1 arg2


CCLRun 是一个非凡的存根程序,它创建 CompilingClassLoader 并用它来装入程序的主类,以确保通过 CompilingClassLoader 来装入整个程序。CCLRun 使用 Java Reflection API 来调用特定类的主方法并把参数传递给它。有关具体信息,请参阅源代码。


运行示例
源码包括了一组小类,它们演示了工作方式。主程序是 Foo 类,它创建类 Bar 的实例。类 Bar 创建另一个类 Baz 的实例,它在 baz 包内,这是为了展示 CCL 是如何处理子包里的代码。Bar 也是通过名称装入的,其名称为 Boo,这用来展示它也能与 CCL 工作。

每个类都声明已被装入并运行。现在用源代码来试一下。编译 CCLRun 和 CompilingClassLoader。确保不要编译其它类(Foo、Bar、Baz 和 Boo),否则将不会使用 CCL,因为这些类已经编译过了。

% java CCLRun Foo arg1 arg2
CCL: Compiling Foo.java...
foo! arg1 arg2
bar! arg1 arg2
baz! arg1 arg2
CCL: Compiling Boo.java...
Boo!


请注重,首先调用编译器,Foo.java 治理 Bar 和 baz.Baz。直到 Bar 通过名称来装入 Boo 时,被调用它,这时 CCL 会再次调用编译器来编译它。
CompilingClassLoader.java


以下是 CompilingClassLoader.java 的源代码

// $Id$
import java.io.*;

public class CompilingClassLoader extends ClassLoader
{
 // Given a filename, read the entirety of that file from disk
 // and return it as a byte array.
 private byte[] getBytes( String filename ) throws IOException {
   // Find out the length of the file
   File file = new File( filename );
   long len = file.length();
   // Create an array that's just the right size for the file's
   // contents
   byte raw[] = new byte[(int)len];
   // Open the file
   FileInputStream fin = new FileInputStream( file );
   // Read all of it into the array; if we don't get all,
   // then it's an error.
   int r = fin.read( raw );
   if (r != len)
     throw new IOException( "Can't read all, "+r+" != "+len );
   // Don't forget to close the file!
   fin.close();
   // And finally return the file contents as an array
   return raw;
 }
 // Spawn a process to compile the java source code file
 // specified in the 'javaFile' parameter.  Return a true if
 // the compilation worked, false otherwise.
 private boolean compile( String javaFile ) throws IOException {
   // Let the user know what's going on
   System.out.println( "CCL: Compiling "+javaFile+"..." );
   // Start up the compiler
   Process p = Runtime.getRuntime().exec( "javac "+javaFile );
   // Wait for it to finish running
   try {
     p.waitFor();
   } catch( InterruptedException ie ) { System.out.println( ie ); }
   // Check the return code, in case of a compilation error
   int ret = p.exitValue();
   // Tell whether the compilation worked
   return ret==0;
 }
 // The heart of the ClassLoader -- automatically compile
 // source as necessary when looking for class files
 public Class loadClass( String name, boolean resolve )
     throws ClassNotFoundException {
   // Our goal is to get a Class object
   Class clas = null;
   // First, see if we've already dealt with this one
   clas = findLoadedClass( name );
   //System.out.println( "findLoadedClass: "+clas );
   // Create a pathname from the class name
   // E.g. java.lang.Object => java/lang/Object
   String fileStub = name.replace( '.', '/' );
   // Build objects pointing to the source code (.java) and object
   // code (.class)
   String javaFilename = fileStub+".java";
   String classFilename = fileStub+".class";
   File javaFile = new File( javaFilename );
   File classFile = new File( classFilename );
   //System.out.println( "j "+javaFile.lastModified()+" c "+
   //  classFile.lastModified() );
   // First, see if we want to try compiling.  We do if (a) there
   // is source code, and either (b0) there is no object code,
   // or (b1) there is object code, but it's older than the source
   if (javaFile.exists() &&
        (!classFile.exists()
         javaFile.lastModified() > classFile.lastModified())) {
     try {
       // Try to compile it.  If this doesn't work, then
       // we must declare failure.  (It's not good enough to use
       // and already-existing, but out-of-date, classfile)
       if (!compile( javaFilename ) !classFile.exists()) {
         throw new ClassNotFoundException( "Compile failed: "+javaFilename );
       }
     } catch( IOException ie ) {
       // Another place where we might come to if we fail
       // to compile
       throw new ClassNotFoundException( ie.toString() );
     }
   }
   // Let's try to load up the raw bytes, assuming they were
   // properly compiled, or didn't need to be compiled
   try {
     // read the bytes
     byte raw[] = getBytes( classFilename );
     // try to turn them into a class
     clas = defineClass( name, raw, 0, raw.length );
   } catch( IOException ie ) {
     // This is not a failure!  If we reach here, it might
     // mean that we are dealing with a class in a library,
     // sUCh as java.lang.Object
   }
   //System.out.println( "defineClass: "+clas );
   // Maybe the class is in a library -- try loading
   // the normal way
   if (clas==null) {
     clas = findSystemClass( name );
   }
   //System.out.println( "findSystemClass: "+clas );
   // Resolve the class, if any, but only if the "resolve"
   // flag is set to true
   if (resolve && clas != null)
     resolveClass( clas );
   // If we still don't have a class, it's an error
   if (clas == null)
     throw new ClassNotFoundException( name );
   // Otherwise, return the class
   return clas;
 }
}

CCRun.java
以下是 CCRun.java 的源代码

// $Id$
import java.lang.reflect.*;

public class CCLRun
{
 static public void main( String args[] ) throws Exception {
   // The first argument is the Java program (class) the user
   // wants to run
   String progClass = args[0];
   // And the arguments to that program are just
   // arguments 1..n, so separate those out into
   // their own array
   String progArgs[] = new String[args.length-1];
   System.arraycopy( args, 1, progArgs, 0, progArgs.length );
   // Create a CompilingClassLoader
   CompilingClassLoader ccl = new CompilingClassLoader();
   // Load the main class through our CCL
   Class clas = ccl.loadClass( progClass );
   // Use reflection to call its main() method, and to
   // pass the arguments in.
   // Get a class representing the type of the main method's argument
   Class mainArgType[] = { (new String[0]).getClass() };
   // Find the standard main method in the class
   Method main = clas.getMethod( "main", mainArgType );
   // Create a list containing the arguments -- in this case,
   // an array of strings
   Object argsArray[] = { progArgs };
   // Call the method
   main.invoke( null, argsArray );
 }
}

Foo.java
以下是 Foo.java 的源代码

// $Id$
public class Foo
{
 static public void main( String args[] ) throws Exception {
   System.out.println( "foo! "+args[0]+" "+args[1] );
   new Bar( args[0], args[1] );
 }
}

Bar.java
以下是 Bar.java 的源代码

// $Id$
import baz.*;
public class Bar
{
 public Bar( String a, String b ) {
   System.out.println( "bar! "+a+" "+b );
   new Baz( a, b );
   try {
     Class booClass = Class.forName( "Boo" );
     Object boo = booClass.newInstance();
   } catch( Exception e ) {
     e.printStackTrace();
   }
 }
}

baz/Baz.java
以下是 baz/Baz.java 的源代码

// $Id$
package baz;
public class Baz
{
 public Baz( String a, String b ) {
   System.out.println( "baz! "+a+" "+b );
 }
}

Boo.java
以下是 Boo.java 的源代码

// $Id$
public class Boo
{
 public Boo() {
   System.out.println( "Boo!" );
 }
}

关于JAVA classloader怎么理解就分享到这里了,希望以上内容可以对大家有一定的帮助,可以学到更多知识。如果觉得文章不错,可以把它分享出去让更多的人看到。

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

JAVA classloader怎么理解

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

JAVA classloader怎么理解

这篇文章给大家介绍JAVA classloader怎么理解,内容非常详细,感兴趣的小伙伴们可以参考借鉴,希望对大家能有所帮助。什么是 ClassLoader?在流行的商业化编程语言中,Java 语言由于在 Java 虚拟机 (JVM) 上运
2023-06-03

Java ClassLoader该如何理解

这篇文章的内容主要围绕Java ClassLoader该如何理解进行讲述,文章内容清晰易懂,条理清晰,非常适合新手学习,值得大家去阅读。感兴趣的朋友可以跟随小编一起阅读吧。希望大家通过这篇文章有所收获!ClassLoader 是 Java
2023-06-02

JAVA的classloader怎么编写

今天小编给大家分享一下JAVA的classloader怎么编写的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。什么是 Clas
2023-06-03

Java的ClassLoader是什么

本文小编为大家详细介绍“Java的ClassLoader是什么”,内容详细,步骤清晰,细节处理妥当,希望这篇“Java的ClassLoader是什么”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来学习新知识吧。ClassLoad
2023-06-16

java类加载器ClassLoader详解

获得ClassLoader的途径1. 获得当前类的ClassLoaderclazz.getClassLoader()2. 获得当前线程上下文的ClassLoaderThread.currentThread().getContextClassLoader();3
java类加载器ClassLoader详解
2019-01-28

Java的ClassLoader有什么用

这篇文章主要介绍“Java的ClassLoader有什么用”,在日常操作中,相信很多人在Java的ClassLoader有什么用问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Java的ClassLoader有
2023-06-17

java中的classloader有什么功能

JavaClassLoader功能ClassLoader是Java虚拟机中负责加载、链接和初始化Java类的核心组件,其功能包括:类加载:加载.class文件,构建类型层次结构。类验证:确保加载的类符合规范,防止安全漏洞。依赖管理:按正确顺序加载和链接类。安全保护:控制对敏感资源的访问。可扩展性:允许创建自定义类加载器以满足特定需求。ClassLoader有四种主要类型:启动类加载器、扩展类加载器、系统类加载器和自定义类加载器,各有其加载位置和功能。通过调用loadClass()方法或创建自定义类加载器,
java中的classloader有什么功能
2024-04-11

Java的ClassLoader机制是什么

本篇内容介绍了“Java的ClassLoader机制是什么”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!JVM在加载类的时候,都是通过Cla
2023-06-17

java classloader的使用方法是什么

Java ClassLoader是Java虚拟机(JVM)的一个重要组成部分,用于加载Java类文件。ClassLoader负责将编译后的Java类文件加载到内存中,并在运行时实例化这些类。使用ClassLoader可以实现动态加载类、实现
java classloader的使用方法是什么
2024-04-09

java classloader的使用方法是什么

Java类加载器使用方法Java类加载器负责加载、链接和初始化类,确保应用程序加载正确的类并防止冲突。使用步骤:加载:从存储库读取类二进制表示。链接:连接类和解析符号引用。初始化:调用静态初始化程序。类型:引导类加载器:加载核心类库。扩展类加载器:加载扩展类路径中的类。系统类加载器:加载系统类路径中的类。自定义类加载器:由开发人员创建。使用:使用默认类加载器。创建自定义类加载器。设置类加载器。最佳实践:使用默认类加载器。仅在需要时创建自定义类加载器。考虑类加载委托的性能影响。测试自定义类加载器的行为。
java classloader的使用方法是什么
2024-04-12

java classloader的工作机制是什么

Java的ClassLoader是Java虚拟机(JVM)的一个重要组成部分,它主要负责加载Java类文件并把字节码文件转换成运行时的Class对象。ClassLoader的工作机制可以简单地描述为以下几个步骤:寻找类文件:ClassLo
java classloader的工作机制是什么
2024-04-09

Java类加载器ClassLoader的使用详解

类加载器负责读取Java字节代码,并转换成java.lang.Class类的一个实例的代码模块。本文主要和大家聊聊JVM类加载器ClassLoader的使用,需要的可以了解一下
2022-12-19

java classloader的工作机制是什么

Java类加载器是一种分层系统,动态加载类定义。它包含三个主要类加载器:Bootstrap、扩展和系统,遵循分层和委托机制。每个加载器在自己的范围内加载类,并委托给父加载器加载不在其范围内的类。Java采用双亲委派模型,保证加载优先级和类隔离。开发人员还可以创建自定义类加载器以修改类加载行为。
java classloader的工作机制是什么
2024-04-10

Java锁怎么理解

本篇内容主要讲解“Java锁怎么理解”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“Java锁怎么理解”吧!自旋?自旋锁如果此时拿不到锁,它不马上进入阻塞状态,而愿意等待一段时间。如果循环一定的次
2023-06-16

java的this怎么理解

java提供了一个this关键字,this关键字总是指向调用该方法的对象。根据this出现位置的不同,this作为对象的默认引用有两种情形。 (推荐学习:java课程)a)、构造器中引用该构造器正在初始化的对象。(this总是引用该构
java的this怎么理解
2017-10-14

java多态怎么理解

这篇文章主要介绍java多态怎么理解,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!在java中,多态是同一个行为具有多个不同表现形式或形态的能力;多态就是同一个接口,使用不同的实例而执行不同操作。多态的优点:1、消除
2023-06-14

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录