Reputation: 6595
TL;TD In Java 8, is there a way for an interface default method to access instance variables?
For instance:
public interface Foo{
default getBazModified(){
return baz * 2;
}
}
public class Bar implements Foo {
int baz;
...
}
I know that sounds like travisty but is there a way to do anything like that in Java 8?
Or is the only way is to have an abstract class that implements Foo, which would have both the instance variable baz declared and have a default implementation of getBazModified()?
Upvotes: 9
Views: 12115
Reputation: 1
In essence your default method when running is running inside the instance of the class you are implementing. Therefore, instead of defining a getter method inside the class, you can always try to access the class parameters inside your default method using getDeclaredField. Keep in mind that you probably need to provide try..catch around it since if there is no parameter defined inside your class you get an exception.
public interface Foo {
default getBazModified() {
//since you have only one declared field in your class,
//you can access it via index number 0
// you can also use getDeclaredField("baz") if you
// prefer but then you are might get issues if you
//rename it. try.. catch would be needed here
int baz = (int) this.getClass().getDeclaredFields()[0].get(this);
return baz * 2;
}
}
public class Bar implements Foo {
int baz;
...
}
Upvotes: -1
Reputation: 132
No. Because variables used inside a default method of an interface have access only to the variables defined inside the interface.
public interface Foo {
int BAZ=1;
default int getBazModified(){
return BAZ * 2;
}
}
Remember variables defined inside an interface are public,static and final and they must be initialized.
Now,
public class Bar implements Foo {
int BAZ = 10;
}
within calling class..
Foo foo = new Bar();
System.out.print(foo.getBazModified()); // output: 2
You can only access instance variable inside "Bar" class by overriding default method. Like so,
public class Bar implements Foo {
int BAZ = 10;
@Override
public int getBazModified() {
return BAZ * 2;
}
}
And know if,
Foo foo = new Bar();
System.out.print(foo.getBazModified()); // output: 20 (access instance variable)
Upvotes: 3
Reputation: 45329
An interface doesn't "see" its implementation's instance fields. This is not specific to interfaces, any super-type wouldn't know about its sub-types' specific fields (or any member type for that matter).
Your solution in this case is to declare an abstract getter method in the interface and use it to read the property of the subclass:
public interface Foo{
default int getBazModified(){
return this.getBaz() * 2;
}
int getBaz();
}
And let the concrete class implement the getter:
public class Bar implements Foo {
int baz;
public int getBaz() {return this.baz;}
}
Upvotes: 17
Reputation: 44210
Did you try to compile it? The answer is no because the interface doesn't know whether there will even be a field named 'baz' or not.
I know that sounds like travisty but is there a way to do anything like that in Java 8?
Yes. An abstract class. That's literally exactly what abstract classes were designed for.
Upvotes: 1