Why cast after an instanceOf?

前端 未结 11 1849
情歌与酒
情歌与酒 2020-11-30 11:05

In the example below (from my coursepack), we want to give to the Square instance c1 the reference of some other object p1, but only i

相关标签:
11条回答
  • 2020-11-30 11:37

    Just to provide an update on this, Java 14 now provides pattern matching for instanceof, this allows you to check and cast in one fell swoop.

    This (old way):

    void outputValue(Object obj) {
        if (obj instanceof String) {                      // Compare
            String aString = (String) obj;                // New variable & explicit casting
            System.out.println(aString.toUpperCase());    // Access member
        }
    }
    

    Can be simplified to this:

    void outputValue(Object obj) {
        if (obj instanceof String aString) {              // Compare and cast (if true)
            System.out.println(aString.toUpperCase());    // Access member
        }
    }
    

    Reference: https://jaxenter.com/pattern-matching-for-instanceof-in-java-14-169830.html

    0 讨论(0)
  • 2020-11-30 11:39

    The variable p1 has whatever type it started with - let's say Shape. p1 is a Shape, and only a Shape, no matter that its current contents happen to be a Square. You can call - let's say - side() on a Square, but not on a Shape. So long as you are identifying the entity in question via the variable p1, whose type is Shape, you can't call side() on it, because of the type of the variable. The way Java's type system works, if you can call p1.side() when you happen to know it's a Square, you can always call p1.side(). But p1 can hold not just Square Shapes, but also (say) Circle Shapes, and it would be an error to call p1.side() when p1 held a Circle. So you need another variable to represent the Shape which you happen to know is a Square, a variable whose type is Square. That's why the cast is necessary.

    0 讨论(0)
  • 2020-11-30 11:45

    Not to be obnoxious, but you have to tell the compiler what you want to do because the alternative would be for it to guess what you're trying to do. Sure, you might think, "If I'm checking the type of an object, OBVIOUSLY that must mean that I want to cast it to that type." But who says? Maybe that's what you're up to and maybe it isn't.

    Sure, in a simple case like

    if (x instanceof Integer)
    {
      Integer ix=(Integer) x;
      ...
    

    My intent is pretty obvious. Or is it? Maybe what I really want is:

    if (x instanceof Integer || x instanceof Double)
    {
      Number n=(Number) x;
    ... work with n ...
    

    Or what if I wrote:

    if (x instanceof Integer || x instanceof String)
    

    What would you expect the compiler to do next? What type should it assume for x?

    RE the comments that instanceof is obsolete or otherwise a bad idea: It can certainly be mis-used. I recently worked on a program where the original author created six classes that all turned out to be pages and pages long, but identical to each other, and the only apparent reason for having them was so he could say "x instanceof classA" versus "x instanceof classB", etc. That is, he used the class as a type flag. It would have been better to just have one class and add an enum for the various types. But there are also plenty of very good uses. Perhaps the most obvious is something like:

    public MyClass
    {
      int foo;
      String bar;
      public boolean equals(Object othat)
      {
        if (!(othat instanceof MyClass))
          return false;
        MyClass that=(MyClass) othat;
        return this.foo==that.foo && this.bar.equals(that.bar); 
      }
      ... etc ...
    }
    

    How would you do that without using instanceof? You could make the parameter be of type MyClass instead of Object. But then there's be no way to even call it with a generic Object, which could be highly desirable in many cases. Indeed, maybe I want a collection to include, say, both Strings and Integers, and I want comparisons of unlike types to simply return false.

    0 讨论(0)
  • 2020-11-30 11:46

    If c1 is declared as a type of Square then casting is required. If it is a declared as an Object then casting is not needed.

    0 讨论(0)
  • 2020-11-30 11:49

    Keep in mind, you could always assign an instance of Square to a type higher up the inheritance chain. You may then want to cast the less specific type to the more specific type, in which case you need to be sure that your cast is valid:

    Object p1 = new Square();
    Square c1;
    
    if(p1 instanceof Square)
        c1 = (Square) p1;
    
    0 讨论(0)
提交回复
热议问题