How to extend a final class in Java

2019-01-19 10:51发布

Here is my problem actually I'm facing now. I have a class, let's say Foo and this class defines a method called getBar that returns a Bar instance. The class Bar is defined inside Foo and and is declared public static final. What I want to do is to define a class MyFoo that extends Foo but I also want to extend Bar with MyBar by adding my own functionalities (methods, properties,etc). I also want getBar to return MyBar.

The problem is Bar is final. Here is an illustration of what I want to do :

public class Foo {

  Bar bar = new Bar();

  public Bar getBar(){
      return bar;
  } 

   ....

   public static final class Bar {

    } 
}

What I want to do is :

public class MyFoo extends Foo {


   public MyBar getBar(){ // here I want to return an instance of MyBar

   }

  public static final class MyBar extends Bar { // That's impossible since Bar is final

  } 
}

How can I achieve that?

[EDIT] I am adding more details to my question. I am actually developing a plugin for Jenkins and after searching, I realize there is a plugin that offers the base functionalities that I want to do, Foo and Bar. I want to custom Foo and Bar, so it can fits my needs. Foo is the main class of the plugin and extends a class called Builder. Foo defines a method called perform which contains all the operations for performing a build. Bar contains operations for configurations purposes and Foo needs Bar to get those informations.

So what I want to do is to extend Foo with MyFoo and add some configurations with MyBar. And as you see, MyFoo will need to get those configuations by calling getBar();

The second thing is I have no control of the instanciation of MyFoo and MyBar

What blocks me, is that I cannot extend Bar. How can I achieve that?

7条回答
神经病院院长
2楼-- · 2019-01-19 11:30

You can't extend a class that is declared final. However you can create an intermediate class, called Wrapper. This Wrapper will basically contain an instance of the original class and provide alternative methods to modify the state of the object according what you want to do.

Of course this will not give access to the private & protected fields of the final class, but you can use its getters and setters methods, combined with the new methods & fields you declared in the Wrapper class to get the result you want to, or something relatively close.

Another solution is not to declare final a class if there is no valid reason to do it.

查看更多
地球回转人心会变
3楼-- · 2019-01-19 11:32

If it implements some interfaces, Decorator Design pattern could solve the Problem.

查看更多
beautiful°
4楼-- · 2019-01-19 11:36

You can't extend a final class - that's the purpose of the final keyword.

However, there are (at least) two work-arounds:

  1. The class is in the Jenkins project, which is open source, so you can just download the project, make the changes you want to the class and re-build the jar
  2. Somewhat of a hack, but you could use a library like Javassist to replace the implementation of the class in memory
查看更多
5楼-- · 2019-01-19 11:39

You cannot extend final classes, that is the point of declaring classes final. You can define an interface that both the final class inherit from and a wrapper class that delegates calls to the wrapped final class. The question is then why to make the class final in first place.

查看更多
Lonely孤独者°
6楼-- · 2019-01-19 11:43

For example we cannot add a StringBuffer in the TreeSet Collection, because TreeSet is not child of the Interface Comparable:

public class Main(){
public static void main(String[] args){
   TreeSet treeSetSB = new TreeSet();
   treeSetSB.add(new StringBuffer("A"));  //error, Comparable is not implemented
}

But we can use a wrapper to implement the Comparable:

class myWrap implements Comparable{
    TreeSet treeset;
    public myWrap() {
        this.treeset=new TreeSet<>();

    }
    public TreeSet getTreeset() {
        return treeset;
    }
    public void setTreeset(TreeSet treeset) {
        this.treeset = treeset;
    }
}


public class Main(){
public static void main(String[] args){

   myWrap myWrap =new myWrap();
   TreeSet myTrs  =myWrap.getTreeset();
   myTrs.add("b");   // no error anymore
   myTrs.add("a");
   myTrs.add("A");
   System.out.println(myTrs);
}
查看更多
倾城 Initia
7楼-- · 2019-01-19 11:48

You can't, basically. It looks like the developer who designed Bar intended for it not to be extended - so you can't extend it. You'll need to look for a different design - possibly one which doesn't use inheritance so much... assuming you can't change the existing code, of course.

(It's hard to give more concrete advice without knowing more about the real goal here - the design of the overall system.)

查看更多
登录 后发表回答