Neo4j-用于列表类型

时间:2019-01-30 00:30:03

标签: neo4j spring-data-neo4j neo4j-ogm

我试图为嵌套对象编写一个自定义转换器,以便将该对象保存为字符串在Neo4j数据库中。

我在自己的字段上使用@Convert批注,并传递了ImageConverter.class类的AttributeConverter

enter image description here

enter image description here

一切都按预期进行,并且我能够将Image类的字符串表示形式保存在Neo4j db中。

但是,现在我想使用List<Image>作为嵌套字段,而不是单个图像。在这种情况下,放置@Convert(ImageConverter.class)无效。

我看到有一个名为ConverterBasedCollectionConverter的类,当我有一个类型为List<LocalDateTime的字段时会被使用。

但是,在自定义转换器的情况下,我找不到如何使用此类的示例。

请任何人帮助我,或者是否有其他方法可以在类型List的字段上使用自定义转换器。

我在我的应用程序中使用Neo4j(3.4.1版)和Spring-data-neo4j(5.0.10.RELEASE)。我也在使用OGM。

PS:我知道建议将嵌套对象存储为与父对象建立关系的单独节点。但是,我的用例要求将对象存储为字符串属性,而不要存储为单独的节点。

此致

V

2 个答案:

答案 0 :(得分:1)

这并不像我想象的那么困难。

提供课程(摘要)

@NodeEntity
public class Actor {

  @Id @GeneratedValue
  private Long id;

  @Convert(MyImageListConverter.class)
  public List<MyImage> images = new ArrayList<>();
  // ....
}

MyImage尽可能简单

public class MyImage {

  public String blob;

  public MyImage(String blob) {
      this.blob = blob;
  }

  public static MyImage of(String value) {
      return new MyImage(value);
  }
}

和一个转换器

public class MyImageListConverter implements AttributeConverter<List<MyImage>, String[]> {

  @Override
  public String[] toGraphProperty(List<MyImage> value) {
      if (value == null) {
          return null;
      }
      String[] values = new String[(value.size())];
      int i = 0;
      for (MyImage image : value) {
          values[i++] = image.blob;
      }
      return values;
    }

    @Override
    public List<MyImage> toEntityAttribute(String[] values) {
      List<MyImage> images = new ArrayList<>(values.length);
      for (String value : values) {
          images.add(MyImage.of(value));
      }
      return images;
    }
}

将在保存时打印以下调试输出,我认为这是您想要的:

UNWIND {rows} as row CREATE (n:Actor) SET n=row.props RETURN row.nodeRef as ref, ID(n) as id, {type} as type with params {type=node, rows=[{nodeRef=-1, props={images=[blobb], name=Jeff}}]}

尤其是图像部分。

与此类似的测试方法

@Test
public void test() {
    Actor jeff = new Actor("Jeff");
    String blobValue = "blobb";
    jeff.images.add(new MyImage(blobValue));
    session.save(jeff);
    session.clear();
    Actor loadedActor = session.load(Actor.class, jeff.getId());
    assertThat(loadedActor.images.get(0).blob).isEqualTo(blobValue);

}

答案 1 :(得分:0)

我想出了解决我问题的方法。因此,如果您需要其他解决方案以及@meistermeier提供的解决方案,可以使用以下代码。

public class ListImageConverter extends ConverterBasedCollectionConverter<Image, String>{

public ListImageConverter() {
    super(List.class, new ImageConverter());
}

@Override
public String[] toGraphProperty(Collection<Image> values) {
    Object[] graphProperties = super.toGraphProperty(values);
    String[] stringArray = Arrays.stream(graphProperties).toArray(String[]::new);
    return stringArray;
}

@Override
public Collection<Image> toEntityAttribute(String[] values) {
    return super.toEntityAttribute(values);
}

}

ImageConverter类仅实现AttributeConverter<Image, String>,在其中我将Image对象序列化和反序列化到json /从json反序列化。

我之所以选择这种方法,是因为我在一个对象中有Image字段,而在另一个对象中有List<Image>。因此,只需将@Convert(ListImageConverter.class)更改为@Convert(ImageConverter.class),我就可以将列表以及单个对象保存在Neo4j数据库中。

注意:如果需要,可以跳过覆盖toEntityAttribute的方法。它并没有增加太多价值。 但是,您必须重写toGraphProperty,因为它会在Neo4j代码中检查名称为toGraphProperty的声明方法的存在。

希望这对某人有帮助!

此致

V