Se você não quiser refatorar o código e puder usar o Java 8, é possível usar referências de método.
Uma demonstração simples primeiro (desculpem as classes internas estáticas)
public class JavaApplication14
{
static class Baz
{
private final int _int;
public Baz(int value){ _int = value; }
public int getInt(){ return _int; }
}
static class Bar
{
private final Baz _baz;
public Bar(Baz baz){ _baz = baz; }
public Baz getBar(){ return _baz; }
}
static class Foo
{
private final Bar _bar;
public Foo(Bar bar){ _bar = bar; }
public Bar getBar(){ return _bar; }
}
static class WSObject
{
private final Foo _foo;
public WSObject(Foo foo){ _foo = foo; }
public Foo getFoo(){ return _foo; }
}
interface Getter<T, R>
{
R get(T value);
}
static class GetterResult<R>
{
public R result;
public int lastIndex;
}
/**
* @param args the command line arguments
*/
public static void main(String[] args)
{
WSObject wsObject = new WSObject(new Foo(new Bar(new Baz(241))));
WSObject wsObjectNull = new WSObject(new Foo(null));
GetterResult<Integer> intResult
= getterChain(wsObject, WSObject::getFoo, Foo::getBar, Bar::getBar, Baz::getInt);
GetterResult<Integer> intResult2
= getterChain(wsObjectNull, WSObject::getFoo, Foo::getBar, Bar::getBar, Baz::getInt);
System.out.println(intResult.result);
System.out.println(intResult.lastIndex);
System.out.println();
System.out.println(intResult2.result);
System.out.println(intResult2.lastIndex);
// TODO code application logic here
}
public static <R, V1, V2, V3, V4> GetterResult<R>
getterChain(V1 value, Getter<V1, V2> g1, Getter<V2, V3> g2, Getter<V3, V4> g3, Getter<V4, R> g4)
{
GetterResult result = new GetterResult<>();
Object tmp = value;
if (tmp == null)
return result;
tmp = g1.get((V1)tmp);
result.lastIndex++;
if (tmp == null)
return result;
tmp = g2.get((V2)tmp);
result.lastIndex++;
if (tmp == null)
return result;
tmp = g3.get((V3)tmp);
result.lastIndex++;
if (tmp == null)
return result;
tmp = g4.get((V4)tmp);
result.lastIndex++;
result.result = (R)tmp;
return result;
}
}
Resultado
241
4
nulo
2
A interface Getter
é apenas uma interface funcional, você pode usar qualquer equivalente.
GetterResult
classe, acessadores retirados para maior clareza, contêm o resultado da cadeia getter, se houver, ou o índice do último getter chamado.
O método getterChain
é um pedaço de código simples e padronizado, que pode ser gerado automaticamente (ou manualmente quando necessário).
Estruturei o código de forma que o bloco de repetição seja evidente.
Essa não é uma solução perfeita, pois você ainda precisa definir uma sobrecarga getterChain
por número de getters.
Em vez disso, eu refatoraria o código, mas se não puder e você se descobrir usando longas cadeias de getter, muitas vezes, você pode considerar construir uma classe com sobrecargas que levam de 2 a, digamos, 10 getters.
null
verificações, poiswsObject.getFoo().getBar().getBaz().getInt()
já é um cheiro de código. Leia o que é a "Lei de Demeter" e prefira refatorar seu código de acordo. Então, o problema com osnull
cheques também desaparecerá. E pense em usarOptional
.