如何将ANY对象序列化为URI?


问题内容

我的基本问题是:是否已构建了可以自动执行此操作的任何内容(不必一定是流行的库/程序包的一部分)?我正在使用的主要工具是Spring(MVC)和Jackson2。

我了解有几种手动方法可以执行此操作:

  1. 在每个类中创建一个将其特定属性序列化为property=value&形式的方法(我觉得有点臭,因为这是一堆逻辑重复)。
  2. 创建一个接受对象的函数,并使用反射来动态读取所有属性(我猜是吸气剂),并通过获取每个属性来构建字符串。我假设这是Jackson一般用于序列化/反序列化的方式,但是我真的不知道。
  3. 使用Jackson的某些功能可自定义序列化对象。我研究了自定义序列化程序,但似乎它们特定于一类(因此我必须为我要序列化的每个类创建一个)。我只是在理解如何将对象普遍应用于对象方面遇到麻烦。一些链接:
  4. 使用ObjectMapper.convertValue(object, HashMap.class);,遍历HashMap的键/值对,然后构建字符串(这是 我现在正在使用 的字符串,但是我觉得转换量过多吗?)。
  5. 我猜还有其他我没想到的。

我的观点是,我希望可以对几个类进行序列化,而不必为每个类指定特定的东西。这就是为什么我认为使用反射(上面的#2)的函数是处理此问题的唯一方法(如果我必须手动执行的话)。

如果有帮助的话,我的意思是举两个例子:

public class C1 {
    private String C1prop1;
    private String C1prop2;
    private String C1prop3;

    // Getters and setters for the 3 properties
}

public class C2 {
    private String C2prop1;
    private String C2prop2;
    private String C2prop3;

    // Getters and setters for the 3 properties
}

(不,属性名称和约定不是我的实际应用程序使用的,这只是一个示例)

序列化的结果是C1prop1=value&C1prop2=value&C1prop3=valueC2prop1=value&C2prop2=value&C2prop3=value,但是只有一个地方定义了序列化的发生方式(已经在某处定义,或者由我手动创建)。

所以我的想法是,我将不得不使用以下形式(取自上面链接的帖子):

public String toString() {
    StringBuilder sb = new StringBuilder();
    try {
        Class c = Class.forName(this.getClass().getName());
        Method m[] = c.getDeclaredMethods();
        Object oo;
        for (int i = 0; i < m.length; i++)
        if (m[i].getName().startsWith("get")) {
            oo = m[i].invoke(this, null);
            sb.append(m[i].getName().substring(3) + ":"
                      + String.valueOf(oo) + "\n");
        }
    } catch (Throwable e) {
        System.err.println(e);
    }
    return sb.toString();
}

并对其进行修改以接受一个对象,并更改附加到的项目的格式StringBuilder。这对我有用,我现在不需要修改它的帮助。

再次重申,我的主要问题是,即使已经必须指定如何处理每个属性和值以及如何组合,也已经有一些东西已经在处理(可能简单的)序列化,而不是我不得不(迅速)修改上面的函数每?

如果有帮助,则其背景是我正在使用RestTemplate(Spring)向其他服务器发出GET请求,并且希望在URL中传递特定对象的属性/值。我了解我可以使用类似的方法:

restTemplate.getForObject("URL?C1prop1={C1Prop1}&...", String.class, C1Object);

我相信属性将被自动映射。但是就像我说的那样,我不想为每种对象类型制作不同的URL模板和方法。我希望有以下类似的东西:

public String getRequest(String url, Object obj) {
    String serializedUri = SERIALIZE_URI(obj);
    String response = restTemplate.getForObject("URL?" + serializedUri, String.class);
    return response;
}

SERIALIZE_URI我在哪里处理它。而且我可以这样称呼它getRequest("whatever", C1Object);getRequest("whateverElse", C2Object);


问题答案:

我认为,解决方案编号4可以。这很容易理解和清除。

我提出了类似的解决方案,其中可以使用@JsonAnySetter批注。请参见以下示例:

import com.fasterxml.jackson.annotation.JsonAnySetter;
import com.fasterxml.jackson.databind.ObjectMapper;

public class JacksonProgram {

    public static void main(String[] args) throws Exception {
        C1 c1 = new C1();
        c1.setProp1("a");
        c1.setProp3("c");

        User user = new User();
        user.setName("Tom");
        user.setSurname("Irg");

        ObjectMapper mapper = new ObjectMapper();
        System.out.println(mapper.convertValue(c1, UriFormat.class));
        System.out.println(mapper.convertValue(user, UriFormat.class));
    }
}

class UriFormat {

    private StringBuilder builder = new StringBuilder();

    @JsonAnySetter
    public void addToUri(String name, Object property) {
        if (builder.length() > 0) {
            builder.append("&");
        }
        builder.append(name).append("=").append(property);
    }

    @Override
    public String toString() {
        return builder.toString();
    }
}

上面的程序打印:

prop1=a&prop2=null&prop3=c
name=Tom&surname=Irg

您的getRequest方法可能如下所示:

public String getRequest(String url, Object obj) {
    String serializedUri = mapper.convertValue(obj, UriFormat.class).toString();
    String response = restTemplate.getForObject(url + "?" + serializedUri, String.class);
    return response;
}