Problem Solving with Data Structures using Java: A Multimedia Approach Chapter 9: Lists and Trees for Structuring Sounds.

Slides:



Advertisements
Similar presentations
Chapter 7. Binary Search Trees
Advertisements

Problem Solving with Data Structures using Java: A Multimedia Approach Chapter 4: Objects as Agents: Manipulating Turtles.
Binary Trees. DCS – SWC 2 Binary Trees Sets and Maps in Java are also available in tree-based implementations A Tree is – in this context – a data structure.
Binary Trees Chapter 6. Linked Lists Suck By now you realize that the title to this slide is true… By now you realize that the title to this slide is.
Georgia Institute of Technology Workshop for CS-AP Teachers Chapter 3 Advanced Object-Oriented Concepts.
Lecture 8 CS203. Implementation of Data Structures 2 In the last couple of weeks, we have covered various data structures that are implemented in the.
©The McGraw-Hill Companies, Inc. Permission required for reproduction or display. ETC - 1 What comes next? Recursion (Chapter 15) Recursive Data Structures.
CS 206 Introduction to Computer Science II 10 / 14 / 2009 Instructor: Michael Eckmann.
CS 106 Introduction to Computer Science I 11 / 15 / 2006 Instructor: Michael Eckmann.
CS 206 Introduction to Computer Science II 02 / 11 / 2009 Instructor: Michael Eckmann.
Chapter 12 Trees. Copyright © 2005 Pearson Addison-Wesley. All rights reserved Chapter Objectives Define trees as data structures Define the terms.
1 Joe Meehean.  Important and common problem  Given a collection, determine whether value v is a member  Common variation given a collection of unique.
Chapter 08 Binary Trees and Binary Search Trees © John Urrutia 2013, All Rights Reserved.
Problem Solving with Data Structures using Java: A Multimedia Approach Chapter 8: Trees of Images.
Test. A software house decides to develop a DVD renting program. The product manager identifies the following requirements: Every DVD will have a title,
CS 206 Introduction to Computer Science II 09 / 30 / 2009 Instructor: Michael Eckmann.
(c) University of Washington04-1 CSC 143 Java Inheritance Example (Review)
More on Trees CS1316: Representing Structure and Behavior.
Problem Solving with Data Structures using Java: A Multimedia Approach Chapter 5: Arrays: A Static Data Structure for Sounds.
MIT AITI 2002 Abstract Classes, Interfaces. Abstract Classes What is an abstract class? An abstract class is a class in which one or more methods is declared,
Lists (and other data structures) that Loop CS1316: Representing Structure and Behavior.
Trees Chapter 8. 2 Tree Terminology A tree consists of a collection of elements or nodes, organized hierarchically. The node at the top of a tree is called.
Chapter Tow Search Trees BY HUSSEIN SALIM QASIM WESAM HRBI FADHEEL CS 6310 ADVANCE DATA STRUCTURE AND ALGORITHM DR. ELISE DE DONCKER 1.
Problem Solving with Data Structures using Java: A Multimedia Approach Chapter 10: Generalizing Lists and Trees.
Tree. Basic characteristic Top node = root Left and right subtree Node 1 is a parent of node 2,5,6. –Node 2 is a parent of node.
Lecture Objectives  To learn how to use a tree to represent a hierarchical organization of information  To learn how to use recursion to process trees.
CS305j Introduction to Computing Inheritance and Polymorphism 1 Topic 26 Introduction to Inheritance and Polymorphism "One purpose of CRC cards [a design.
REVIEW On Friday we explored Client-Server Applications with Sockets. Servers must create a ServerSocket object on a specific Port #. They then can wait.
COP3530 Data Structures600 Stack Stack is one the most useful ADTs. Like list, it is a collection of data items. Supports “LIFO” (Last In First Out) discipline.
Data Structures : Project 5 Data Structures Project 5 – Expression Trees and Code Generation.
Spring 2010CS 2251 Trees Chapter 6. Spring 2010CS 2252 Chapter Objectives Learn to use a tree to represent a hierarchical organization of information.
Structuring Music CS1316: Representing Structure and Behavior.
Priority Queues and Binary Heaps Chapter Trees Some animals are more equal than others A queue is a FIFO data structure the first element.
Tree (new ADT) Terminology:  A tree is a collection of elements (nodes)  Each node may have 0 or more successors (called children)  How many does a.
Starting at Binary Trees
Topic 15 The Binary Search Tree ADT Binary Search Tree A binary search tree (BST) is a binary tree with an ordering property of its elements, such.
Binary Search Trees Nilanjan Banerjee. 2 Goal of today’s lecture Learn about Binary Search Trees Discuss the first midterm.
Topic 1 Object Oriented Programming. 1-2 Objectives To review the concepts and terminology of object-oriented programming To discuss some features of.
Problem Solving with Data Structures using Java: A Multimedia Approach Chapter 12: Circular Linked Lists And Graphs: Lists and Trees That Loop.
Starting Out with C++ Early Objects Seventh Edition by Tony Gaddis, Judy Walters, and Godfrey Muganda Modified for use at Midwestern State University Chapter.
CE 221 Data Structures and Algorithms Chapter 4: Trees (Binary) Text: Read Weiss, §4.1 – 4.2 1Izmir University of Economics.
CS 206 Introduction to Computer Science II 10 / 02 / 2009 Instructor: Michael Eckmann.
Week 10 - Friday.  What did we talk about last time?  Graph representations  Adjacency matrix  Adjacency lists  Depth first search.
Coming up: Inheritance
Binary Tree. Some Terminologies Short review on binary tree Tree traversals Binary Search Tree (BST)‏ Questions.
Finally! Making the Villagers CS1316: Representing Structure and Behavior.
Week 15 – Wednesday.  What did we talk about last time?  Review up to Exam 1.
(c) University of Washington06-1 CSC 143 Java Inheritance Tidbits.
GameDevClub CODE CHEAT SHEET NOTE: ALL OF THE CODE IS CASE-SENSITIVE AND THE SYNTAX IS STRICT SO A LOT OF YOUR ERRORS WILL PROBABLY COME FROM TYPOS If.
1. What is it? It is a queue that access elements according to their importance value. Eg. A person with broken back should be treated before a person.
CS Prelim Review – 10/15/09  First prelim is TOMORROW at 7:30 PM  Review session – Tonight 7:30 PM, Phillips 101  Things you should do:  Review every.
OOP Basics Classes & Methods (c) IDMS/SQL News
INF3110 Group 2 EXAM 2013 SOLUTIONS AND HINTS. But first, an example of compile-time and run-time type checking Imagine we have the following code. What.
CSE 143 Lecture 9 Recursion slides created by Alyssa Harding
Chapter 9: Continuing Classes By Matt Hirsch. Table Of Contents 1.Static Fields and Methods 2.Inheritance I. Recycle Code with Inheritance II. Overriding.
Stacks as an Abstract Data Type CS1316: Representing Structure and Behavior.
Georgia Institute of Technology More on Creating Classes Barb Ericson Georgia Institute of Technology June 2006.
Trees Lecture 12 CS2110 – Fall 2017.
Creating and Modifying Text part 2
CS1316: Representing Structure and Behavior
CS1316: Representing Structure and Behavior
Find in a linked list? first last 7  4  3  8 NULL
CS1316: Representing Structure and Behavior
Finally! Making the Villagers
Stacks as an Abstract Data Type
Tree.
Problem Solving with Data Structures using Java: A Multimedia Approach
Tree (new ADT) Terminology: A tree is a collection of elements (nodes)
Presentation transcript:

Problem Solving with Data Structures using Java: A Multimedia Approach Chapter 9: Lists and Trees for Structuring Sounds

Chapter Objectives

Story Structuring sounds into songs (? Collections?) Version 1: Representing linearity through elements order. Repeating and weaving with sounds Finding and replacing sounds in a list How do trace and debug what we’re doing with sounds and lists? Structuring sounds into songs Version 2: Creating trees of sounds. Traversing in a tree Pre-order and post-order

SoundElement: Creating a linked list of sounds /** * Sounds for a linked list **/ public class SoundElement { /** * The sound this element is associated with **/ Sound mySound; /** * The next element to process **/ public SoundElement next;

Constructing an element /** * Constructor sets next to null * and references the input sound. **/ public SoundElement(Sound aSound){ next = null; mySound = aSound; }

Linked List blah-blah-blah /** * Methods to set and get next elements nextOne next element in list **/ public void setNext(SoundElement nextOne){ this.next = nextOne; } public SoundElement getNext(){ return this.next; }

Printing a SoundElement list /** * Provide a printable representation of me the information string */ public String toString(){ return "SoundElement with sound: " + mySound + " (and next: " + next + ")."; } Think about it: What is “next”? What method gets called to convert next into a string? What method is this?

Methods for playing /** * The method to get the sound the sound associated with this node */ public Sound getSound(){return mySound;} /** * Play JUST me, blocked. */ public void playSound(){mySound.blockingPlay();} /** * Play JUST me, blocked. */ public void blockingPlay(){mySound.blockingPlay();}

Methods for Playing /** * Play the list of sound elements * after me **/ public void playFromMeOn(){ this.collect().play(); } /** * Collect all the sounds from me on, * recursively. **/ public Sound collect(){ if (this.getNext() == null) {return mySound;} else {return mySound.append(this.getNext().collect());} }

Starting to explore > Sound s = new Sound("D:/cs1316/mediasources/shh-a-h.wav"); > Sound t = new Sound("D:/cs1316/mediasources/croak-h.wav"); > Sound u = new Sound("D:/cs1316/mediasources/clap-q.wav"); > SoundElement e1 = new SoundElement(s); > SoundElement e2 = new SoundElement(t); > SoundElement e3 = new SoundElement(u); > e1.playFromMeOn(); > e1 SoundElement with sound: Sound file: D:/cs1316/mediasources/shh-a- h.wav number of samples: (and next: null).

How did that happen? How were we able to get that string from typing in e1? By defining the method toString() on a class, we can define how it should appear. VERY useful in debugging!

Growing the list > e1.setNext(e2) > e1 SoundElement with sound: Sound file: D:/cs1316/mediasources/shh-a- h.wav number of samples: (and next: SoundElement with sound: Sound file: D:/cs1316/mediasources/croak-h.wav number of samples: 8808 (and next: null).). > e2.setNext(e3) > e1 SoundElement with sound: Sound file: D:/cs1316/mediasources/shh-a- h.wav number of samples: (and next: SoundElement with sound: Sound file: D:/cs1316/mediasources/croak-h.wav number of samples: 8808 (and next: SoundElement with sound: Sound file: D:/cs1316/mediasources/clap-q.wav number of samples: 4584 (and next: null).).). Where’s all that coming from?!? When we add in next to our toString(), it calls next’s toString()

e1.playFromMeOn() e1 mySound: shh-a- h.wav next: mySound: croak- h.wav next: mySound: clap- q.wav next: null e2e3 public void playFromMeOn(){ this.collect().play(); }

public Sound collect(){ if (this.getNext() == null) {return mySound;} e1 mySound: shh-a- h.wav next: mySound: croak- h.wav next: mySound: clap- q.wav next:null e2e3 this

else {return mySound.append(this.getNext().collect());} e1 mySound: shh-a- h.wav next: mySound: croak- h.wav next: mySound: clap- q.wav next:null e2e3 this We “freeze” this call to e1.collect()

public Sound collect(){ if (this.getNext() == null) {return mySound;} mySound: croak- h.wav next: mySound: clap- q.wav next:null e2e3 this Where’s e1? What e1? From e2’s view, there is no e1

else {return mySound.append(this.getNext().collect());} mySound: croak- h.wav next: mySound: clap- q.wav next:null e2e3 this We “freeze” this call to e2.collect()

public Sound collect(){ if (this.getNext() == null) {return mySound;} mySound: clap- q.wav next:null e3 this We return clap-q.wav

else {return mySound.append(this.getNext().collect());} mySound: croak- h.wav next: mySound: clap- q.wav next:null e2e3 this Back in e2.collect(), We return croak- h.wav appended with clap-q.wav

else {return mySound.append(this.getNext().collect());} e1 mySound: shh-a- h.wav next: mySound: croak- h.wav next: mySound: clap- q.wav next:null e2e3 this Back in e2.collect(), We return shh-a- h.wav appended with croak-h.wav appended with clap-q.wav

e1.playFromMeOn() e1 mySound: shh-a- h.wav next: mySound: croak- h.wav next: mySound: clap- q.wav next: null e2e3 public void playFromMeOn(){ this.collect().play(); } Now we play shh- a-h.wav appended with croak-h.wav appended with clap-q.wav

Testing Sound Lists public class SoundListTest { SoundElement root; public SoundElement root(){ return root;} public void setUp(){ FileChooser.setMediaPath("D:/cs1316/MediaSources/"); Sound s = new Sound(FileChooser.getMediaPath( "scratch- h.wav")); root = new SoundElement(s); s = new Sound(FileChooser.getMediaPath( "gonga-2.wav")); SoundElement one = new SoundElement(s); root.repeatNext(one,10); s = new Sound(FileChooser.getMediaPath( "scritch-q.wav")); SoundElement two = new SoundElement(s); root.weave(two,3,3); s = new Sound(FileChooser.getMediaPath( "clap-q.wav")); SoundElement three = new SoundElement(s); root.weave(three,5,2); root.playFromMeOn(); }

Repeating /** * Repeat the input phrase for the number of times specified. * It always appends to the current node, NOT insert. nextOne node to be copied in to list count number of times to copy it in. */ public void repeatNext(SoundElement nextOne,int count) { SoundElement current = this; // Start from here SoundElement 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 } Shockingly similar to what we saw before!

Copying a node (for repeat and weave) /** * copyNode returns a copy of this element another element with the same sound */ public SoundElement copyNode() { Sound copySound = new Sound(mySound); SoundElement returnMe = new SoundElement(copySound); return returnMe; }

Weaving /** * Weave the input sound count times every skipAmount elements nextOne SoundElement to be copied into the list count how many times to copy skipAmount how many nodes to skip per weave */ public void weave(SoundElement nextOne, int count, int skipAmount) { SoundElement current = this; // Start from here SoundElement copy; // Where we keep the one to be weaved in SoundElement 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.getNext() != null) && (skipped < skipAmount)) { current = current.getNext(); skipped++; }; oldNext = current.getNext(); // Save its next current.insertAfter(copy); // Insert the copy after this one current = oldNext; // Continue on with the rest if (current == null) // Did we actually get to the end early? break; // Leave the loop } Again, shockingly similar to what we saw before! Should we break before the last insert (when we get to the end) or after?

What? You don’t like gongs? So, what happens when you have umpteen copies of gong, but you don’t like gongs? Yes, we can remake the list, but what if you couldn’t?

Degong-ing the SoundListTest public void degong(){ Sound gong = new Sound(FileChooser.getMediaPath( "gonga- 2.wav")); Sound snap = new Sound(FileChooser.getMediaPath( "snap- tenth.wav")); root.replace(gong,snap); }

Replacing one sound by another in the list (recursively) /** * Replace the one sound with the other sound * in all the elements from me on. * Decide two sounds are equal if come from same filename oldSound sound to be replaced newSOund sound to put in its place **/ public void replace(Sound oldSound, Sound newSound){ if (mySound.getFileName() != null) {if (mySound.getFileName().equals(oldSound.getFileName())) {mySound = newSound;}} if (next != null) {next.replace(oldSound,newSound);} }

Imagine: e1.replace(croak,clap) e1 mySound: shh-a- h.wav next: mySound: croak- h.wav next: mySound: clap- q.wav next: null e2e3

public void replace(Sound oldSound, Sound newSound){ if (mySound.getFileName() != null) {if (mySound.getFileName().equals(oldSound.getFileName())) {mySound = newSound;}} e1 mySound: shh-a- h.wav next: mySound: croak- h.wav next: mySound: clap- q.wav next: null e2e3 this

if (next != null) {next.replace(oldSound,newSound);} e1 mySound: shh-a- h.wav next: mySound: croak- h.wav next: mySound: clap- q.wav next: null e2e3 this

public void replace(Sound oldSound, Sound newSound){ if (mySound.getFileName() != null) {if (mySound.getFileName().equals(oldSound.getFileName())) {mySound = newSound;}} mySound: croak- h.wav clap- q.wav next: mySound: clap- q.wav next: null e2e3 this This is a match!

if (next != null) {next.replace(oldSound,newSound);} mySound: croak- h.wav clap- q.wav next: mySound: clap- q.wav next: null e2e3 this

public void replace(Sound oldSound, Sound newSound){ if (mySound.getFileName() != null) {if (mySound.getFileName().equals(oldSound.getFileName())) {mySound = newSound;}} mySound: clap- q.wav next: null e3 this

if (next != null) {next.replace(oldSound,newSound);} mySound: clap- q.wav next: null e3 this And we’re done! We do return all the way back up again, but there’s nothing else to be done.

Tracing what’s happening Use “Debug Mode” from Debugger menu Type in variable names to trace. Set Breakpoints (Control-B) on the lines you want to stop in. Step over—skips to the next line Step in—skips in to that method call Resume—go to the next breakpoint

Version 2: When lists aren’t good enough Do we think about music as a linear list of sounds? That is how we experience it. Or do we tend to cluster the sounds? Verse, chorus, verse? This motif, that motif, this motif? And what about this cool idea of embedding operations into the data structure?

Creating a tree of sounds SoundBranch ScaleBranchSoundBranch childrennext Clap,snap,snap Rest, snap, clap Clink,clave, gong Rest, chirp, clap Clap, rest, snap Rest, snap, rest

Creating and playing our tree of sounds Welcome to DrJava. > SoundTreeExample tree = new SoundTreeExample(); > tree.play() > tree.playScaled(2.0); > tree.play();

SoundTreeExample public class SoundTreeExample { ScaleBranch scaledBranch; // Needed between methods SoundBranch root; public SoundBranch getRoot() {return root;} /** * Constructor to set up the example */ public SoundTreeExample() { setUp(); }

Setting up the basic sounds public void setUp() { Sound clap = new Sound(FileChooser.getMediaPath("clap-q.wav")); Sound chirp = new Sound(FileChooser.getMediaPath("chirp-2.wav")); Sound rest = new Sound(FileChooser.getMediaPath("rest-1.wav")); Sound snap = new Sound(FileChooser.getMediaPath("snap- tenth.wav")); Sound clink = new Sound(FileChooser.getMediaPath("clink- tenth.wav")); Sound clave = new Sound(FileChooser.getMediaPath("clave- twentieth.wav")); Sound gong = new Sound(FileChooser.getMediaPath("gongb-2.wav"));

Build the root and first branch root = new SoundBranch(); SoundNode sn; SoundBranch branch1 = new SoundBranch(); sn = new SoundNode(clap.append(rest).append(snap)); branch1.addChild(sn); sn = new SoundNode(rest.append(snap).append(rest)); branch1.addChild(sn);

A ScaleBranch and last Branch scaledBranch = new ScaleBranch(1.0); sn = new SoundNode(clink.append(clave).append(gong)); scaledBranch.addChild(sn); sn = new SoundNode(rest.append(chirp).append(clap)); scaledBranch.addChild(sn); SoundBranch branch2 = new SoundBranch(); sn = new SoundNode(clap.append(snap).append(snap)); branch2.addChild(sn); sn = new SoundNode(rest.append(snap).append(clap)); branch2.addChild(sn);

Building the whole tree root.addChild(branch1); root.addChild(scaledBranch); root.addChild(branch2); }

Playing the tree, and changing the scale public void play(){ root.playFromMeOn(); } public void playScaled(double factor){ scaledBranch.setFactor(factor); root.playFromMeOn(); }

Method to test the whole tree /** * Method to test the tree */ public static void main(String[] args) { SoundTreeExample tree = new SoundTreeExample(); tree.getRoot().collect().play(); }

What a tree “looks like” (printed, e.g., toString()) > tree > tree.root() SoundBranch (with child: SoundBranch (with child: SoundNode (with sound: Sound number of samples: and next: SoundNode (with sound: Sound number of samples: and next: null and next: ScaleBranch (1.0) SoundBranch (with child: SoundNode (with sound: Sound number of samples: and next: SoundNode (with sound: Sound number of samples: and next: null and next: SoundBranch (with child: SoundNode (with sound: Sound number of samples: 8452 and next: SoundNode (with sound: Sound number of samples: and next: null and next: No next))) and next: No next) Obviously, this doesn’t help us much, but the root() does

Going deeper: How’d we do that? How we build a tree of sounds is very much like a tree of pictures. Set up a general node abstract class that all other nodes inherit from. Create a leaf node that will store our data of interest (sounds). Create a branch node that will store collections of leaves and references to other branches. Create (as we wish) branch nodes with operations that do things to the children of this branch.

Our SoundTree Class Structure abstract CollectableNode Knows next Knows How to do basic list operations, and defines abstract sound operations (can collect() its sound(s)) SoundBranch Knows children Knows How add children, and collect all the sounds from its children and next SoundNode Knows mySound Knows How collect itself and its next The subclasses extend CollectableNode

Our SoundTree Class Structure (a little further) abstract CollectableNode Knows next Knows How to do basic list operations, and collect() SoundBranch Knows children Knows How add children, and collect() SoundNode Knows mySound Knows How collect() ScaleBranch Knows a scaling factor Knows How to access scaling factor, and to collect from children then scale the resultant sound

CollectableNode /** * Stuff that all nodes and branches in the * sound tree know. **/ abstract public class CollectableNode { /** * The next branch/node/whatever to process **/ public CollectableNode next; /** * Constructor for CollectableNode just sets * next to null **/ public CollectableNode(){ next = null; }

CollectableNode’s know about linked lists /** * Methods to set and get next elements nextOne next element in list **/ public void setNext(CollectableNode nextOne){ this.next = nextOne; } public CollectableNode getNext(){ return this.next; } The rest of it is there, too: add(), last(), insertAfter(), remove()… But you’ve seen that all before.

CollectableNodes know only a little about sound /** * Play the list of sound elements * after me **/ public void playFromMeOn(){ this.collect().play(); } /** * Collect all the sounds from me on **/ abstract public Sound collect(); /** * Collect all the sounds from me on, * but if there's processing, do it after. **/ abstract public Sound collectAfter();

SoundNodes know sounds /* * SoundNode is a class representing a drawn picture * node in a scene tree. **/ public class SoundNode extends CollectableNode { /** * The sound I'm associated with **/ Sound mySound; /* * Make me with this sound sound the Sound I'm associated with **/ public SoundNode(Sound sound){ super(); // Call superclass constructor mySound = sound; }

What happens if you print a SoundNode /** * Method to return a string with informaiton * about this node */ public String toString() { return "SoundNode (with sound: "+mySound+" and next: "+next; }

How SoundNodes collect their sounds /** * Collect all the sounds from me on, * recursively. **/ public Sound collect(){ if (this.getNext() == null) {return mySound;} else {return mySound.append(this.getNext().collect());} } “Hang on, how does that work?!?” Wait until we see the rest, then we’ll trace it.

SoundBranches know about children public class SoundBranch extends CollectableNode { /* * A list of children to draw */ public CollectableNode children; /* * Construct a branch with children and * next as null **/ public SoundBranch(){ super(); // Call superclass constructor children = null; } Notice: Children could be SoundNodes or other branches— any kind of CollectableNode

SoundBranches know about children (contd.) /** * Method to add nodes to children **/ public void addChild(CollectableNode child){ if (children != null) {children.add(child);} else {children = child;} }

How SoundBranches print /** * Method to return a string with informaiton * about this branch */ public String toString() { String childString = "No children", nextString="No next"; if (children != null) {childString = children.toString();} if (next != null) {nextString = next.toString();} return "SoundBranch (with child: "+childString+" and next: "+ nextString+")"; }

Collecting sounds from SoundBranches /** * Collect all the sound from our children, * then collect from next. pen Turtle to draw with **/ public Sound collect(){ Sound childSound; if (children != null) {childSound = children.collect();} else {childSound = new Sound(1);} // Collect from my next if (this.getNext() != null) {childSound=childSound.append(this.getNext().collect());} return childSound; } This collects from the children (or creates a default sound), then from the next’s, and returns them all together

ScaleBranches know about scaling (and a factor) public class ScaleBranch extends SoundBranch { /** Amount to scale **/ double factor; /** * Construct a branch with this factor **/ public ScaleBranch(double nufactor){ super(); // Call superclass constructor this.factor = nufactor; } /** Accessors **/ public double getFactor() {return this.factor;} public void setFactor(double nufactor) {this.factor = nufactor;}

Scaling the children in a ScaleBranch /** * Collect all the sound from our children, * then collect from next. Scale the children pen Turtle to draw with **/ public Sound collect(){ Sound childSound; if (children != null) {childSound = children.collect().scale(factor);} else {childSound = new Sound(1);} // Collect from my next if (this.getNext() != null) {Sound nextSound=this.getNext().collect(); childSound = childSound.append(nextSound);} return childSound; }

Printing a ScaleBranch /** * Method to return a string with informaiton * about this branch */ public String toString() { return "ScaleBranch ("+factor+") "+super.toString(); }

Walking the data structure: At the root > tree.getRoot() SoundBranch (with child: SoundBranch (with child: SoundNode (with sound: Sound number of samples: and next: SoundNode (with sound: Sound number of samples: and next: null and next: ScaleBranch (1.0) SoundBranch (with child: SoundNode (with sound: Sound number of samples: and next: SoundNode (with sound: Sound number of samples: and next: null and next: SoundBranch (with child: SoundNode (with sound: Sound number of samples: 8452 and next: SoundNode (with sound: Sound number of samples: and next: null and next: No next))) and next: No next)

Down the children’s next path > tree.getRoot().children SoundBranch (with child: SoundNode (with sound: Sound number of samples: and next: SoundNode (with sound: Sound number of samples: and next: null and next: ScaleBranch (1.0) SoundBranch (with child: SoundNode (with sound: Sound number of samples: and next: SoundNode (with sound: Sound number of samples: and next: null and next: SoundBranch (with child: SoundNode (with sound: Sound number of samples: 8452 and next: SoundNode (with sound: Sound number of samples: and next: null and next: No next))) > tree.getRoot().children.getNext() ScaleBranch (1.0) SoundBranch (with child: SoundNode (with sound: Sound number of samples: and next: SoundNode (with sound: Sound number of samples: and next: null and next: SoundBranch (with child: SoundNode (with sound: Sound number of samples: 8452 and next: SoundNode (with sound: Sound number of samples: and next: null and next: No next)) > tree.getRoot().children.getNext().getNext() SoundBranch (with child: SoundNode (with sound: Sound number of samples: 8452 and next: SoundNode (with sound: Sound number of samples: and next: null and next: No next) > tree.getRoot().children.getNext().getNext().getNext() null

Tracing a tree traversal Welcome to DrJava. > SoundTreeExample tree = new SoundTreeExample(); > tree.getRoot().collect().play() This code: (1)Creates a tree object, and fills it up. (2)Then gets the root, collects the sound from the whole tree (traverses the tree), then plays the collected sound. This code: (1)Creates a tree object, and fills it up. (2)Then gets the root, collects the sound from the whole tree (traverses the tree), then plays the collected sound.

Tracing the traversal: tree.getRoot().collect().play SoundBranch ScaleBranchSoundBranch childrennext Clap,snap,snap Rest, snap, clap Clink,clave, gong Rest, chirp, clap Clap, rest, snap Rest, snap, rest Tree.getRoot()

public Sound collect(){ Sound childSound; if (children != null) {childSound = children.collect();} else {childSound = new Sound(1);} SoundBranch ScaleBranchSoundBranch Clap,snap,snap Rest, snap, clap Clink,clave, gong Rest, chirp, clap Clap, rest, snap Rest, snap, rest this Yes, the root has children, so we ask the first of our children to collect()

public Sound collect(){ Sound childSound; if (children != null) {childSound = children.collect();} else {childSound = new Sound(1);} SoundBranch ScaleBranchSoundBranch Clap,snap,snap Rest, snap, clap Clink,clave, gong Rest, chirp, clap Clap, rest, snap Rest, snap, rest this Yes, we’re in the same method, but now with a new this. It’s really a new call. The call to getRoot().collect() is now paused…

public Sound collect(){ if (this.getNext() == null) {return mySound;} else {return mySound.append(this.getNext().collect());} } SoundBranch ScaleBranchSoundBranch Clap,snap,snap Rest, snap, clap Clink,clave, gong Rest, chirp, clap Clap, rest, snap Rest, snap, rest this Now, we’re in SoundNode’s collect() The call to getRoot().children.collect() is now paused… We have a next, so we collect from there, too.

public Sound collect(){ if (this.getNext() == null) {return mySound;} else {return mySound.append(this.getNext().collect());} } SoundBranch ScaleBranchSoundBranch Clap,snap,snap Rest, snap, clap Clink,clave, gong Rest, chirp, clap Clap, rest, snap Rest, snap, rest this We’re in SoundNode’s collect(), with a new SoundNode The call to getRoot().children.children.collect( ) is now paused… We have no next here, so we return “Rest,snap, rest”

public Sound collect(){ if (this.getNext() == null) {return mySound;} else {return mySound.append(this.getNext().collect());} } SoundBranch ScaleBranchSoundBranch Clap,snap,snap Rest, snap, clap Clink,clave, gong Rest, chirp, clap Clap, rest, snap Rest, snap, rest this We’re back in SoundNode’s collect() We now return “Clap, rest, snap” appended with “Rest, snap, rest”

// Collect from my next if (this.getNext() != null) {Sound nextSound=this.getNext().collect(); childSound = childSound.append(nextSound);} return childSound; } SoundBranch ScaleBranchSoundBranch Clap,snap,snap Rest, snap, clap Clink,clave, gong Rest, chirp, clap Clap, rest, snap Rest, snap, rest this We’re now in the second half of SoundBranch’s collect() We have a next, so we move on to call collect() on that.

public Sound collect(){ Sound childSound; if (children != null) {childSound = children.collect().scale(factor);} else {childSound = new Sound(1);} SoundBranch ScaleBranchSoundBranch Clap,snap,snap Rest, snap, clap Clink,clave, gong Rest, chirp, clap Clap, rest, snap Rest, snap, rest this Now we’re in ScaleBranch’s collect() Let’s skip the path down the children. childSound will become “Clink,clave,gong,rest,chirp,clap” scaled by the factor

// Collect from my next if (this.getNext() != null) {Sound nextSound=this.getNext().collect(); childSound = childSound.append(nextSound);} return childSound; } SoundBranch ScaleBranchSoundBranch Clap,snap,snap Rest, snap, clap Clink,clave, gong Rest, chirp, clap Clap, rest, snap Rest, snap, rest this Second half of ScaleBranch— we get the sounds from next

public Sound collect(){ Sound childSound; if (children != null) {childSound = children.collect();} else {childSound = new Sound(1);} SoundBranch ScaleBranchSoundBranch Clap,snap,snap Rest, snap, clap Clink,clave, gong Rest, chirp, clap Clap, rest, snap Rest, snap, rest this We’re in the first half of SoundBranch childSound will become “Clap, snap, snap, rest, snap, clap”

// Collect from my next if (this.getNext() != null) {Sound nextSound=this.getNext().collect(); childSound = childSound.append(nextSound);} return childSound; } SoundBranch ScaleBranchSoundBranch Clap,snap,snap Rest, snap, clap Clink,clave, gong Rest, chirp, clap Clap, rest, snap Rest, snap, rest this We’re in the second half of SoundBranch There is no next so we return “Clap, snap, snap, rest, snap, clap”

// Collect from my next if (this.getNext() != null) {Sound nextSound=this.getNext().collect(); childSound = childSound.append(nextSound);} return childSound; } SoundBranch ScaleBranchSoundBranch Clap,snap,snap Rest, snap, clap Clink,clave, gong Rest, chirp, clap Clap, rest, snap Rest, snap, rest this Back to second half of ScaleBranch getNext().collect() returned “Clap, snap, snap, rest, snap, clap” This returns “Clink,clave,gong,rest,chirp,clap” scaled by the factor, plus “Clap, snap, snap, rest, snap, clap”

// Collect from my next if (this.getNext() != null) {Sound nextSound=this.getNext().collect(); childSound = childSound.append(nextSound);} return childSound; } SoundBranch ScaleBranchSoundBranch Clap,snap,snap Rest, snap, clap Clink,clave, gong Rest, chirp, clap Clap, rest, snap Rest, snap, rest this Back in the second half of SoundBranch’s collect() This returns “Clap,rest,snap,rest,snap,rest”, plus “Clink,clave,gong,rest,chirp,clap” scaled by the factor, plus “Clap, snap, snap, rest, snap, clap”

// Collect from my next if (this.getNext() != null) {Sound nextSound=this.getNext().collect(); childSound = childSound.append(nextSound);} return childSound; } SoundBranch ScaleBranchSoundBranch Clap,snap,snap Rest, snap, clap Clink,clave, gong Rest, chirp, clap Clap, rest, snap Rest, snap, rest this FINAL RETURN: “Clap,rest,snap,rest,snap,rest”, plus “Clink,clave,gong,rest,chirp,clap” scaled by the factor, plus “Clap, snap, snap, rest, snap, clap” Now we’re back to the second half of this. There is no next

That was one kind of tree traversal There are other ways of thinking about traversing a tree! Go down the next before going to the children? Maybe do the scaling to the next instead of to the children? Processing first is called pre-order traversal. Processing last (doing the next) is called post- order traversal.

SoundBranch’s collectAfter() /** * Collect all the sound from our children, * then collect from next. * If there's processing, do to Next, not to Children pen Turtle to draw with **/ public Sound collectAfter(){ Sound childSound; if (children != null) {childSound = children.collectAfter();} else {childSound = new Sound(1);} // Collect from my next if (this.getNext() != null) {childSound=childSound.append(this.getNext().collectAfter());} return childSound; }

ScaleBranch’s collectAfter() /** * Collect all the sound from our children, * then collect from next. * Scale the next list, not the children pen Turtle to draw with **/ public Sound collectAfter(){ Sound childSound; if (children != null) {childSound = children.collect();} else {childSound = new Sound(1);} // Collect from my next if (this.getNext() != null) {Sound nextSound=this.getNext().collect(); childSound = childSound.append(nextSound.scale(factor));} return childSound; } Notice that we’re scaling what getNext().collect() returns, not what children.collect() returns

Trying this in SoundTreeList public void playAfter(){ root.collectAfter().play(); } > tree.playAfter()

Where else could we go with this? How about a kind of SoundBranch that knows how to repeat its children a certain number of time? How about a kind of SoundBranch that can mix its children with the next, with a certain percentage of each? Back with pictures: How about a kind of branch that can rotate its children? Or scale them, so you could zoom in or out, or grow/shrink the monster, across frames?

Arrays, and Lists, and Trees, oh my! Where we’re going next: We’ve explored lists and trees in a good deal of depth now! You obviously saw lots of similarities between these implementations. “Couldn’t we do this once, instead of duplicating all this code?!?” YES! Let us show you… “Man, what I really want to do is user interfaces with Java—real applications! What does this all have to do with that?!?” A user interface is a tree. Let us show you…