SlideShare una empresa de Scribd logo
1 de 87
Problem Solving with Data
Structures using Java:
A Multimedia Approach
Chapter 6: Structuring
Music Using Linked Lists
Chapter Objectives
JMusic: Java Music library
 JMusic knows about WAV files and many other
formats, too (e.g., QuickTime)
• http://explodingart.com/jmusic
• http://sourceforge.net/projects/jmusic/
 We’ll use it for manipulating MIDI
• Musical Instrument Digital Interface, an industry-
standard interface used on electronic musical
keyboards and PCs for computer control of musical
instruments and devices.
 MIDI is about recording music, not sound.
JMusic pieces
 import jm.music.data.*;
• // To get Note and Phrase classes
• // .* means “All the classes in the package.”
 import jm.util.*;
• // To get the View class that knows how to
draw music notation from MIDI.
 import jm.JMC;
• // The class containing constants like
JMC.FLUTE and JMC.C4
MIDI notes
Relationship between Music
and MIDI
MIDI
Instruments
Different ways of creating
Phrases
> Phrase phr2 = new
Phrase("Phrase
2",4.0,JMC.FLUTE);
> phr2.addNoteList(notes2)
A Phrase that starts later
> Phrase phr2 = new Phrase("Phrase
2",4.0,JMC.FLUTE);
> phr2.addNoteList(notes2)
> View.notate(phr2)
Amazing
Grace
> AmazingGraceSong song1 =
new AmazingGraceSong();
> song1.showMe();
> AmazingGraceSong song1 =
new AmazingGraceSong();
> song1.showMe();
import jm.music.data.*;
import jm.JMC;
import jm.util.*;
import jm.music.tools.*;
/**
* Class that represents the song Amazing Grace
* @author Mark Guzdial
* @author Barb Ericson
*/
public class AmazingGraceSong {
/** the score for the song */
private Score myScore =
new Score("Amazing Grace");
/**
* Constructor to set up the song
*/
public AmazingGraceSong() {
myScore.setTimeSignature(3,4);
double[] phrase1Data = {
JMC.G4, JMC.QN,
JMC.C5, JMC.HN, JMC.E5,JMC.EN, JMC.C5,JMC.EN,
JMC.E5,JMC.HN,JMC.D5,JMC.QN,JMC.C5,JMC.HN,JMC.A4,JMC.QN,
JMC.G4,JMC.HN,JMC.G4,JMC.EN,JMC.A4,JMC.EN,
JMC.C5,JMC.HN,JMC.E5,JMC.EN,JMC.C5,JMC.EN,
JMC.E5,JMC.HN,JMC.D5,JMC.EN,JMC.E5,JMC.EN,
JMC.G5,JMC.DHN
};
double[] phrase2Data = {
JMC.G5,JMC.HN,JMC.E5,JMC.EN,JMC.G5,JMC.EN,
JMC.G5,JMC.HN,JMC.E5,JMC.EN,JMC.C5,JMC.EN,
JMC.E5,JMC.HN,JMC.D5,JMC.QN,
JMC.C5,JMC.HN,JMC.A4,JMC.QN,
JMC.G4,JMC.HN,JMC.G4,JMC.EN,JMC.A4,JMC.EN,
JMC.C5,JMC.HN,JMC.E5,JMC.EN,JMC.C5,JMC.EN,
JMC.E5,JMC.HN,JMC.D5,JMC.QN,
JMC.C5,JMC.DHN
};
Phrase myPhrase = new Phrase();
myPhrase.addNoteList(phrase1Data);
myPhrase.addNoteList(phrase2Data);
// create a new part and add the phrase to it
Part aPart = new Part("Parts",
JMC.FLUTE, 1);
aPart.addPhrase(myPhrase);
// add the part to the score
myScore.addPart(aPart);
}
/**
* Method to show the song
*/
public void showMe() {
View.notate(myScore);
}}
Imports and some private data
import jm.music.data.*;
import jm.JMC;
import jm.util.*;
import jm.music.tools.*;
public class AmazingGraceSong {
private Score myScore = new Score("Amazing
Grace");
 myScore is private instance data
Filling the
Score
public AmazingGraceSong() {
myScore.setTimeSignature(3,4);
double[] phrase1Data = {
JMC.G4, JMC.QN,
JMC.C5, JMC.HN, JMC.E5,JMC.EN, JMC.C5,JMC.EN,
JMC.E5,JMC.HN,JMC.D5,JMC.QN,
JMC.C5,JMC.HN,JMC.A4,JMC.QN,
JMC.G4,JMC.HN,JMC.G4,JMC.EN,JMC.A4,JMC.EN,
JMC.C5,JMC.HN,JMC.E5,JMC.EN,JMC.C5,JMC.EN,
JMC.E5,JMC.HN,JMC.D5,JMC.EN,JMC.E5,JMC.EN,
JMC.G5,JMC.DHN
};
double[] phrase2Data = {
JMC.G5,JMC.HN,JMC.E5,JMC.EN,JMC.G5,JMC.EN,
JMC.G5,JMC.HN,JMC.E5,JMC.EN,JMC.C5,JMC.EN,
JMC.E5,JMC.HN,JMC.D5,JMC.QN,
JMC.C5,JMC.HN,JMC.A4,JMC.QN,
JMC.G4,JMC.HN,JMC.G4,JMC.EN,JMC.A4,JMC.EN,
JMC.C5,JMC.HN,JMC.E5,JMC.EN,JMC.C5,JMC.EN,
JMC.E5,JMC.HN,JMC.D5,JMC.QN,
JMC.C5,JMC.DHN
};
Phrase myPhrase = new Phrase();
myPhrase.addNoteList(phrase1Data);
myPhrase.addNoteList(phrase2Data);
// create a new part and add the phrase to it
Part aPart = new Part("Parts",
JMC.FLUTE, 1);
aPart.addPhrase(myPhrase);
// add the part to the score
myScore.addPart(aPart);
}
Each array is note,
duration, note,
duration, note,
duration, etc.
Broke it roughly into
halves, just to make
it easier to
manipulate.
This is a
constructor that will
create the data for
the object when it is
created.
Showing the Score
public void showMe(){
View.notate(myScore);
};
Part: Instrument &
Phrase: startingTime &
The Organization of JMusic
Objects
Phrase: startingTime &
Note
(pitch,duration)
Note
(pitch,duration)
Note
(pitch,duration)
Note
(pitch,duration)
Note
(pitch,duration)
Part: Instrument &
Score: timeSignature, tempo, &
Thought Experiment
 How are they doing that?
 How can there be any number of Notes
in a Phrase, Phrases in a Part, and Parts
in a Score?
• (Hint: They ain’t usin’ arrays!)
How do we explore composition
here?
 We want to quickly and easily throw
together notes in different groupings and
see how they sound.
 The current JMusic structure models
music.
• Let’s try to create a structure that models
thinking about music as bunches of
riffs/SongElements that we want to combine
in different ways.
Introducing the Linked List
 A linked list is information broken into
smaller pieces, where each piece knows
the next piece, but none other.
Another example of a linked list
 Non-linear video
editing (like in iMovie)
• You have a collection
of video clips
(information)
• You drag them into a
timeline.
• Each clip still doesn’t
know all clips, but it
knows the next one.
Why use linked lists versus
arrays?
 Just two reasons now, more later:
1. Can grow to any size (well, as long as
memory permits)
• Just create a new element and poke it into the
list.
1. MUCH easier to insert (and delete)!
SongNode and SongPhrase
 SongNode instances will hold pieces
(phrases) from SongPhrase.
 SongNode instances will be the nodes in
the linked list
• Each one will know its next.
 Ordering will encode the order in the Part.
• Each one will get appended after the last.
Using SongNode and
SongPhrase
Welcome to DrJava.
> import jm.JMC;
> SongNode first = new SongNode();
> first.setPhrase(SongPhrase.riff1());
> first.showFromMeOn(JMC.FLUTE);
> SongNode second = new SongNode();
> second.setPhrase(SongPhrase.riff2());
> second.showFromMeOn(JMC.PIANO);
> first.setNext(second);
> first.showFromMeOn(JMC.SAX);
Inserting a third node
> SongNode third = new SongNode();
> third.setPhrase(SongPhrase.riff3());
> third.showFromMeOn(JMC.PIANO);
> first.setNext(third);
> third.setNext(second);
> first.showFromMeOn(JMC.FLUTE);
How to think about it
 To start:
first
myPhrase: riff1
next: second
second
myPhrase: riff2
next: null
How to
think about it
first
myPhrase: riff1
next: third
second
myPhrase: riff2
next: null
myPhrase: riff3
next: second
third
Declarations for SongNode
import jm.music.data.*;
import jm.JMC;
import jm.util.*;
import jm.music.tools.*;
public class SongNode {
/**
* the next SongNode in the list
*/
private SongNode next;
/**
* the Phrase containing the notes and durations associated with this node
*/
private Phrase myPhrase;
SongNode’s know their
Phrase and the next
node in the list
Constructor for SongNode
/**
* When we make a new element, the next part is
empty, and ours is a blank new part
*/
public SongNode(){
this.next = null;
this.myPhrase = new Phrase();
}
Setting the phrase
/**
* setPhrase takes a Phrase and makes it the one for
this node
* @param thisPhrase the phrase for this node
*/
public void setPhrase(Phrase thisPhrase){
this.myPhrase = thisPhrase;
}
Linked list methods
/**
* Creates a link between the current node and the input node
* @param nextOne the node to link to
*/
public void setNext(SongNode nextOne){
this.next = nextOne;
}
/**
* Provides public access to the next node.
* @return a SongNode instance (or null)
*/
public SongNode getNext(){
return this.next;
}
insertAfter
/**
* Insert the input SongNode AFTER this node,
* and make whatever node comes NEXT become the next of the input
node.
* @param nextOne SongNode to insert after this one
*/
public void insertAfter(SongNode nextOne)
{
SongNode oldNext = this.next(); // Save its next
this.setNext(nextOne); // Insert the copy
nextOne.setNext(oldNext); // Make the copy point on to the rest
}
Question:
// Assume we’ve just created all 4 nodes
node1.setNext(node2); node2.setNext(node3);
node1.insert(node4);
 Which of these is true? (More than one.)
 A. node4’s next is null.
 B. node3’s next is null.
 C. node1’s next is node2
 D. node4’s next is node2
After Group Question:
// Assume we’ve just created all 4 nodes
node1.setNext(node2); node2.setNext(node3);
node1.insert(node4);
 Which of these is true? (More than one.)
 A. node4’s next is null.
 B. node3’s next is null.
 C. node1’s next is node2
 D. node4’s next is node2
Using and tracing insertAfter()
> SongNode nodeA = new SongNode();
> SongNode nodeB = new SongNode();
> nodeA.setNext(nodeB);
> SongNode nodeC = new SongNode()
> nodeA.insertAfter(nodeC);
public void insertAfter(SongNode nextOne)
{
SongNode oldNext = this.next(); // Save its next
this.setNext(nextOne); // Insert the copy
nextOne.setNext(oldNext); // Make the copy point
on to the rest
}
Traversing
the list
/**
* Collect all the notes from this node on
* in an part (then a score) and open it up for viewing.
* @param instrument MIDI instrument (program) to be used in playing this list
*/
public void showFromMeOn(int instrument){
// Make the Score that we'll assemble the elements into
// We'll set it up with a default time signature and tempo we like
// (Should probably make it possible to change these -- maybe with inputs?)
Score myScore = new Score("My Song");
myScore.setTimeSignature(3,4);
myScore.setTempo(120.0);
// Make the Part that we'll assemble things into
Part myPart = new Part(instrument);
// Make a new Phrase that will contain the notes from all the phrases
Phrase collector = new Phrase();
// Start from this element (this)
SongNode current = this;
// While we're not through...
while (current != null)
{
collector.addNoteList(current.getNotes());
// Now, move on to the next element
current = current.next();
};
// Now, construct the part and the score.
myPart.addPhrase(collector);
myScore.addPart(myPart);
// At the end, let's see it!
View.notate(myScore);
}
The Core of the Traversal
// Make a new Phrase that will contain the notes from all the phrases
Phrase collector = new Phrase();
// Start from this element (this)
SongNode current = this;
// While we're not through...
while (current != null)
{
collector.addNoteList(current.getNotes());
// Now, move on to the next element
current = current.next();
};
Step 1:
// Start from this element (this)
SongNode current = this;
first
myPhrase: riff1
next: third
second
myPhrase: riff2
next: null
myPhrase: riff3
next: second
third
current
// While we're not through...
while (current != null)
{
collector.addNoteList(current.getNotes());
first
myPhrase: riff1
next: third
second
myPhrase: riff2
next: null
myPhrase: riff3
next: second
third
Collector:
current
// Now, move on to the next element
current = current.next();
};
first
myPhrase: riff1
next: second
second
myPhrase: riff2
next: null
myPhrase: riff3
next: second
third
Collector:
current
// While we're not through...
while (current != null)
{
collector.addNoteList(current.getNotes());
first
myPhrase: riff1
next: third
second
myPhrase: riff2
next: null
myPhrase: riff3
next: second
third
Collector:
current
// Now, move on to the next element
current = current.next();
};
first
myPhrase: riff1
next: node2
second
myPhrase: riff2
next: null
myPhrase: riff3
next: second
third
Collector:
current
// While we're not through...
while (current != null)
{
collector.addNoteList(current.getNotes());
first
myPhrase: riff1
next: third
second
myPhrase: riff2
next: null
myPhrase: riff3
next: second
third
Collector:
current
// Now, move on to the next element
current = current.next();
};
first
myPhrase: riff1
next: third
second
myPhrase: riff2
next: null
myPhrase: riff2
next: second
third
Collector:
currentnull
// While we're not through...
while (current != null)
first
myPhrase: riff1
next: node2
second
myPhrase: riff2
next: null
myPhrase: riff2
next: second
third
Collector:
currentnull
Then return what you collected
// Now, construct the part and the score.
myPart.addPhrase(collector);
myScore.addPart(myPart);
// At the end, let's see it!
View.notate(myScore);
}
Question
 At the end of showFromMeOn(), which of
these is true? (More than one)
 A. this is the last node in the list.
 B. The nodes in the list are unchanged.
 C. current is null.
 D. next on the last node in the list is null.
 E. The first node on the list points to
current.
After-Group Question
 At the end of showFromMeOn(), which of
these is true? (More than one)
 A. this is the last node in the list.
 B. The nodes in the list are unchanged.
 C. current is null.
 D. next on the last node in the list is null.
 E. The first node on the list points to
current.
SongPhrase
 SongPhrase is a collection of static
methods.
 We don’t ever need an instance of
SongPhrase.
 Instead, we use it to store methods that
return phrases.
• It’s not very object-oriented, but it’s useful
here.
SongPhrase.riff1()
import jm.music.data.*;
import jm.JMC;
import jm.util.*;
import jm.music.tools.*;
public class SongPhrase {
//Little Riff1
static public Phrase riff1() {
double[] phrasedata =
{JMC.G3,JMC.EN,JMC.B3,JMC.EN,JMC.C4,JMC.EN,JMC.D4,JMC.EN};
Phrase myPhrase = new Phrase();
myPhrase.addNoteList(phrasedata);
return myPhrase;
SongPhrase.riff2()
//Little Riff2
static public Phrase riff2() {
double[] phrasedata =
{JMC.D4,JMC.EN,JMC.C4,JMC.EN,JMC.E4,JMC.E
N,JMC.G4,JMC.EN};
Phrase myPhrase = new Phrase();
myPhrase.addNoteList(phrasedata);
return myPhrase;
}
Computing a phrase
//Larger Riff1
static public Phrase pattern1() {
double[] riff1data =
{JMC.G3,JMC.EN,JMC.B3,JMC.EN,JMC.C4,JMC.EN,JMC.D4,JMC.EN};
double[] riff2data =
{JMC.D4,JMC.EN,JMC.C4,JMC.EN,JMC.E4,JMC.EN,JMC.G4,JMC.EN};
Phrase myPhrase = new Phrase();
// 3 of riff1, 1 of riff2, and repeat all of it 3 times
for (int counter1 = 1; counter1 <= 3; counter1++)
{for (int counter2 = 1; counter2 <= 3; counter2++)
myPhrase.addNoteList(riff1data);
myPhrase.addNoteList(riff2data);
};
return myPhrase;
}
Question
 How many times does addNoteList end
up being called in that method?
 A. 6 times
 B. 12 times
 C. 4 times
 D. 8 times
After-Group Question
 How many times does addNoteList end
up being called in that method?
 A. 6 times
 B. 12 times
 C. 4 times
 D. 8 times
As long as it’s a phrase…
 The way that we use SongNote and
SongPhrase, any method that returns a
phrase is perfectly valid SongPhrase
method.
10 Random Notes
(Could be less random…)
/*
* 10 random notes
**/
static public Phrase random() {
Phrase ranPhrase = new Phrase();
Note n = null;
for (int i=0; i < 10; i++) {
n = new Note((int) (128*Math.random()),0.1);
ranPhrase.addNote(n);
}
return ranPhrase;
}
10 Slightly Less Random Notes
/*
* 10 random notes above middle C
**/
static public Phrase randomAboveC() {
Phrase ranPhrase = new Phrase();
Note n = null;
for (int i=0; i < 10; i++) {
n = new Note((int) (60+(5*Math.random())),0.25);
ranPhrase.addNote(n);
}
return ranPhrase;
}
Going beyond connecting nodes
 So far, we’ve just created nodes and
connected them up.
 What else can we do?
 Well, music is about repetition and
interleaving of themes.
• Let’s create those abilities for SongNodes.
Repeating a Phrase
Welcome to DrJava.
> SongNode node = new SongNode();
> node.setPhrase(SongPhrase.randomAboveC());
> SongNode node1 = new SongNode();
> node1.setPhrase(SongPhrase.riff1());
> node.repeatNext(node1,10);
> import jm.JMC;
> node.showFromMeOn(JMC.PIANO);
What it looks like
node node1 node1 node1 …
Repeating
/**
* Repeat the input phrase for the number of times
specified.
* It always appends to the current node, NOT insert.
* @param nextOne node to be copied in to list
* @param count number of times to copy it in.
*/
public void repeatNext(SongNode nextOne,int count) {
SongNode current = this; // Start from here
SongNode copy; // Where we keep the current copy
for (int i=1; i <= count; i++)
{
copy = nextOne.copyNode(); // Make a copy
current.setNext(copy); // Set as next
current = copy; // Now append to copy
}
}
Note! What
happens to this’s
next? How
would you create
a looong repeat
chain of several
types of phrases
with this?
Here’s making a copy
/**
* copyNode returns a copy of this node
* @return another song node with the same notes
*/
public SongNode copyNode(){
SongNode returnMe = new SongNode();
returnMe.setPhrase(this.getPhrase());
return returnMe;
}
Step 1:
public void repeatNext(SongNode nextOne,int count) {
SongNode current = this; // Start from here
SongNode copy; // Where we keep the current copy
node
phrase:
10
random
notes
next: null
current
node1
phrase:
riff1()
next: null
nextOne
Step 2:
copy = nextOne.copyNode(); // Make a copy
node
phrase:
10
random
notes
next: null
current
node1
phrase:
riff1()
next: null
phrase:
riff1()
next: null
copy nextOne
Step 3:
current.setNext(copy); // Set as next
node
phrase:
10
random
notes
next:
current
node1
phrase:
riff1()
next: null
phrase:
riff1()
next: null
copy nextOne
Step 4:
current = copy; // Now append to copy
node
phrase:
10
random
notes
next:
current
node1
phrase:
riff1()
next: null
phrase:
riff1()
next: null
copy nextOne
Step 5 & 6:
copy = nextOne.copyNode(); // Make a copy
current.setNext(copy); // Set as next
node
phrase:
10
random
notes
next:
current
node1
phrase:
riff1()
next: null
phrase:
riff1()
next:
copy
phrase:
riff1()
next: null
nextOne
Step 7 (and so on):
current = copy; // Now append to copy
node
phrase:
10
random
notes
next:
current
node1
phrase:
riff1()
next: null
phrase:
riff1()
next:
copy
phrase:
riff1()
next: null
nextOne
What happens if the node already
points to something?
 Consider repeatNext and how it inserts:
It simply sets the next value.
 What if the node already had a next?
 repeatNext will erase whatever used to
come next.
 How can we fix it?
repeatNextInserting
/**
* Repeat the input phrase for the number of times specified.
* But do an insertion, to save the rest of the list.
* @param nextOne node to be copied into the list
* @param count number of times to copy it in.
**/
public void repeatNextInserting(SongNode nextOne, int count){
SongNode current = this; // Start from here
SongNode copy; // Where we keep the current copy
for (int i=1; i <= count; i++)
{
copy = nextOne.copyNode(); // Make a copy
current.insertAfter(copy); // INSERT after current
current = copy; // Now append to copy
}
}
Weaving
/**
* Weave the input phrase count times every skipAmount nodes
* @param nextOne node to be copied into the list
* @param count how many times to copy
* @param skipAmount how many nodes to skip per weave
*/
public void weave(SongNode nextOne, int count, int skipAmount)
{
SongNode current = this; // Start from here
SongNode copy; // Where we keep the one to be weaved in
SongNode oldNext; // Need this to insert properly
int skipped; // Number skipped currently
for (int i=1; i <= count; i++)
{
copy = nextOne.copyNode(); // Make a copy
//Skip skipAmount nodes
skipped = 1;
while ((current.next() != null) && (skipped < skipAmount))
{
current = current.next();
skipped++;
};
oldNext = current.next(); // Save its next
current.insertAfter(copy); // Insert the copy after this one
current = oldNext; // Continue on with the rest
if (current.next() == null) // Did we actually get to the end early?
break; // Leave the loop
}
}
Just one of many
possible weaves.
Lots of design
choices.
For example, should
we break before the
last insert (when we
get to the end) or
after?
We should do
this!
Creating a node to weave
> SongNode node2 = new SongNode();
> node2.setPhrase(SongPhrase.riff2());
> node2.showFromMeOn(JMC.PIANO);
Doing a weave
> node.weave(node2,4,2);
> node.showFromMeOn(JMC.PIANO);
Weave Results
Before:
After
Walking the Weave
public void weave(SongNode nextOne, int count, int
skipAmount)
{
SongNode current = this; // Start from here
SongNode copy; // Where we keep the one to be
weaved in
SongNode oldNext; // Need this to insert properly
int skipped; // Number skipped currently
Skip forward
for (int i=1; i <= count; i++)
{
copy = nextOne.copyNode(); // Make a copy
//Skip skipAmount nodes
skipped = 1;
while ((current.next() != null) && (skipped < skipAmount))
{
current = current.next();
skipped++;
};
Then do an insert
if (current.next() == null) // Did we actually get to the end
early?
break; // Leave the loop
oldNext = current.next(); // Save its next
current.insertAfter(copy); // Insert the copy after this one
current = oldNext; // Continue on with the rest
}
More on Arrays vs. Lists
 Arrays
• Much easier to traverse
• Very fast to access a specific (nth
) element
• But really a pain to insert and delete.
• Hard to write the code
• Can take a long time if it’s a big array
 Lists
• More complex to traverse
• Slower to access a specific element
• Very easy to insert (and later we’ll see, delete)
• Simple code
• Takes no time at all to run
Another Version:
Creating a tree of song parts, each
with its own instrument
 SongNode and SongPhrase offer us
enormous flexibility in exploring musical
patterns.
 But it’s only one part!
 We’ve lost the ability of having different
parts starting at different times!
 Let’s get that back.
The Structure We’re Creating
Song
SongPart
SongPart
SongNode SongNode
SongNode
SongNode
SongNode
SongNode
SongNode
Starting to look like a tree…
Example
Song
import jm.music.data.*;
import jm.JMC;
import jm.util.*;
import jm.JMC;
public class MyFirstSong {
public static void main(String [] args) {
Song songroot = new Song();
SongNode node1 = new SongNode();
SongNode riff3 = new SongNode();
riff3.setPhrase(SongPhrase.riff3());
node1.repeatNext(riff3,16);
SongNode riff1 = new SongNode();
riff1.setPhrase(SongPhrase.riff1());
node1.weave(riff1,7,1);
SongPart part1 = new SongPart(JMC.PIANO, node1);
songroot.setFirst(part1);
SongNode node2 = new SongNode();
SongNode riff4 = new SongNode();
riff4.setPhrase(SongPhrase.riff4());
node2.repeatNext(riff4,20);
node2.weave(riff1,4,5);
SongPart part2 = new SongPart(JMC.STEEL_DRUMS, node2);
songroot.setSecond(part2);
songroot.show();
}
}
Making the Song() class:
Root of a song tree
import jm.music.data.*; import jm.JMC; import jm.util.*; import
jm.music.tools.*;
/**
* Class that represents a song
* @author Mark Guzdial
* @author Barb Ericson
*/
public class Song {
/** first Channel */
private SongPart first;
/** second Channel */
private SongPart second;
Our Song will
have two
branches (parts)
Our Song will
have two
branches (parts)
/**
* Take in a SongPart to make the first channel in the song
* @param channel1 the first channel in the song
*/
public void setFirst(SongPart channel1) {
first = channel1;
first.getMyPart().setChannel(1);
}
/**
* Take in a SongPart to make the second channel in the song
* @param channel2 the second channel in the song
*/
public void setSecond(SongPart channel2) {
second = channel2;
first.getMyPart().setChannel(2);
}
Simple accessorsSimple accessors
/**
* Make the score and show it
*/
public void show() {
// Make the Score that we'll assemble the parts into
// We'll set it up with a default time signature and tempo we like
Score myScore = new Score("My Song");
myScore.setTimeSignature(3,4);
myScore.setTempo(120.0);
// Now, construct the part and the score.
first.getMyPart().addPhrase(first.collect());
second.getMyPart().addPhrase(second.collect());
myScore.addPart(first.getMyPart());
myScore.addPart(second.getMyPart());
// At the end, let's see it!
View.notate(myScore);
}
}
To show a song is
to collect the two
parts and
assemble them
into the score.
To show a song is
to collect the two
parts and
assemble them
into the score.
SongPart class:
How each branch is implemented
import jm.music.data.*; import jm.JMC; import jm.util.*; import
jm.music.tools.*;
/**
* Class that represents a part of a song
* @author Mark Guzdial
* @author Barb Ericson
*/
public class SongPart {
/** SongPart has a Part */
private Part myPart;
/** the first node in the linked list */
private SongNode myList;
/**
* Construct a SongPart
* @param instrument MIDI instrument (program)
* @param startNode where the song list starts from
*/
public SongPart(int instrument, SongNode startNode) {
myPart = new Part(instrument);
myList = startNode;
}
/**
* Method to get the part
* @ return the part
*/
public Part getMyPart() {
return myPart;
}
A SongPart has a Jmusic part,
and a list of SongNodes
A SongPart has a Jmusic part,
and a list of SongNodes
Let SongNode do the work here
/**
* Collect parts of this SongPart
* @return all the notes in a phrase
*/
public Phrase collect() {
// delegate to SongNode's collect
return this.myList.collect();
}
/**
* Collect all notes in this SongPart and open it up for viewing.
*/
public void show() {
// Make the Score that we'll assemble the part into
// We'll set it up with a default time signature and tempo we like
// (Should probably make it possible to change these --
// maybe with inputs?)
Score myScore = new Score("My Song");
myScore.setTimeSignature(3,4);
myScore.setTempo(120.0);
// Now, construct the part and the score.
this.myPart.addPhrase(this.collect());
myScore.addPart(this.myPart);
// At the end, let's see it!
View.notate(myScore);
}
}
A SongPart can be
shown by itself, but
we’ll mostly use
Song’s show()
A SongPart can be
shown by itself, but
we’ll mostly use
Song’s show()
Structuring Music. Next?
 Linked lists (and a simple tree) are
powerful representations for music.
 Next, we’ll explore the use of this
representation for manipulating images.

Más contenido relacionado

La actualidad más candente

ikh331-06-distributed-programming
ikh331-06-distributed-programmingikh331-06-distributed-programming
ikh331-06-distributed-programming
Anung Ariwibowo
 
Error Control in Multimedia Communications using Wireless Sensor Networks report
Error Control in Multimedia Communications using Wireless Sensor Networks reportError Control in Multimedia Communications using Wireless Sensor Networks report
Error Control in Multimedia Communications using Wireless Sensor Networks report
Muragesh Kabbinakantimath
 
Distributed Data Structures
Distributed Data StructuresDistributed Data Structures
Distributed Data Structures
PDX Web & Design
 
Степан Кольцов — Rust — лучше, чем C++
Степан Кольцов — Rust — лучше, чем C++Степан Кольцов — Rust — лучше, чем C++
Степан Кольцов — Rust — лучше, чем C++
Yandex
 
How To Crack RSA Netrek Binary Verification System
How To Crack RSA Netrek Binary Verification SystemHow To Crack RSA Netrek Binary Verification System
How To Crack RSA Netrek Binary Verification System
Jay Corrales
 
Dns server clients (actual program)
Dns server clients (actual program)Dns server clients (actual program)
Dns server clients (actual program)
Youssef Dirani
 

La actualidad más candente (20)

What the &~#@&lt;!? (Pointers in Rust)
What the &~#@&lt;!? (Pointers in Rust)What the &~#@&lt;!? (Pointers in Rust)
What the &~#@&lt;!? (Pointers in Rust)
 
The Curious Clojurist - Neal Ford (Thoughtworks)
The Curious Clojurist - Neal Ford (Thoughtworks)The Curious Clojurist - Neal Ford (Thoughtworks)
The Curious Clojurist - Neal Ford (Thoughtworks)
 
Go a crash course
Go   a crash courseGo   a crash course
Go a crash course
 
Kotlin Coroutines. Flow is coming
Kotlin Coroutines. Flow is comingKotlin Coroutines. Flow is coming
Kotlin Coroutines. Flow is coming
 
Sasha Romijn - Everything I always wanted to know about crypto, but never tho...
Sasha Romijn - Everything I always wanted to know about crypto, but never tho...Sasha Romijn - Everything I always wanted to know about crypto, but never tho...
Sasha Romijn - Everything I always wanted to know about crypto, but never tho...
 
Protocol handler in Gecko
Protocol handler in GeckoProtocol handler in Gecko
Protocol handler in Gecko
 
Introduction to Rust
Introduction to RustIntroduction to Rust
Introduction to Rust
 
ikh331-06-distributed-programming
ikh331-06-distributed-programmingikh331-06-distributed-programming
ikh331-06-distributed-programming
 
Error Control in Multimedia Communications using Wireless Sensor Networks report
Error Control in Multimedia Communications using Wireless Sensor Networks reportError Control in Multimedia Communications using Wireless Sensor Networks report
Error Control in Multimedia Communications using Wireless Sensor Networks report
 
SSL Failing, Sharing, and Scheduling
SSL Failing, Sharing, and SchedulingSSL Failing, Sharing, and Scheduling
SSL Failing, Sharing, and Scheduling
 
Distributed Data Structures
Distributed Data StructuresDistributed Data Structures
Distributed Data Structures
 
Степан Кольцов — Rust — лучше, чем C++
Степан Кольцов — Rust — лучше, чем C++Степан Кольцов — Rust — лучше, чем C++
Степан Кольцов — Rust — лучше, чем C++
 
Distributed Computing Patterns in R
Distributed Computing Patterns in RDistributed Computing Patterns in R
Distributed Computing Patterns in R
 
Introduction to kotlin coroutines
Introduction to kotlin coroutinesIntroduction to kotlin coroutines
Introduction to kotlin coroutines
 
The Ring programming language version 1.5.1 book - Part 24 of 180
The Ring programming language version 1.5.1 book - Part 24 of 180The Ring programming language version 1.5.1 book - Part 24 of 180
The Ring programming language version 1.5.1 book - Part 24 of 180
 
Good Code
Good CodeGood Code
Good Code
 
Udp socket programming(Florian)
Udp socket programming(Florian)Udp socket programming(Florian)
Udp socket programming(Florian)
 
How To Crack RSA Netrek Binary Verification System
How To Crack RSA Netrek Binary Verification SystemHow To Crack RSA Netrek Binary Verification System
How To Crack RSA Netrek Binary Verification System
 
Dns server clients (actual program)
Dns server clients (actual program)Dns server clients (actual program)
Dns server clients (actual program)
 
Naïveté vs. Experience
Naïveté vs. ExperienceNaïveté vs. Experience
Naïveté vs. Experience
 

Destacado

Love in-ubud-marketing-smk-a ervan jonathan
Love in-ubud-marketing-smk-a ervan jonathanLove in-ubud-marketing-smk-a ervan jonathan
Love in-ubud-marketing-smk-a ervan jonathan
Ervan123
 
Perladangan organik
Perladangan organikPerladangan organik
Perladangan organik
muzammir1992
 
Knowledge management
Knowledge managementKnowledge management
Knowledge management
NEHA SHARMA
 

Destacado (20)

Love in-ubud-marketing-smk-a ervan jonathan
Love in-ubud-marketing-smk-a ervan jonathanLove in-ubud-marketing-smk-a ervan jonathan
Love in-ubud-marketing-smk-a ervan jonathan
 
Raspberry Pi
Raspberry PiRaspberry Pi
Raspberry Pi
 
Perladangan organik
Perladangan organikPerladangan organik
Perladangan organik
 
CIRC products
CIRC products CIRC products
CIRC products
 
A new beginning quiz!
A new beginning quiz!A new beginning quiz!
A new beginning quiz!
 
Knowledge management
Knowledge managementKnowledge management
Knowledge management
 
United Nations and what it has to do with ไทยแลนด์
United Nations and what it has to do with ไทยแลนด์United Nations and what it has to do with ไทยแลนด์
United Nations and what it has to do with ไทยแลนด์
 
The 8
The 8The 8
The 8
 
Дж. Максвел – батько класичної електродинаміки (до 185-річчя від дня народження)
Дж. Максвел – батько класичної електродинаміки (до 185-річчя від дня народження)Дж. Максвел – батько класичної електродинаміки (до 185-річчя від дня народження)
Дж. Максвел – батько класичної електродинаміки (до 185-річчя від дня народження)
 
Harvard Graduate School Education: teaching cs to teachers
Harvard Graduate School Education: teaching cs to teachersHarvard Graduate School Education: teaching cs to teachers
Harvard Graduate School Education: teaching cs to teachers
 
CS Teachers are Different than Software Developers: WIPSCE 2014 Keynote
CS Teachers are Different than Software Developers: WIPSCE 2014 KeynoteCS Teachers are Different than Software Developers: WIPSCE 2014 Keynote
CS Teachers are Different than Software Developers: WIPSCE 2014 Keynote
 
Question 4
Question 4Question 4
Question 4
 
MIT to the NYSE: Journey from University Tech to M&A
MIT to the NYSE: Journey from University Tech to M&AMIT to the NYSE: Journey from University Tech to M&A
MIT to the NYSE: Journey from University Tech to M&A
 
Mga krusada
Mga krusadaMga krusada
Mga krusada
 
Rutgers Invited Talk: Creative Expression to Motivate Interest in Computing
Rutgers Invited Talk: Creative Expression to Motivate Interest in ComputingRutgers Invited Talk: Creative Expression to Motivate Interest in Computing
Rutgers Invited Talk: Creative Expression to Motivate Interest in Computing
 
Sinister scuptor
Sinister scuptorSinister scuptor
Sinister scuptor
 
Metų laikai ir jų spalvos
Metų laikai ir jų spalvosMetų laikai ir jų spalvos
Metų laikai ir jų spalvos
 
Ervan jonathan
Ervan jonathanErvan jonathan
Ervan jonathan
 
Sinister sculptor quiz 2
Sinister sculptor quiz 2Sinister sculptor quiz 2
Sinister sculptor quiz 2
 
Zabytki
ZabytkiZabytki
Zabytki
 

Similar a Teaching linked lists data structures using MIDI

Structuring Music Using Linked Lists
Structuring Music Using Linked ListsStructuring Music Using Linked Lists
Structuring Music Using Linked Lists
Cynthia Marcello
 
Chapter 2 Part2 C
Chapter 2 Part2 CChapter 2 Part2 C
Chapter 2 Part2 C
ececourse
 
Sample Exam Questions on Python for revision
Sample Exam Questions on Python for revisionSample Exam Questions on Python for revision
Sample Exam Questions on Python for revision
afsheenfaiq2
 
Gdc09 Minigames
Gdc09 MinigamesGdc09 Minigames
Gdc09 Minigames
Susan Gold
 
Pointcuts and Analysis
Pointcuts and AnalysisPointcuts and Analysis
Pointcuts and Analysis
Wiwat Ruengmee
 

Similar a Teaching linked lists data structures using MIDI (20)

Structuring Music Using Linked Lists
Structuring Music Using Linked ListsStructuring Music Using Linked Lists
Structuring Music Using Linked Lists
 
DeVry GSP 115 All Assignments latest
DeVry GSP 115 All Assignments latestDeVry GSP 115 All Assignments latest
DeVry GSP 115 All Assignments latest
 
NativeBoost
NativeBoostNativeBoost
NativeBoost
 
Java Keeps Throttling Up!
Java Keeps Throttling Up!Java Keeps Throttling Up!
Java Keeps Throttling Up!
 
What Shazam doesn't want you to know
What Shazam doesn't want you to knowWhat Shazam doesn't want you to know
What Shazam doesn't want you to know
 
The Effect of Hierarchical Memory on the Design of Parallel Algorithms and th...
The Effect of Hierarchical Memory on the Design of Parallel Algorithms and th...The Effect of Hierarchical Memory on the Design of Parallel Algorithms and th...
The Effect of Hierarchical Memory on the Design of Parallel Algorithms and th...
 
Chapter 2 Part2 C
Chapter 2 Part2 CChapter 2 Part2 C
Chapter 2 Part2 C
 
Developing with Cello
Developing with CelloDeveloping with Cello
Developing with Cello
 
Machine language
Machine languageMachine language
Machine language
 
Tut7 4
Tut7 4Tut7 4
Tut7 4
 
Network simulator 2
Network simulator 2Network simulator 2
Network simulator 2
 
Sample Exam Questions on Python for revision
Sample Exam Questions on Python for revisionSample Exam Questions on Python for revision
Sample Exam Questions on Python for revision
 
C Programming Homework Help
C Programming Homework HelpC Programming Homework Help
C Programming Homework Help
 
DeVry GSP 115 Week 3 Assignment latest
DeVry GSP 115 Week 3 Assignment latestDeVry GSP 115 Week 3 Assignment latest
DeVry GSP 115 Week 3 Assignment latest
 
Erlang bootstrap course
Erlang bootstrap courseErlang bootstrap course
Erlang bootstrap course
 
Swug July 2010 - windows debugging by sainath
Swug July 2010 - windows debugging by sainathSwug July 2010 - windows debugging by sainath
Swug July 2010 - windows debugging by sainath
 
Gdc09 Minigames
Gdc09 MinigamesGdc09 Minigames
Gdc09 Minigames
 
III MCS python lab (1).pdf
III MCS python lab (1).pdfIII MCS python lab (1).pdf
III MCS python lab (1).pdf
 
Android and cpp
Android and cppAndroid and cpp
Android and cpp
 
Pointcuts and Analysis
Pointcuts and AnalysisPointcuts and Analysis
Pointcuts and Analysis
 

Más de Mark Guzdial

The Role of CS Departments in The US President’s “CS for All” Initiative: Pan...
The Role of CS Departments in The US President’s “CS for All” Initiative: Pan...The Role of CS Departments in The US President’s “CS for All” Initiative: Pan...
The Role of CS Departments in The US President’s “CS for All” Initiative: Pan...
Mark Guzdial
 

Más de Mark Guzdial (12)

Inventing computing education to meet
 all undergraduates’ needs
Inventing computing education to meet
 all undergraduates’ needsInventing computing education to meet
 all undergraduates’ needs
Inventing computing education to meet
 all undergraduates’ needs
 
Computing Education as a Foundation for 21st Century Literacy
Computing Education as a Foundation for 21st Century LiteracyComputing Education as a Foundation for 21st Century Literacy
Computing Education as a Foundation for 21st Century Literacy
 
Priming the Computer Science Teacher Pump Report: Finding a Home for Computer...
Priming the Computer Science Teacher Pump Report: Finding a Home for Computer...Priming the Computer Science Teacher Pump Report: Finding a Home for Computer...
Priming the Computer Science Teacher Pump Report: Finding a Home for Computer...
 
Using Learning Sciences Research to Improve Computing Teaching: Predictions, ...
Using Learning Sciences Research to Improve Computing Teaching: Predictions, ...Using Learning Sciences Research to Improve Computing Teaching: Predictions, ...
Using Learning Sciences Research to Improve Computing Teaching: Predictions, ...
 
The Role of CS Departments in The US President’s “CS for All” Initiative: Pan...
The Role of CS Departments in The US President’s “CS for All” Initiative: Pan...The Role of CS Departments in The US President’s “CS for All” Initiative: Pan...
The Role of CS Departments in The US President’s “CS for All” Initiative: Pan...
 
LaTICE 2016: Learner-Centered Design of Computing Education for All
LaTICE 2016: Learner-Centered Design of Computing Education for AllLaTICE 2016: Learner-Centered Design of Computing Education for All
LaTICE 2016: Learner-Centered Design of Computing Education for All
 
Critiquing CS Assessment from a CS for All lens: Dagstuhl Seminar Poster
Critiquing CS Assessment from a CS for All lens: Dagstuhl Seminar PosterCritiquing CS Assessment from a CS for All lens: Dagstuhl Seminar Poster
Critiquing CS Assessment from a CS for All lens: Dagstuhl Seminar Poster
 
VL/HCC 2015 Keynote: Requirements for a Computing Literate Society
VL/HCC 2015 Keynote:  Requirements for a Computing Literate SocietyVL/HCC 2015 Keynote:  Requirements for a Computing Literate Society
VL/HCC 2015 Keynote: Requirements for a Computing Literate Society
 
Providing learning and reflection opportunities to develop in-service CS teac...
Providing learning and reflection opportunities to develop in-service CS teac...Providing learning and reflection opportunities to develop in-service CS teac...
Providing learning and reflection opportunities to develop in-service CS teac...
 
Cyber securityeducation may2015
Cyber securityeducation may2015Cyber securityeducation may2015
Cyber securityeducation may2015
 
Flash Talk for the ECEP Alliance from the NSF BPC Community Meeting
Flash Talk for the ECEP Alliance from the NSF BPC Community MeetingFlash Talk for the ECEP Alliance from the NSF BPC Community Meeting
Flash Talk for the ECEP Alliance from the NSF BPC Community Meeting
 
Talk on Ebooks at the NSF BPC/CE21/STEM-C Community Meeting
Talk on Ebooks at the NSF BPC/CE21/STEM-C Community MeetingTalk on Ebooks at the NSF BPC/CE21/STEM-C Community Meeting
Talk on Ebooks at the NSF BPC/CE21/STEM-C Community Meeting
 

Último

Último (20)

Making communications land - Are they received and understood as intended? we...
Making communications land - Are they received and understood as intended? we...Making communications land - Are they received and understood as intended? we...
Making communications land - Are they received and understood as intended? we...
 
ICT role in 21st century education and it's challenges.
ICT role in 21st century education and it's challenges.ICT role in 21st century education and it's challenges.
ICT role in 21st century education and it's challenges.
 
ICT Role in 21st Century Education & its Challenges.pptx
ICT Role in 21st Century Education & its Challenges.pptxICT Role in 21st Century Education & its Challenges.pptx
ICT Role in 21st Century Education & its Challenges.pptx
 
Unit 3 Emotional Intelligence and Spiritual Intelligence.pdf
Unit 3 Emotional Intelligence and Spiritual Intelligence.pdfUnit 3 Emotional Intelligence and Spiritual Intelligence.pdf
Unit 3 Emotional Intelligence and Spiritual Intelligence.pdf
 
General Principles of Intellectual Property: Concepts of Intellectual Proper...
General Principles of Intellectual Property: Concepts of Intellectual  Proper...General Principles of Intellectual Property: Concepts of Intellectual  Proper...
General Principles of Intellectual Property: Concepts of Intellectual Proper...
 
Understanding Accommodations and Modifications
Understanding  Accommodations and ModificationsUnderstanding  Accommodations and Modifications
Understanding Accommodations and Modifications
 
2024-NATIONAL-LEARNING-CAMP-AND-OTHER.pptx
2024-NATIONAL-LEARNING-CAMP-AND-OTHER.pptx2024-NATIONAL-LEARNING-CAMP-AND-OTHER.pptx
2024-NATIONAL-LEARNING-CAMP-AND-OTHER.pptx
 
Graduate Outcomes Presentation Slides - English
Graduate Outcomes Presentation Slides - EnglishGraduate Outcomes Presentation Slides - English
Graduate Outcomes Presentation Slides - English
 
80 ĐỀ THI THỬ TUYỂN SINH TIẾNG ANH VÀO 10 SỞ GD – ĐT THÀNH PHỐ HỒ CHÍ MINH NĂ...
80 ĐỀ THI THỬ TUYỂN SINH TIẾNG ANH VÀO 10 SỞ GD – ĐT THÀNH PHỐ HỒ CHÍ MINH NĂ...80 ĐỀ THI THỬ TUYỂN SINH TIẾNG ANH VÀO 10 SỞ GD – ĐT THÀNH PHỐ HỒ CHÍ MINH NĂ...
80 ĐỀ THI THỬ TUYỂN SINH TIẾNG ANH VÀO 10 SỞ GD – ĐT THÀNH PHỐ HỒ CHÍ MINH NĂ...
 
Holdier Curriculum Vitae (April 2024).pdf
Holdier Curriculum Vitae (April 2024).pdfHoldier Curriculum Vitae (April 2024).pdf
Holdier Curriculum Vitae (April 2024).pdf
 
Accessible Digital Futures project (20/03/2024)
Accessible Digital Futures project (20/03/2024)Accessible Digital Futures project (20/03/2024)
Accessible Digital Futures project (20/03/2024)
 
How to Create and Manage Wizard in Odoo 17
How to Create and Manage Wizard in Odoo 17How to Create and Manage Wizard in Odoo 17
How to Create and Manage Wizard in Odoo 17
 
Single or Multiple melodic lines structure
Single or Multiple melodic lines structureSingle or Multiple melodic lines structure
Single or Multiple melodic lines structure
 
Mehran University Newsletter Vol-X, Issue-I, 2024
Mehran University Newsletter Vol-X, Issue-I, 2024Mehran University Newsletter Vol-X, Issue-I, 2024
Mehran University Newsletter Vol-X, Issue-I, 2024
 
Introduction to Nonprofit Accounting: The Basics
Introduction to Nonprofit Accounting: The BasicsIntroduction to Nonprofit Accounting: The Basics
Introduction to Nonprofit Accounting: The Basics
 
Spatium Project Simulation student brief
Spatium Project Simulation student briefSpatium Project Simulation student brief
Spatium Project Simulation student brief
 
How to Give a Domain for a Field in Odoo 17
How to Give a Domain for a Field in Odoo 17How to Give a Domain for a Field in Odoo 17
How to Give a Domain for a Field in Odoo 17
 
Food safety_Challenges food safety laboratories_.pdf
Food safety_Challenges food safety laboratories_.pdfFood safety_Challenges food safety laboratories_.pdf
Food safety_Challenges food safety laboratories_.pdf
 
REMIFENTANIL: An Ultra short acting opioid.pptx
REMIFENTANIL: An Ultra short acting opioid.pptxREMIFENTANIL: An Ultra short acting opioid.pptx
REMIFENTANIL: An Ultra short acting opioid.pptx
 
Sensory_Experience_and_Emotional_Resonance_in_Gabriel_Okaras_The_Piano_and_Th...
Sensory_Experience_and_Emotional_Resonance_in_Gabriel_Okaras_The_Piano_and_Th...Sensory_Experience_and_Emotional_Resonance_in_Gabriel_Okaras_The_Piano_and_Th...
Sensory_Experience_and_Emotional_Resonance_in_Gabriel_Okaras_The_Piano_and_Th...
 

Teaching linked lists data structures using MIDI

  • 1. Problem Solving with Data Structures using Java: A Multimedia Approach Chapter 6: Structuring Music Using Linked Lists
  • 3. JMusic: Java Music library  JMusic knows about WAV files and many other formats, too (e.g., QuickTime) • http://explodingart.com/jmusic • http://sourceforge.net/projects/jmusic/  We’ll use it for manipulating MIDI • Musical Instrument Digital Interface, an industry- standard interface used on electronic musical keyboards and PCs for computer control of musical instruments and devices.  MIDI is about recording music, not sound.
  • 4. JMusic pieces  import jm.music.data.*; • // To get Note and Phrase classes • // .* means “All the classes in the package.”  import jm.util.*; • // To get the View class that knows how to draw music notation from MIDI.  import jm.JMC; • // The class containing constants like JMC.FLUTE and JMC.C4
  • 8. Different ways of creating Phrases > Phrase phr2 = new Phrase("Phrase 2",4.0,JMC.FLUTE); > phr2.addNoteList(notes2)
  • 9. A Phrase that starts later > Phrase phr2 = new Phrase("Phrase 2",4.0,JMC.FLUTE); > phr2.addNoteList(notes2) > View.notate(phr2)
  • 10. Amazing Grace > AmazingGraceSong song1 = new AmazingGraceSong(); > song1.showMe(); > AmazingGraceSong song1 = new AmazingGraceSong(); > song1.showMe(); import jm.music.data.*; import jm.JMC; import jm.util.*; import jm.music.tools.*; /** * Class that represents the song Amazing Grace * @author Mark Guzdial * @author Barb Ericson */ public class AmazingGraceSong { /** the score for the song */ private Score myScore = new Score("Amazing Grace"); /** * Constructor to set up the song */ public AmazingGraceSong() { myScore.setTimeSignature(3,4);
  • 11. double[] phrase1Data = { JMC.G4, JMC.QN, JMC.C5, JMC.HN, JMC.E5,JMC.EN, JMC.C5,JMC.EN, JMC.E5,JMC.HN,JMC.D5,JMC.QN,JMC.C5,JMC.HN,JMC.A4,JMC.QN, JMC.G4,JMC.HN,JMC.G4,JMC.EN,JMC.A4,JMC.EN, JMC.C5,JMC.HN,JMC.E5,JMC.EN,JMC.C5,JMC.EN, JMC.E5,JMC.HN,JMC.D5,JMC.EN,JMC.E5,JMC.EN, JMC.G5,JMC.DHN }; double[] phrase2Data = { JMC.G5,JMC.HN,JMC.E5,JMC.EN,JMC.G5,JMC.EN, JMC.G5,JMC.HN,JMC.E5,JMC.EN,JMC.C5,JMC.EN, JMC.E5,JMC.HN,JMC.D5,JMC.QN, JMC.C5,JMC.HN,JMC.A4,JMC.QN, JMC.G4,JMC.HN,JMC.G4,JMC.EN,JMC.A4,JMC.EN, JMC.C5,JMC.HN,JMC.E5,JMC.EN,JMC.C5,JMC.EN, JMC.E5,JMC.HN,JMC.D5,JMC.QN, JMC.C5,JMC.DHN }; Phrase myPhrase = new Phrase(); myPhrase.addNoteList(phrase1Data); myPhrase.addNoteList(phrase2Data); // create a new part and add the phrase to it Part aPart = new Part("Parts", JMC.FLUTE, 1); aPart.addPhrase(myPhrase); // add the part to the score myScore.addPart(aPart); } /** * Method to show the song */ public void showMe() { View.notate(myScore); }}
  • 12. Imports and some private data import jm.music.data.*; import jm.JMC; import jm.util.*; import jm.music.tools.*; public class AmazingGraceSong { private Score myScore = new Score("Amazing Grace");  myScore is private instance data
  • 13. Filling the Score public AmazingGraceSong() { myScore.setTimeSignature(3,4); double[] phrase1Data = { JMC.G4, JMC.QN, JMC.C5, JMC.HN, JMC.E5,JMC.EN, JMC.C5,JMC.EN, JMC.E5,JMC.HN,JMC.D5,JMC.QN, JMC.C5,JMC.HN,JMC.A4,JMC.QN, JMC.G4,JMC.HN,JMC.G4,JMC.EN,JMC.A4,JMC.EN, JMC.C5,JMC.HN,JMC.E5,JMC.EN,JMC.C5,JMC.EN, JMC.E5,JMC.HN,JMC.D5,JMC.EN,JMC.E5,JMC.EN, JMC.G5,JMC.DHN }; double[] phrase2Data = { JMC.G5,JMC.HN,JMC.E5,JMC.EN,JMC.G5,JMC.EN, JMC.G5,JMC.HN,JMC.E5,JMC.EN,JMC.C5,JMC.EN, JMC.E5,JMC.HN,JMC.D5,JMC.QN, JMC.C5,JMC.HN,JMC.A4,JMC.QN, JMC.G4,JMC.HN,JMC.G4,JMC.EN,JMC.A4,JMC.EN, JMC.C5,JMC.HN,JMC.E5,JMC.EN,JMC.C5,JMC.EN, JMC.E5,JMC.HN,JMC.D5,JMC.QN, JMC.C5,JMC.DHN }; Phrase myPhrase = new Phrase(); myPhrase.addNoteList(phrase1Data); myPhrase.addNoteList(phrase2Data); // create a new part and add the phrase to it Part aPart = new Part("Parts", JMC.FLUTE, 1); aPart.addPhrase(myPhrase); // add the part to the score myScore.addPart(aPart); } Each array is note, duration, note, duration, note, duration, etc. Broke it roughly into halves, just to make it easier to manipulate. This is a constructor that will create the data for the object when it is created.
  • 14. Showing the Score public void showMe(){ View.notate(myScore); };
  • 15. Part: Instrument & Phrase: startingTime & The Organization of JMusic Objects Phrase: startingTime & Note (pitch,duration) Note (pitch,duration) Note (pitch,duration) Note (pitch,duration) Note (pitch,duration) Part: Instrument & Score: timeSignature, tempo, &
  • 16. Thought Experiment  How are they doing that?  How can there be any number of Notes in a Phrase, Phrases in a Part, and Parts in a Score? • (Hint: They ain’t usin’ arrays!)
  • 17. How do we explore composition here?  We want to quickly and easily throw together notes in different groupings and see how they sound.  The current JMusic structure models music. • Let’s try to create a structure that models thinking about music as bunches of riffs/SongElements that we want to combine in different ways.
  • 18. Introducing the Linked List  A linked list is information broken into smaller pieces, where each piece knows the next piece, but none other.
  • 19. Another example of a linked list  Non-linear video editing (like in iMovie) • You have a collection of video clips (information) • You drag them into a timeline. • Each clip still doesn’t know all clips, but it knows the next one.
  • 20. Why use linked lists versus arrays?  Just two reasons now, more later: 1. Can grow to any size (well, as long as memory permits) • Just create a new element and poke it into the list. 1. MUCH easier to insert (and delete)!
  • 21. SongNode and SongPhrase  SongNode instances will hold pieces (phrases) from SongPhrase.  SongNode instances will be the nodes in the linked list • Each one will know its next.  Ordering will encode the order in the Part. • Each one will get appended after the last.
  • 22. Using SongNode and SongPhrase Welcome to DrJava. > import jm.JMC; > SongNode first = new SongNode(); > first.setPhrase(SongPhrase.riff1()); > first.showFromMeOn(JMC.FLUTE); > SongNode second = new SongNode(); > second.setPhrase(SongPhrase.riff2()); > second.showFromMeOn(JMC.PIANO); > first.setNext(second); > first.showFromMeOn(JMC.SAX);
  • 23. Inserting a third node > SongNode third = new SongNode(); > third.setPhrase(SongPhrase.riff3()); > third.showFromMeOn(JMC.PIANO); > first.setNext(third); > third.setNext(second); > first.showFromMeOn(JMC.FLUTE);
  • 24. How to think about it  To start: first myPhrase: riff1 next: second second myPhrase: riff2 next: null
  • 25. How to think about it first myPhrase: riff1 next: third second myPhrase: riff2 next: null myPhrase: riff3 next: second third
  • 26. Declarations for SongNode import jm.music.data.*; import jm.JMC; import jm.util.*; import jm.music.tools.*; public class SongNode { /** * the next SongNode in the list */ private SongNode next; /** * the Phrase containing the notes and durations associated with this node */ private Phrase myPhrase; SongNode’s know their Phrase and the next node in the list
  • 27. Constructor for SongNode /** * When we make a new element, the next part is empty, and ours is a blank new part */ public SongNode(){ this.next = null; this.myPhrase = new Phrase(); }
  • 28. Setting the phrase /** * setPhrase takes a Phrase and makes it the one for this node * @param thisPhrase the phrase for this node */ public void setPhrase(Phrase thisPhrase){ this.myPhrase = thisPhrase; }
  • 29. Linked list methods /** * Creates a link between the current node and the input node * @param nextOne the node to link to */ public void setNext(SongNode nextOne){ this.next = nextOne; } /** * Provides public access to the next node. * @return a SongNode instance (or null) */ public SongNode getNext(){ return this.next; }
  • 30. insertAfter /** * Insert the input SongNode AFTER this node, * and make whatever node comes NEXT become the next of the input node. * @param nextOne SongNode to insert after this one */ public void insertAfter(SongNode nextOne) { SongNode oldNext = this.next(); // Save its next this.setNext(nextOne); // Insert the copy nextOne.setNext(oldNext); // Make the copy point on to the rest }
  • 31. Question: // Assume we’ve just created all 4 nodes node1.setNext(node2); node2.setNext(node3); node1.insert(node4);  Which of these is true? (More than one.)  A. node4’s next is null.  B. node3’s next is null.  C. node1’s next is node2  D. node4’s next is node2
  • 32. After Group Question: // Assume we’ve just created all 4 nodes node1.setNext(node2); node2.setNext(node3); node1.insert(node4);  Which of these is true? (More than one.)  A. node4’s next is null.  B. node3’s next is null.  C. node1’s next is node2  D. node4’s next is node2
  • 33. Using and tracing insertAfter() > SongNode nodeA = new SongNode(); > SongNode nodeB = new SongNode(); > nodeA.setNext(nodeB); > SongNode nodeC = new SongNode() > nodeA.insertAfter(nodeC); public void insertAfter(SongNode nextOne) { SongNode oldNext = this.next(); // Save its next this.setNext(nextOne); // Insert the copy nextOne.setNext(oldNext); // Make the copy point on to the rest }
  • 34. Traversing the list /** * Collect all the notes from this node on * in an part (then a score) and open it up for viewing. * @param instrument MIDI instrument (program) to be used in playing this list */ public void showFromMeOn(int instrument){ // Make the Score that we'll assemble the elements into // We'll set it up with a default time signature and tempo we like // (Should probably make it possible to change these -- maybe with inputs?) Score myScore = new Score("My Song"); myScore.setTimeSignature(3,4); myScore.setTempo(120.0); // Make the Part that we'll assemble things into Part myPart = new Part(instrument); // Make a new Phrase that will contain the notes from all the phrases Phrase collector = new Phrase(); // Start from this element (this) SongNode current = this; // While we're not through... while (current != null) { collector.addNoteList(current.getNotes()); // Now, move on to the next element current = current.next(); }; // Now, construct the part and the score. myPart.addPhrase(collector); myScore.addPart(myPart); // At the end, let's see it! View.notate(myScore); }
  • 35. The Core of the Traversal // Make a new Phrase that will contain the notes from all the phrases Phrase collector = new Phrase(); // Start from this element (this) SongNode current = this; // While we're not through... while (current != null) { collector.addNoteList(current.getNotes()); // Now, move on to the next element current = current.next(); };
  • 36. Step 1: // Start from this element (this) SongNode current = this; first myPhrase: riff1 next: third second myPhrase: riff2 next: null myPhrase: riff3 next: second third current
  • 37. // While we're not through... while (current != null) { collector.addNoteList(current.getNotes()); first myPhrase: riff1 next: third second myPhrase: riff2 next: null myPhrase: riff3 next: second third Collector: current
  • 38. // Now, move on to the next element current = current.next(); }; first myPhrase: riff1 next: second second myPhrase: riff2 next: null myPhrase: riff3 next: second third Collector: current
  • 39. // While we're not through... while (current != null) { collector.addNoteList(current.getNotes()); first myPhrase: riff1 next: third second myPhrase: riff2 next: null myPhrase: riff3 next: second third Collector: current
  • 40. // Now, move on to the next element current = current.next(); }; first myPhrase: riff1 next: node2 second myPhrase: riff2 next: null myPhrase: riff3 next: second third Collector: current
  • 41. // While we're not through... while (current != null) { collector.addNoteList(current.getNotes()); first myPhrase: riff1 next: third second myPhrase: riff2 next: null myPhrase: riff3 next: second third Collector: current
  • 42. // Now, move on to the next element current = current.next(); }; first myPhrase: riff1 next: third second myPhrase: riff2 next: null myPhrase: riff2 next: second third Collector: currentnull
  • 43. // While we're not through... while (current != null) first myPhrase: riff1 next: node2 second myPhrase: riff2 next: null myPhrase: riff2 next: second third Collector: currentnull
  • 44. Then return what you collected // Now, construct the part and the score. myPart.addPhrase(collector); myScore.addPart(myPart); // At the end, let's see it! View.notate(myScore); }
  • 45. Question  At the end of showFromMeOn(), which of these is true? (More than one)  A. this is the last node in the list.  B. The nodes in the list are unchanged.  C. current is null.  D. next on the last node in the list is null.  E. The first node on the list points to current.
  • 46. After-Group Question  At the end of showFromMeOn(), which of these is true? (More than one)  A. this is the last node in the list.  B. The nodes in the list are unchanged.  C. current is null.  D. next on the last node in the list is null.  E. The first node on the list points to current.
  • 47. SongPhrase  SongPhrase is a collection of static methods.  We don’t ever need an instance of SongPhrase.  Instead, we use it to store methods that return phrases. • It’s not very object-oriented, but it’s useful here.
  • 48. SongPhrase.riff1() import jm.music.data.*; import jm.JMC; import jm.util.*; import jm.music.tools.*; public class SongPhrase { //Little Riff1 static public Phrase riff1() { double[] phrasedata = {JMC.G3,JMC.EN,JMC.B3,JMC.EN,JMC.C4,JMC.EN,JMC.D4,JMC.EN}; Phrase myPhrase = new Phrase(); myPhrase.addNoteList(phrasedata); return myPhrase;
  • 49. SongPhrase.riff2() //Little Riff2 static public Phrase riff2() { double[] phrasedata = {JMC.D4,JMC.EN,JMC.C4,JMC.EN,JMC.E4,JMC.E N,JMC.G4,JMC.EN}; Phrase myPhrase = new Phrase(); myPhrase.addNoteList(phrasedata); return myPhrase; }
  • 50. Computing a phrase //Larger Riff1 static public Phrase pattern1() { double[] riff1data = {JMC.G3,JMC.EN,JMC.B3,JMC.EN,JMC.C4,JMC.EN,JMC.D4,JMC.EN}; double[] riff2data = {JMC.D4,JMC.EN,JMC.C4,JMC.EN,JMC.E4,JMC.EN,JMC.G4,JMC.EN}; Phrase myPhrase = new Phrase(); // 3 of riff1, 1 of riff2, and repeat all of it 3 times for (int counter1 = 1; counter1 <= 3; counter1++) {for (int counter2 = 1; counter2 <= 3; counter2++) myPhrase.addNoteList(riff1data); myPhrase.addNoteList(riff2data); }; return myPhrase; }
  • 51. Question  How many times does addNoteList end up being called in that method?  A. 6 times  B. 12 times  C. 4 times  D. 8 times
  • 52. After-Group Question  How many times does addNoteList end up being called in that method?  A. 6 times  B. 12 times  C. 4 times  D. 8 times
  • 53. As long as it’s a phrase…  The way that we use SongNote and SongPhrase, any method that returns a phrase is perfectly valid SongPhrase method.
  • 54. 10 Random Notes (Could be less random…) /* * 10 random notes **/ static public Phrase random() { Phrase ranPhrase = new Phrase(); Note n = null; for (int i=0; i < 10; i++) { n = new Note((int) (128*Math.random()),0.1); ranPhrase.addNote(n); } return ranPhrase; }
  • 55. 10 Slightly Less Random Notes /* * 10 random notes above middle C **/ static public Phrase randomAboveC() { Phrase ranPhrase = new Phrase(); Note n = null; for (int i=0; i < 10; i++) { n = new Note((int) (60+(5*Math.random())),0.25); ranPhrase.addNote(n); } return ranPhrase; }
  • 56. Going beyond connecting nodes  So far, we’ve just created nodes and connected them up.  What else can we do?  Well, music is about repetition and interleaving of themes. • Let’s create those abilities for SongNodes.
  • 57. Repeating a Phrase Welcome to DrJava. > SongNode node = new SongNode(); > node.setPhrase(SongPhrase.randomAboveC()); > SongNode node1 = new SongNode(); > node1.setPhrase(SongPhrase.riff1()); > node.repeatNext(node1,10); > import jm.JMC; > node.showFromMeOn(JMC.PIANO);
  • 58. What it looks like node node1 node1 node1 …
  • 59. Repeating /** * Repeat the input phrase for the number of times specified. * It always appends to the current node, NOT insert. * @param nextOne node to be copied in to list * @param count number of times to copy it in. */ public void repeatNext(SongNode nextOne,int count) { SongNode current = this; // Start from here SongNode copy; // Where we keep the current copy for (int i=1; i <= count; i++) { copy = nextOne.copyNode(); // Make a copy current.setNext(copy); // Set as next current = copy; // Now append to copy } } Note! What happens to this’s next? How would you create a looong repeat chain of several types of phrases with this?
  • 60. Here’s making a copy /** * copyNode returns a copy of this node * @return another song node with the same notes */ public SongNode copyNode(){ SongNode returnMe = new SongNode(); returnMe.setPhrase(this.getPhrase()); return returnMe; }
  • 61. Step 1: public void repeatNext(SongNode nextOne,int count) { SongNode current = this; // Start from here SongNode copy; // Where we keep the current copy node phrase: 10 random notes next: null current node1 phrase: riff1() next: null nextOne
  • 62. Step 2: copy = nextOne.copyNode(); // Make a copy node phrase: 10 random notes next: null current node1 phrase: riff1() next: null phrase: riff1() next: null copy nextOne
  • 63. Step 3: current.setNext(copy); // Set as next node phrase: 10 random notes next: current node1 phrase: riff1() next: null phrase: riff1() next: null copy nextOne
  • 64. Step 4: current = copy; // Now append to copy node phrase: 10 random notes next: current node1 phrase: riff1() next: null phrase: riff1() next: null copy nextOne
  • 65. Step 5 & 6: copy = nextOne.copyNode(); // Make a copy current.setNext(copy); // Set as next node phrase: 10 random notes next: current node1 phrase: riff1() next: null phrase: riff1() next: copy phrase: riff1() next: null nextOne
  • 66. Step 7 (and so on): current = copy; // Now append to copy node phrase: 10 random notes next: current node1 phrase: riff1() next: null phrase: riff1() next: copy phrase: riff1() next: null nextOne
  • 67. What happens if the node already points to something?  Consider repeatNext and how it inserts: It simply sets the next value.  What if the node already had a next?  repeatNext will erase whatever used to come next.  How can we fix it?
  • 68. repeatNextInserting /** * Repeat the input phrase for the number of times specified. * But do an insertion, to save the rest of the list. * @param nextOne node to be copied into the list * @param count number of times to copy it in. **/ public void repeatNextInserting(SongNode nextOne, int count){ SongNode current = this; // Start from here SongNode copy; // Where we keep the current copy for (int i=1; i <= count; i++) { copy = nextOne.copyNode(); // Make a copy current.insertAfter(copy); // INSERT after current current = copy; // Now append to copy } }
  • 69. Weaving /** * Weave the input phrase count times every skipAmount nodes * @param nextOne node to be copied into the list * @param count how many times to copy * @param skipAmount how many nodes to skip per weave */ public void weave(SongNode nextOne, int count, int skipAmount) { SongNode current = this; // Start from here SongNode copy; // Where we keep the one to be weaved in SongNode oldNext; // Need this to insert properly int skipped; // Number skipped currently for (int i=1; i <= count; i++) { copy = nextOne.copyNode(); // Make a copy //Skip skipAmount nodes skipped = 1; while ((current.next() != null) && (skipped < skipAmount)) { current = current.next(); skipped++; }; oldNext = current.next(); // Save its next current.insertAfter(copy); // Insert the copy after this one current = oldNext; // Continue on with the rest if (current.next() == null) // Did we actually get to the end early? break; // Leave the loop } } Just one of many possible weaves. Lots of design choices. For example, should we break before the last insert (when we get to the end) or after? We should do this!
  • 70. Creating a node to weave > SongNode node2 = new SongNode(); > node2.setPhrase(SongPhrase.riff2()); > node2.showFromMeOn(JMC.PIANO);
  • 71. Doing a weave > node.weave(node2,4,2); > node.showFromMeOn(JMC.PIANO);
  • 73. Walking the Weave public void weave(SongNode nextOne, int count, int skipAmount) { SongNode current = this; // Start from here SongNode copy; // Where we keep the one to be weaved in SongNode oldNext; // Need this to insert properly int skipped; // Number skipped currently
  • 74. Skip forward for (int i=1; i <= count; i++) { copy = nextOne.copyNode(); // Make a copy //Skip skipAmount nodes skipped = 1; while ((current.next() != null) && (skipped < skipAmount)) { current = current.next(); skipped++; };
  • 75. Then do an insert if (current.next() == null) // Did we actually get to the end early? break; // Leave the loop oldNext = current.next(); // Save its next current.insertAfter(copy); // Insert the copy after this one current = oldNext; // Continue on with the rest }
  • 76. More on Arrays vs. Lists  Arrays • Much easier to traverse • Very fast to access a specific (nth ) element • But really a pain to insert and delete. • Hard to write the code • Can take a long time if it’s a big array  Lists • More complex to traverse • Slower to access a specific element • Very easy to insert (and later we’ll see, delete) • Simple code • Takes no time at all to run
  • 77. Another Version: Creating a tree of song parts, each with its own instrument  SongNode and SongPhrase offer us enormous flexibility in exploring musical patterns.  But it’s only one part!  We’ve lost the ability of having different parts starting at different times!  Let’s get that back.
  • 78. The Structure We’re Creating Song SongPart SongPart SongNode SongNode SongNode SongNode SongNode SongNode SongNode Starting to look like a tree…
  • 79. Example Song import jm.music.data.*; import jm.JMC; import jm.util.*; import jm.JMC; public class MyFirstSong { public static void main(String [] args) { Song songroot = new Song(); SongNode node1 = new SongNode(); SongNode riff3 = new SongNode(); riff3.setPhrase(SongPhrase.riff3()); node1.repeatNext(riff3,16); SongNode riff1 = new SongNode(); riff1.setPhrase(SongPhrase.riff1()); node1.weave(riff1,7,1); SongPart part1 = new SongPart(JMC.PIANO, node1); songroot.setFirst(part1); SongNode node2 = new SongNode(); SongNode riff4 = new SongNode(); riff4.setPhrase(SongPhrase.riff4()); node2.repeatNext(riff4,20); node2.weave(riff1,4,5); SongPart part2 = new SongPart(JMC.STEEL_DRUMS, node2); songroot.setSecond(part2); songroot.show(); } }
  • 80. Making the Song() class: Root of a song tree import jm.music.data.*; import jm.JMC; import jm.util.*; import jm.music.tools.*; /** * Class that represents a song * @author Mark Guzdial * @author Barb Ericson */ public class Song { /** first Channel */ private SongPart first; /** second Channel */ private SongPart second; Our Song will have two branches (parts) Our Song will have two branches (parts)
  • 81. /** * Take in a SongPart to make the first channel in the song * @param channel1 the first channel in the song */ public void setFirst(SongPart channel1) { first = channel1; first.getMyPart().setChannel(1); } /** * Take in a SongPart to make the second channel in the song * @param channel2 the second channel in the song */ public void setSecond(SongPart channel2) { second = channel2; first.getMyPart().setChannel(2); } Simple accessorsSimple accessors
  • 82. /** * Make the score and show it */ public void show() { // Make the Score that we'll assemble the parts into // We'll set it up with a default time signature and tempo we like Score myScore = new Score("My Song"); myScore.setTimeSignature(3,4); myScore.setTempo(120.0); // Now, construct the part and the score. first.getMyPart().addPhrase(first.collect()); second.getMyPart().addPhrase(second.collect()); myScore.addPart(first.getMyPart()); myScore.addPart(second.getMyPart()); // At the end, let's see it! View.notate(myScore); } } To show a song is to collect the two parts and assemble them into the score. To show a song is to collect the two parts and assemble them into the score.
  • 83. SongPart class: How each branch is implemented import jm.music.data.*; import jm.JMC; import jm.util.*; import jm.music.tools.*; /** * Class that represents a part of a song * @author Mark Guzdial * @author Barb Ericson */ public class SongPart { /** SongPart has a Part */ private Part myPart; /** the first node in the linked list */ private SongNode myList;
  • 84. /** * Construct a SongPart * @param instrument MIDI instrument (program) * @param startNode where the song list starts from */ public SongPart(int instrument, SongNode startNode) { myPart = new Part(instrument); myList = startNode; } /** * Method to get the part * @ return the part */ public Part getMyPart() { return myPart; } A SongPart has a Jmusic part, and a list of SongNodes A SongPart has a Jmusic part, and a list of SongNodes
  • 85. Let SongNode do the work here /** * Collect parts of this SongPart * @return all the notes in a phrase */ public Phrase collect() { // delegate to SongNode's collect return this.myList.collect(); }
  • 86. /** * Collect all notes in this SongPart and open it up for viewing. */ public void show() { // Make the Score that we'll assemble the part into // We'll set it up with a default time signature and tempo we like // (Should probably make it possible to change these -- // maybe with inputs?) Score myScore = new Score("My Song"); myScore.setTimeSignature(3,4); myScore.setTempo(120.0); // Now, construct the part and the score. this.myPart.addPhrase(this.collect()); myScore.addPart(this.myPart); // At the end, let's see it! View.notate(myScore); } } A SongPart can be shown by itself, but we’ll mostly use Song’s show() A SongPart can be shown by itself, but we’ll mostly use Song’s show()
  • 87. Structuring Music. Next?  Linked lists (and a simple tree) are powerful representations for music.  Next, we’ll explore the use of this representation for manipulating images.

Notas del editor

  1. No, they don’t have to write these all down. They’re in the book.
  2. Phrases can have instruments, but not the general design principle. Can have any number of notes in a Phrase, any number of Phrases in a Part, any number of Parts in a Score.