1

我今天早些时候问了一个类似的问题,但现在我被告知要求已经改变。

我有 3 个课程 - ProductProductOffering(PO)和ProductOfferingLists(POL)。产品包含 PO,PO 包含 POL。所有这些类都扩展了一个名为 CMS 的抽象类。

我想将 PO 附加到 P,并将 POL 附加到 PO,但我正在寻找一种可重用的方法,因为除了类名之外,逻辑是相同的。

不幸的是,我的测试器类需要两个方法,而层次类需要强制转换来实现抽象 CMS 契约。

这让我发疯了。我怎样才能使它更可重用,所以我没有多种方法和铸造?

package puzzler;
import java.util.ArrayList;
import java.util.List;

public class Tester {
    public static void main(String[] args) {
        Product p = new Product();
        PO pa = new PO();
        POL po = new POL();

        List<PO> lpa = new ArrayList<PO>();
        List<POL> lpo = new ArrayList<POL>();

        attachChildToParent(lpa, p);
        attachChildToParent(lpo, pa);
    }

    static void attachChildToParent(List<PO> listChild, Product parent) {
        for (PO po : listChild) {
            parent.attach(po);
        }
    }

    static void attachChildToParent(List<POL> listChild, PO parent) {
        for (POL po : listChild) {
            parent.attach(po);
        }
    }
}

CMS 类:

package puzzler;

abstract class CMS {
    abstract void attach(CMS childNode);
}

产品类 - 拥有 PO 的层次结构的顶部:

package puzzler;
import java.util.List;

public class Product extends CMS {
    List<PO> lpo;

    public List<PO> getLpo() {
        return lpo;
    }

    public void setLpo(List<PO> lpo) {
        this.lpo = lpo;
    }

    @Override
    void attach(CMS childNode) {
        this.getLpo().add((PO)childNode);
    }
}

产品提供,由产品拥有,反过来又拥有一个产品提供列表

package puzzler;
import java.util.List;

public class PO extends CMS {
    List<POL> lpol;

    public List<POL> getLpo() {
        return lpol;
    }

    public void setLpo(List<POL> lpol) {
        this.lpol = lpol;
    }


    @Override
    void attach(CMS childNode) {
        this.getLpo().add((POL)childNode);
    }
}    

PO 拥有的产品提供列表 (POL)

package puzzler;

public class POL extends CMS {
    @Override
    void attach(CMS childNode) {
        throw new UnsupportedOperationException("Not supported yet.");
    }
}
4

3 回答 3

1

(使用 Java 7 语法,但很容易适应 Java 6)

如下定义您的 CMS 类:

public abstract class CMS<C extends CMS<?>> {
    public abstract void attach(C childNode);
}

然后您的 Product 类可以如下所示:

public class Product extends CMS<PO> {
    private List<PO> lpo = new ArrayList<>();

    public List<PO> getLpo() {
        return lpo;
    }

    public void setLpo(List<PO> lpo) {
        this.lpo = lpo;
    }

    @Override
    public void attach(PO childNode) {
        getLpo().add(childNode);
    }
}

然后你的 PO 类可以如下所示:

public class PO extends CMS<POL> {
    private List<POL> lpol = new ArrayList<>();

    public List<POL> getLpo() {
        return lpol;
    }

    public void setLpo(List<POL> lpol) {
        this.lpol = lpol;
    }

    @Override
    public void attach(POL childNode) {
        getLpo().add(childNode);

    }
}

然后你的 POL 类可以如下所示:

public class POL extends CMS<CMS<?>> {
    @Override
    public void attach(CMS<?> child) {
        throw new UnsupportedOperationException("Not supported yet.");
    }
}

最后,您的 Tester 类可能如下所示:

public class Tester {
      public static void main(String[] args) {
            Product p = new Product();
            PO po = new PO();

            List<PO> pos = new ArrayList<PO>();
            List<POL> pols = new ArrayList<POL>();

            attachChildToParent(pos, p);
            attachChildToParent(pols, po);
        }

    private static <C extends CMS<?>> void attachChildToParent(List<C> childNodes, CMS<C> parent) {
        for (C childNode : childNodes) {
            parent.attach(childNode);
        }
    }
}

只需要一种方法,不需要强制转换。

于 2012-08-17T14:51:24.780 回答
0

只需将泛型与组合模式一起使用,例如

  • 将 CMS 定义为接口
  • 定义单个类 PO,按类型参数化
  • 在 CMSImpl 或类似的东西中实现一次方法

http://en.wikipedia.org/wiki/Generics_in_Java

http://en.wikipedia.org/wiki/Composite_pattern

于 2012-08-17T14:21:55.687 回答
0

抽象类,如在实现类中使用的接口,需要在任何地方使用。

删除了额外的演员表和两种方法替换为一种。

import java.util.ArrayList;
import java.util.List;

public class Tester {
  public static void main(String[] args) {
    Product p = new Product();
    PO pa = new PO();
    POL po = new POL();

    List<CMS> lpa = new ArrayList<CMS>();
    List<CMS> lpo = new ArrayList<CMS>();

    attachChildToParent(lpa, p);
    attachChildToParent(lpo, pa);
  }

  static void attachChildToParent(List<CMS> listChild, CMS parent) {
    for (CMS po : listChild) {
      parent.attach(po);
    }
  }
}

abstract class CMS {
  abstract void attach(CMS childNode);
}

class Product extends CMS {
  List<CMS> lpo;

  public List<CMS> getLpo() {
    return lpo;
  }

  public void setLpo(List<CMS> lpo) {
    this.lpo = lpo;
  }

  @Override
  void attach(CMS childNode) {
    this.getLpo().add(childNode);
  }
}

class PO extends CMS {
  List<CMS> lpol;

  public List<CMS> getLpo() {
    return lpol;
  }

  public void setLpo(List<CMS> lpol) {
    this.lpol = lpol;
  }


  @Override
  void attach(CMS childNode) {
    this.getLpo().add(childNode);
  }
}
class POL extends CMS {
  @Override
  void attach(CMS childNode) {
    throw new UnsupportedOperationException("Not supported yet.");
  }
}
于 2012-08-17T15:27:49.370 回答