5.1. Plan of Action5.1.1. Program TasksWe'll use Linrg unchanged for the computational part of the program. Our front-end program will therefore have the tasks of
Keeping Linrg as a separate executable is admittedly a strange decision: The code is so simple, it would be easier to fold it into the main application. We're doing it this way in order to illustrate a few additional points. If Linrg were a large-address-space application, using a 64-bit address space under Mac OS X 10.4 or later, it might make sense to build it as a 64-bit utility, with a 32-bit application running its human interface. 5.1.2. Model-View-ControllerCocoa applications are built around the design pattern called Model-View-Controller (MVC). The pattern asserts that three kinds of things comprise an interactive program.
The ModelIt seems plain that the first task of our programstoring a list of data pairsis the sort of task a model object performs. Similarly, the task of calculating the linear regression is purely a function of the data we present to the calculation and would be the same no matter how we managed the data points beforehand or presented the results afterward. What we want, then, is a document containing a set of data points and the results of calculating a linear regression on them. This simple design is shown in Figure 5.1. Figure 5.1. The minimal data model for a program that manages linear regressions. The document contains a list of data points and the results of the regression done on them. The regression refers to the list of data points.We will be working in Objective-C, which provides the easiest access to Cocoa. From the figure, it's natural to imagine the interface for a DataPoint object: @interface DataPoint : NSObject <NSCoding> { This code segment declares DataPoint to be a subclass of the basic NSObject class, promises that DataPoint will be able to read and write itself in data streams according to the NSCoding protocol, and says that its data consists of two double-precision numbers: x and y. The code then declares a default initializer (init) and an initializer that sets the instance values (initWithX:Y:). After that come accessors for reading and setting the x and y values. Simple. The interface for the Regression class is mostly the same concept, applied to the four data members instead of two: @interface Regression : NSObject <NSCoding> { Once again, we see the descent from NSObject, the promise of NSCoding, the four data members, and accessor methods for those members. We make dataPoints an NSMutableArray, which is a Cocoa class that keeps ordered lists of objects. There are two additional public methods.
One data member, linrgTask, didn't figure in our sketch model. This data member is an NSTask, an object used for running command line tools from Cocoa applications. We'll be using an NSTask to run Linrg on our data points. The ControllerThe controller object we create will be an instance of MyDocument, a subclass of Cocoa's NSDocument. Xcode's template for a new Cocoa document-based application automatically includes a MyDocument class with skeleton code in the project. NSDocuments are automatically placed in the command-response chain of the application and bring with them the basics of loading and storing a document's contents in the file system. We can expect our documents to be told to load and store themselves, to compute the linear regression, and to add, edit, and remove data points. Therefore, we will want to provide methods for loading and storing and for computing. These tasks can be done through Regression, our top-level model object, so we conclude that MyDocument need add a data member only for a Regression object.
Strangely, we won't be providing any methods for managing the data points. More on this later. @class Regression; By the vagaries of the Objective-C language, it is not necessary to declare in the interface every method a class implements. One method we do need to declare is compute:, the method that triggers calculation of the regression. As a method that responds to commands from the application's human interface, compute: follows a strict signaturetaking one anonymous object (type id), the sender of the command, and returning IBAction, which is #defined as void but serves to inform Interface Builder that this command may be issued to this class by the human interface. The View(s)The view layer of our program is best specified by how we want it to look and behave. Figure 5.2 shows what we're aiming for. Figure 5.2. Planned layout of the main window of our linear-regression program. The Add and Remove buttons insert and delete points in the data set, which can be edited directly in the table at left. The Compute button passes the points to Linrg and causes the results to be displayed in the read-only fields below it. |
Wednesday, October 14, 2009
Section 5.1. Plan of Action
Subscribe to:
Post Comments (Atom)
No comments:
Post a Comment