Accessing implementation specific methods on an object that is returned to its API

208 Views Asked by At

Let me start by abstractly formulating the problem: I have two public interface types. One of them contains a method which receives at least two instances of the other interface type. The implementation of the method depends on the implementation of the passed objects.

Consider the following public API, which consists of two interfaces:

public interface Node {
}

public interface Tree {
    void connect(Node parent, Node child);
}

Now, I want to implement that API, like so:

public class NodeImpl implements Node {
    private final Wrapped wrapped;

    public NodeImpl(Wrapped wrapped) {
        this.wrapped = wrapped;
    }

    public Wrapped getWrapped() {
        return wrapped;
    }
}

public class TreeImpl implements Tree {
    @Override
    public void connect(Node parent, Node child) {
        // connect parent and child using the wrapped object
    }
}

public class Wrapped {
    // wrapped object which actually represents the node internally
}

I need to access the wrapped objects in the connect method, which is impossible, because the getWrapped method is not part of the API. It is an implementation detail.

So the question is: How can I implement the connect method without leaking implementation detail to the API?

Here is what I tried so far:

  • Put the connect method in the Node interface and call parent.connect(child). This gives me access to the wrapped object of the parent, however the wrapped object of the child is still not available.

  • Just assume the passed Node is of type NodeImpl and use a downcast. This seems wrong to me. There might be other Node implementations.

  • Don't put the wrapped object in the node, but use a map in the TreeImpl that maps Node's to Wrapped objects. This is basically the same as above. It breaks down as soon as a Node instance is passed to the connect method, which has no associated mapping.

Please note, that the Node interface might contain methods. However, this is unimportant for this question.

Also, please note that I am in control of both: The interface declaration as well as the implementation.


Another attempt to solve this is to convert the connect method to a addChild method in the Node interface and to make the Node interface generic:

public interface Node<T extends Node<T>> {
    void addChild(Node<T> child);
}

public class NodeImpl implements Node<NodeImpl> {
    private final Wrapped wrapped;

    public NodeImpl(Wrapped wrapped) {
        this.wrapped = wrapped;
    }

    public Wrapped getWrapped() {
        return wrapped;
    }

    @Override
    public void addChild(Node<NodeImpl> child) {
    }
}

public class Wrapped {
    // wrapped object which actually represents the node internally
}

public Node<NodeImpl> createNode() {
    return new NodeImpl(new Wrapped());
}

private void run() {
    Node<NodeImpl> parent = createNode();
    Node<NodeImpl> child = createNode();
    parent.addChild(child);
}

Node and createNode are part of the public API. NodeImpl and Wrapped should be hidden. run is the client code. As you can see, NodeImpl has to be visible to the client, so this is still a leaking abstraction.

1

There are 1 best solutions below

8
On

if connect method needs to access the Wrapped object in each node, that means NodeImpl can be only connected to a NodeImpl, so no need to make it complex, add a method addChild or connect to Node interface, in NodeImpl implementation you can down cast the argument to NodeImpl, if there is a type mismatch you may throw a exception.

without down-casting you can use generics like, but i think the simple solution is to down cast the

interface NodeConnector<T extends Node>
{
  void  connect(T parent,T child);
}


public abstract class AbstractNode implements Node
{
  @Override
  public void connect(Node node)
  {

    NodeConnector<Node> nodeConnector = getNodeConnector();
    nodeConnector.connect(this, node);
    Node parent = this;
  }

  protected abstract NodeConnector<Node> getNodeConnector();


}

class NodeImpl extends AbstractNode
{
  @SuppressWarnings("unchecked")
  protected NodeConnector<Node> getNodeConnector()
  {
    return (NodeConnector) new NodeConnectorImpl();
  }
}



class NodeConnectorImpl implements NodeConnector<NodeImpl>
{
  @Override
  public void connect(NodeImpl parent, NodeImpl child)
  {

  }
}