如何部分更新 Google 应用引擎数据存储中的实体
How to Update an Entity in a Google app engine data store partially
我在 Google App Engine for Data Store 中使用 Objectify。我有一个具有 6 个属性的实体。
例如,
密钥 (id)
名字
姓氏
年龄
mobile_Number
电子邮件
我需要为实体编写一个更新端点。我希望更新端点应该能够更新具有指定字段的实体,而不是整个实体。例如,如果我想更新 mobile_number,它应该单独更新手机号码。或者如果我想更新 firstName 它应该只更新那个。
为此,我需要编写一个通用方法来根据字段更新实体。
非常感谢!
您在事务中加载和保存实体。
ofy().transact(() -> {
Thing thing = ofy().load().key(key).now();
thing.setWhatever();
ofy().save().entity(thing);
});
事务保证所有工作单元都有效地串行执行。并非实际上 串行执行,但数据将保持一致,就好像所有工作单元都已串行化一样。
是的,我通过以下方式修复了此更新。
将输入参数作为 Hashmap 获取,我可以在其中获取要更新的用户属性。哈希映射的键应该与实体的 属性 值相同。
public Response update(HashMap<String, String> ProfileEntity) {
//update logic
}
并根据以下 if 条件更新值:
ProfileEntity existingEntity = getById(ProfileEntity.get("key"));
if(existingEntity == null)
System.out.println("Invalid Profile Key");
if(ProfileEntity.containsKey("firstName"))
existingEntity.setFirstName(ProfileEntity.get("firstName"));
if(ProfileEntity.containsKey("lastName"))
existingEntity.setLastName(ProfileEntity.get("lastName"));
if(ProfileEntity.containsKey("age"))
existingEntity.setAge(ProfileEntity.get("age")));
if(ProfileEntity.containsKey("mobile_Number"))
existingEntity.setMobileNumber(ProfileEntity.get("mobile_Number"));
super.save(existingEntity);
我为此使用了 spring BeanUtils。
public static String[] getNullPropertyNames(Object source) {
final BeanWrapper src = new BeanWrapperImpl(source);
java.beans.PropertyDescriptor[] pds = src.getPropertyDescriptors();
Set<String> emptyNames = new HashSet<String>();
for (java.beans.PropertyDescriptor pd : pds) {
Object srcValue = src.getPropertyValue(pd.getName());
if (srcValue == null) emptyNames.add(pd.getName());
}
String[] result = new String[emptyNames.size()];
return emptyNames.toArray(result);
}
public static void copyProperties(Object src, Object target) {
BeanUtils.copyProperties(src, target, getNullPropertyNames(src));
}
然后像这样使用你的更新
public Publisher update(@Named("id") Long id, Publisher publisher) throws Exception {
checkExists(id);
Publisher destination = get(id);
copyProperties(publisher, destination);
return insert(destination);
}
注意: 此方法不会覆盖 null 和列表属性。
我在 Google App Engine for Data Store 中使用 Objectify。我有一个具有 6 个属性的实体。 例如,
密钥 (id)
名字
姓氏
年龄
mobile_Number
电子邮件
我需要为实体编写一个更新端点。我希望更新端点应该能够更新具有指定字段的实体,而不是整个实体。例如,如果我想更新 mobile_number,它应该单独更新手机号码。或者如果我想更新 firstName 它应该只更新那个。
为此,我需要编写一个通用方法来根据字段更新实体。
非常感谢!
您在事务中加载和保存实体。
ofy().transact(() -> {
Thing thing = ofy().load().key(key).now();
thing.setWhatever();
ofy().save().entity(thing);
});
事务保证所有工作单元都有效地串行执行。并非实际上 串行执行,但数据将保持一致,就好像所有工作单元都已串行化一样。
是的,我通过以下方式修复了此更新。
将输入参数作为 Hashmap 获取,我可以在其中获取要更新的用户属性。哈希映射的键应该与实体的 属性 值相同。
public Response update(HashMap<String, String> ProfileEntity) {
//update logic
}
并根据以下 if 条件更新值:
ProfileEntity existingEntity = getById(ProfileEntity.get("key"));
if(existingEntity == null)
System.out.println("Invalid Profile Key");
if(ProfileEntity.containsKey("firstName"))
existingEntity.setFirstName(ProfileEntity.get("firstName"));
if(ProfileEntity.containsKey("lastName"))
existingEntity.setLastName(ProfileEntity.get("lastName"));
if(ProfileEntity.containsKey("age"))
existingEntity.setAge(ProfileEntity.get("age")));
if(ProfileEntity.containsKey("mobile_Number"))
existingEntity.setMobileNumber(ProfileEntity.get("mobile_Number"));
super.save(existingEntity);
我为此使用了 spring BeanUtils。
public static String[] getNullPropertyNames(Object source) {
final BeanWrapper src = new BeanWrapperImpl(source);
java.beans.PropertyDescriptor[] pds = src.getPropertyDescriptors();
Set<String> emptyNames = new HashSet<String>();
for (java.beans.PropertyDescriptor pd : pds) {
Object srcValue = src.getPropertyValue(pd.getName());
if (srcValue == null) emptyNames.add(pd.getName());
}
String[] result = new String[emptyNames.size()];
return emptyNames.toArray(result);
}
public static void copyProperties(Object src, Object target) {
BeanUtils.copyProperties(src, target, getNullPropertyNames(src));
}
然后像这样使用你的更新
public Publisher update(@Named("id") Long id, Publisher publisher) throws Exception {
checkExists(id);
Publisher destination = get(id);
copyProperties(publisher, destination);
return insert(destination);
}
注意: 此方法不会覆盖 null 和列表属性。