A structure is a collection of data where each element of data is addressed by a name.
{% hint style="success" %} Tip Underneath the hood, all CFML structures are based on the java.util.Map interface. So if you are coming from a Java background, structures are just HashMaps. {% endhint %}
As an analogy, think about a refrigerator. If we’re keeping track of the produce inside the fridge, we don’t really care about where in the fridge the produce is in or basically: order doesn’t matter. Instead we organize things by name, which are unique, and each name can have any value. Like the name grapes might have the value 2, then the name lemons might have the value 1, and eggplants the value 6.
{% hint style="info" %}
All CFML structures are passed to functions as memory references, not values. Keep that in mind, when working with structures. There is also the passby=reference|value
attribute to function arguments where you can decide if you want to pass by reference or value.
{% endhint %}
A structure is an unordered collection where the data gets organized as a key and value pair. CFML syntax for structures follows the following syntax:
produce = {
grapes = 2,
lemons = 1,
eggplants = 6
};
{% hint style="success" %}
Tip Please note that =
sign and :
are interchangeable in CFML. So you can use any to define your structures.
{% endhint %}
Since CFML is a case-insensitive language, the structure defined above will store the keys all in uppercase. If you want the exact casing to be preserved in the structure, then surround the keys with quotes ("
).
produce = {
"grapes" = 2,
"lemons" = 1,
"eggplants" = 6
};
The key is used as the address and the value is the data at that address. Please note that the value can be ANYTHING. It can be an array, an object, a simple value or even an embedded structure. It doesn't matter.
Retreiving values are also simple, just reference the structure by the key name:
writeOutput( "I have #produce[ "grapes" ]# grapes in my fridge!" );
writeOutput( "I have #produce[ "eggplants" ]# eggplants in my fridge!" );
I can also set new or override values of the structure a-la-carte:
// new key using default uppercase notation
produce.apples = 3;
// new key using case sensitive key
produce[ "apples" ] = 3;
// I just ate one grape, let's reduce it
produce[ "grapes" ] = 1; // or
produce.grapes--;
{% hint style="success" %}
Tip You can use the toString()
call on any structure to get a string representation of its keys+values: produce.toString()
{% endhint %}
You can also declare new structures via the structNew()
function. This just basically assigns a struct to a variable. You will then be responsible for filling that structure out with data.
produce = {};
produce = structnew();
But why would I ever use the function if I can do the {}
notation which looks nicer? The answer is that in CFML you can create different types of structures:
linked/ordered
- a struct with ordered keys that maintain insertion ordernormal
- an unordered structuresoft
- a struct with Java soft referenced values, which can be cleared by the garbage collector if memory is needed.weak
- a struct with Java weak referenced values, which do not prevent their referents from being garbage collected.
linkedList = structNew( 'ordered' );
cache = structnew( 'soft' );
Once you create structures you can use them in many funky ways. Please check out all the structure functions and all the structure modern member functions that are available to you.
As you can see, there are many cool methods from detecting keys, values, length, counts, etc. A very cool method is keyArray()
which gives you the listing of keys as an array:
You can use different constructs for looping over structures:
for
loopsloop
constructseach()
closures
for( var key in produce ){
systemOutput( "I just had #produce[ key ]# #key#" );
}
produce.each( function( key, value ){
systemOutput( "I just had #value# #key#" );
} );
As of now only Lucee allows you to leverage the each()
operations in a multi-threaded fashion. The structEach()
or each()
functions allows for a parallel
and maxThreads
arguments so the iteration can happen concurrently on as many maxThreads
as supported by your JVM.
structEach( struct, callback, parallel:boolean, maxThreads:numeric );
each( collection, callback, parallel:boolean, maxThreads:numeric );
This is incredibly awesome as now you callback will be called concurrently! However, please note that once you enter concurrency land, you should shiver and tremble. Thread concurrency will be of the utmost importance and you must make sure that var scoping is done correctly and that appropriate locking strategies are in place.
myStruct.each( function( key, value ){
myservice.process( value );
}, true, 20 );