XStream short dynamic aliases

2019-02-14 19:05发布

I want to have short names for classes, now i can do it with aliases

XStream x = new XStream();
x.alias("dic", Dic.class);

but i have to define alias manually for every class, is there any way to configure xstream to do it automatically?

3条回答
Luminary・发光体
2楼-- · 2019-02-14 19:32

The only other alternative is to use XStream annotations:

package com.my.incredibly.long.package.name;

@XStreamAlias("dic")
public class Dic {
  ...

Then, in your code where you configure XStream:

xstream.processAnnotations(Dic.class);
// OR
xstream.autodetectAnnotations(true);

The problem, however, is that in order to deserialize your classes XStream has to know their aliases already, so autodetectAnnotations(true) is NOT going to help unless you can guarantee that you will serialize the class prior to deserializing it. Plus (and this may or may not be a concern for you) you're introducing an explicit XStream dependency to your objects.

I've ended up tagging all the classes I need serialized (several choices here: annotate them via XStream or your own annotation; have them implement marker interface; grab all the classes from particular package(s)), autodetecting them on load and explicitly configuring XStream instance to alias them as class name without package name.

查看更多
Luminary・发光体
3楼-- · 2019-02-14 19:35

They way I solved is:

1.- When I create xstream I override its wrapmapper method

XStream xstream = new XStream() {
    @Override
    protected MapperWrapper wrapMapper(MapperWrapper next) {
        return new MyClassAliasingMapper(next);
    }
};

2.- with

public class MyClassAliasingMapper extends ClassAliasingMapper {

    public MyClassAliasingMapper(Mapper wrapped) {
        super(wrapped);
    }

    @Override
    public Class realClass(String elementName) {
        try {
            return <… your own class …>
        } catch (Exception e) {
            // do nothing we fall back on super's implementation
        }
        return super.realClass(elementName);
    }

    @Override
    public String serializedClass(Class type) {
        try {
            return <… your own element name …>
        } catch (Exception e) {
            // do nothing we fall back on super's implementation
        }
        return super.serializedClass(type);
    }
}
查看更多
Melony?
4楼-- · 2019-02-14 19:41

Internally, XStream uses its Mapper interface to handle the mapping of classes and fields to their corresponding names in the XML. There are a large number of implementations of this interface. The XStream class itself can take a Mapper in its constructor. You might want to check out the source code of that class to see which Mapper implementation it uses by default, and then write your own implementation that automatically does your aliasing for you. ClassAliasingMapper looks useful, for example.

查看更多
登录 后发表回答