Or perhaps slightly more accurately, both subclassing and convention over configuration would appear to be a form of programming by difference or "refinement".
Wednesday, January 11, 2012
Convention over configuration is subclassing?
Friday, December 9, 2011
Ruby (and Rails) scalability?
Node.js did really well on tasks that have lots of concurrent requests that are mostly waiting, did OK on basic static serving tasks and not so well on compute-intensive tasks.
Having developed an interest in minimal web-servers, I wondered how Sinatra and, by association, Ruby on Rails would do.
For Sinatra I used the scanty blog engine and the basic "Hello World" example:
require 'sinatra'
get '/hi' do
"Hello World!"
end
For Ruby on Rails, I used the blog tutorial "out of the box", invoking it with "
rails s"
to start the server. In addition I also had RoR just serving a static file instead of the database-backed blog. All this on my new dev machine, a 2011 MacBook Air with 1.8 GHz Intel Core i7 and 4 GB of DRAM. I also discovered that httperf is a much better benchmark program for
my needs than ab. I used
it with 100 requests per connection, a burst length of 100 and a sufficient number
of connections to get stable results without taking all day.| Platform | # requests/sec |
| Apache | 5884 |
| Sinatra Hello World | 357 |
| Ruby on Rails static page | 312 |
| Sinatra scanty blog | 101 |
| Ruby on Rails blog | 17 |
Is this considered normal or am I doing something wrong?
Thursday, July 21, 2011
The surprising thing about Objective-C...
Alex Payne: I'm constantly surprised at the popularity and success of Objective-C. Almost everyone I know tells the same story about Objective-C: they started learning it and they hated it. They thought it was the worst of C and the worst of dynamic languages. And then eventually, they learned to love it. Most of the time, that software works pretty darn well, so who am I to judge? I'm pleasantly surprised by the continued success of Objective-C, and I think it should be taken as a lesson for the language designers out there.This is echoed by the the first (and as of this writing only) comment to the post:
Alasdair Allan [18 July 2011 10:09 AM] I certainly agree with Alex about Objective-C, when I was initially learning the language I deeply despised it. Now I love it, and think it's one of the more elegant and powerful of the (many) languages I know. Definitely a lesson to language designers, do what you think is right and ignore the crowds. If you are right people will grow to love your language, just as soon as they figure it out.I actually liked Objective-C pretty much from the start, but then again at that time (1986) there simply wasn't anything close that I had access to, and writing an Objective-C pre-processor and runtime on my Amiga was simply more feasible than a C++ frontend or a complete Smalltalk VM.
Modifying the sentiment expressed slightly, I'd say that from a theoretical point of view, I hate Objective-C and think it's a bad joke, a trainwreck. However, from practical experience, I love it and find it's one of the most productive languages out there for actually building stuff. And no, it's not just about the frameworks, as I've used Objective-C in non-NeXT, non-Apple environments where we had to build most of our own frameworks.
So while I support Alasdair's comment, my lesson for language designers is that our theory appears to not be particularly good at predicting reality. In other words: our theory sucks has many research opportunities.
Wednesday, April 27, 2011
Lazy Initialization
So couldn't agree more.
Saturday, March 12, 2011
Speed matters
The change was an increase from 10 to 30 search results, which was expected to produce an increase in user satisfaction, because users had asked for more search results. Instead, there was a completely unexpected and at first inexplicable 20% drop in traffic after the change was implemented. Only after some time did the team discover that the new results page took half a second longer to display, and in further testing they found that every 100 ms delay caused a measurable drop in clicks.
While I am not aware of similar research on desktop apps, I am sure that the same principle applies: speed matters, a lot; and it matters pre-consciously, that is, long before users will mention speed as an issue.
Thursday, February 17, 2011
The experienced craftsman plans less
The essence of this process is very fundamental indeed. We may understand it best by comparing the work of a fifty-year-old carpenter with the work of a novice. The experienced carpenter keeps going. He doesn’t have to keep stopping, because every action he performs, is calculated in such a way that some later action can put it right to the extent that it is imperfect now. What is critical here, is the sequence of events. The carpenter never takes a step which he cannot correct later; so he can keep working, confidently, steadily.The novice by comparison, spends a great deal of his time trying to figure out what to do. He does this essentially because he knows that an action he takes now may cause unretractable problems a little further down the line; and if he is not careful, he will find himself with a joint that requires the shortening of some crucial member – at a stage when it is too late to shorten that member. The fear of these kinds of mistakes forces him to spend hours trying to figure ahead: and it forces him to work as far as possible to exact drawings because they will guarantee that he avoids these kinds of mistakes.
The difference between the novice and the master is simply that the novice has not learnt, yet, how to do things in such a way that he can afford to make small mistakes. The master knows that the sequence of his actions will always allow him to cover his mistakes a little further down the line. It is this simple but essential knowledge which gives the work of a master carpenter its wonderful, smooth, relaxed, and almost unconcerned simplicity.
Mac App Store won't let me buy apps: solution
The solution turned out to be manually signing in using the Store menu (manually sign out if you are already signed in). At that point I was allowed to update/verify my billing information and subsequent purchase attempts worked.
In previous attempts, I had not signed in manually, but rather had the App Store do the sign-in after I attempted to purchase.
So needs a little more work...
Train wreck management
Tuesday, February 15, 2011
Only 1 GHz
Hmm
1 GHz actually seems like quite a lot to me. 1000 times an Apple ][, 40 times a NeXT, and the latter was driving a megapixel display. I guess we have gotten used to wasted cycles.
PhoneGeometry.h
One way to rectify this situation would be to start using CG* structs and functions on the desktop as well. However, this introduces a dependency on CoreGraphics that shouldn't be there for Foundation-based code.
My alternative is to standardize on NSPoint and friends, and map those to their CG alternatives on iOS. That way, I have minimized my dependencies, with only a small header file to pay for it: PhoneGeomtry.h
This is now part of MPWFoundation (on github).
//
// PhoneGeometry.h
// MPWFoundation
//
// Created by Marcel Weiher on 11/11/10.
// Copyright 2010-2011 Marcel Weiher. All rights reserved.
//
#if TARGET_OS_IPHONE
#ifndef PHONE_GEOMETRY
#define PHONE_GEOMETRY
#import <CoreGraphics/CoreGraphics.h>
typedef CGRect NSRect;
typedef CGPoint NSPoint;
typedef CGSize NSSize;
#define NSMakeRect CGRectMake
#define NSMakePoint CGPointMake
#define NSMakeSize CGSizeMake
#define NSEqualPoints CGPointEqualToPoint
#define NSEqualRects CGRectEqualToRect
#define NSIntersectsRect CGRectIntersectsRect
static inline NSString *NSStringFromRect( CGRect r ) { return [NSString stringWithFormat:@"(%g,%g - %g,%g)",r.origin.x,r.origin.y,r.size.width,r.size.height]; }
static inline NSString *NSStringFromPoint( CGPoint p ) { return [NSString stringWithFormat:@"(%g,%g)",p.x,p.y]; }
static inline NSString *NSStringFromSize( CGSize s ) { return [NSString stringWithFormat:@"(%g,%g)",s.width,s.height]; }
#endif
#endif
Tuesday, February 1, 2011
Objective-XML and MPWFoundation now available on github
Thanks to Todd Blanchard for providing the necessary impetus to learn git.
Tuesday, January 18, 2011
On switching away from CoreData
Rather, the issues we have had with CoreData were additional complexity and more importantly gratuitous dependencies that, at least for our application, were not offset by noticeable benefits.
One of the most significant structural dependencies is that CoreData requires all your model classes to be subclasses of NSManagedObject, a class provided by CoreData. This may not seem like a big problem at first, but it gets in the way of defining a proper DomainModel, which should always be independent. The Java community actually figured this out a while ago, which is why there was a recent move to persistence frameworks supporting POJOs. (Of course, POOO doesn't have quite the same ring to it, and also the Java frameworks were a lot more heavy-handed than CoreData). The model is where your value is, it should be unencumbered. For example, when we started looking at the iPhone, there was no CoreData there, so we faced the prospect of duplicating all our model code.
In addition to initially not having CoreData, the iPhone app also used (and still uses) a completely different persistence mechanism (more feed oriented), and there were other applications where yet a third persistence mechanism was used (more document centric than DB-centric, with an externally defined file format). A proper class hierarchy would have had an abstract superclass without any reference to a specific persistence mechanism, but capturing the domain knowledge of our model. With CoreData, this hierarchy was impossible.
Since we had externally defined file formats in every case, we had to write an Atomic Store adapter and thus also couldn't really benefit from CoreData's change management. When we did the move, it turned out that the Atomic Store adapter we had written was significantly more code than just serializing and de-serializing the XML ourselves.
Another benefit of CoreData is its integration with Bindings, but that also turned out to be of little use to us. The code we managed to save with Bindings was small and trivial, whereas the time and effort to debug bindings when they went wrong or to customize them for slightly specialized needs was very, very large. So we actually ditched Bindings a long time before we got rid of CoreData.
So why was CoreData chosen in the first place? Since I wasn't around for that decision, I don't know 100%, but as far as I can tell it was mostly "Shiny Object Syndrome". CoreData and Bindings were new Apple technologies at the time, therefore they had to be used.
So are there any lessons here? The first would be to avoid Shiny Object Syndrome. By all means have fun and play around, but not in production code. Second and related is to really examine your needs. CoreData is probably highly appropriate in many contexts, it just wasn't in ours. Finally, it would be a huge improvement if CoreData were to support Plain Old Objective-C Objects. In fact, if that were the case we probably would not have to ditch it.
Monday, January 10, 2011
Little Message Dispatch
I feel much the same way, that is although I think Grand Central Dispatch is awesome, I simply haven't been able to justify spending much time with it, because it usually turns out that my own threading needs so far have been far more modest than what GCD provides. In fact, I find that an approach that's even more constrained than the one based on NSOperationQueue that Brent describes has been working really well in a number of projects.
Instead of queueing up operations and letting them unwind however, I just spawn a single I/O thread (at most a few) and then have that perform the I/O deterministically. This is paired with a downloader that uses the NSURL loading system to download any number of requests in parallel.
This loads 3 types of objects: first the thumbnails, then article content, then images associated with the articles. The sequencing is both deliberate (thumbs first, article images cannot be loaded before the article content is present) and simply expressed in the code by the well-known means of just writing the actions one after the other, rather than having those dependencies expressed in call-backs, completion blocks or NSOperation subclasses.
- (void)downloadNewsContent { id pool=[NSAutoreleasePool new]; [[self downloader] downloadRequests:[self thumbnailRequests]]; [[self downloader] downloadRequests:[self contentRequests]]; [[self downloader] downloadOnlyRequests:[self imageRequests]]; [pool release]; }
So work is done semi-sequentially in the background, while coordination is done on the main thread, with liberal use of performSelectorOnMainThread. Of course, I make that a little simpler with a couple of HOMs that dispatch messages to threads:
- async runs the message on a new thread, I use it for long-running, intrinsically self contained work. It is equivalent to performSelectorInBackground: except for being able to take an arbitrary message.
- asyncOnMainThread and syncOnMainThread are the equivalents of performSelectorOnMainThread, with the waitUntilDone flag set to YES or NO
- afterDelay: sends he message after the specified delay
Brent sums it up quite well in his post:
-(void)loadSections { [[self asyncOnMainThread] showSyncing]; [[[self sections] do] downloadNewsContent]; [[self asyncOnMainThread] showDoneSyncing]; } ... -(IBAction)syncButtonClicked { [[self async] loadSections]; }
Here’s the thing about code: the better it is, the more it looks and reads like a children’s book.Yep.
Tuesday, January 4, 2011
Node.js performance? µhttpd performance!
Of course, there is also a significant body of research on this topic, showing for example that user-level thread implementations tend to get very similar performance to event-based servers. There is also the issue that the purity of "no blocking APIs" is somewhat naive on a modern Unix, because blocking on I/O can happen in lots of different non-obvious places. At the very least, you may encounter a page-fault, and this may even be desirable in order to use memory mapped files.
In those cases, the fact that you have purified all your APIs makes no difference, you are still blocked on I/O, and if you've completely foregone kernel threads like node.js appears to do, then your entire server is now blocked!
Anyway, baving seen some interesting node.js benchmarking, I was obviously curious to see how my little embedded Objective-C http-server based on the awesome GNU microhttp stacked up.
The baseline is a typical static serving test, where Apache (out-of-the box configuration on Mac OS X client) serves a small static file and the two app servers serve a small static string.
| Platform | # requests/sec |
| Static (via Apache) | 6651.58 |
| Node.js | 5793.44 |
| MPWHttp | 8557.83 |
| Platform | # requests/sec |
| Static (via Apache) | - |
| Node.js | 88.48 |
| MPWHttp | 47.04 |
| Platform | # requests/sec |
| Static (via Apache) | - |
| Node.js | 9.62 |
| MPWHttp | 7698.65 |
To make the comparison a little bit more fair, I added an xor with a randomly initialized value so that the optimizer could not remove the loop (verified by varying the loop count).
| Platform | # requests/sec |
| Static (via Apache) | - |
| Node.js | 9.62 |
| MPWHttp | 222.9 |
Cross-checking on my 8 core Mac Pro gave the following results:
| Platform | # requests/sec |
| Static (via Apache) | - |
| Node.js | 10.72 |
| MPWHttp | 1011.86 |
In conclusion, I think it is fair to say that node.js succeeds admirably in a certain category of tasks: lots of concurrency, lots of blocked I/O, very little computation, very little memory use so we don't page fault. In more typical mixes with some concurrency, some computation some I/O and a bit of memory use (so chances of paging), a more balanced approach may be better.
Thursday, December 2, 2010
This is not LISP
(apply + (take 1000 (iterate inc 1)))Hmm..I find the following preferable:
(1 to: 1000 ) reduce + 0
Sunday, May 9, 2010
iPhone XML: from DOM to incremental SAX
In my last post, I extended Ray Wenderlich's XML parser comparison to MAX, and performance seemed to be about 2x better than the nearest competitor, TBXML and around 7x faster than NSXMLParser, Cocoa Touch's built-in XML parser.
While the resulting code has been posted here, I haven't yet explained how I got there. First, I downloaded both Ray's project and Objective-XML 5.3. iPhone OS requires a bit of trickiness to get a reusable library, partly due to the fact that frameworks or dynamic libraries are not supported, partly due to the fact that simulator and device are not just different Xcode architectures of a single SDK, and not even just different SDKs, but actually different platforms. If anyone can tell me how to create a normal target that can compile/links for both the simulator and the device, I'd love to hear about it!
So, in order to compile for iPhoneOS, you'll need to select the iXmlKit target:
You'll need to build twice, changing Active SDK settings once to the the device and once to the simulator. You will then have two copies of the libiXmlKit.a library, one in the directory Release-iphoneos, the other in Release-iphonsimulator (both relative to your base build directory):
marcel@mpwls[LD]ls -la ~/programming/Build/Release-iphoneos/
-rw-r--r-- 1 marcel staff 521640 May 9 19:52 libiXmlKit.a
These two copies then need to be joined together using the lipo command to create a single library that can be used both with the simulator and the device.
lipo -create Release-iphoneos/libiXmlKit.a Release-iphonesimulator/libiXmlKit.a -output Release/libiXmlKit.a
(Newer Objective-XML versions will have a shell-script target that automates this process). Once I had the fat libiXmlKit.a library, I created a new MAX group in the XMLPerformance project, and copied both the library and the MAX header file into that group:
I then created a new MAX Song parser class:
#import "iTunesRSSParser.h"
@interface MAXSongParser : iTunesRSSParser {
id parser;
NSDateFormatter *parseFormatter;
}
@end
The implementation is also fairly straightforward, with your basic init method:
#import "MAXSongParser.h"
#import "MPWMAXParser.h"
#import "Song.h"
@implementation MAXSongParser
-init
{
self=[super init];
parseFormatter = [[NSDateFormatter alloc] init];
[parseFormatter setDateStyle:NSDateFormatterLongStyle];
[parseFormatter setTimeStyle:NSDateFormatterNoStyle];
MPWMAXParser *newParser=[MPWMAXParser parser];
[newParser setUndefinedTagAction:MAX_ACTION_NONE];
[newParser setHandler:self forElements:[NSArray arrayWithObjects: @"item",@"album",@"title",@"channel",@"rss",@"category",nil]
inNamespace:nil prefix:@"" map:nil];
[newParser setHandler:self forElements:[NSArray arrayWithObjects:
@"releasedate",@"artist",@"album",nil]
inNamespace:@"http://phobos.apple.com/rss/1.0/modules/itms/"
prefix:@"" map:nil];
parser=[newParser retain];
return self;}
The MAX parser is initialized in this init method. We define the elements we care about using the two "setHandler:forElements:inNamespace:prefix:map:" messages, one for each namespace we will be dealing with. In the default (RSS) namespace, we are interested in the "item", "album", "title", "channel", "rss" and "category" elements. In Apple's special "itms" namespace, we will handle "releasdate", "artist" and "album". Setting MAX_ACTION_NONE as the undefined tag action means that the parser will ignore elements not listed as interesting and all their sub-elements.
Songs are created in the -itemElement:... method, which turns the relevant child-elements of the item element into Song attributes:
-itemElement:children attributes:attributes parser:parser
{Song *song=[[Songalloc] init];
[song setAlbum:[children objectForUniqueKey:@"album"]];
[song setTitle:[children objectForUniqueKey:@"title"]];
[song setArtist:[children objectForUniqueKey:@"artist"]];
[song setAlbum:[children objectForUniqueKey:@"album"]];
[song setCategory:[children objectForUniqueKey:@"category"]];
[song setReleaseDate:[parseFormatter dateFromString:[children objectForUniqueKey:@"releasedate"]] ];
return song;}
Two more methods make the actual parse process complete: <channel> elements have one or more <item> elements, so we want to return all of them, using "objectsForKey:":
-channelElement:children attributes:attributes parser:parser{return [[children objectsForKey:@"item"] retain];
}
Finally, there are a bunch of elements that we have defined interest in but treat identically...these can be handled using the "default" element handler:
-defaultElement:children attributes:attributes parser:parser{return [[children lastObject] retain];
}
That concludes the routines that actually parse the XML into objects, now for kicking off the parser. With the timing code removed, the method is fairly straightforward:
- (void)downloadAndParse:(NSURL *)url {
id pool=[NSAutoreleasePool new];
[parserparse: [NSData dataWithContentsOfURL:url]];for ( id song in [parserparseResult] ) {
[self performSelectorOnMainThread:@selector(parsedSong:)
withObject:song waitUntilDone:NO];
}
[pool release];}
With the timing code, it all gets a bit messier:
- (void)downloadAndParse:(NSURL *)url {
id pool=[NSAutoreleasePool new];
[self performSelectorOnMainThread:@selector(downloadStarted) withObject:nil waitUntilDone:NO];
NSData *data=[NSData dataWithContentsOfURL:url];
[self performSelectorOnMainThread:@selector(downloadEnded) withObject:nil waitUntilDone:NO];NSTimeInterval start = [NSDate timeIntervalSinceReferenceDate];
[parserparse:data];
for ( id song in [parserparseResult] ) {
[selfperformSelectorOnMainThread:@selector(parsedSong:) withObject:song waitUntilDone:NO];
}
NSTimeInterval duration = [NSDatetimeIntervalSinceReferenceDate] - start;
[selfperformSelectorOnMainThread:@selector(addToParseDuration:) withObject:[NSNumbernumberWithDouble:duration] waitUntilDone:NO];
[selfperformSelectorOnMainThread:@selector(parseEnded) withObject:nilwaitUntilDone:NO];
[[NSURLCachesharedURLCache] removeAllCachedResponses];
[pool release];
}
This produces a non-incremental DOM-style parser, so we first download the data, then process it into a DOM and finally transfer the processed objects to the main thread for display. It differs from other DOM-syle XML parsers in that it actually produces domain objects (as a Domain Object Model parser arguably should), rather than a generic XML DOM that must then be converted to objects.
Turning the DOM-style parser into a SAX-stye parser is almost completely trivial. Instead of returning the Song objects at the end of itemElement:..
[song setReleaseDate:[parseFormatterdateFromString:[children objectForUniqueKey:@"releasedate"]] ];
return song;}
we instead pass them to the delegate there and return nil so no tree is constructed:
[song setReleaseDate:[parseFormatterdateFromString:[children objectForUniqueKey:@"releasedate"]] ];
[self performSelectorOnMainThread:@selector(parsedSong:)
withObject:song waitUntilDone:NO];
[song release];
return nil;
}
This means we can also remove the "channelElement" method and the for loop in downloadAndParse: that passed the Song objects to the main thread. This is a SAX-style parser (though it doesn't use the SAX methods and does produce domain objects), but it is still non-incremental because it first downloads all the data and then parses it. If we want to turn the SAX parser into an incremental parser that overlaps processing with downloading, there is one final tweak that fortunately further simplifies the downloadAndParse method (again with timing code removed):
- (void)downloadAndParse:(NSURL *)url {
id pool=[NSAutoreleasePoolnew];
[parserparseDataFromURL:url];[[NSURLCachesharedURLCache] removeAllCachedResponses];
[pool release];}
While this is probably best not only in terms of performance and responsiveness, but also in terms of code size, it doesn't play well with the XMLPerformance example, because there are no external measurement hooks that allow us to separate downloading from parsing for performance measurement purposes.
In addition, the XMLPerformance example is odd in that is both multi-threaded and measure real time rather than CPU time when measuring parse performance. The reason this is odd is that when both parsing and display are active, the scheduler can take the CPU away from the XML parsing thread at any time and switch to the display thread, but this time will be counted as XML parsing time by the measurement routines. This is obviously incorrect and penalizes all the incremental parsers, which is why Ray's comparison showed all the DOM parsers as performing better than the SAX parsers.
I hope these explanations show how to create different styles of parsers using MAX.
Sunday, May 2, 2010
iPhone XML Performance Revisited
Ray Wenderlich has done a great comparison of iPhone XML parsers, using the same sample I had looked at earlier in the context of responsiveness.
As Ray was comparing performance, my hobby-horse, I was obviously curious as to how MAX stacked up against all the upstart competition. Quite well, as it turns out (average of 5 runs on an iPad with 3.2):
Figure 1: total times (seconds)
MAX was about 50% faster than the closest competition, TBXML, at 0.43s vs. 0.61s.
However, the XMLPerformance sample is a bit weird in that it measures elapsed time, not CPU time, and is multi-threaded, updating the display as results come in.
In order to account for this overhead that has nothing to do with the XML parsers, I added a "Dummy" parser that doesn't actually parse anything, but rather just generates dummy Song entries as quickly as possible. This takes around 0.2 seconds all by itself. Subtracting this non-XML overhead from the total times yields the following results:
Figure 2: XML parse times (seconds)
When measuring just the XML parsers themselves, MAX is around twice as fast as the closest competition and seven times as fast as the built in NSXMLParser.
Sweet.
[Update] I forgot the link to where I had uploaded the source: XMLPerformance.tgz at http://www.metaobject.com/downloads/Objective-C/
Friday, April 23, 2010
This blog has moved
Monday, January 25, 2010
Objective-XML 5.3
- Cocotron targets for Windows support.
- XMLRPC support.
- No longer uses 'private' API that was causing AppStore rejections for some iPhone apps using Objective-XML.
- Support for numeric entitites.
Sunday, December 13, 2009
The Responder Chain is a Collection
NSEnumerator *responderEnumerator = [[firstResponder mapToNextObjectFromMessage] nextResponder];
I am currently abstracting from the intricate delegate mapping and other ops, these could be handled in an analog fashion. With the enumerator in place, we can obviously snapshot it to get the current state of the responder chain, and also log that.
NSArray *responders = [responderEnumerator allObjects]; NSLog(@"full responder chain: %@",responders);
Now we can express both current features and possible variations of the Responder Chain architecture compactly as common collection operations. The current dispatch mechanism simply sends the message to the first object that is capable of responding. This corresponds to using the first object of a -select, which is expressed in the -selectFirst convenience method.
Current dispatch
[[[responders selectFirst] respondsToSelector:action] performSelector:action withObject:sender];
If I understood him correctly, Tim wants the objects in the responder chain to return an object that they would like to respond to the message. This turns the -select into a -collect (without a -collectFirst), but is otherwise very similar.
Tim's dispatch
possibleResponders = [[responders collect] responsibleTargetForAction:theAction sender:sender]]; [[possibleResponders objectAtIndex:0] performSelector:action withObject:sender];
I hope this does Tim's ideas justice, but I think the succinct formulation should make it easy to tell wether it does or not.
In terms of combining validation with target/action, I'd be somewhat wary of accidentally triggering actions when validation was meant, though I do appreciate the advantages of combining the two operations. I am not sure what value the block is adding over just having an additional BOOL parameter in the target/action method.
Combined action and validation
typedef BOOL IBAction;
-(IBAction)delete:sender :(BOOL)onlyValidate
{
NSArray *selection = [self selectedItems];
if ( onlyValidate || [selection count] == 0 ) {
return NO;
}
// perform the action
}
// or if you're worried about the naming issues
-(IBAction)delete:sender
{
}
