15 – Sequential Containers

Slides:



Advertisements
Similar presentations
Object Oriented Programming COP3330 / CGS5409.  C++ Automatics ◦ Copy constructor () ◦ Assignment operator =  Shallow copy vs. Deep copy  DMA Review.
Advertisements

Lecture 3 Feb 4 summary of last week’s topics and review questions (handout) Today’s goals: Chapter 1 overview (sections 1.4 to 1.6) c++ classes constructors,
An Array-Based Implementation of the ADT List public class ListArrayBased implements ListInterface { private static final int MAX_LIST = 50; private Object.
Pointers1 Pointers & Dynamic Arrays Allocating memory at run-time.
ARRAYS AND POINTERS Although pointer types are not integer types, some integer arithmetic operators can be applied to pointers. The affect of this arithmetic.
CHAPTER 4 Sequential Containers. Chapter Objectives  To become familiar with the Standard Template Library (STL) and template classes  To understand.
1 Chapter 3 Lists, Stacks, and Queues Abstract Data Types, Vectors Sections 3.1, 3.2, 3.3, 3.4 Abstract Data Types (ADT) Iterators Implementation of Vector.
Pointers Pointer a data type stores a memory address points to whatever the memory location contains A pointer is a variable that can store a memory address.
Pointers and Dynamic Memory Allocation Copyright Kip Irvine 2003, all rights reserved. Revised 10/28/2003.
CPSC 252 The Big Three Page 1 The “Big Three” Every class that has data members pointing to dynamically allocated memory must implement these three methods:
C arrays are limited: -they are represented by pointers (which may or may not be valid); -Indexes not checked (which means you can overrun your array);
CMSC 202, Version 3/02 1 Copy Constructors and Overloaded Assignment.
Data Structures Using C++1 Chapter 3 Pointers and Array-Based Lists.
1 Chapter 3 Lists, Stacks, and Queues Reading: Sections 3.1, 3.2, 3.3, 3.4 Abstract Data Types (ADT) Iterators Implementation of Vector.
Chapter 1 C++ Basics Review (Section 1.4). Classes Defines the organization of a data user-defined type. Members can be  Data  Functions/Methods Information.
Nirmalya Roy School of Electrical Engineering and Computer Science Washington State University Cpt S 223 – Advanced Data Structures C++ Review 2.
An Array-Based Implementation of the ADT List
Memory Management.
Motivation for Generic Programming in C++
Pointers and Dynamic Arrays
Chapter 13: Pointers, Classes, Virtual Functions, and Abstract Classes
Motivation and Overview
Vectors Holds a set of elements, like an array
Dr. Bernard Chen Ph.D. University of Central Arkansas
Operator overloading Conversions friend inline
Class: Special Topics Copy Constructors Static members Friends this
Dynamic Memory CSCE 121 J. Michael Moore.
Wednesday, February 28, 2018 Announcements… For Today…
Sequential Containers
Memberwise Assignment / Initialization
This pointer, Dynamic memory allocation, Constructors and Destructor
The dirty secrets of objects
Tuesday, February 20, 2018 Announcements… For Today… 4+ For Next Time…
Collections Intro What is the STL? Templates, collections, & iterators
Dynamically Allocated Memory
C++ Templates L03 - Iterator 10 – Iterator.
10 – Iterators C++ Templates 4.6 The Iterator pgs
Chapter 12: Pointers, Classes, Virtual Functions, and Abstract Classes
Array Lists Chapter 6 Section 6.1 to 6.3
2.5 Reasoning about Programs: Assertions and Loop Invariants
Abstract Data Types Iterators Vector ADT Sections 3.1, 3.2, 3.3, 3.4
Object Oriented Programming COP3330 / CGS5409
Classes with Dynamically Allocated Data
Lab 03 - Iterator.
understanding memory usage by a c++ program
Automatics, Copy Constructor, and Assignment Operator
Chapter 14: Pointers, Classes, Virtual Functions, and Abstract Classes
Chapter 3 Lists, Stacks, and Queues Abstract Data Types, Vectors
14 – Sequential Containers
Constructors and Other Tools
The Standard Template Library
Lists - I The List ADT.
Lists - I The List ADT.
Lecture 8 : Intro. to STL (Standard Template Library)
C++ Templates L03 - Iterator 10 – Iterator.
C++ Templates L03 - Iterator 10 – Iterator.
CSC212 Data Structure - Section RS
COP 3330 Object-oriented Programming in C++
Class: Special Topics 2 For classes using memory allocation
Collections Intro What is the STL? Templates, collections, & iterators
Chapter 3 Lists, Stacks, and Queues
CS 144 Advanced C++ Programming April 30 Class Meeting
8.3 Vectors Copyright © 2008 Pearson Addison-Wesley. All rights reserved. 1.
Copy Constructors and Overloaded Assignment
14 – Sequential Containers
Chapter 6 – Queues and Deques
15 – Sequential Containers
4.1 Introduction Arrays A few types Structures of related data items
SPL – PS3 C++ Classes.
Data Structures & Programming
Presentation transcript:

15 – Sequential Containers 4.3 Implementation of a Vector Class 4.4 The Copy Constructor, Assignment Operator, and Destructor 15 – Sequential Containers

Attendance Quiz #14 Sequential Containers

Tip #15: C++ Enum's Sequential Containers enum wind_directions_t { NO_WIND, NORTH_WIND, SOUTH_WIND, EAST_WIND, WEST_WIND }; wind_directions_t windDirection = NO_WIND; windDirection = 453; // doesn't work, we get a compiler error! The idea behind enumerated types is to create new data types that can take on only a restricted range of values. Moreover, these values are all expressed as constants rather than magic numbers--in fact, there should be no need to know the underlying values. The names of the constants should be sufficient for the purposes of comparing values.

Tip #15  We believe all that God has revealed, all that He does now reveal, and we believe that He will yet reveal many great and important things pertaining to the Kingdom of God. —Articles of Faith 1:9

4.3, pgs. 240-246 4.3 Implementation of a Vector Class The Default Constructor The swap Function The Subscripting Operator The push_back Function The insert Function The erase Function The reserve Function Performance of the KW::Vector 4.3, pgs. 240-246

Lab 03 Iterator MyArray Iterator Program Correctness #include <iostream> #include <string> using namespace std; #define MAX_ARRAY_SIZE 1000 template<typename T> class MyArray { private: size_t size_; T* array_; public: MyArray(const size_t array_size) : size_(0), array_(new T[array_size]) {} ~MyArray() { delete{} array_; } void push_back(T item) { array_[size_++] = item; } class Iterator size_t index_; Iterator(T* a) : array_(a), index_(0) {} T& operator[](int i) const { return array_[index_ + i]; } }; Iterator begin() const { return MyArray<T>::Iterator(array_); } int main(int argc, char * argv[]) { MyArray<int> numbers(MAX_ARRAY_SIZE); numbers.push_back(1); numbers.push_back(2); numbers.push_back(3); numbers.push_back(4); MyArray<int>::Iterator iter = numbers.begin(); for (int i = 0; i < 4; ++i) cout << iter[i] << ' '; cout << endl << endl; return 0; } MyArray Iterator

A Vector Class Object Sequential Containers template<typename T> class Vector { private: // Data fields /** The initial capacity of the array */ static const size_t INITIAL_CAPACITY = 10; /** The current capacity of the array */ size_t current_capacity; /** The current num_items of the array */ size_t num_items; /** The array to contain the data */ T* the_data; public: // Member Functions ... }; We will implement a simplified version of the vector class. The physical size of the array is indicated by the data field current_capacity (type size_t because it must be non-negative) The number of data items stored is indicated by the data field num_items

Vector<int> myVector(); Sequential Containers template<typename T> class Vector { private: // Data fields ... public: // Member Functions /** Construct an empty vector w/default capacity */ Vector<T>() : current_capacity(INITIAL_CAPACITY), the_data(new T[INITIAL_CAPACITY]), num_items(0) {} }; Initializer List. current_capacity = INITIAL_CAPACITY; the_data = new T[INITIAL_CAPACITY]; num_items = 0;

myVector[1]; Sequential Containers template<typename T> class Vector { private: // Data fields ... public: // Member Functions /** Subscripting operator */ T& operator[](size_t index) // Verify valid index. if (index < 0 || index >= num_items) throw std::out_of_range("index to operator[] is out of range"); } return the_data[index]; }; Because this is a template class, all of the code must be in the header or in a file included by the header . In the STL vector, only the at function validates the index.

myVector.push_back(10); Sequential Containers void push_back(const T& the_value) { // Make sure there is space for the new item. if (num_items == current_capacity) reserve(2 * current_capacity); /* Allocate an expanded array */ } // Insert the new item. the_data[num_items] = the_value; num_items++;

The reserve(size) Function Sequential Containers void reserve(size_t new_capacity) { if (new_capacity > current_capacity) if (new_capacity > 2 * current_capacity) current_capacity = new_capacity; else current_capacity *= 2; // Double the capacity. T* new_data = new T[current_capacity]; for (size_t i = 0; i < num_items; i++) new_data[i] = the_data[i]; // Copy the data over. delete[] the_data; // Free old memory. the_data = new_data; // Point to the new data. } Doubling spreads out the cost of copying.  Doubling an array of size n allows us to add n more items before another array copy.  Therefore, we can add n new items after we have copied over n existing items.  Although reallocation is O(n), we have to do a reallocation only after n items are added.  This averages out to 1 copy per add, so reallocation is effectively an O(1) operation.  push_back() is performed in amortized constant time.

myVector.insert(2, 20); void insert(size_t index, const T& the_value) Sequential Containers void insert(size_t index, const T& the_value) { if (index > num_items) // Validate index. throw std::out_of_range("index to insert is out of range"); } if (num_items == current_capacity) // Check for room reserve(2 * current_capacity); // Allocate an expanded array for (size_t i = num_items; i > index; i--) // Open a slot the_data[i] = the_data[i - 1]; the_data[index] = the_value; // Insert new item num_items++;

myVector.erase(3); void erase(size_t index) { // Validate index. Sequential Containers void erase(size_t index) { // Validate index. if (index > num_items) throw std::out_of_range ("index to erase is out of range"); } // Move items below the removed one up. for (size_t i = index + 1; i < num_items; i++) the_data[i - 1] = the_data[i]; num_items--;

Performance of Vector Sequential Containers The functions operator[] and at are each a few lines of code and contain no loops - execute in constant time, or O(1). If we insert into (or remove from) the middle of a KW::vector, then at most n items have to be shifted which is O(n). What if we have to reallocate before we can insert? Recall that we spread out the cost of copying so that effectively it is an O(1) operation, so the insertion is still O(n) . Even if we don’t spread out the cost of copying, the copy operation is still O(n), so the worst case merely doubles the cost.

4.4 The Copy Constructor, Assignment Operator, and Destructor Copying Objects and the Copy Constructor Shallow Copy versus Deep Copy Assignment Operator The Destructor 4.4, pgs. 247-251

Copying Objects / Copy Constructor Sequential Containers We want a copy of an object to be an independent copy, which means we should be able to modify one of the objects without affecting the other. Copying of primitive types is straightforward—the values are duplicated and placed in the target locations . For class types, copying is done by a class's default copy constructor: ClassName(const ClassName& old_obj); The copy constructor is invoked automatically: when an object is passed to a function by value. when an object is returned from a function. when an object is initialized with another object of the same class. when the compiler generates a temporary object.

Shallow Copy versus Deep Copy Sequential Containers Referring back to our vector class, a copy of the size_t fields current_capacity and num_items creates independent copies of the variables. However, copying the pointer value the_data does not create an independent copy. Copying a pointer this way is considered a shallow copy. 1 2 3 4 5 Vector<int> v1 num_items = 5 current_capacity = 8 int* the_data = _____ Vector<int> v2 = v1; Vector<int> v2 num_items = 5 current_capacity = 8 int* the_data = _____

Shallow Copy versus Deep Copy Sequential Containers Referring back to our vector class, a copy of the size_t fields current_capacity and num_items creates independent copies of the variables. However, copying the pointer value the_data does not create an independent copy. Copying a pointer this way is considered a shallow copy. 1 2 3 4 5 Vector<int> v2 num_items = 5 current_capacity = 8 int* the_data = _____ Vector<int> v2 = v1;

Deep Copy Sequential Containers We need to create an independent copy or a deep copy of the underlying array so that v1.the_data and v2.the_data point to different arrays, making vector v2 a deep copy of vector. /** Make a (deep) copy of a vector. @param other The vector to be copied */ Vector<T>(const Vector<T>& other) : current_capacity(other.capacity), num_items(other.num_items), the_data(new T[other.current_capacity]) { for (size_t i = 0; i < num_items; i++) the_data[i] = other.the_data[i]; } Vector<int> v1(); Vector<int> v2(v1); Vector<int> v3 = v2;

Assignment Operator Sequential Containers Assignment creates an independent copy of an object. Copy constructor is called when a new object is created from an existing object, as a copy of the existing object. Assignment operator is called when a already initialized object is assigned a new value from another existing object. MyClass t1, t2; MyClass t3 = t1; // use copy constructor t2 = t1; // use assignment operator. t2.operator=(t1); // (equivalent) Assignment of primitive types is straightforward—the value is duplicated and placed in the target location. Every class has a default assignment operator which makes a copy of each data field of the source and places it into the corresponding data field of the target . If a class type, that class’s assignment operator is used. The default assignment operator makes a shallow copy of an object, so we must override it if we want truly independent copies.

Assignment Operator Sequential Containers /** Assign the contents of one vector to another. @param other The vector to be assigned to this vector @return This vector with a copy of the other vector's contents */ Vector<T>& operator=(const vector<T>& other) { // Make a copy of the other vector. Vector<T> the_copy(other); // Swap contents of self with the copy. swap(the_copy); // Return -- upon return the copy will be destroyed. return *this; } Vector<int> v1(); Vector<int> v2(); v2 = v1; Upon return, the destructor for the temporary object is invoked automatically by the compiler, deleting the old data.

The Destructor Sequential Containers The purpose of the destructor is to undo what the constructor does. The constructor takes a block of uninitialized memory and sets it to a known state, thus creating an object. When the destructor is finished, the object is no longer considered to be in a valid state, and the memory it occupies can be reused for creating other objects. If this is not done, the program will have memory leaks. Each class has a default destructor, which effectively invokes the destructor for each data field. If the pointer references a dynamically allocated object (such as the array referenced by pointer the_data), the memory allocated to that object must be freed.

The swap Function Sequential Containers The swap function exchanges the value of this vector object with another vector object It uses the standard template function to swap the values /** Exchanges the contents of this vector with another. @param other The other vector */ void swap(Vector<T>& other) { std::swap(num_items, other.num_items); std::swap(current_capacity, other.current_capacity); std::swap(the_data, other.the_data); }