Project Home |
Image by
Steve Buissinne
from Pixabay
Sometimes we want to group different kinds of data together. For example, let's think about a bank account. What kind of information does your bank account store about you? Most important are your bank account number (an integer) and your account balance (a float). It also stores your name and address (both strings). Often in programming, we want to group related data like this together in a single construct, and we can do that with objects.
A class defines an object type, the variables inside the object (such as bank account number and balance), and functions you can perform on the object (such as deposit() and withdraw()). So we begin by defining a class.
The first step is easy enough: type the keyword class, and then choose a class name beginning with an uppercase letter (this is not required, but is the convention followed by programmers). For our bank account example, we'll name the class BankAccount:
class BankAccount { };
You'll notice the class definition will be encompassed by curly braces and has a trailing semi-colon. Now we need to add some variables and functions.
Variables inside a class are called member variables, so the member variables for this class are the account number, account balance, your name, and your address. Adding these to our skeleton definition we get the following:
class BankAccount { public: int accountNumber; float accountBalance; string name; string address; };
The public keyword indicates that the member variables (and functions—we'll add those later) that follow are available to the rest of the program to use. You can also declare a class's variables and functions to be private, but we'll cover that later. Notice, too, that the variables are declared in the same way as before, indicating the type and giving it a descriptive name.
In order to create a BankAccount object, we first need to define a constructor. The constructor is a piece of code that executes when a new object of this class is created. Inside the constructor, we want to set initial values for any member variables we have, otherwise our program may not work as we intend. A constructor is defined in the the same way as a function; the only differences are: 1) the name of the constructor must match the name of the class and 2) the constructor has no return type.
For a brand new BankAccount, when we don't yet know anything about the data that will be put into it, default values of zeroes and empty strings will be fine; we can always set them to something else later. Thus, our BankAccount constructor is defined as follows:
class BankAccount { public: int accountNumber; float accountBalance; string name; string address; BankAccount() { accountNumber = 0; accountBalance = 0.0; name = ""; address = ""; } };
We have now defined a basic data structure that holds bank account information, and so we can now write a program that creates a BankAccount object and sets the values of its fields. Notice the class must be declared before the main function where we create our object.
#include <iostream> using namespace std; class BankAccount { public: int accountNumber; float accountBalance; string name; string address; BankAccount() { accountNumber = 0; accountBalance = 0.0; name = ""; address = ""; } }; int main() { BankAccount alisonBankAccount; //make a bankAccount for Alison alisonBankAccount.accountNumber = 14538; //assign the account a number alisonBankAccount.accountBalance = 1000000.01; //give Alison some money alisonBankAccount.name = "Alison Norman"; //assign name and address alisonBankAccount.address = "1600 Pennsylvania Avenue NW Washington, DC 20500"; }
Look at the first line in main. We've created a BankAccount object called alisonBankAccount, and it has been initialized using the default constructor.
We use dot notation to access and set the public variables of an object, as seen in the remaining lines in main. In dot notation, we access member variables by first writing the name of the object (e.g., alisonBankAccount), then a dot, and then the name of the field we want (e.g., accountNumber). From here, we see that we can create multiple different BankAccounts and populate them all with different balances and information. For example, let's say that in addition to Alison's bank account, we'll make an account for Angie. Here's the same code as above, but now we've added a second instance of the BankAccount class:
//make two different bank accounts BankAccount alisonBankAccount; BankAccount angieBankAccount; //assign two different account numbers alisonBankAccount.accountNumber = 14538; //assign the account a number angieBankAccount.accountNumber = 54324; //assign different account balances alisonBankAccount.accountBalance = 1000000.01; angieBankAccount.accountBalance = 2.52; //give Angie some money to get out of debt //i.e. add 300.95 to the current value of accountBalance angieBankAccount.accountBalance += 300.95; cout << "Angie has $" << angieBankAccount.accountBalance << ". Yay Angie!\n";
We can give Alison and Angie's BankAccount objects completely different information, and we can also modify the values that the fields hold. Notice how Angie made a timely deposit with the line:
angieBankAccount.accountBalance += 300.95;The += operator means that you increase the value of the variable to the left of the += by the value on the right side of the += and save it as the same variable. In other words, the following two lines are equivalent:
angieBankAccount.accountBalance += 300.95; angieBankAccount.accountBalance = angieBankAccount.accountBalance + 300.95;These objects do a great job of cleaning up our code—consider having to keep track of Alison and Angie's bank account info without them and what happens when we want to add a third bank account—all those variables add up!
Alison says, "Thank you!".
Before moving on, don't forget to turn your cups to
red so that a member of the camp staff can check your code, and don't forget to
switch drivers!
In the program above, we declared the member variables as public, but, as mentioned before, another option is private. Private member variables can only be accessed by other objects of their type—in our case, our BankAccountobjects. There are a number of reasons to declare member variables as private, one of which is to maintain quality control on their values—so that account balances can't be set to negative, for instance. In general, you will want to declare variables and functions necessary for the inner workings of the class private and make public the variables and functions necessary for object to be useful to the rest of the program.
Consider the usefulness of using public functions to modify private data, as opposed to leaving the data public. What if we wanted to apply the same set of checks to make sure the account balance is always updated properly—for example, checking for overdrafts or account deposit limits. With our data safely hidden behind the private wall and the use of functions, we can say with confidence our bank account balances are correct.
Returning to our BankAccount class, let's go ahead and declare our member variables as private, as shown below.
class BankAccount { public: BankAccount() { accountNumber = 0; accountBalance = 0.0; name = ""; address = ""; } private: int accountNumber; float accountBalance; string name; string address; };
Now that these variables are private, our object is actually fairly useless right now. We left the constructor public, so we can create new objects, but all our data is inaccessible from outside the class because it's declared private. To make our object useful, we'll introduce member functions, which define actions that can be performed on or with the member variables.
Consider what behavior a BankAccount should exhibit. What sorts of actions are necessary? [Think a minute and then read on!]
People need to be able to deposit to and withdraw from their accounts, as well as see their current account balance. The latter we can solve with a simple getAccountBalance function that just returns the value stored in accountBalance. The deposit and withdraw operations are a little more complex; both modify the account balance by a given amount. What might the deposit function look like? Would it take any arguments? What kind? Does it need to return anything? [Again, think a minute before continuing!]
deposit() would need a float as an argument, which is the amount being deposited, and then would need to update the accountBalance variable appropriately. It doesn't need to return anything. Since this is a member function, we write its instructions inside the class definition, like this:
#include <iostream> using namespace std; class BankAccount { public: float getAccountBalance() { return accountBalance; } void deposit(float amount) { accountBalance += amount; } BankAccount() { accountNumber = 0; accountBalance = 0.0; name = ""; address = ""; } private: int accountNumber; float accountBalance; string name; string address; };Everything under the public keyword and above the private keyword is our class's public members, and everything below the private keyword is the class's private members. We want both functions to be public; they are the rest of the program's interface with our BankAccount data.
Now when we want to deposit money in main, we won't need to type:
angieBankAccount.accountBalance += 300.95;Instead, we'll be able to say:
angieBankAccount.deposit(300.95);Thus, our modified main looks like this:
int main(){ BankAccount alisonBankAccount; BankAccount angieBankAccount; angieBankAccount.deposit(600.95); }There are a few new things to note.
Add another bank account to the example above. Deposit $500 dollars into Alison's account, then withdraw $400 from Angie's account and place it in the account you created. Then print out the balances from all three accounts. (You'll need to write the withdraw function. How is it similar to the deposit function? How is it different? What happens if you attempt to withdraw an amount greater than your balance?)
Good job!
Don't forget to turn your cups to red so that a member of the camp staff can
check your code, and don't forget to switch drivers!
Now, let's move on to decisions.