对未导入的类进行操作

时间:2018-10-01 12:55:54

标签: java javafx import

我的问题是我们如何在Java中对未在程序中导入的类的对象进行操作?这是我的JavaFX应用程序中的一个示例:

for(String item: participantsLabel.getItems())

这是我的for循环,其中participantsLabel是ListView类的对象(只是某种表,我们可以在其中存储元素)。方法getItems()向我们返回participantsLabel对象中包含的元素的列表(特别是ObservableList对象,它是List类的子类)。因此,重点是-程序看不到任何错误-我们可以轻松导入列表,而无需导入List类。但是,例如,当我们想通过创建以下内容将此对象分配给某个变量时:

ObservableList ourList = participantsLabel.getItems();

编译器向我们显示一个错误-我们必须导入一个ObservedList类。有人可以解释一下我,然后我们如何在for循环中解释列表,而不导入其返回类型?

2 个答案:

答案 0 :(得分:1)

在编译器中,始终将完全限定的类名写入其输出的字节码中。导入只是语法糖,它使您可以使用不带包名的类名代替完全限定名。

仅当在代码中声明一个类型的变量,在强制转换中使用它或用于访问static成员或构造函数时,才需要导入。其他类型不需要导入,因为编译器能够根据所涉及类的声明/签名来确定表达式的类型。

在类文件上使用javap应该提供一些有关编译结果的见解:

源代码

import javafx.collections.FXCollections;
import java.util.ArrayList; // some unused import

public class SimpleProgram {

    public static void main(String[] args) {
        System.out.println(FXCollections.observableArrayList("a").get(0));
    }

}

javap -v SimpleProgram.class

[...]
public class SimpleProgram
  [...]
  this_class: #8                          // SimpleProgram
  super_class: #9                         // java/lang/Object
  interfaces: 0, fields: 0, methods: 2, attributes: 1
Constant pool:
   #1 = Methodref          #9.#23         // java/lang/Object."<init>":()V
   #2 = Fieldref           #24.#25        // java/lang/System.out:Ljava/io/PrintStream;
   #3 = Class              #26            // java/lang/String
   #4 = String             #27            // a
   #5 = Methodref          #28.#29        // javafx/collections/FXCollections.observableArrayList:([Ljava/lang/Object;)Ljavafx/collections/ObservableList;
   #6 = InterfaceMethodref #30.#31        // javafx/collections/ObservableList.get:(I)Ljava/lang/Object;
   #7 = Methodref          #32.#33        // java/io/PrintStream.println:(Ljava/lang/String;)V
   #8 = Class              #34            // SimpleProgram
   #9 = Class              #35            // java/lang/Object
  #10 = Utf8               <init>
  #11 = Utf8               ()V
  #12 = Utf8               Code
  #13 = Utf8               LineNumberTable
  #14 = Utf8               LocalVariableTable
  #15 = Utf8               this
  #16 = Utf8               LSimpleProgram;
  #17 = Utf8               main
  #18 = Utf8               ([Ljava/lang/String;)V
  #19 = Utf8               args
  #20 = Utf8               [Ljava/lang/String;
  #21 = Utf8               SourceFile
  #22 = Utf8               SimpleProgram.java
  #23 = NameAndType        #10:#11        // "<init>":()V
  #24 = Class              #36            // java/lang/System
  #25 = NameAndType        #37:#38        // out:Ljava/io/PrintStream;
  #26 = Utf8               java/lang/String
  #27 = Utf8               a
  #28 = Class              #39            // javafx/collections/FXCollections
  #29 = NameAndType        #40:#41        // observableArrayList:([Ljava/lang/Object;)Ljavafx/collections/ObservableList;
  #30 = Class              #42            // javafx/collections/ObservableList
  #31 = NameAndType        #43:#44        // get:(I)Ljava/lang/Object;
  #32 = Class              #45            // java/io/PrintStream
  #33 = NameAndType        #46:#47        // println:(Ljava/lang/String;)V
  #34 = Utf8               SimpleProgram
  #35 = Utf8               java/lang/Object
  #36 = Utf8               java/lang/System
  #37 = Utf8               out
  #38 = Utf8               Ljava/io/PrintStream;
  #39 = Utf8               javafx/collections/FXCollections
  #40 = Utf8               observableArrayList
  #41 = Utf8               ([Ljava/lang/Object;)Ljavafx/collections/ObservableList;
  #42 = Utf8               javafx/collections/ObservableList
  #43 = Utf8               get
  #44 = Utf8               (I)Ljava/lang/Object;
  #45 = Utf8               java/io/PrintStream
  #46 = Utf8               println
  #47 = Utf8               (Ljava/lang/String;)V
{
  public SimpleProgram();
    descriptor: ()V
    flags: (0x0001) ACC_PUBLIC
    Code:
      stack=1, locals=1, args_size=1
         0: aload_0
         1: invokespecial #1                  // Method java/lang/Object."<init>":()V
         4: return
      [...]
      LocalVariableTable:
        Start  Length  Slot  Name   Signature
            0       5     0  this   LSimpleProgram;

  public static void main(java.lang.String[]);
    descriptor: ([Ljava/lang/String;)V
    flags: (0x0009) ACC_PUBLIC, ACC_STATIC
    Code:
      stack=5, locals=1, args_size=1
         0: getstatic     #2                  // Field java/lang/System.out:Ljava/io/PrintStream;
         3: iconst_1
         4: anewarray     #3                  // class java/lang/String
         7: dup
         8: iconst_0
         9: ldc           #4                  // String a
        11: aastore
        12: invokestatic  #5                  // Method javafx/collections/FXCollections.observableArrayList:([Ljava/lang/Object;)Ljavafx/collections/ObservableList;
        15: iconst_0
        16: invokeinterface #6,  2            // InterfaceMethod javafx/collections/ObservableList.get:(I)Ljava/lang/Object;
        21: checkcast     #3                  // class java/lang/String
        24: invokevirtual #7                  // Method java/io/PrintStream.println:(Ljava/lang/String;)V
        27: return
      [...]
      LocalVariableTable:
        Start  Length  Slot  Name   Signature
            0      28     0  args   [Ljava/lang/String;
}

首先请注意,没有对应于导入的部分。 java.util.ArrayList完全丢失,并且所使用的所有类型/成员的名称都存储在常量池中。

   #6 = InterfaceMethodref #30.#31        // javafx/collections/ObservableList.get:(I)Ljava/lang/Object;
  #30 = Class              #42            // javafx/collections/ObservableList
  #31 = NameAndType        #43:#44        // get:(I)Ljava/lang/Object;
[...]
  #41 = Utf8               ([Ljava/lang/Object;)Ljavafx/collections/ObservableList;
  #42 = Utf8               javafx/collections/ObservableList
[...]
16: invokeinterface #6,  2            // InterfaceMethod javafx/collections/ObservableList.get:(I)Ljava/lang/Object;

这是ObservableList.get的调用。使用的方法存储在常量池中,并且由编译器确定,即使我们没有导入它。这只是编译器自动执行的操作。

您可能会注意到,甚至不需要使用全限定名称导入的类型:

   #9 = Class              #35            // java/lang/Object
  [...]
  #35 = Utf8               java/lang/Object

答案 1 :(得分:0)

在方法调用返回的值中,从方法的(getItems())签名可以清楚地看出编译器应使用哪个类。

相反,当您使用(非完全限定)符号时,需要导入它。或将其更改为完全合格-编译器再次完全知道要使用什么,而无需导入。