我正在尝试创建一个像通用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的对象可以以一种常见的方式访问它们的属性。 如果有一种设计模式可以更好地解决这个问题,那么我可以实现它。
答案 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
注释还会生成equals
和hashCode
方法,这些方法在对象存储在Set或Map集合中时非常重要。如果要检查两个对象是否相同(相同的字段值),也可能需要方法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。这通常是坏消息。
使用您的通用POJO,您在调用方法时无法获得IDE的支持(因为它们根本不存在)。此外,如果您忘记了某个字段或拼错了它,您将遇到问题。如果不担心字符串中的名称存在问题,也不可能进行重构。
如果您不是单独参与项目,您的同事是否能够理解并使用此解决方案?你能在6个月,一年,5年内回到它吗?
尽可能简单地使用属性方法(getXyz,setXyz)并根据其用途在具有不同类的对象上调用它们。
查看您的IDE,它可能会为您的字段自动生成getter和setter方法,因此实质上,您只需设置字段然后按一个按钮即可完成剩下的工作。 (IntelliJ IDEA和Eclipse也这样做,我也提到了两个。我非常肯定Netbeans也这么做。)
如果您不确定所需的课程,请为您的问题创建Use Cases。一个好的开始就是在用例文本和动词方法中为名词创建类。之后,您可以开始概括。 (虽然要注意API,框架和适合所有的解决方案,即使它们很诱人......我也喜欢它!直到我意识到我正在构建一个静态站点生成器而不是打印一些HTML-文件...通常几周后......)
说完这一切之后,可能会有框架从数据库模式(Hibernate?),Web服务等生成类,也许还有这样的方法。无论如何,这些仍然是具有实际属性方法等的类。
希望这有帮助!