Reputation: 113
I tried to implement an Enum styled factory pattern as inner Enum, but it didn't work. Is there any solution without separating inner Enum into a new file? In other words, is it possible of inner Enum styled factory pattern?
The code is below.
public class SampleParent {
private class InnerChild { }
private class InnerChildA extends InnerChild { }
private class InnerChildB extends InnerChild { }
private class InnerChildC extends InnerChild { }
enum InnerChildEnum {
CHILD_A {
@Override
public InnerChild getInstance() {
return new InnerChildA(); // compile error
}
},
CHILD_B {
@Override
public InnerChild getInstance() {
return new SampleParent.InnerChildB(); // compile error
}
},
CHILD_C {
@Override
public InnerChild getInstance() {
return SampleParent.new InnerChildC(); // compile error
}
},
;
public abstract InnerChild getInstance();
}
private static class InnerChildFactoryEnumStyled {
public static InnerChild getInnerChild(InnerChildEnum child) {
return child.getInstance();
}
}
public static void main(String[] args) {
// I want to write this way
InnerChild child = InnerChildFactoryEnumStyled.getInnerChild(InnerChildEnum.CHILD_A);
}
}
The compile error message is below
$ javac SampleParent.java
SampleParent.java:12: error: non-static variable this cannot be referenced from a static context
return new InnerChildA();
^
SampleParent.java:18: error: non-static variable this cannot be referenced from a static context
return new SampleParent.InnerChildB();
^
SampleParent.java:24: error: cannot find symbol
return SampleParent.new InnerChildC();
^
symbol: variable SampleParent
3 errors
Upvotes: 6
Views: 383
Reputation: 113
The improvements code is here, in case.
public class SampleParent {
private String foo = "foo";
private abstract class InnerChild {
public abstract void doSomething();
}
private class InnerChildA extends InnerChild {
public void doSomething() {
System.out.println(foo); // refer sampleParent.foo, not original's foo
}
}
private class InnerChildB extends InnerChild {
public void doSomething() {
System.out.println(foo); // refer sampleParent.foo, not original's foo
}
}
private class InnerChildC extends InnerChild {
public void doSomething() {
System.out.println(outer().foo); // refer sampleParent.foo, not original's foo
}
}
enum InnerChildEnum {
CHILD_A {
@Override
public InnerChild getInstance() {
return sampleParent.new InnerChildA();
}
},
CHILD_B {
@Override
public InnerChild getInstance() {
return sampleParent.new InnerChildB();
}
},
CHILD_C {
@Override
public InnerChild getInstance() {
return sampleParent.new InnerChildC();
}
},
;
private static SampleParent sampleParent = new SampleParent();
public abstract InnerChild getInstance();
}
private static class InnerChildFactoryEnumStyled {
public static InnerChild getInnerChild(InnerChildEnum child) {
return child.getInstance();
}
}
public static void main(String[] args) {
System.out.println("Hello World.");
// I want to write this way
InnerChild childA = InnerChildFactoryEnumStyled.getInnerChild(InnerChildEnum.CHILD_A);
childA.doSomething();
InnerChild childB = InnerChildFactoryEnumStyled.getInnerChild(InnerChildEnum.CHILD_B);
childB.doSomething();
InnerChild childC = InnerChildFactoryEnumStyled.getInnerChild(InnerChildEnum.CHILD_C);
childC.doSomething();
}
public SampleParent outer() {
return SampleParent.this;
}
}
Upvotes: 1
Reputation: 131486
To instantiate a no static inner class(InnerChildA
, InnerChildB
, etc...), you need to qualify the instantiation with an instance of the enclosing type (SampleParent
).
You have to do something like that : new ExternalClass().new InternalClass()
In your case you could declare and instance a static SampleParent
instance in the InnerChildEnum
enum and reuse it in each implemented getInstance()
method of the enum values declaration.
private class InnerChild {
}
private class InnerChildA extends InnerChild {
}
private class InnerChildB extends InnerChild {
}
private class InnerChildC extends InnerChild {
}
enum InnerChildEnum {
CHILD_A {
@Override
public InnerChild getInstance() {
return sampleParent.new InnerChildA();
}
},
CHILD_B {
@Override
public InnerChild getInstance() {
return sampleParent.new InnerChildB(); // compile error
}
},
CHILD_C {
@Override
public InnerChild getInstance() {
return sampleParent.new InnerChildC();
}
};
private static SampleParent sampleParent = new SampleParent();
public abstract InnerChild getInstance();
}
Upvotes: 3
Reputation: 1835
Your inner classes are not static so they must refer to an instance of the enclosing class SampleParent
. Change your class declarations to
private static class InnerChild { }
private static class InnerChildA extends InnerChild { }
private static class InnerChildB extends InnerChild { }
private static class InnerChildC extends InnerChild { }
and you can e. g. return new InnerChildA();
in your enum.
Upvotes: 5