For instance, an ID3 tag defined in the 2.2 version of the specification consists of a header made up of a three-character ISO-8859-1 string, which is always “ID3”; two one-byte unsigned integers that specify the major version and revision of the specification; eight bits worth of boolean flags; and four bytes that encode the size of the tag in an encoding particular to the ID3 specification. Following the header is a list of frames, each of which has its own internal structure. After the frames are as many null bytes as are necessary to pad the tag out to the size specified in the header.
An instance of this class would make a perfect repository to hold the data needed to represent an ID3 tag. You could then write functions to read and write instances of this class. For example, assuming the existence of certain other functions for reading the appropriate primitive data types, a function might look like this:
(defun read-id3-tag (in)
(let ((tag (make-instance 'id3-tag)))
(setf identifier (read-iso-8859-1-string in :length 3))
(setf major-version (read-u1 in))
(setf flags (read-u1 in))
(setf size (read-id3-encoded-size in))
tag))
It’s not hard to see how you could write the appropriate classes to represent all the composite data structures in a specification along with read-foo
and write-foo
functions for each class and for necessary primitive types. But it’s also easy to tell that all the reading and writing functions are going to be pretty similar, differing only in the specifics of what types they read and the names of the slots they store them in. It’s particularly irksome when you consider that in the ID3 specification it takes about four lines of text to specify the structure of an ID3 tag, while you’ve already written eighteen lines of code and haven’t even written write-id3-tag
yet.