文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

Nebula Graph和SpringBoot环境连接及查询如何实现

2023-07-06 12:35

关注

这篇文章主要讲解了“Nebula Graph和SpringBoot环境连接及查询如何实现”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“Nebula Graph和SpringBoot环境连接及查询如何实现”吧!

说明

Nebula Graph 是一款开源的、分布式的、易扩展的原生图数据库,能够承载包含数千亿个点和数万亿条边的超大规模数据集,并且提供毫秒级查询。

当前Nebula Graph的最新版本是3.2.1, 根据官方的文档进行配置

GQL 常用查询

下面列出一些常用的查询

-- 列出图空间SHOW SPACES;-- 列出tag(点类型)和edge(边类型), 需要先 USE 一个图空间SHOW TAGS;SHOW EDGES;

列出某一类型的点和边

MATCH ()-[e:follow]-() RETURN eMATCH (v:player) RETURN v

带条件的查询, 在结果数量较多时必须带limit, 否则Nebula会报错

match (v:ADDRESS)-[e]-() where id(v)==\"ADD:82388116\" return v,e limit 100

基础配置和使用

在上面的链接中, 提供了最小的配置和测试代码

pom.xml 增加包依赖

对于Nebula Graph 3.2.1, 需要使用3.0.0的版本. client的每个版本只能对应特定的一两个服务端版本

<dependency><groupId>com.vesoft</groupId><artifactId>client</artifactId><version>3.0.0</version></dependency>

Java调用

Java调用主要是三部分, 创建连接池, 创建会话, 执行查询

创建 NebulaPool 连接池

连接到地址127.0.0.1, 端口9669, 连接池大小100. 注意地址和端口是一个列表, Nebula是支持集群的. 连接时不需要用户和密码

NebulaPool pool = new NebulaPool();try {NebulaPoolConfig nebulaPoolConfig = new NebulaPoolConfig();nebulaPoolConfig.setMaxConnSize(100);List<HostAddress> addresses = Arrays.asList(new HostAddress("127.0.0.1", 9669));Boolean initResult = pool.init(addresses, nebulaPoolConfig);if (!initResult) {log.error("pool init failed.");return;}} catch ()//...
创建 Session 会话

创建会话时需要用户名和密码

Session session = pool.getSession("root", "nebula", false);
执行查询

创建一个SPACE, 然后使用这个SPACE, 创建一个TAG person, 创建一个EDGE like

String createSchema = "CREATE SPACE IF NOT EXISTS test(vid_type=fixed_string(20)); "+ "USE test;"+ "CREATE TAG IF NOT EXISTS person(name string, age int);"+ "CREATE EDGE IF NOT EXISTS like(likeness double)";ResultSet resp = session.execute(createSchema);if (!resp.isSucceeded()) {log.error(String.format("Execute: `%s', failed: %s",createSchema, resp.getErrorMessage()));System.exit(1);}

添加一个点记录

String insertVertexes = "INSERT VERTEX person(name, age) VALUES "+ "'Bob':('Bob', 10), "+ "'Lily':('Lily', 9), "+ "'Tom':('Tom', 10), "+ "'Jerry':('Jerry', 13), "+ "'John':('John', 11);";ResultSet resp = session.execute(insertVertexes);if (!resp.isSucceeded()) {log.error(String.format("Execute: `%s', failed: %s",insertVertexes, resp.getErrorMessage()));System.exit(1);}

查询

String query = "GO FROM \"Bob\" OVER like "+ "YIELD $^.person.name, $^.person.age, like.likeness";ResultSet resp = session.execute(query);if (!resp.isSucceeded()) {log.error(String.format("Execute: `%s', failed: %s",query, resp.getErrorMessage()));System.exit(1);}printResult(resp);

在 SpringBoot 项目中使用 Nebula Graph

pom.xml 增加包依赖
<dependency><groupId>com.vesoft</groupId><artifactId>client</artifactId><version>3.0.0</version></dependency>
Session工厂: NebulaSessionFactory.java

配合@Bean(destroyMethod = "close"), 创建一个工厂类, 接收pool并实现close()方法

public class NebulaSessionFactory {    private final NebulaPool pool;    private final String username;    private final String password;    public NebulaSessionFactory(NebulaPool pool, String username, String password) {        this.pool = pool;        this.username = username;        this.password = password;    }    public Session getSession() {        try {            return pool.getSession(username, password, false);        } catch (NotValidConnectionException|IOErrorException|AuthFailedException|ClientServerIncompatibleException e) {            throw new RuntimeException("Nebula session exception", e);        }    }    public void close() {        pool.close();    }}

为什么不直接将 NebulaPool 配置为Bean? 因为 Session 每次创建时需要带用户名密码, 将密码作为config注入到每个Service中肯定是大家都不愿意看到的.

配置修改: application.yml
myapp:  nebula:    hosts: @nebula.hosts@    username: @nebula.username@    password: @nebula.password@    max-conn: @nebula.max-conn@
Spring启动配置: NebulaGraphConfig.java

应用启动时读取配置, 创建 NebulaPool, 并实例化 NebulaSessionFactory, destroyMethod = "close", 这个表示在项目shutdown时会调用Bean的close方法释放资源.

@Configurationpublic class NebulaGraphConfig {    @Value("${myapp.nebula.hosts}")    private String hosts;    @Value("${myapp.nebula.max-conn}")    private int maxConn;    @Value("${myapp.nebula.username}")    private String username;    @Value("${myapp.nebula.password}")    private String password;    @Bean(destroyMethod = "close")    public NebulaSessionFactory nebulaSessionFactory() {        List<HostAddress> hostAddresses = new ArrayList<>();        String[] hostList = hosts.split(",[ ]*");        for (String host : hostList) {            String[] hostParts = host.split(":");            if (hostParts.length != 2 || !hostParts[1].matches("\\d+")) {                throw new RuntimeException("Invalid host name set for Nebula: " + host);            }            hostAddresses.add(new HostAddress(hostParts[0], Integer.parseInt(hostParts[1])));        }        NebulaPoolConfig poolConfig = new NebulaPoolConfig();        poolConfig.setMaxConnSize(maxConn);        NebulaPool pool = new NebulaPool();        try {            pool.init(hostAddresses, poolConfig);        } catch (UnknownHostException e) {            throw new RuntimeException("Unknown Nebula hosts");        }        return new NebulaSessionFactory(pool, username, password);    }}
Service调用

在 Service 中进行调用

@Service@Slf4jpublic class GraphServiceImpl implements GraphService {    @Autowired    private NebulaSessionFactory sessionFactory;    @Override    public <T> NebulaResult<T> query(String graphSpace, String gql) {        Session session = null;        try {            log.info("GQL: {}", gql);            session = sessionFactory.getSession();            NebulaResult<Void> res = query(session, "USE " + graphSpace);            if (!res.isSuccess() || res.getResults() == null || res.getResults().size() == 0) {                log.error("Failed to use space:{}", graphSpace);                return null;            }            if (!graphSpace.equals(res.getResults().get(0).getSpaceName())) {                log.error("Failed to use space:{}, result:{}", graphSpace, res.getResults().get(0).getSpaceName());                return null;            }            return query(session, gql);        } catch (IOErrorException e) {            log.error(e.getMessage(), e);            return null;        } finally {            if (session != null) {                session.release();            }        }    }    private <T> NebulaResult<T> query(Session session, String gql) throws IOErrorException {        String json = session.executeJson(gql);        return JacksonUtil.extractByType(json, new TypeReference<>() {});    }}
辅助类 NebulaResult.java 等
外层结构

这里定义了 json 格式响应的外层结构

@Datapublic class NebulaResult<T> implements Serializable {    private List<Error> errors;    private List<Result<T>> results;    @JsonIgnore    public boolean isSuccess() {        return (errors != null && errors.size() == 1 && errors.get(0).getCode() == 0);    }    @Data    public static class Error implements Serializable {        private int code;    }    @Data    @JsonIgnoreProperties(ignoreUnknown = true)    @JsonInclude(JsonInclude.Include.NON_NULL)    public static class Result<T> implements Serializable {        private String spaceName;        private List<Element<T>> data;        private List<String> columns;        private Error errors;        private long latencyInUs;    }    @Data    public static class Element<T> implements Serializable {        private List<Meta<T>> meta;        private List<Serializable> row;    }    @Data    public static class Meta<T> implements Serializable {        private String type;        private T id;    }}

内层因为区分Edge和Vertex, 结构不一样. 如果是混合返回的结果, 可以用 Serializable

String gql = "match (v:ADDR)-[e]-() where id(v)==\"ADD:123123\" return v,e limit 100";        NebulaResult<Serializable> res = graphService.query("insurance", gql);        log.info(JacksonUtil.compress(res));        Assertions.assertThat(res).isNotNull();

对于边, 需要使用结构化的ID

@Data@JsonIgnoreProperties(ignoreUnknown = true)@JsonInclude(JsonInclude.Include.NON_NULL)public class EdgeId implements Serializable {    private int ranking;    private int type;    private String dst;    private String src;    private String name;}

用这个结构进行查询

NebulaResult<EdgeId> res3 = graphService.query("t_test1", "MATCH ()-[e:follow]-() RETURN e");

对于点, ID就是String

NebulaResult<String> res2 = graphService.query("t_test1", "MATCH (v:player) RETURN v");

感谢各位的阅读,以上就是“Nebula Graph和SpringBoot环境连接及查询如何实现”的内容了,经过本文的学习后,相信大家对Nebula Graph和SpringBoot环境连接及查询如何实现这一问题有了更深刻的体会,具体使用情况还需要大家实践验证。这里是编程网,小编将为大家推送更多相关知识点的文章,欢迎关注!

阅读原文内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     813人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     354人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     318人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     435人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     224人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯