- Feature Articles
- CodeSOD
-
Error'd
- Most Recent Articles
- Secret Horror
- Not Impossible
- Monkeys
- Killing Time
- Hypersensitive
- Infallabella
- Doubled Daniel
- It Figures
- Forums
-
Other Articles
- Random Article
- Other Series
- Alex's Soapbox
- Announcements
- Best of…
- Best of Email
- Best of the Sidebar
- Bring Your Own Code
- Coded Smorgasbord
- Mandatory Fun Day
- Off Topic
- Representative Line
- News Roundup
- Editor's Soapbox
- Software on the Rocks
- Souvenir Potpourri
- Sponsor Post
- Tales from the Interview
- The Daily WTF: Live
- Virtudyne
Admin
Admin
Admin
On the topic of getters and setters in PHP, they're mostly pointless and they're a waste of resources especially without an opcode cache. On my setup, I made two crappy classes, one with five private fields encapsulated by five getter/setter pairs proxying the field (which looked almost exactly like java code, funnily enough) and another with five public fields, and called memory_get_usage() at the end after creating an instance. The script with the getter/setters used 59708 bytes of memory and the script with the public fields used 49244 bytes.
In the context of a class library of any significant size, such as a web site framework, these useless getters and setters can add up to a HUGE black hole for memory. I have been developing a framework for my employer in PHP (their choice, not mine. i wouldn't use it for this if i had the choice but having said that, PHP is not imposing any insurmountable restrictions on us) and when I refactored the class library to use public fields instead of getters/setters, the whole shebang ended up using 25% less memory per request at least, and when I replaced all calls to /(require|include)(_once)?/i with calls to a class loader which only includes files if the class is actually instantiated or extended, it was another 25% saving again. We went from 4 to 6.5 meg of memory per request to 1.5 to 3.2, and we have about 350 classes in our library and growing. Why interpret classes on every request if you're not actually using them?
I do something with __get() and __set() (and __isset() and __unset()) that the language should (but doesn't for backward compatibility's sake) - I put them in a class from which EVERY other class I write extends, and make them throw a damn exception if the field doesn't exist. I hate it how you can have this situation:
YUCK!
Ahh, that's better.
So we can have it both ways - crappy old code gets to benefit from PHP5's improved object support, and our nice new code can fail as it should. That's not a bad thing.
But the __get(), __set() and __call() 'magic' methods really shine for handling interface changes. With an interpreted language it's a bit harder to find all usages of a field or method even with the reasonably good support for code sensitivity provided by Eclipse PDT or Netbeans. Say we want to rename 'field' from the prior example to 'fieldWithBetterName', because 'field' was inappropriate, or no longer described the use accurately, or was just plain wrong. And say we wanted to change field2 to lazy load its value from the database because it isn't known initially...
And you can go backwards with __call(). This example is much more brittle, but it's not hard to clean up:
Getter and setter methods in PHP are good if the setter has to do something else, or if the getter has to lazy load something, or ensure something has been created, or whatever, but they're unnecessary and wasteful if they do nothing other than proxy the field.
PHP gets a lot of crap - especially around here - for being a toy language. I think it has a similar reputation to VB.NET. There are a lot of crappy crappy crappy developers writing the lowest imaginable quality code for them because the barrier to entry is so low, but if you've got no choice but to use the technology, you've your wits about you, you care about how your code looks and works, and you learn a few really simple techniques like the ones I hackily mentioned above, there's no reason for them to function as the toys they're perceived to be. VB.NET is very nearly functionally equivalent to C#. It's certainly trivial to port C# to VB.NET (not so much so the other way around depending on how much the Microsoft.VisualBasic namespace has been abused).
PHP has a LOT of faults, but if you're a halfway decent developer, there's no way your PHP code will look or act anything like all of the shit that informs the stereotype.
Admin
A watch doesn't allow you to break whenever a field changes. And in fact with .net the generated debug code doesn't support breakpoints based on memory locations changing, which is all that happens when a field is set.
BUT... as with all things programming, there are no dogmatic answers. Encapsulation is not equally important in your tiny desktop app (where you're recreating Notepad for some inexplicable reason) as it is in your library code. If all the code is in one assembly and app-specific a purist may argue that you don't need the members to be public at all. I'd be more pragmatic and say it doesn't matter in such a case. If you've used fields and the need arises to change what happens, or even to get that breakpoint possibility, just change it to a property and rebuild.
Obviously whenever you cannot be 100% sure that you can easily rebuild any and all code that might depend on the member, you should use properties and not fields, as the cost is extremely small (an extra "{ get; set; }" in current C#) and the potential benefit is huge.
Admin
Try to repeat your memory-usage test with 100 instances of each type and see what percentage difference you then get. As you know, methods (recalling that properties are pairs of methods following certain conventions that allow using the same syntax as that of reading and writing fields) are loaded into memory once for ALL instances, while each instance necessarily has it's own storage that the methods operate on, such as the field backing a property.
Personally I doubt I would ever encounter a situation in which the total memory consumed by all instances of a type increasing by 10K would make me even consider dropping the accessors in favor of public fields.
In order not to be a hypocrite I shall admit that I sometimes use public fields, but only for things like playing around, for code that isn't supposed to live long, and where I know I can rebuild all the code using the member whenever I'd like to, hence making it easy to turn a field into a property should the need arise.
Admin
Admin
People who make a bunch of private variables and then expose them all by get and set methods and think they are writing good code is the real WTF. The whole point of object orientation is encapsulation.
Think about your design first and keep rethinking it all the time and you will eventually learn how to properly encapsulate your data and where get and setters are appropriate.
Personally I've always been of the opinion that you make data private/protected , write the methods you need and add get and setters as needed oh and just because you've got a get or a set method doesn't mean you need the other one for that variable.
BTW all those folks bitching about C++ , the upcoming C++0x has accessors/properties from memory.
Admin
A prevailing point of view is 3 people? It's not like you've addressed every point here.
Admin
It's more about interpretation overhead than execution. I had reservations the overhead would increase drastically based on the number of instances created, and the quick test I ran seems to bear this out: the public field one uses 95k with 100 instances in memory and the getter/setter one uses 110k with 100 instances in memory.
Here's the code for the field one. You can guess what the getter/setter one looks like ;)
This is for just one class too. So without the instances in memory, we're looking at a 10k difference and with them we're looking at 15k, which also includes 100 timestamps.
but the whole point of my previous ramble was to demonstrate that getters/setters in php are not only a waste of resources, but they're also completely unnecessary because the language provides all the tools you need to transition between a public field and a getter/setter seamlessly and efficiently when you decide that the getter/setter needs to do more than simply proxy a field.
So it's basically a moot point... you can have it both ways with public fields and the __get()/__set() magic methods, so why not favour performance? if you've got a large enough class library with dozens to hundreds of fields, there is no disputing that the memory saving is significant when we're talking about 2-3k less per field per request compared with getter/setters.
PHP is also markedly slower when pushing another method on to the stack, so all of those unnecessary gets and sets are a waste of CPU cycles as well.
Of course! How do you think ClassLoader knows when to load ;) ?
(require|include)(_once)? all read the file and interpret the type immediately. If you register a class loader with spl_autoload_register, then register all your includes with the class loader, they all only get interpreted on demand.
Admin
Admin
Admin
he he... FatCatException ..
Admin
That comment is not a troll. It isn't an argument against making public fields private with public accessors. It is an argument against having every field of every class you make private and those fields having public accessors. The original poster says this limits encapsulation and it does! I (and probably the OP) would argue that such classes should "do more", while hiding more from other classes.
Admin
This actually came up as a response in an interview for a candiate recently. They were completely unaware that such a change is a breaking change. If you have other code that is expecting a field and you change to a property then everything is broken unless you recompile all of the client code. Even worse in languages/platforms that provide Reflection (such as .NET) where all of a suden the item will no longer be found.