java - 有没有直接的方法来计算原始字段的大小?

标签 java serialization reflection primitive-types

我正在编写一个小型序列化实用程序,我想计算将对象写入字节数组需要保留多少字节。 假设我有一个包含一些字段/getter 的类,比如

int age;
String name;
Colour colour;
boolean active;
...

我以为我可以使用反射来遍历字段并查看它们的大小,但我所能做的就是检查它们是否是原始的。这是一个开始,因为我会使用 short 作为对象的偏移量。好的,我可以用

if(field.getType() == int.class) {
    size = Integer.BYTES;
}

或使用 map 。这并不难,因为我只需要知道图元的大小。但出于好奇:有什么直接的方法吗?

编辑澄清: 我用它来序列化数据,基本上类似于 DataOutput/InputStream,但我想对自己进行一些调整。

API 类如下所示:

public final void write(int i) {
    checkBounds(Integer.BYTES);
    PrimitiveWriteUtil.write(i, data, marker);
    advance(Integer.BYTES);
}

实际工作由实用类完成:

public static void write(int i, byte[] target, int pos) {
    target[pos] =       (byte) ((i >>> 24) & 0xFF);
    target[pos + 1] =   (byte) ((i >>> 16) & 0xFF);
    target[pos + 2] =   (byte) ((i >>>  8) & 0xFF);
    target[pos + 3] =   (byte) ((i >>>  0) & 0xFF);
}

所以,我想知道我的字节数组所需的大小,这样我就可以避免以后检查和调整大小的工作。

最佳答案

好吧,您已经了解了 BYTES 字段,问题是,这些字段是在相应的包装器类型中声明的,并且没有简单的方法来获取原始类型的包装器类型。就代码大小而言,获取它的最简单方法是通过 get 方法读取字段,这意味着原始值的包装:

public class FieldSize {
    static final int BOOLEAN_SIZE = 1; // adjust to your storage method
    static final int REF_SIZE = 2;     // dito
    private static int getSize(Field f, Object o) {
        try {
        return f.getType().isPrimitive()? f.getType()==boolean.class? BOOLEAN_SIZE:
               f.get(o).getClass().getField("BYTES").getInt(null): REF_SIZE;
        } catch(ReflectiveOperationException ex) {
            throw new AssertionError(ex);
        }
    }
    boolean z;
    byte b;
    short s;
    char c;
    int i;
    long l;
    float f;
    double d;
    String str;

    public static void main(String[] args) {
        System.out.println("total: "+calculateSize(new FieldSize())+" size");
    }
    static int calculateSize(Object o) {
        int total=0;
        for(Class cl=o.getClass(); cl!=null; cl=cl.getSuperclass()) {
            for(Field f: cl.getDeclaredFields()) {
                if(Modifier.isStatic(f.getModifiers())) continue;
                int size=getSize(f, o);
                System.out.println(f.getName()+" ("+f.getType()+"): "+size);
                total+=size;
            }
        }
        return total;
    }
}

但是在底层操作上,这并不是最简单的方案,如果你不喜欢,我完全理解。

在那种情况下,答案是没有简单/直接的方法来获取这些数字,因此无论是从类型到数字的映射还是对这八种可能性的线性探测,都是不可避免的,因此也是最简单的(可用)解决方案。

关于java - 有没有直接的方法来计算原始字段的大小?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/38302257/

相关文章:

java - 如何将 Flux<List<T>> 扁平化为 Flux<T>?

java - Android 消息/处理程序机制 - 安全吗?

c# - 如何在 C# 中使用反射获取类型的完全限定名称

java - 在可序列化对象上获取 NotSerializableException

serialization - 如何为 kotlin 异常生成 serialVersionUID?

c# - 通过字符串名称设置/获取类属性

java - 如何获取注解上方的注解

java - 访问/更改 JEditorPane 的 html 加载元素 + Unicode (Java) 的 HTMLEditorKit 问题

java - URL 的解析 Intent 有时为 null,抛出 NullPointerException

android - 如何使用 parcelable 将对象从一个 Android Activity 发送到另一个?