Graph Data Modelling in Java

Alberto Venturini
The Startup
Published in
12 min readSep 21, 2020

Modelling data is a crucial aspect of software engineering. Choosing appropriate data structures or databases is fundamental to the success of an application or a service.

In this article, I will discuss some techniques related to modelling data domains with graphs. In particular, I will show how labelled property graphs, and graph databases, can be an effective solution to some of the challenges we sometimes encounter with other models, such as relational databases, when we deal with highly-connected data.

By the end of this article we will have a simple — but fully functional — implementation of an in-memory labelled-property graph in Java. We’ll use this graph to run some queries on a sample dataset.

All the code presented here can be found on GitHub.

Sample domain: book reviews

Before writing this article, I headed over to kaggle.com and browsed through some of the data sets available there. Eventually, I picked this book review data set which we’ll use as a running example throughout this article.

This data set contains the following CSV files:

  • BX-Users.csv, with anonymised user data. Each user has a unique id, location and age.
  • BX-Books.csv, which contains each book’s ISBN, title, author, publisher and year of publication. This file also contains links to thumbnail pictures, but we won’t use those here.
  • BX-Book-Ratings.csv, which contains a row for each book review.

If we picture this data set as an entity-relationship (ER) diagram, this is how it would look like:

Some fields are denormalized, e.g. Author, Publisher, and Location. While this might or might not be what we eventually want (depending on the performance characteristics of certain queries), at this point let's assume we want our data in normalized form. After normalization, this is our updated ER diagram:

Note that we split Location into 3 tables (City, State, and Country) because the original field contained strings such as "San Francisco, California, USA".

Now, suppose we store this data into a relational database. As a thought exercise, how easily can we express these two queries with SQL code?

  • Query 1: get the average rating for each author.
  • Query 2: get all books reviewed by users from a specific country.

If you’re familiar with SQL, you’re probably thinking joins: whenever we have relations that span multiple tables, we typically navigate the relations by joining pairs of tables via primary and foreign keys.

However, both queries involve a non-trivial amount of joins — especially the second one. This is not necessarily a bad thing — relational databases can be quite good at executing joins efficiently — but it might lead to SQL code that is difficult to read, maintain, and optimize.

Moreover, in order to navigate many-to-many relationships, we had to create join tables (e.g. BookRating or BookPublished). This is a common pattern which, however, adds some complexity to the overall schema.

We could denormalize some of the data, but this would have the side-effect of locking us into a specific view of our data and causing our model to be less flexible.

Domain modelling with Java classes

Now let’s suppose we want to store our entire data set in memory. The sample data set that we got from Kaggle contains less than a million entries, so it will easily fit in memory.

Storing our data in memory is a trivial but perfectly valid approach, especially if we want to support a read-only workload and we don’t need to guarantee write consistency and persistency. If we need to offer those guarantees, we can still store the data in memory, but we’ll probably need to ensure that our data structures are thread-safe and that we somehow persist the changes to non-volatile memory.

We’ll start with some classes, for example:

Soon, though, we are confronted with a question: how do we link these classes together? How do we establish relationships between books and authors?

In the case of Book and Author, we might consider that a book has one author (let's keep things simple and suppose each book has only one author), and use a direct reference:

This makes it trivial to get the author of a book. However, the reverse (getting all books written by an author) is more expensive, because we need to scan the entire list of books. For example, if we’re looking for all books by Dan Brown, we need to write something like this:

When we have many (e.g. millions) books, this will perform poorly. We could store references to books in the Author class itself:

However, this solution doesn’t “feel” good. Every time we want to change a book’s author, we have two places to change.

Besides, how should we handle relationships that contain extra data? For example, the relationship between a book and its publisher contains a piece of extra data, i.e. the year of publication. It’s not clear where we should put this field: should we declare it in the Book class? Then how do we handle cases when a book has multiple publishers and/or years of publication? Should we create a new class (e.g. BookPublished), essentially adopting the join table pattern of relational databases?

Labelled property graphs

I would argue that both the relational and the Java reference models share a common trait: they represent relationships as entity data.

In both cases, not only does an entity contain attributes about itself (e.g. a Book table contains its title and ISBN code) but it also contains data about how it's connected to other entities.

This entity-centric (or table-centric) model, which is adopted by RDBMSs, has been traditionally successful thanks to its efficiency in storing and retrieving huge amounts of entities. If our data has a lot of loosely-connected entities, a table-centric model works well. However, when our data has a lot of relationships, representing them as data might result in more complex queries and, overall, in a less flexible data model.

Graph models adopt a different approach: in a graph, relationships are modelled explicitly and are treated as first-class citizens of the data model, just like entities.

You may recall from mathematics that a graph is a collection of nodes (also known as vertices) and edges (sometimes called relationships). Each node stores some data, and each edge connects two nodes. Here’s a picture of a sample graph with 6 nodes and 6 edges:

Mathematical graph

On top of this, a labelled property graph model adds a few extra features:

  • Each node and each edge have a label that identifies their role in the data model.
  • Each node and each edge store a set of key-value properties.
  • Each edge has a direction, i.e. it is a directed graph (as opposed to undirected graphs, where edges don’t have directions)

This is essentially the model adopted by production graph databases such as Neo4j and Titan.

So how can we model the book review domain as a labelled property graph? Here’s my first take at it:

Nodes have labels (e.g. the Book node is labelled "Book") and some properties -- for example, the Book node has two properties, isbn and title. These properties correspond to entity attributes in our ER diagram.

Edges have labels too (e.g. the edge connecting User and Book is labelled "Reviewed"). Some edges also have properties -- for example, the Reviewed edge has the rating property, which stores the rating a user gave to a book, and the Published by edge has a year property, i.e. the year the book was published. Other edges don't have properties -- for example, the In City edge which connects a user to the city where they live doesn't have properties because we don't need to store extra data on that relationship.

The picture above represents the schema of our graph model. When we create a graph instance and store some data in it, here is how it could be pictured (this is just a subset of the entire graph):

This could have easily been drawn on a whiteboard. In fact, when it comes to connected data, adopting a graph model is arguably quite natural and intuitive.

For more information about labelled property graphs, and how to model data with them, I recommend checking out Neo4j’s Graph Modeling Guidelines or Kelvin Lawrence’s free book on Apache Gremlin.

Implementing a labelled property graph in Java

Let’s see how we can implement a labelled property graph in Java. All the code in this section can be found in this repository on Github (the repository also contains code to parse the CSV files from Kaggle.)

We’ll start by defining the Node and Edge classes. We'll also define a common superclass, GraphElement, which represents elements that have a label and properties.

Nothing surprising here, except maybe those outgoingEdges and incomingEdges fields in the Node class. This is essentially how we connect nodes and edges together, and how we'll navigate the graph to extract meaningful data (we'll see that soon.)

I chose to represent outgoingEdges and incomingEdges as lists, but these might as well be sets (e.g. hash sets or tree sets) or other structures. The choice depends on a number of factors (e.g. whether we need to guarantee uniqueness of each edge.) However, these consideration are beyond the scope of this article -- if you are looking for efficient in-memory graph databases, you might want to consider products such as Memgraph or Neo4j embedded. For this example I decided to keep things simple and use plain array lists.

Also, each node has an id field. Unsurprisingly, the primary function of this field is to guarantee uniqueness of each node.

Next, we’ll define the Graph class which exposes methods for creating nodes and edges:

The two maps nodeIdToNode and nodeLabelToNode allow us retrieve nodes by their id and label, respectively. This will become especially useful when we start writing queries.

We define a BookReviewGraph class as a subclass of Graph:

We don’t use random-generated ids (e.g. UUIDs); instead, we repurpose entity data as ids. This has a couple of advantages:

  • it’s easier to debug, and
  • it’s easier to query (if we’re looking for books by author, we can just get the node with id = "author-" + authorName)

Sample queries

Now that we have a basic graph implementation for our book review domain, let’s see how we can implement the two queries we introduced above:

  • Query 1: get the average rating for each author.
  • Query 2: get all books reviewed by users from a specific country.

The pattern for both queries will be the same: starting with a node (or a set of nodes), we navigate through edges and nodes, until we reach the data that we want to return.

In general, querying a graph means extracting a subgraph, i.e. a specific pattern of nodes and edges, that yields the desired answer.

Query 1: average rating for each author

Let’s start by writing a query that returns the average rating for a specific author.

The query will traverse a section of the graph in order to extract the data we want (average rating for each author.) We can break it down into 4 steps:

  1. Start from the node corresponding to the author we are interested in.
  2. Navigate through the “Written by” edges that point to the author node.
  3. Get the source nodes of the “Written by” edges. These will be book nodes.
  4. Navigate through the “Reviewed” edges that point to the book nodes and extract the rating property from these edges.

As a picture:

In code:

Using this query, we can easily find the average rating for each author:

Here’s the output of the query with the data set we got from Kaggle:

Gustavs Miller 10.0
World Wildlife Fund 10.0
Alessandra Redies 10.0
Christopher J. Cramer 10.0
Ken Carlton 10.0
Jane Dunn 10.0
Michael Easton 10.0
Howard Zehr 10.0
ROGER MACBRIDE ALLEN 10.0
Michael Clark 10.0
...

Query 2: books reviewed by users from a specific country

This query adopts the same pattern as the previous one. We start with country nodes and traverse the graph until we get to book nodes, then we collect book titles:

In code:

Here’s the output we get with the Kaggle data set, when we query for books reviewed in Italy:

La Tregua
What She Saw
Diario di un anarchico foggiano (Le formiche)
Kept Woman OME
Always the Bridesmaid
A Box of Unfortunate Events: The Bad Beginning/The Reptile Room/The Wide Window/The Miserable Mill (A Series of Unfortunate Events)
Aui, Language of Space: Logos of Love, Pentecostal Peace, and Health Thru Harmony, Creation and Truth
Pet Sematary
Maria (Letteratura)
Potemkin Cola (Ossigeno)
...

Improving performance with indices

In both queries above, the starting point into the graph is a precise node which we get via the graph.getNodeById() method. This works when we know the id of the starting node.

However, in many cases we might not have a starting node — instead, we might need to start with a set of nodes.

For example, suppose we want to code the following query:

  • Query 3: given a book title, get the average age of users who reviewed a book with that title.

Our starting set of nodes is made up of the books with the given title. We could go through each book node and filter out books with a non-matching title:

This is essentially what databases refer to as a full table scan: in order to find the books we want, we have to go through all of them sequentially. The cost of this operation is linear in the number of books, so this can potentially take a long time.

In almost all cases (except maybe when the sequence is very short) we can improve performance by using a map (essentially a simplified version of an index in a database).

First, we create a map (booksByTitleIndex) which we update every time we add a new book:

Now we can use this index at the beginning of our query:

When we run this query on the Kaggle data set, here’s the output we get with title “Dracula”:

30.338983050847457

A HashMap is a good choice when we want to find exact matches on a certain property, because the average cost of finding exact matches is constant (O(1)).

When we want to find entities based on the total order of a certain property (e.g. get all users whose age is above 25), we can use a different data structure, e.g. a TreeMap, where the cost of finding elements is logarithmic.

If we wanted to find elements by prefix, a Trie would be the right choice to implement an index.

Conclusion

In this article we’ve seen how we can model a data domain using graphs, and how to implement a simple labelled property graph in Java. The code is quite basic and lacks error handling and thread-safety, but it’s a starting point to understand and apply graph modelling to data domains.

In the past year, I have worked on a project that has involved the analysis of high volumes of interconnected data. Modelling this data as graphs has allowed my team and myself to have an intuitive understanding of how the data is connected: transitioning from whiteboard brainstorming to data model implementation was a straightforward process. Initially we were unsure how the data would eventually be queried; adopting a graph model has allowed us to keep our model flexible and easy to query throughout the project.

The last decade has seen a growing interest in graph databases in the software community. Most of the technology powering these databases is not really new; what’s new is the fact that we now have a lot of interconnected data, and we want to query this data based on how elements are connected together, rather than just the values they hold.

--

--