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

时间:2013-08-23 17:32:09

标签: java spring spring-mvc reflection jackson

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

我知道有一些手动方法可以做到这一点:

  1. 在每个类中创建一个方法,将其特定属性序列化为property=value&形式(有点臭,因为它是一堆逻辑重复,我觉得)。
  2. 创建一个接受对象的函数,并使用反射动态读取所有属性(我猜是getters),并通过获取每个属性来构建字符串。我假设这是杰克逊一般用于序列化/反序列化的方式,但我真的不知道。
  3. 使用Jackson的某些功能来自定义序列化对象。我已经研究过自定义序列化程序,但它似乎是特定于一个类(所以我必须为每个我正在尝试序列化的类创建一个),而我希望有一个通用的方法。我只是无法理解如何将一个普遍应用于对象。一些链接:
  4. 使用ObjectMapper.convertValue(object, HashMap.class);,迭代HashMap的键/值对,然后构建字符串(我现在正在使用的,但我觉得转换过多?)。
  5. 我猜我还有其他的想法。
  6. 我调查过的主要帖子是Java: Getting the properties of a class to construct a string representation

    我的观点是,我有几个类,我希望能够序列化,而无需为每个类指定特定的内容。这就是为什么我认为使用反射的函数(上面的#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);

2 个答案:

答案 0 :(得分:15)

我认为,解决方案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;
}

答案 1 :(得分:2)

让我们 c1

c1.setC1prop1("C1prop1");
c1.setC1prop2("C1prop2");
c1.setC1prop3("C1prop3");

c1 转换为URI

    UriComponentsBuilder.fromHttpUrl("http://test.com")
            .queryParams(new ObjectMapper().convertValue(c1, LinkedMultiValueMap.class))
            .build()
            .toUri());

我们之后

    http://test.com?c1prop1=C1prop1&c1prop2=C1prop2&c1prop3=C1prop3