Heim >Java >javaLernprogramm >Eingehende Analyse der Java-System-Systemklasse
Tiantian sagte, ich solle System.out.println für die Ausgabe verwenden, also habe ich eine kleine Frage: Ist out eine Variable oder eine interne Klasse? Umfangreiches und systematisches Wissen wird in verschiedenen Themen ausführlich erläutert. Wir können diese verstreuten Wissenspunkte nicht ignorieren, sonst wird es sehr peinlich, wenn uns im Interview eine so einfache Frage gestellt wird und wir sie nicht beantworten können.
System als Systemklasse, im java.lang-Paket von JDK ist ersichtlich, dass es sich auch um eine handelt Kernsprache der Java-Eigenschaft. Der Konstruktor der Systemklasse ist mit „Private“ versehen und darf nicht instanziiert werden. Daher sind die Methoden in der Klasse auch statische modifizierte statische Methoden.
public final static InputStream in; //标准输入流 public final static PrintStream out; //标准输出流 public final static PrintStream err; //标准错误流
Es ist zu erkennen, dass es sich bei Out und In im System nicht um interne Klassen handelt, sondern um echte Feldvariablen . out ist das Variablenfeld, das durch die letzte Statik von PrintStream geändert wurde, was bedeutet, dass Methoden der PrintStream-Klasse aufgerufen werden können. Println ist eine Ausgabemethode von PrintStream, daher verwenden wir normalerweise System.out.println(), um Inhalte auf der Konsole auszugeben.
public static void main(String[] args) { int[] arr1 = { 0, 1, 2, 3, 4 }; int[] arr2 = { 9, 9, 9, 9, 9 }; System.arraycopy(arr1, 2, arr2, 0, 3); arr1[3] = 8; for (int i = 0; i < 5; i++) System.out.print(arr2[i] + " "); //2 3 4 9 9 }
Die Arraycopy-Methode verfügt über fünf Parameter: das zu kopierende Array, die zu kopierende Startposition, das zu kopierende Array und die Startposition des zu kopierenden Arrays, an das Ende dieses Arrays kopiert. Diese Methode ähnelt copyOf und copyOfRange in Arrays. Sie verfügt über mehr Parameter und kann bei Bedarf verwendet werden.
Dies ist kein Beispiel, die Methode currentTimeMillis und die Methode getTime im Datum Klasse Es ist genau das Gleiche. Wenn Sie nur Millisekunden benötigen, ist ein solcher Anruf auch sehr praktisch. Es ist jedoch zu beachten, dass currentTimeMillis nicht direkt das Ergebnis von getTime erhält. Da die minimale Genauigkeit der Zeit einiger Betriebssysteme 10 Millisekunden beträgt, kann es zu Problemen kommen einige Abweichungen.
Wir rufen diese Methode auf und geben den StringRufen Sie die Eigenschaften des Systems ab.
键 | 相关值的描述 |
---|---|
java.version | Java 运行时环境版本 |
java.vendor | Java运行时环境供应商 |
java.vendor.url | Java 供应商的 URL |
java.home | Java 安装目录 |
java.vm.specification.version | Java 虚拟机规范版本 |
java.vm.specification.vendor | Java 虚拟机规范供应商 |
java.vm.specification.name | Java 虚拟机规范名称 |
java.vm.version | Java 虚拟机实现版本 |
java.vm.vendor | Java 虚拟机实现供应商 |
java.vm.name | Java 虚拟机实现名称 |
java.specification.version | Java 运行时环境规范版本 |
java.specification.vendor | Java 运行时环境规范供应商 |
java.specification.name | Java 运行时环境规范名称 |
java.class.version | Java 类格式版本号 |
java.class.path | Java 类路径 |
java.library.path | 加载库时搜索的路径列表 |
java.io.tmpdir | 默认的临时文件路径 |
java.compiler | 要使用的 JIT 编译器的名称 |
java.ext.dirs | 一个或多个扩展目录的路径 |
os.name | 操作系统的名称 |
os.arch | 操作系统的架构 |
os.version | 操作系统的版本 |
file.separator | 文件分隔符(在 UNIX 系统中是“/”) |
path.separator | 路径分隔符(在 UNIX 系统中是“:”) |
line.separator | 行分隔符(在 UNIX 系统中是“/n”) |
user.name | 用户的账户名称 |
user.home | 用户的主目录 |
user.dir | 用户的当前工作目录 |
在我们操作文件的时候很可能需要使用到我们的当前工作目录,可以用这个方法来获得。
public static void main(String[] args) { String dirPath = System.getProperty("user.dir"); System.out.println(dirPath); //输出工作目录 D:\Workspaces\MyEclipse 10\Algorithms(这是我的目录,每个人都不同) }
上面的表中就不再举例了,比较常用的是后几个key。
调用 gc 方法暗示着 Java 虚拟机做了一些努力来回收未用对象或失去了所有引用的对象,以便能够快速地重用这些对象当前占用的内存。当控制权从方法调用中返回时,虚拟机已经尽最大努力从所有丢弃的对象中回收了空间。
public static void main(String[] args) { Date d = new Date(); d = null; System.gc(); // 在调用这句gc方法时,上面已经失去了d引用的new Date()被回收 }
实际上我们并不一定需要调用gc()方法,让编译器自己去做好了。如果调用gc方法,会在对象被回收之前调用finalize()方法,但是我们也知道finalize()方法不一定会被调用。总之java在这回收方面做的远不如c和c++。我们可以规避有关回收方面的问题。当需要了解的时候最好专门的去看JVM回收机制的文章。
exit(int)方法终止当前正在运行的 Java 虚拟机,参数解释为状态码。根据惯例,非 0 的状态码表示异常终止。 而且,该方法永远不会正常返回。 这是唯一一个能够退出程序并不执行finally的情况。
public static void main(String[] args) { try { System.out.println("this is try"); System.exit(0); } catch (Exception e) { // TODO Auto-generated catch block e.printStackTrace(); } finally { System.out.println("this is finally"); } }
这段程序最后只会输出 this is try 这一句话,而不会输出 this is finally 。退出虚拟机会直接杀死整个程序,已经不是从代码的层面来终止程序了,所以finally不会执行。
看完了表面的方法,我们来继续学习一下System的源代码。还是老样子,找个jdk包打开rt.jar找到java.lang.System类。
首先映入眼帘的就是一个静态块:
/* register the natives via the static initializer. * * VM will invoke the initializeSystemClass method to complete * the initialization for this class separated from clinit. * Note that to use properties set by the VM, see the constraints * described in the initializeSystemClass method. */ private static native void registerNatives(); static { registerNatives(); }
native不用看了,本机方法。这是可以猜得到的,因为System类要使用输入和输出流可能会用到和操作系统相关的一些本机方法。那么在static块中调用了registerNatives()方法,这个方法是本地方法我们看不到具体实现。但是注释说了:“VM will invoke the initializeSystemClass method to complete the initialization for this class separated from clinit”。
那么JVM调用的initializeSystemClass方法是怎么实现的呢?
private static void initializeSystemClass() { props = new Properties(); initProperties(props); sun.misc.VM.saveAndRemoveProperties(props); lineSeparator = props.getProperty("line.separator"); sun.misc.Version.init(); FileInputStream fdIn = new FileInputStream(FileDescriptor.in); FileOutputStream fdOut = new FileOutputStream(FileDescriptor.out); FileOutputStream fdErr = new FileOutputStream(FileDescriptor.err); setIn0(new BufferedInputStream(fdIn)); setOut0(newPrintStream(fdOut, props.getProperty("sun.stdout.encoding"))); setErr0(newPrintStream(fdErr, props.getProperty("sun.stderr.encoding"))); loadLibrary("zip"); Terminator.setup(); sun.misc.VM.initializeOSEnvironment(); Thread current = Thread.currentThread(); current.getThreadGroup().add(current); setJavaLangAccess(); sun.misc.VM.booted(); }
这个方法就在System类中,但是我们刚才没有介绍,因为是private的方法,只是用来自己做注册使用。我整理了一下源代码去掉了无用的部分。这个方法的大概意思是说:1.初始化Properties 2.初始化输入、输出、错误流 3.进行一大堆配置。
可以注意其中的几行,setIn0,setOut0,setErr0这三个方法。这三个方法是System中public方法setIn,setOut,setErr内部调用的子方法。我们用这几个方法来设置这三个流。
public static void setIn(InputStream in) { checkIO(); setIn0(in); }
比如这是setIn方法,我们使用这个方法来设置输入流(此方法被使用的频率不是很高)。checkIO是检查IO流是否正确,setIn0是native方法,做真正的输入流替换工作。
private static native void setIn0(InputStream in);
Das obige ist der detaillierte Inhalt vonEingehende Analyse der Java-System-Systemklasse. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!