模拟在我的类中使用的通用pojo

时间:2018-06-17 21:36:34

标签: java design-patterns pojo dto

我正在尝试创建一个像通用POJO一样工作的对象,因为我必须通过不同的对象传递它,我需要一个接口来访问它的属性。

目前我用一个带有getField方法的基础对象来做,我在其中输入参数的名称,我通过反射得到它,它是这样的:

public abstract class BasePOJO{

    public Object getField(String fieldName){

        try {
            return this.getClass().getField(fieldName).get(this);

        } catch (NoSuchFieldException ex) {
            Logger.getLogger(BaseRequest.class.getName()).log(Level.SEVERE, null, ex);
        } catch (SecurityException ex) {
            Logger.getLogger(BaseRequest.class.getName()).log(Level.SEVERE, null, ex);
        } catch (IllegalArgumentException ex) {
            Logger.getLogger(BaseRequest.class.getName()).log(Level.SEVERE, null, ex);
        } catch (IllegalAccessException ex) {
            Logger.getLogger(BaseRequest.class.getName()).log(Level.SEVERE, null, ex);
        }

        return null;
    }
}

我用这个扩展了这个类:

public class CredentialsPOJO extends BasePOJO{
    public String username;
    public String password;
}

并调用如下属性:

credentialsPojo.getField("username");

我已经快速创建了这个,它很糟糕但它确实有效。但我想改变它以更好地模拟那个对象。

Tambiénhecreado el siguiente:

public class BasePOJO{

    private Map<String, Object> parameters = new HashMap<>();


    public void addParameter(String paramName, Object value){
        parameters.put(paramName, value);
    }

    public Object getParameter(String paramName){
        return parameters.get(paramName);
    }
}

我更喜欢这个对象,它更多的OO解决方案,这个问题是它只返回一个Object类型

这个想法是使用这个POJO的对象可以以一种常见的方式访问它们的属性。 如果有一种设计模式可以更好地解决这个问题,那么我可以实现它。

3 个答案:

答案 0 :(得分:1)

这样的东西在PHP中起作用(例如Magento - 魔术方法),老实说我认为这是一种糟糕的机制。

为什么?

如果字段不存在,则没有IDE控件。您的类API不清楚,因此错误的风险非常高。您必须查看实现以了解可以使用的字段。在你的例子中,你也通过公开字段来打破封装,所以任何人都可以直接使用它,而无需你的方法。

在这些日子里,使用自动生成的getter / setter,没有必要这样做。如果你想摆脱样板getter / setter代码,你可以使用Lombok作为例子。

答案 1 :(得分:1)

术语澄清

POJO =普通的旧Java对象,也就是说,它不使用某些用于“装饰”的第三方注释。对象行为(Hibernate的实体注释,或者已经提到过生成方法的Lombok)。与你的情况相匹配的是 Bean =类,每个类字段都有getter和setter。

解决您的问题

// derive type from left side of the assignement operator
public <T> T getProperty(String paramName){
    return (T) parameters.get(paramName);
}

这样,您可以键入以下内容:

String name = pojo.getProperty("name");
Integer age = pojo.getProperty("age");

缺点

由于给定的方法只是显式地将地图值转换为&#34;赋值运算符左侧的任何类型&#34;,如何实现编译时检查将无法实现写完后立即给你警告:Integer age = getProperty("name");。注意,在执行代码时,还有运行时类型检查。这样的分配将抛出ClassCastException

请记住,只要您将映射定义为Map<String,Object>,就无法进行编译时检查 - 给定密钥下的值几乎可以是任何值。这也适用于反射方法,因为getField(String)返回隐藏在Object类型中的属性值。

要进行编译时检查,请阅读推荐部分。

<强>建议

我强烈建议您使用项目Lombok,正如@Peteef已经提到的那样。确保您还在IDE中安装了lombok插件,否则自动生成的方法会生成语法错误。

注释@Data将为每个非最终字段的每个字段和设置器生成getter。此外,@Data注释还会生成equalshashCode方法,这些方法在对象存储在SetMap集合中时非常重要。如果要检查两个对象是否相同(相同的字段值),也可能需要方法equals

@Data
public class Credentials {
    public String username;
    public String password;
}

注意,如果这些类应该用作简单数据持有者,您仍然可以使用带有公共字段的简单类而不使用getter / setter。当然,它不是OOP,但你不必盲目地遵循每一个约定(有时候它们会更加高效)。

public class Credentials {
    public String username;
    public String password;
}

答案 2 :(得分:0)

您即将创建inner platform。这通常是坏消息。

使用您的通用PO​​JO,您在调用方法时无法获得IDE的支持(因为它们根本不存在)。此外,如果您忘记了某个字段或拼错了它,您将遇到问题。如果不担心字符串中的名称存在问题,也不可能进行重构。

如果您不是单独参与项目,您的同事是否能够理解并使用此解决方案?你能在6个月,一年,5年内回到它吗?

尽可能简单地使用属性方法(getXyz,setXyz)并根据其用途在具有不同类的对象上调用它们。

查看您的IDE,它可能会为您的字段自动生成getter和setter方法,因此实质上,您只需设置字段然后按一个按钮即可完成剩下的工作。 (IntelliJ IDEA和Eclipse也这样做,我也提到了两个。我非常肯定Netbeans也这么做。)

如果您不确定所需的课程,请为您的问题创建Use Cases。一个好的开始就是在用例文本和动词方法中为名词创建类。之后,您可以开始概括。 (虽然要注意API,框架和适合所有的解决方案,即使它们很诱人......我也喜欢它!直到我意识到我正在构建一个静态站点生成器而不是打印一些HTML-文件...通常几周后......)

说完这一切之后,可能会有框架从数据库模式(Hibernate?),Web服务等生成类,也许还有这样的方法。无论如何,这些仍然是具有实际属性方法等的类。

希望这有帮助!