What's the point in having an abstract class with no abstract methods?

后端 未结 9 1366
一整个雨季
一整个雨季 2021-01-02 21:11

Can have an abstract class implementing all of its methods-- with no abstract methods in it.

Eg.:

public abstract class someClass          


        
相关标签:
9条回答
  • 2021-01-02 21:28
    • you can declare to implement an interface and don't provide implementation and then each child implicitly gets interface extended

    • you prevent to create instance of this class

    • you in future provide common implementation to all children
    0 讨论(0)
  • 2021-01-02 21:29

    It has a conceptual meaning: this class has a behaviour which makes no sense on its own.

    Granted, it's difficult to imagine such a scenario without well-defined extension points (i.e. abstract methods), but occasionally it will be a reasonably accurate model of your problem.

    You can have something like this:

    public abstract class ObjectWithId {
        private final String id;
    
        public ObjectWithId( String id ) {
           this.id = id;
        }
    
        public final String getId() {
           return id;
        }
    }
    

    And then you can extend it to declare different types of objects that have ids. Here you have a fully specified and implemented behaviour but no restriction on any other behaviours subclasses may exhibit.

    Note though that a much neater way to model the same thing is to use composition instead of inheritance.

    public final class ObjectWithId<T> {
        private final String id;
        private final T ob;
    
        public ObjectWithId( String id, T ob ) {
           this.id = id;
           this.ob = ob;
        }
    
        public String getId() {
           return id;
        }
    
        public T getObject() {
           return ob;
        }
    }
    

    But before generics were introduced (up to Java version 1.4), this wouldn't have been as elegant and obviously better than the abstract class solution because you'd have had to trade in type safety.

    0 讨论(0)
  • 2021-01-02 21:34

    The practical difference is that you can't create an instance of it. You would have to subclass it and create an instance of the subclass.

    As to WHY you would want to do this, in practice ... I'm hard pressed to think of a good reason. You could say that the class is only meaningful if someone creates a subclass that implements some function. But then why not make that function abstract in the super-class?

    I wouldn't rule out the possibility that someone might come up with some example where this makes sense, but I can't think of one. Just because it's possible to write a piece of code and that code compiles successfully doesn't mean that that it makes sense. I mean, I can write "total_price = item_price * zip_code + customer_height_in_cubits - 7.879", but that doesn't mean such a line of code would be meaningful.

    0 讨论(0)
  • 2021-01-02 21:36

    As you pointed out, you can prevent the class from being instantiated by making it's constructor private. Othere than that, there is no benefit whatsoever. This is probably supported just to provide language completeness.

    0 讨论(0)
  • 2021-01-02 21:44

    We generally use Abstraction concept with inheritance

    Consider using abstract classes if any of these statements apply to your situation:

    • You want to share code among several closely related classes.

    To answer your question,

    Why declare a class with concrete methods Abstract?

    One possible reason is to support inheritance without actually creating objects


    Assume you have two classes one Abstract and other Concrete

    Abstract class : AbsClass

    abstract class AbsClass {
      int a = 5;
    
      //Constructor
      public AbsClass() {
        System.out.println(a);
      }
    
      void methodA() {
        System.out.println(a + 10);
    
      }
    
    }
    

    and Concrete class : ConcreteClass

    class ConcreteClass {
      int a = 10;
    
     //Made the constructor Private to prevent from creating objects of this class
      private ConcreteClass() {
        System.out.println(a);
    
      }
    
      void methodA() {
        System.out.println(a + 10);
      }
    

    }

    The above two classes should function similarly (?) Until you try to Subclass them

    class AbsImplementer extends AbsClass {
    //Works fine
    
    }
    
    class ConcImplementer extends ConcreteClass {
    //Compilation Error Implicit super constructor ConcreteClass() is not visible
    
    
    }
    
    0 讨论(0)
  • 2021-01-02 21:47

    This can be useful for cases when the classes derived from the abstract base class must have some behaviour that is different from each other but that behaviour can not be abstracted as residing within a method that has the same signature for all the classes. Being unable to share a signature can occur if the different behaviour requires methods that are passed different primitive types. Because they use primitive types you can not use generics to express the similarity.

    An abstract base class without any abstract methods is acting a bit like a marker interface, in that it is declaring that implementing classes must provide some behaviour without having that behaviour encapsulated within a new method with a signature that is the same for all implementations. You would use an abstract base class rather than a marker interface when the implementing classes have some behaviour in common, especially if the base class can implement it for the derived classes.

    For example:

    abstract class Sender {
       protected final void beginMessage() {
          ...
       }
    
       protected final void endMessage() {
          ...
       }
    
       protected final void appendToMessage(int x) {
          ...
       }
     }
    
     final class LongSender extends Sender {
        public void send(int a, int b, int c) {
           beginMessage();
           appendToMessage(a);
           appendToMessage(b);
           appendToMessage(c);
           endMessage();
        }
     }
    
     final class ShortSender extends Sender {
        public void send(int a) {
           beginMessage();
           appendToMessage(a);
           endMessage();
        }
     }
    
    0 讨论(0)
提交回复
热议问题