
Thunder-Struct
1266 words. Time to Read: About 12 minutes.I just found out about this super handy built-in class in Ruby called a Struct
, and I wanted to share.
What is a Struct
Very simply put, a Struct
is a data object that is simulates a quick way to declare a class. A common need in a program is to glue a bunch of related data together. You can think of our options for that as a sort of continuum.
Let’s say, for example, that you have a student grade that you want to keep track of. You could define a class to explicitly lay out your data structure.
Or, seeing all of the typing, you could violently overreact and go in the opposite direction.
For a simple data object like this where there aren’t really methods to define yet, I think a lot of people’s first thought is to try to be more in the middle of the spectrum.
But there’s another, even awesomer (yeah, I said it) way: the Struct
.
What Do I Do With It?
Structs
are cool because they give you a lot of flexibility in how you access and use the data without giving up any explicit-ness.
They also come pre-built with some of the features that you’d have to build yourself if you used an actual class
definition.
And if you need your own methods, you can add them via a block when creating them.
Neat, right? I bet right now, you’re feeling… inde-struct-ible?
But now, you might have the same questions that I did once I got to this point.
What’s the Benefit?
Why would I want to use a Struct
instead of a Class
or a Hash
? When is one better than the other? It’s all a matter of what you’re trying to do.
Static vs. Dynamic Attributes
You can’t easily add attributes to a Struct
, so in applications where your keys/attributes need to be dynamic or you don’t know what they’ll be ahead of time (think “word-counter”), you might be better suited with a Hash
. On the other hand, if you know exactly what keys you’ll need, the object-based “dot access” looks nice, and having a constructor can save you a lot of time typing. It can be a good, clean way of signaling your design intent. A good example of having well-defined attributes is if you work with an Address.
Less Error Prone
Structs
tend to be more rigid, which can help protect from errors and uncaught typos.
They’re Fast
Structs
can also be much faster to create than hashes.
They’re Compact
And, because they don’t have all of the methods that come with Hashes
, they end up using less memory. This isn’t a huge savings, but maybe it counts if you’re working in a constrained environment or with a bazillion data points.
However, if you need some of those methods, maybe you’re better off with a Hash
.
But Don’t Go Crazy
If you have a lot of methods and custom functionality, or if the object is a larger part of your application, it’s probably better to stick with our good old friend, the Class. While the upside of Structs
is that they are quick and easy to create on the fly, that’s also their downfall. Having too many structs
, or ones that are too large, can be hard to read and might miss out on the benefits of defining a Class the normal, idiomatic way. Just like most things in programming, it all comes down to trade-offs.
One Last Thing: OpenStructs
I thought I should add this in because, when I initially started researching, I thought OpenStructs
were the same as regular structs
. Some StackOverflow answers can be misleading. OpenStructs
get created in a similar manner to regular ones, except for the fact that they’re essentially anonymous.
The main difference is that you can dynamically add attributes to this kind.
If you tried that with a regular struct
, you’d get a NoMethodError
.
What’s the trade-off here? You don’t get to name it. Also, OpenStructs are outrageously slow and memory intensive . Check out these benchmarks.
Wrap-Up
So that’s it. I thought it was a super handy tool — kind of like a NamedTuple
in Python, but with even nicer syntax! Now we’re one trick closer to Ruby mastery!
More Resources:
Author: Ryan Palo | Tags: ruby tricks design-intent struct |Like my stuff? Have questions or feedback for me? Want to mentor me or get my help with something? Get in touch! To stay updated, subscribe via RSS