gpt4 book ai didi

neo4j - Neo4j/Spring-Data 中的懒惰/急切加载/获取

转载 作者:行者123 更新时间:2023-12-03 20:42:54 25 4
gpt4 key购买 nike

我有一个简单的设置并遇到了一个令人费解的(至少对我而言)问题:

我有三个相互关联的 pojo:

@NodeEntity
public class Unit {
@GraphId Long nodeId;
@Indexed int type;
String description;
}


@NodeEntity
public class User {
@GraphId Long nodeId;
@RelatedTo(type="user", direction = Direction.INCOMING)
@Fetch private Iterable<Worker> worker;
@Fetch Unit currentUnit;

String name;

}

@NodeEntity
public class Worker {
@GraphId Long nodeId;
@Fetch User user;
@Fetch Unit unit;
String description;
}

所以你有一个带有“currentunit”的 User-Worker-Unit,它在用户中标记,允许直接跳转到“当前单位”。每个User可以有多个worker,但是一个worker只能分配到一个unit(一个unit可以有多个worker)。

我想知道的是如何控制 @Fetch “User.worker”上的注释。我实际上希望仅在需要时才使用它,因为大部分时间我只与“ worker ”一起工作。

我经历了 http://static.springsource.org/spring-data/data-neo4j/docs/2.0.0.RELEASE/reference/html/我不太清楚:
  • worker 是可迭代的,因为它应该是只读的(传入关系) - 在文档中清楚地说明了这一点,但在示例中大部分时间都使用了“Set”。为什么?还是没关系...
  • 如何让工作人员仅在访问时加载? (懒加载)
  • 为什么我需要用@Fetch 注释甚至是简单的关系(worker.unit)。没有更好的办法吗?我有另一个具有许多这样简单关系的实体 - 我真的想避免仅仅因为我想要一个对象的属性而不得不加载整个图。
  • 我是否缺少 Spring 配置以便它可以用于延迟加载?
  • 有没有办法通过额外调用加载任何关系(未标记为@Fetch)?

  • 从我的角度来看,只要我想要一个 Worker,这个构造就会加载整个数据库,即使我大部分时间都不关心 User。

    我发现的唯一解决方法是使用存储库并在需要时手动加载实体。

    ------- 更新 -------

    我已经使用 neo4j 工作了很长时间,并找到了解决上述问题的方法,该解决方案不需要一直调用 fetch(因此不会加载整个图形)。唯一的缺点:它是一个运行时方面:
    import org.aspectj.lang.ProceedingJoinPoint;
    import org.aspectj.lang.annotation.Around;
    import org.aspectj.lang.annotation.Aspect;
    import org.aspectj.lang.annotation.Pointcut;
    import org.springframework.beans.factory.annotation.Autowired;
    import org.springframework.data.mapping.model.MappingException;
    import org.springframework.data.neo4j.annotation.NodeEntity;
    import org.springframework.data.neo4j.support.Neo4jTemplate;

    import my.modelUtils.BaseObject;

    @Aspect
    public class Neo4jFetchAspect {

    // thew neo4j template - make sure to fill it
    @Autowired private Neo4jTemplate template;

    @Around("modelGetter()")
    public Object autoFetch(ProceedingJoinPoint pjp) throws Throwable {
    Object o = pjp.proceed();
    if(o != null) {
    if(o.getClass().isAnnotationPresent(NodeEntity.class)) {
    if(o instanceof BaseObject<?>) {
    BaseObject<?> bo = (BaseObject<?>)o;
    if(bo.getId() != null && !bo.isFetched()) {
    return template.fetch(o);
    }
    return o;
    }
    try {
    return template.fetch(o);
    } catch(MappingException me) {
    me.printStackTrace();
    }
    }
    }
    return o;
    }

    @Pointcut("execution(public my.model.package.*.get*())")
    public void modelGetter() {}

    }

    您只需要调整应该应用方面的类路径:my.model.package..get())")

    我将方面应用于我的模型类上的 ALL get 方法。这需要一些先决条件:
  • 您必须在模型类中使用 getter(该方面不适用于公共(public)属性 - 您无论如何都不应该使用)
  • 所有模型类都在同一个包中(因此您需要稍微调整代码)-我想您可以调整过滤器
  • aspectj 作为运行时组件是必需的(使用 tomcat 时有点棘手)-但它可以工作:)
  • 所有模型类都必须实现 BaseObject 接口(interface),该接口(interface)提供:

    公共(public)接口(interface) BaseObject {
    公共(public) bool isFetched();
    }

  • 这可以防止双重提取。我只是检查一个强制性的子类或属性(即名称或除 nodeId 之外的其他内容)以查看它是否被实际获取。 Neo4j 将创建一个对象,但只填充 nodeId 并保持其他所有内容不变(因此其他所有内容均为 NULL)。

    IE。
    @NodeEntity
    public class User implements BaseObject{
    @GraphId
    private Long nodeId;

    String username = null;

    @Override
    public boolean isFetched() {
    return username != null;
    }
    }

    如果有人找到了一种无需那种奇怪的解决方法就可以做到这一点的方法,请添加您的解决方案:) 因为这个方法有效,但我会喜欢没有aspectj 的方法。

    不需要自定义字段检查的基础对象设计

    一种优化是创建一个基类而不是一个实际使用 bool 字段(加载 bool 值)并检查它的接口(interface)(因此您无需担心手动检查)
    public abstract class BaseObject {
    private Boolean loaded;
    public boolean isFetched() {
    return loaded != null;
    }
    /**
    * getLoaded will always return true (is read when saving the object)
    */
    public Boolean getLoaded() {
    return true;
    }

    /**
    * setLoaded is called when loading from neo4j
    */
    public void setLoaded(Boolean val) {
    this.loaded = val;
    }
    }

    这是有效的,因为在保存对象时返回“true”以进行加载。当方面查看对象时,它使用 isFetched() 这 - 当对象尚未检索时将返回 null。检索到对象后,将调用 setLoaded 并将加载的变量设置为 true。

    如何防止jackson触发延迟加载?

    (作为对评论中问题的回答 - 请注意,我还没有尝试过,因为我没有这个问题)。

    对于 jackson,我建议使用自定义序列化程序(参见即 http://www.baeldung.com/jackson-custom-serialization )。这允许您在获取值之前检查实体。您只需检查它是否已被获取,然后继续整个序列化或仅使用 id:
    public class ItemSerializer extends JsonSerializer<BaseObject> {
    @Override
    public void serialize(BaseObject value, JsonGenerator jgen, SerializerProvider provider)
    throws IOException, JsonProcessingException {
    // serialize the whole object
    if(value.isFetched()) {
    super.serialize(value, jgen, provider);
    return;
    }
    // only serialize the id
    jgen.writeStartObject();
    jgen.writeNumberField("id", value.nodeId);
    jgen.writeEndObject();
    }
    }

    Spring 配置

    这是我使用的示例 Spring 配置 - 您需要将包调整到您的项目:
    <?xml version="1.0" encoding="UTF-8" standalone="no"?>
    <beans xmlns="http://www.springframework.org/schema/beans"
    xmlns:context="http://www.springframework.org/schema/context"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xmlns:neo4j="http://www.springframework.org/schema/data/neo4j"
    xmlns:tx="http://www.springframework.org/schema/tx"
    xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.0.xsd
    http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-3.0.xsd
    http://www.springframework.org/schema/data/neo4j http://www.springframework.org/schema/data/neo4j/spring-neo4j-2.0.xsd http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-2.5.xsd">

    <context:annotation-config/>
    <context:spring-configured/>

    <neo4j:repositories base-package="my.dao"/> <!-- repositories = dao -->

    <context:component-scan base-package="my.controller">
    <context:exclude-filter type="annotation" expression="org.springframework.stereotype.Controller"/> <!-- that would be our services -->
    </context:component-scan>
    <tx:annotation-driven mode="aspectj" transaction-manager="neo4jTransactionManager"/>
    <bean class="corinis.util.aspects.Neo4jFetchAspect" factory-method="aspectOf"/>
    </beans>

    AOP 配置

    这是用于此工作的/META-INF/aop.xml:
    <!DOCTYPE aspectj PUBLIC
    "-//AspectJ//DTD//EN" "http://www.eclipse.org/aspectj/dtd/aspectj.dtd">
    <aspectj>
    <weaver>
    <!-- only weave classes in our application-specific packages -->
    <include within="my.model.*" />
    </weaver>
    <aspects>
    <!-- weave in just this aspect -->
    <aspect name="my.util.aspects.Neo4jFetchAspect" />
    </aspects>
    </aspectj>

    最佳答案

    自己找到了所有问题的答案:

    @Iterable:是的,iterable 可用于只读

    @load 访问:默认情况下不加载任何内容。并且自动延迟加载不可用(至少据我所知)

    对于其余的:
    当我需要建立关系时,我必须使用 @Fetch 或使用 neo4jtemplate.fetch 方法:

    @NodeEntity
    public class User {
    @GraphId Long nodeId;
    @RelatedTo(type="user", direction = Direction.INCOMING)
    private Iterable<Worker> worker;
    @Fetch Unit currentUnit;

    String name;

    }

    class GetService {
    @Autowired private Neo4jTemplate template;

    public void doSomethingFunction() {
    User u = ....;
    // worker is not avaiable here

    template.fetch(u.worker);
    // do something with the worker
    }
    }

    关于neo4j - Neo4j/Spring-Data 中的懒惰/急切加载/获取,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/8852491/

    25 4 0
    Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
    广告合作:1813099741@qq.com 6ren.com