Hi Tideon,
I too came to Cypher and Neo4J about a year ago and have had similar "attunement" issues (I have deep OO/Python/Javascript/Smalltalk experience). I call the structure that you contemplate for gd.BankAccounts
an OrderedDictionary
-- a list where each element is a pair and where order is preserved. As others have suggested, I think you've inadvertently created what we EE's would call an "impedance mismatch" between your contemplated structure and the natural semantics of Neo4J.
I've found it helpful to draw the data structure on a piece of paper or whiteboard. When I did this just now, I drew a circle for "gd", and an arc to another circle that I labeled "BankAccount". I draw each property as a horizontal rectangle divided into left and right cells. The left contains the property name, the right contains the property value. For each BankAccount
node, I added the following properties: name
, accountCode
, and accountNumber
. I drew two instances of BankAccount
(each a node), one for each element of your list. I connected those with a bidirectional "next/previous" relationship.
Now I transliterate that drawing into a Cypher query. I like to use the Neo4J browser (https://your.domain.name:7474) as a REPL so that I fiddle with it. I haven't tried it, but I think it looks something like:
MATCH (gd: Party)-[BANK_ACCOUNTS: LINK]->(abn_amro: BankAccount {name: "ABN-AMRO", accountInformation: "NL56 ABNA 123.456.789"})-[NEXT: LINK]->(ing: BankAccount {name: "ING", accountInformation: "123.456.789"}) RETURN gd
The advantage of doing it this way is that you gain all the power of Neo4J in managing these lists. In particular, each list can be queried and manipulated. After you've created a few thousand gd
instances, it's going to be tedious if you find yourself needing to add an additional field to each BankAccount
element and you've built out your original approach.
It's analogous to a design question that comes up in Python (and other OO languages) all the time -- when I want to model a dictionary of key-value pairs, do I use a dictionary or create a new class/object. Most languages (Cypher, Python, PHP, etc) have at least an implicit bias towards the answer to that question. In a language like C, Java, PHP and others, creating and then instantiating a new class is a heavyweight operation and might lead to the approach you've contemplated. In Neo4J/Cypher, creating nodes and relationships is MUCH easier.
One final thought that might help -- consider how you'd store this BankAccounts
field in relational DB like MySQL. While it might be tempting to express it as, for example, a JSON string that is stored in a string-valued column (VARCHAR
or whatever), that approach would give your DBAs apoplexy. If you were designing a SQL schema for this, you'd almost immediately start thinking about a separate "BankAccount" table, foreign keys, and joins. THAT is your tipoff that your best bet in Neo4J is promote the BankAccount
property to a relationship named bankAccounts
added to gd
and then connecting instances of a BankAccount
node in an list using instances of Relationship
.
I hope this helps!