David Notkin Autumn 2009 CSE303 Lecture 26 If you think C++ is not overly complicated, just what is a protected abstract virtual base pure virtual private.

Slides:



Advertisements
Similar presentations
Copyright 2008 by Pearson Education Building Java Programs Chapter 8 Lecture 8-3: Encapsulation, toString reading: self-checks: #13-18,
Advertisements

David Notkin Autumn 2009 CSE303 Lecture 13 This space for rent.
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,
David Notkin Autumn 2009 CSE303 Lecture 25. The plan 11/30 C++ intro12/2 C++ intro12/4 12/712/912/11 Final prep, evaluations 12/15 Final CSE303 Au092.
תכנות מונחה עצמים Object Oriented Programming (OOP) אתגר מחזור ב ' Classes – - המשך.
1 Classes Overview l Classes as Types l Declaring Instance Variables l Implementing Methods l Constructors l Accessor and Mutator Methods.
Rossella Lau Lecture 8, DCO10105, Semester B, DCO10105 Object-Oriented Programming and Design  Lecture 8: Polymorphism & C++ pointer  Inheritance.
Lecture 5-6 OOP Overview. Outline A Review of C++ Classes (Lecture 5) OOP, ADTs and Classes Class Definition, Implementation and Use Constructors and.
1 CSE 303 Lecture 21 Classes and Objects in C++ slides created by Marty Stepp
Shallow Versus Deep Copy and Pointers Shallow copy: when two or more pointers of the same types point to the same memory – They point to the same data.
Copyright 2010 by Pearson Education Building Java Programs Chapter 8 Lecture 8-2: Object Behavior (Methods) and Constructors reading:
11 Values and References Chapter Objectives You will be able to: Describe and compare value types and reference types. Write programs that use variables.
1 CSE 303 Lecture 22 Advanced Classes and Objects in C++ slides created by Marty Stepp
1 CISC181 Introduction to Computer Science Dr. McCoy Lecture 19 Clicker Questions November 3, 2009.
CSE 333 – SECTION 4. Overview Pointers vs. references Const Classes, constructors, new, delete, etc. More operator overloading.
Pointer Data Type and Pointer Variables
Object Oriented Programming Concepts OOP – reasoning about a program as a set of objects rather than as a set of actions Object – a programming entity.
Copyright 2008 by Pearson Education Building Java Programs Chapter 8 Lecture 8-2: Constructors and Encapsulation reading: self-checks: #10-17.
CS212: Object Oriented Analysis and Design Lecture 6: Friends, Constructor and destructors.
 2006 Pearson Education, Inc. All rights reserved Classes: A Deeper Look, Part 2.
Copyright 2010 by Pearson Education Building Java Programs Chapter 8 Lecture 8-2: Object Behavior (Methods) and Constructors, Encapsulation, this reading:
1 Overloading Overloading allows a function or operator to have a different meaning depending on the type of objects it is used on. Examples: operator+
C++ Memory Overview 4 major memory segments Key differences from Java
Pointers and Dynamic Memory Allocation Copyright Kip Irvine 2003, all rights reserved. Revised 10/28/2003.
Object-Oriented Programming in C++
The Big Three Based on Weiss “Data Structures and algorithm Analysis CS240 Computer Science II.
Data Structures Using C++ 2E Chapter 3 Pointers. Data Structures Using C++ 2E2 Objectives Learn about the pointer data type and pointer variables Explore.
CS 376b Introduction to Computer Vision 01 / 23 / 2008 Instructor: Michael Eckmann.
Lecture 3 Classes, Structs, Enums Passing by reference and value Arrays.
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:
Object-Based Programming Mostly Review. Objects Review what is object? class? member variables? member functions? public members? private members? friend.
CMSC 202, Version 3/02 1 Copy Constructors and Overloaded Assignment.
CS 11 C++ track: lecture 1 Administrivia Need a CS cluster account sysadmin/account_request.cgi Need to know UNIX (Linux)
1 Chapter 1 C++ Basics Review Reading: Sections 1.4 and 1.5.
Object Management. Constructors –Compiler-generated –The Initializer List –Copy Constructors –Single-arg (conversion ctors) The Assignment Operator.
CS-1030 Dr. Mark L. Hornick 1 Basic C++ State the difference between a function/class declaration and a function/class definition. Explain the purpose.
Effective C++, 2nd Ed. By Scott Myers. Constructors, Destructors, and Assignment Operators 11.Define a copy constructor and an assignment operator for.
Copyright © 2009 Pearson Education, Inc. Publishing as Pearson Addison-Wesley Chapter 14: More About Classes.
1 Classes II Chapter 7 2 Introduction Continued study of –classes –data abstraction Prepare for operator overloading in next chapter Work with strings.
1 Mr. Muhammad Hanif Lecturer Information Technology MBBS Campus Dadu University of SIndh.
Engineering Classes. Objectives At the conclusion of this lesson, students should be able to: Explain why it is important to correctly manage dynamically.
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.
Java & C++ Comparisons How important are classes and objects?? What mechanisms exist for input and output?? Are references and pointers the same thing??
Nirmalya Roy School of Electrical Engineering and Computer Science Washington State University Cpt S 223 – Advanced Data Structures C++ Review 2.
Dynamic Memory Management & Static Class Members Lecture No 7 Object Oriented Programming COMSATS Institute of Information Technology.
Liang, Introduction to C++ Programming, (c) 2007 Pearson Education, Inc. All rights reserved X 1 Chapter 9 Introduction of Object Oriented Programming.
Copyright 2010 by Pearson Education Building Java Programs Chapter 8 Lecture 8-3: Constructors; Encapsulation reading: self-checks: #13-18,
Copyright 2010 by Pearson Education Building Java Programs Chapter 8 Lecture 8-2: Object Behavior (Methods) and Constructors, Encapsulation, this reading:
Memory Management.
Learning Objectives Pointers as dada members
Overview 4 major memory segments Key differences from Java stack
Friend Class Friend Class A friend class can access private and protected members of other class in which it is declared as friend. It is sometimes useful.
"A class is where we teach an object how to behave." --Rich Pattis
System Programming Practical Session 8 C++ classes.
Memberwise Assignment / Initialization
The dirty secrets of objects
Overview 4 major memory segments Key differences from Java stack
understanding memory usage by a c++ program
Building Java Programs
Building Java Programs
Building Java Programs
Building Java Programs
Building Java Programs
CSE 142 Lecture Notes Defining New Types of Objects, cont'd.
CS410 – Software Engineering Lecture #5: C++ Basics III
Lecture 8-2: Object Behavior (Methods) and Constructors
Building Java Programs
Copy Constructors and Overloaded Assignment
SPL – PS3 C++ Classes.
Building Java Programs
Presentation transcript:

David Notkin Autumn 2009 CSE303 Lecture 26 If you think C++ is not overly complicated, just what is a protected abstract virtual base pure virtual private destructor and when was the last time you needed one? — Tom Cargill If C++ has taught me one thing, it’s this: Just because the system is consistent doesn’t mean it’s not the work of Satan. — Andrew Plotkin

The plan 11/30 C++ intro12/2 C++ intro12/4 C++ intro 12/7 Social 12/9 Implications 12/11 Final prep, evaluations 12/15 Final CSE303 Au092

Constructing objects client code creating stack-allocated object: type name(parameters); Point p1(4, -2); creating heap allocated (pointer to) object: type* name = new type(parameters); Point* p2 = new Point(5, 17); in Java, all objects are allocated on the heap in Java, all variables of object types are references (pointers)

A client program #include #include "Point.h" using namespace std; int main() { Point p1(1, 2); Point p2(4, 6); cout << "p1 is: (" << p1.getX() << ", " << p1.getY() << ")" << endl; // p1 is: (1, 2) cout << "p2 is: (" << p2.getX() << ", " << p2.getY() << ")" << endl; // p2 is: (4, 6) cout << "dist : " << p1.distance(p2) << endl; return 0; // dist : 5 }

Client with pointers #include #include "Point.h" using namespace std; int main() { Point* p1 = new Point(1, 2); Point* p2 = new Point(4, 6); cout getX() << ", " getY() << ")" << endl; // p1 is: (1, 2) cout getX() << ", " getY() << ")" << endl; // p2 is: (4, 6) cout distance(*p2) << endl; delete p1; // dist : 5 delete p2; // free return 0; }

Stack vs. heap objects which is better, stack or pointers? –if it needs to live beyond function call (e.g. returning), use a pointer –if allocating a whole bunch of objects, use pointers "primitive semantics" can be used on objects –stack objects behave use primitive value semantics (like ints) new and delete replace malloc and free –new does all of the following: allocates memory for a new object calls the class's constructor, using the new object as this returns a pointer to the new object –must call delete on any object you create with new, else it leaks

Why doesn't this code change p1? int main() { Point p1(1, 2); cout << p1.getX() << "," << p1.getY() << endl; example(p1); cout << p1.getX() << "," << p1.getY() << endl; return 0; } void example(Point p) { p.setLocation(40, 75); cout << "ex:" << p.getX() << "," << p.getY() << endl; } // 1,2 // ex:40,75 // 1,2

Object copying a stack-allocated object is copied whenever you: –pass it as a parameter foo(p1); –return itreturn p; –assign one object to another p1 = p2; the above rules do not apply to pointers –object's state is still (shallowly) copied if you dereference/assign *ptr1 = *ptr2; You can control how objects are copied by redefining the = operator for your class (ugh)

Objects as parameters We generally don't pass objects as parameters like this: double Point::distance(Point p) { int dx = x - p.getX(); int dy = y - p.getY(); return sqrt(dx * dx + dy * dy); } on every call, the entire parameter object p will be copied this is slow and wastes time/memory it also would prevent us from writing a method that modifies p

References to objects Instead, we pass a reference or pointer to the object: double Point::distance(Point& p) { int dx = x - p.getX(); int dy = y - p.getY(); return sqrt(dx * dx + dy * dy); } now the parameter object p will be shared, not copied are there any potential problems with this code?

const object references If the method will not modify its parameter, make it const double Point::distance(const Point& p) { int dx = x - p.getX(); int dy = y - p.getY(); return sqrt(dx * dx + dy * dy); } the distance method is promising not to modify p –if it does, a compiler error occurs –clients can pass Points via references without fear that their state will be changed

const methods If the method will not modify the object itself, make the method const: double Point::distance(const Point& p) const { int dx = x - p.getX(); int dy = y - p.getY(); return sqrt(dx * dx + dy * dy); } a const after the parameter list signifies that the method will not modify the object upon which it is called (this) –helps clients know which methods aren't mutators and helps the compiler optimize method calls a const reference only allows const methods to be called on it

const and pointers const Point* p –p points to a Point that is const; cannot modify that Point's state –can reassign p to point to a different Point (as long as it is const) Point* const p –p is a constant pointer; cannot reassign p to point to a different object –can change the Point object's state by calling methods on it const Point* const p –p points to a Point that is const; cannot modify that Point's state –p is a constant pointer; cannot reassign p to point to a different object (This is not one of the more beloved features of C++.)

Pointer, reference, etc.? How do you decide whether to pass a pointer, reference, or object? Some principles: –Minimize the use of object pointers as parameters. (C++ introduced references for a reason. They are safer and saner.) –Minimize passing objects by value, because it is slow, it has to copy the entire object and put it onto the stack, etc. –In other words, pass objects as references as much as possible; but if you really want a copy, pass it as a normal object. –Objects as fields are usually pointers (why not references?). –If you are not going to modify an object, declare it as const. –If your method returns a pointer/object field that you don't want the client to modify, declare its return type as const.

Operator overloading operator overloading: Redefining the meaning of a C++ operator in particular contexts. –example: the string class overloads + to do concatenation –example: the stream classes overload > to do I/O it is legal to redefine almost all C++ operators –() [] ^ % ! | & > = == != and many others –intended for when that operator "makes sense" for your type example: a Matrix class's * operator would do matrix multiplication allows your classes to be "first class citizens" like primitives –cannot redefine operators between built-in types ( int + int ) a useful, but very easy to abuse, feature of C++

Overloading syntax public: // declare in.h returntype operator op(parameters); returntype classname::operator op(parameters) { statements; // define in.cpp } most overloaded operators are placed inside a class –example: overriding Point + Point some overloaded operators don't go inside your class –example: overriding int + Point

Overloaded comparison ops Override == to make objects comparable like Java's equals –comparison operators like == return type bool –by default == doesn’t work on objects (what about Point* ?) –if you override ==, you must also override != // Point.h bool Point::operator==(const Point& p); // Point.cpp bool Point::operator==(const Point& p) { return x == p.getX() && y == p.getY(); } Override < etc. to make comparable like Java's compareTo –even if you override < and ==, you must still manually override <=

Overriding << Override << to make your objects printable like Java's toString –<< goes outside your class (not a member) –<< takes a stream reference and your object –returns a reference to the same stream passed in // Point.cpp std::ostream& operator<<(std::ostream& out, const Point& p) { out << "(" << p.getX() << ", " << p.getY() << ")"; return out; }

Designing a class Suppose we want to design a class LineSegment, where each object represents a 2D line segment between two points. We should be able to: –create a segment between two pairs of coordinates, –ask a segment for its endpoint coordinates, –ask a segment for its length, –ask a segment for its slope, and –translate (shift) a line segment's position.

LineSegment.h #include "Point.h" class LineSegment { private: Point* p1; // endpoints of line Point* p2; public: LineSegment(int x1, int y1, int x2, int y2); double getX1() const; double getY1() const; double getX2() const; double getY2() const; double length() const; double slope() const; void translate(int dx, int dy); };

LineSegment.cpp #include "LineSegment.h" LineSegment::LineSegment(int x1, int y1, int x2, int y2) { p1 = new Point(x1, y1); p2 = new Point(x2, y2); } double LineSegment::length() const { return p1->distance(*p2); } double LineSegment::slope() const { int dy = p2->getY() - p1->getY(); int dx = p2->getX() - p1->getX(); return (double) dy / dx; } void LineSegment::translate(int dx, int dy) { p1->setLocation(p1->getX() + dx, p1->getY() + dy); p2->setLocation(p2->getX() + dx, p2->getY() + dy); }...

Problem: memory leaks if we create LineSegment objects, we'll leak memory: LineSegment* line = new LineSegment(1, 2, 5, 4);... delete line; the two Point objects p1 and p2 inside line are not freed –the delete operator is a "shallow" delete operation –it doesn't recursively delete/free pointers nested inside the object why not?

Destructors public: ~classname(); // declare in.h classname::~classname() { // define in.cpp statements; } destructor: Code that manages the deallocation of an object. –usually not needed if the object has no pointer fields –called by delete and when a stack object goes out of scope –the default destructor frees the object's memory, but no pointers Java has a very similar feature to destructors, called a finalizer

Destructor example // LineSegment.h class LineSegment { private: Point* p1; Point* p2; public: LineSegment(int x1, int y1, int x2, int y2); double getX1() const;... ~LineSegment(); }; // LineSegment.cpp LineSegment::~LineSegment() { delete p1; delete p2; }

Shallow copy bug A subtle problem occurs when we copy LineSegment objects: –LineSegment line1(0, 0, 10, 20); –LineSegment line2 = line1; –line2.translate(5, 3); –cout << line1.getX2() << endl; // 15 !!! When you declare one object using another, its state is copied –it is a shallow copy; any pointers in the second object will store the same address as in the first object (both point to same place) –if you change what's pointed to by one, it affects the other Even worse: the same p1, p2 above are freed twice!

Copy constructors copy constructor: Copies one object's state to another. –called when you assign one object to another at declaration LineSegment line2 = line1; –can be called explicitly (same behavior as above) LineSegment line2(line1); –called when an object is passed as a parameter foo(line1); // void foo(LineSegment l)... if your class doesn't have a copy constructor, –the default one just copies all members of the object –if any members are objects, it calls their copy constructors (but not pointers)

Copy constructor example // LineSegment.h class LineSegment { private: Point* p1; Point* p2; public: LineSegment(int x1, int y1, int x2, int y2); LineSegment(const LineSegment& line); … // LineSegment.cpp LineSegment::LineSegment(const LineSegment& line) { p1 = new Point(line.getX1(), line.getY1()); p2 = new Point(line.getX2(), line.getY2()); }

Assignment bug Another problem with assigning LineSegment objects: LineSegment line1(0, 0, 10, 20); LineSegment line2(9, 9, 50, 80);... line2 = line1; line2.translate(5, 3); cout << line1.getX2() << endl; // 15 again !!! When you assign one object to another, its state is copied –it is a shallow copy; if you change one, it affects the other –assignment with = does NOT call the copy constructor We wish the = operator behaved differently...

Overloading = // LineSegment.h class LineSegment { private: Point* p1; Point* p2; void init(int x1, int y1, int x2, int y2); public: LineSegment(int x1, int y1, int x2, int y2); LineSegment(const LineSegment& line);... const LineSegment& operator=(const LineSegment& rhs);...

Overloading =, cont'd. // LineSegment.cpp void LineSegment::init(int x1, int y1, int x2, int y2) { p1 = new Point(x1, y1); // common helper init function p2 = new Point(x2, y2); } LineSegment::LineSegment(int x1, int y1, int x2, int y2) { init(x1, y1, x2, y2); } LineSegment::LineSegment(const LineSegment& line) { init(line.getX1(), line.getY1(), line.getX2(), line.getY2()); } const LineSegment& LineSegment::operator=(const LineSegment& rhs) { init(rhs.getX1(), rhs.getY1(), rhs.getX2(), rhs.getY2()); return *this; // always return *this from = }

An extremely subtle bug if your object was storing pointers to two Points p1, p2 but is then assigned to have new state using =, the old pointers will leak! Instead const LineSegment& LineSegment::operator=(const LineSegment& rhs) { delete p1; delete p2; init(rhs.getX1(), rhs.getY1(), rhs.getX2(), rhs.getY2()); return *this; // always return *this from = }

Another subtle bug if an object is assigned to itself, our = operator will crash! LineSegment line1(10, 20, 30, 40);... line1 = line1; Instead const LineSegment& LineSegment::operator=(const LineSegment& rhs) { if (this != &rhs) { delete p1; delete p2; init(rhs.getX1(), rhs.getY1(), rhs.getX2(), rhs.getY2()); } return *this; // always return *this from = }

Recap When writing a class with pointers as fields, you must define: –a destructor –a copy constructor –an overloaded operator = Point p1; calls 0-argument constructor Point p2(17, 5); calls 2-argument constructor Point p3 = p2; calls copy constructor Point p4(p3); calls copy constructor foo(p4); calls copy constructor p4 = p1; calls operator =

Questions? CSE303 Au0934