- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
有什么办法可以告诉ObjectOutputStream
应该在不使用关键字 transient
的情况下序列化可序列化类的哪些字段并且没有定义 serialPersistentFields
-大批?
背景:我需要使用注释来定义类的哪些成员应该被序列化(或者更好:不被序列化)。涉及的类必须实现接口(interface)Serializable
,但不是 Externalizable
,所以我不想为每个对象实现序列化/反序列化算法,而只想为它使用注释。我无法使用 transient
关键字,因为注释需要一些进一步的检查来确定一个字段是否应该被序列化。这些检查必须由 ObjectOutputStream
完成。 (或在我自己的 ObjectOutputStream
子类中)。我也无法定义 serialPersistentFields
-array 在每个类中,因为如前所述,在编译时并没有定义哪些字段应该被序列化。
因此,在受影响的类中唯一应该注意的是字段级别的注释( @Target(ElementType.FIELD)
)。
在过去的几天里,我尝试了很多方法,但没有找到一种有效的方法:ObjectOutputStream
有一个方法 writeObjectOverride(Object)
可用于在扩展 ObjectOutputStream
时定义自己的序列化过程实现.这仅适用于 ObjectOutputStream
用无参数构造函数初始化,否则 writeObjectOverride
永远不会被调用。但是这种方法需要我自己实现整个序列化过程,我不想这样做,因为它非常复杂并且默认已经实现了 ObjectOutputStream
.我正在寻找一种方法来修改默认的序列化实现。
另一种方法是扩展 ObjectOutputStream
再次覆盖 writeObjectOverride(Object)
(在拨打 enableReplaceObject(true)
之后)。在这种方法中,我尝试使用某种 SerializationProxy(请参阅 What is the Serialization Proxy Pattern? )将序列化对象封装在代理中,该代理定义了应序列化的字段列表。但是这种方法也失败了,因为 writeObjectOverride 然后也被称为代理中的字段列表( List<SerializedField> fields
),从而导致无限循环。
例子:
public class AnnotationAwareObjectOutputStream extends ObjectOutputStream {
public AnnotationAwareObjectOutputStream(OutputStream out)
throws IOException {
super(out);
enableReplaceObject(true);
}
@Override
protected Object replaceObject(Object obj) throws IOException {
try {
return new SerializableProxy(obj);
} catch (Exception e) {
return new IOException(e);
}
}
private class SerializableProxy implements Serializable {
private Class<?> clazz;
private List<SerializedField> fields = new LinkedList<SerializedField>();
private SerializableProxy(Object obj) throws IllegalArgumentException,
IllegalAccessException {
clazz = obj.getClass();
for (Field field : getInheritedFields(obj.getClass())) {
// add all fields which don't have an DontSerialize-Annotation
if (!field.isAnnotationPresent(DontSerialize.class))
fields.add(new SerializedField(field.getType(), field
.get(obj)));
}
}
public Object readResolve() {
// TODO: reconstruct object of type clazz and set fields using
// reflection
return null;
}
}
private class SerializedField {
private Class<?> type;
private Object value;
public SerializedField(Class<?> type, Object value) {
this.type = type;
this.value = value;
}
}
/** return all fields including superclass-fields */
public static List<Field> getInheritedFields(Class<?> type) {
List<Field> fields = new ArrayList<Field>();
for (Class<?> c = type; c != null; c = c.getSuperclass()) {
fields.addAll(Arrays.asList(c.getDeclaredFields()));
}
return fields;
}
}
// I just use the annotation DontSerialize in this example for simlicity.
// Later on I want to parametrize the annotation and do some further checks
@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
public @interface DontSerialize {
}
Field.class.getDeclaredField(...)
每次调用时都返回受影响字段的新实例:
public void setTransientTest() throws SecurityException,
NoSuchFieldException, IllegalArgumentException,
IllegalAccessException {
Class<MyClass> clazz = MyClass.class;
// anyField is defined as "private String anyField"
Field field = clazz.getDeclaredField("anyField");
System.out.println("1. is "
+ (Modifier.isTransient(field.getModifiers()) ? "" : "NOT ")
+ "transient");
Field modifiersField = Field.class.getDeclaredField("modifiers");
boolean wasAccessible = modifiersField.isAccessible();
modifiersField.setAccessible(true);
modifiersField.setInt(field, field.getModifiers() | Modifier.TRANSIENT);
modifiersField.setAccessible(wasAccessible);
System.out.println("2. is "
+ (Modifier.isTransient(field.getModifiers()) ? "" : "NOT ")
+ "transient");
Field field2 = clazz.getDeclaredField("anyField");
System.out.println("3. is "
+ (Modifier.isTransient(field2.getModifiers()) ? "" : "NOT ")
+ "transient");
}
1. is NOT transient
2. is transient
3. is NOT transient
Field field2 = clazz.getDeclaredField("anyField");
) 后,它已经丢失了 transient 修饰符。
ObjectOutputStream
并覆盖
ObjectOutputStream.PutField putFields()
并定义自己的 PutField 实现。 PutField 允许您指定序列化哪些(附加)字段,但不幸的是,该接口(interface)只有许多形式为
put(String name, <type> val)
的方法。并且在实现这些时,我无法将方法调用与调用它的类字段相关联。例如,当序列化声明为
private String test = "foo"
的字段时方法
put("test", "foo")
被调用,但我无法关联
name
的值(即
test
),类包含字段
test
因为没有对包含类的引用可用,因此无法阅读字段
test
的注释。 .
DontSerialize
的字段之外,我无法成功序列化所有字段。展示。
最佳答案
我会重新考虑“序列化”是否真的是你想做的事情。鉴于序列化规则依赖于在运行时定义的一些逻辑,反序列化过程将是一场噩梦。
有趣的问题,虽然。
关于java - 指定在 ObjectOutputStream 中(不)序列化哪些字段而不使用 transient 或 serialPersistentFields,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/5259647/
我知道这个问题已经被问过很多次了,但我找不到适合我的答案。 我在 Spring Roo 应用程序中有两个实体,它们处于多对多关系、发布和组件中。 首先,我通过以下方式获取现有版本的实例 selecte
我正在尝试将用户详细信息存储到下表中:user、role、user_role。尝试保存详细信息时,它会引发以下错误。 Error during managed flush [org.hibernate
我有两个 hibernate 实体 Coupon 和 CouponHistory,在 CouponHistory 和 Coupon 之间具有单向关系。 @Entity @Table(name = "v
我在外键 dimension_id 之一的表中有表 product。所以在服务层编写我的测试用例时它显示了错误。 这是我的测试用例 @Transactional(propagation = Propa
在 ARM 架构手册中提到缓存可以是 transient 的和非 transient 的,并且它是由实现定义的。我无法理解关于缓存的 transient 内存的概念和使用。我正在尝试编写启用 MMU
我有 2 个域模型和一个 Spring REST Controller ,如下所示: @Entity public class Customer{ @Id private Long id; @OneT
我知道这个问题在Stackoverflow中有很多问题,但是即使有很多答案,这些答案也帮不了我什么,也没有找到答案。 在我的WebAPP中,它可以正常工作,但是当我将其转换为API时,它失败了(主题标
我有以下域名 class User { Boolean accountLocked String password Boolean passwordExpired Bo
我写了一个 elisp 宏,在 transient-mark-mode 中保留区域: (defmacro keep-region (command) "Wrap command in code t
这是我的员工类(class): @Entity public class Employee { @Id @GeneratedValue private int id; private String f
我正在通读 Windows Phone 7.5 Unleashed,有很多代码看起来像这样(在页面的代码隐藏中): bool loaded; protected override void OnNav
我有一个自连接员工实体类,其中包含与其自身相关的 id、name 和 ref 列。我想创建它的新实例并将其保存到数据库。 首先我创建了一个 Employee 类的实例并将其命名为 manager。然后
我有一个用于添加新公寓的表单,在该表单中我有一个下拉列表,用户可以在其中选择负责的人员。 显然,当您从下拉列表中选择并尝试保存公寓时,我的应用程序认为该人已被修改。它给了我下面的错误,指示我应该首先保
我正在尝试保存一个复杂的对象,该对象内部有许多引用元素,而且它在大多数情况下都能完美运行。 但是在某些情况下,我们会遇到以下异常, object references an unsaved trans
我遇到了一些可能的问题答案,但这是关于从 Hibernate 3.4.0GA 升级到 Hibernate 4.1.8 的问题。所以这曾经在以前的版本下工作,我已经四处搜索了为什么它在这个新版本中出现了
似乎一遍又一遍地问这个问题,我仍然找不到解决我问题的答案。我在下面有一个域模型。每个新创建或更新的“安全用户”都需要我确保其具有配置文件,如果没有,则创建一个新的配置文件并分配给它。 配置文件的要求相
我很难调试为什么 JPA 不级联我的 @ManyToMany 关系。我发现的所有答案都与缺少级联语句有关。但我确实拥有它们并且仍然得到: Caused by: org.hibernate.Transi
例如,当我使用 transient 通过更改 translate(x, y) 的值来实现 2s 持续时间的动画时。如何获取0.5s时刻translate(x, y)的当前值? 最佳答案 我认为你做不到
我在尝试保存属于多对多关联的对象时收到 TransientObjectException。我有点理解为什么会这样,但想了解如何正确完成我正在尝试做的事情。 简而言之,我正在尝试做的事情: 我的应用程序
transient final int 和 transient final Integer 有什么不同。 使用 int: transient final int a = 10; 序列化前: a = 1
我是一名优秀的程序员,十分优秀!