Chapter Two: Objects and Classes PDF

Summary

This document introduces object-oriented programming (OOP) concepts using Java examples. It explains objects, classes, and constructors in OOP programming. It also demonstrates how to create and use object references in Java.

Full Transcript

**Chapter Two** **Objects and Classes** **3.1. Defining Classes for Objects** *Object-oriented programming (OOP)* involves programming using objects. An object represents an entity in the real world that can be distinctly identified. For example, a student, a desk, a circle, a button, and even a...

**Chapter Two** **Objects and Classes** **3.1. Defining Classes for Objects** *Object-oriented programming (OOP)* involves programming using objects. An object represents an entity in the real world that can be distinctly identified. For example, a student, a desk, a circle, a button, and even a loan can all be viewed as objects. An object has a unique identity, state, and behaviors. - The state of an object is represented by data fields (also known as properties) with their current values. - The behavior of an object is defined by a set of methods. Invoking a method on an object means that you ask the object to perform a task. A circle object, for example, has a data field, *radius*, which is the property that characterizes a circle. One behavior of a circle is that its area can be computed using the method *getArea().* Objects of the same type are defined using a common class. A class is a template or blueprint that defines what an object\'s data and methods will be. An object is an instance of a class. You can create many instances of a class. Creating an instance is referred to as *instantiation*. The terms object and *instance* are often interchangeable. The relationship between classes and objects is analogous to the relationship between apple pie recipes and apple pies. You can make as many apple pies as you want from a single recipe. Figure 3.1. shows a class named *Circle* and its three objects. **Figure 3.1. A class is a template for creating objects.** A Java *class* uses variables to define data fields and methods to define behaviors. Additionally, a class provides methods of a special type, known as *constructors*, which are invoked when a new object is created. A constructor is a special kind of method. A constructor can perform any action, but constructors are designed to perform initializing actions, such as initializing the data fields of objects. Figure3.2. shows an example of the class for *Circle* objects. ![](media/image2.png) **Figure 3.2. A class is a construct that defines objects of the same types** The *Circle* class does not have a *main* method and therefore cannot be run; it is merely a definition used to declare and create *Circle* objects. For convenience, the class that contains the *main* method will be referred to as the main class. The illustration of class templates and objects in Figure 3.1.can be standardized using UML (Unified Modeling Language) notations. This notation, as shown in Figure 3.3 is called a UML class diagram, or simply a class diagram. **Figure 3.3. Classes and objects can be represented using UML notation** In the class diagram, the data field is denoted as dataFieldName: dataFieldType The constructor is denoted as ClassName(parameterName: parameterType) The method is denoted as methodName(parameterName: parameterType): returnType ### 3.2. Accessing Objects via Reference Variables Newly created objects are allocated in the memory. How can they be accessed? The answer is given in this section. #### 3.2.1. Reference Variables and Reference Types Objects are accessed via object *reference variables*, which contain references to the objects. Such variables are declared using the following syntax: ClassName objectRefVar; A class defines a type, known as a *reference type*. Any variable of the class type can reference to an instance of the class. The following statement declares the variable myCircle to be of the Circletype: Circle myCircle; The variable myCircle can reference a Circle object. The next statement creates an object and assigns its reference to myCircle. myCircle = new Circle(); Using the syntax shown below, you can write one statement that combines the declaration of an object reference variable, the creation of an object, and the assigning of an object reference to the variable. ClassName objectRefVar = new ClassName(); Here is an example: Circle myCircle = new Circle(); The variable myCircle holds a reference to a Circle object. Note -- ----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- An object reference variable that appears to hold an object actually contains a reference to that object. Strictly speaking, an object reference variable and an object are different, but most of the time the distinction between them can be ignored. So it is fine, for simplicity, to say that myCircle is a Circle object rather than a more long-winded phrase stating that myCircle is a variable that contains a reference to a Circle object. When the distinction makes a subtle difference, the long phrase should be used. -- ----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- Note -- ------------------------------------------------------------------------------------------------------------------------------------------------------------------- Arrays are treated as objects in Java. Arrays are created using the new operator. An array variable is actually a variable that contains a reference to an array. -- ------------------------------------------------------------------------------------------------------------------------------------------------------------------- #### 3.2.2. Accessing an Object\'s Data and Methods After an object is created, its data can be accessed and its methods invoked using the *dot operator (.*), also known as the object member access operator: - objectRefVar.dataField references a data field in the object. - objectRefVar.method(arguments)invokes a method on the object. For example, myCircle.radius references the radius in myCircle, and myCircle.getArea() invokes the getArea method on myCircle. Methods are invoked as operations on objects. The data field radius is referred to as an instance variable because it is dependent on a specific instance. For the same reason, the method getArea is referred to as an *instance method*, because you can only invoke it on a specific instance. The object on which an instance method is invoked is referred to as a calling object. #### 3.2.3. Example: Declaring Classes and Creating Objects Listing 3.1 is a program that declares a *circle* class. The program construct object with radius 5 and displays the radius and area of the circle. ##### Listing 3.1. TestCircle1.java +-----------------------------------------------------------------------+ | 1 public class TestCircle1 { | | | | 2 /\*\* Main method \*/ | | | | 3 public static void main(String\[\] args) { | | | | 4 // Create a circle | | | | 5 Circle myCircle = new Circle(); | | | | 6 // initialized radius | | | | 7 myCircle.radius=5; | | | | 8System.out.println(\"The area of the circle of radius \" | | | | 9 + myCircle.radius + \" is \" + myCircle.getArea()); | | | | 10 | | | | 11 } | | | | 12 } | | | | 13 | | | | 14// Define the circle class with two constructors | | | | 15class Circle { | | | | 16double radius; | | | | 17 | | | | 18//Return the area of this circle | | | | 19double getArea() { | | | | 20return radius \* radius \* Math.PI; | | | | 21 } | | | | 22 } | +-----------------------------------------------------------------------+ The program contains two classes. The first class, TestCircle1, is the main class. Its sole purpose is to test the second class, Circle. Every time you run the program, the JVM invokes the main method in the main class. You can put the two classes into one file, but only one class in the file can be a public class. Furthermore, the public class must have the same name as the file name and the main method must be in a public class. Therefore, the file name is TestCircle1.java if the TestCircle1 and Circle classes are both in the same file. The getArea method is an instance method that is always invoked by an instance in which the radius is specified. #### 3.2.4. Reference Data Fields and the null Value The data fields can be of reference types. For example, the following Student class contains a data field name of the String type. String is a predefined Java class. class Student { String name; // name has default value null int age; // age has default value 0 Boolean isScienceMajor; // isScienceMajor has default value false char gender; // c has default value \'\\u0000\' } If a data field of a reference type does not reference any object, the data field holds a special Java value, **null**. null is a literal just like true and false. While true and false are Boolean literals, null is a literal for a reference type. The default value of a data field is null for a reference type, 0 for a numeric type, false for a boolean type, and \'\\u0000\' for a char type. However, Java assigns no default value to a local variable inside a method. The following code displays the default values of data fields name, age, isScienceMajor, and gender for a Student object: class Test { public static void main(String\[\] args) { Student student = new Student(); System.out.println(\"name? \" + student.name); System.out.println(\"age? \" + student.age); System.out.println(\"isScienceMajor? \" + student.isScienceMajor); System.out.println(\"gender? \" + student.gender); } } The following code has a compilation error because local variables x and y are not initialized: class Test { public static void main(String\[\] args) { int x;// x has no default value String y;// y has no default value System.out.println(\"x is \" + x); System.out.println(\"y is \" + y); } } Caution -- -------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- NullPointerException is a common runtime error. It happens when you invoke a method on a reference variable with null value. Make sure you assign an object reference to the variable before invoking the method through the reference variable. -- -------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- #### 3.2.5. Differences between Variables of Primitive Types and Reference Types ##### Every variable represents a memory location that holds a value. When you declare a variable, you are telling the compiler what type of value the variable can hold. For a variable of a primitive type, the value is of the primitive type. For a variable of a reference type, the value is a reference to where an object is located. For example, as shown in [Figure 3.5](mk:@MSITStore:C:%5CUsers%5CAlex%5CDesktop%5CIntroduction%20to%20Java%20Programming(BETTER).chm::/7.4.%20Accessing%20Objects%20via%20Reference%20Variables.htm#ch07fig05), the value of int variable i is int value 1, and the value of Circle object c holds a reference to where the contents of the Circle object are stored in the memory. ![](media/image4.png) **Figure 3.5.Primitive and Reference Type variables** When you assign one variable to another, the other variable is set to the same value. For a variable of a primitive type, the real value of one variable is assigned to the other variable. For a variable of a reference type, the reference of one variable is assigned to the other variable. As shown in Figure 3.6, the assignment statement i = j copies the contents of j into i for primitive variables. As shown in Figure 3.7, the assignment statement c1 = c2 copies the reference of c2 into c1 for reference variables. After the assignment, variables c1 and c2 refer to the same object. **Figure 3.6. Primitive variable j is copied to variable i.** ![](media/image6.png) **Figure 3.7. Reference variable c2 is copied to variable c1.** Note -- ---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- As shown in Figure 3.7, after the assignment statement c1 = c2, c1 points to the same object referenced by c2. The object previously referenced by c1 is no longer useful and therefore is now known as garbage. Garbage occupies memory space. The JVM detects garbage and automatically reclaims the space it occupies. This process is called garbage collection. -- ---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- Tip -- -------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- If you know that an object is no longer needed, you can explicitly assign null to a reference variable for the object. The JVM will automatically collect the space if the object is not referenced by any reference variable. -- -------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- **3.3. Constructors** The constructor has exactly the same name as the defining class. Like regular methods, constructors can be overloaded (i.e., multiple constructors with the same name but different signatures), making it easy to construct objects with different initial data values. To construct an object from a class, invoke a constructor of the class using the new operator, as follows: new ClassName(arguments); For example, new Circle() creates an object of the Circle class using the first constructor defined in the Circle class, and new Circle(5) creates an object using the second constructor defined in the Circle class. A class normally provides a constructor without arguments (e.g., Circle()). Such a constructor is called a no-arg or no-argument constructor. A class may be declared without constructors. In this case, a no-arg constructor with an empty body is implicitly declared in the class. This constructor, called *a default constructor*, is provided automatically only if no constructors are explicitly declared in the class. Note +-----------------------------------+-----------------------------------+ | | Constructors are special kind of | | | methods, with three differences: | | | | | | - Constructors must have the | | | same name as the class | | | itself. | | | | | | - Constructors do not have a | | | return type---not even void. | | | | | | - Constructors are invoked | | | using the new operator when | | | an object is created. | | | | | | - Constructors play the role of | | | initializing objects. | +-----------------------------------+-----------------------------------+ Caution +-----------------------------------+-----------------------------------+ | | It is a common mistake to put the | | | void keyword in front of a | | | constructor. For example, | | | | | | Public void Circle() { | | | | | | } | | | | | | In this case, Circle() is a | | | method, not a constructor. | | | | | | **Listing 3.2 TestCircle2.java** | | | | | | +------------------------------+ | | | | 1 public class TestCircle2 { | | | | | | | | | | 2 /\*\* Main method \*/ | | | | | | | | | | 3 public static void | | | | | main(String\[\] args) { | | | | | | | | | | 4 // Create a circle with | | | | | radius 5.0 | | | | | | | | | | 5 Circle1 myCircle = new | | | | | Circle1(5.0); | | | | | | | | | | 6 System.out.println(\"The | | | | | area of the circle of radius | | | | | \" | | | | | | | | | | 7 + myCircle.radius + \" is | | | | | \" + myCircle.getArea()); | | | | | | | | | | 8 | | | | | | | | | | 9 // Create a circle with | | | | | radius 1 | | | | | | | | | | 10 Circle1 yourCircle = new | | | | | Circle1(); | | | | | | | | | | 11 System.out.println(\"The | | | | | area of the circle of radius | | | | | \" | | | | | | | | | | 12 + yourCircle.radius + \" | | | | | is \" + | | | | | yourCircle.getArea()); | | | | | | | | | | 13 | | | | | | | | | | 14 // Modify circle radius | | | | | | | | | | 15 yourCircle.radius = 100; | | | | | | | | | | 16 System.out.println(\"The | | | | | area of the circle of radius | | | | | \" | | | | | | | | | | 17 + yourCircle.radius + \" | | | | | is \" + | | | | | yourCircle.getArea()); | | | | | | | | | | 18 } | | | | | | | | | | 19 } | | | | | | | | | | 20 // Define the circle | | | | | class with two constructors | | | | | | | | | | 21 class Circle1 { | | | | | | | | | | 22double radius; | | | | | | | | | | 23 | | | | | | | | | | 24 /\*\* Construct a circle | | | | | with radius 1 \*/ | | | | | | | | | | 25Circle1() { | | | | | | | | | | 26 radius = 1.0; | | | | | | | | | | 27 } | | | | | | | | | | 28 | | | | | | | | | | 29 /\*\* Construct a circle | | | | | with a specified radius \*/ | | | | | | | | | | 30Circle1(double newRadius) | | | | | { | | | | | | | | | | 31 radius = newRadius; | | | | | | | | | | 32 } | | | | | | | | | | 33/\*\* Return the area of | | | | | this circle \*/ | | | | | | | | | | 34doublegetArea() { | | | | | | | | | | 35return radius \* radius \* | | | | | Math.PI; | | | | | | | | | | 36 } | | | | | | | | | | 37} | | | | +------------------------------+ | | | | | | The main class contains the main | | | method (line 3) that creates two | | | objects. The constructor | | | Circle1(5.0) was used to create | | | myCircle with a radius of 5.0 | | | (line 5), and the constructor | | | Circle1() was used to create | | | yourCircle with a radius of 1.0 | | | (line 10). | | | | | | These two objects (referenced by | | | myCircle and yourCircle) have | | | different data but share the same | | | methods. Therefore, you can | | | compute their respective areas by | | | using the getArea() method. | | | | | | ### 3.4. The this Keyword | | | | | | Sometimes you need to reference a | | | class\'s hidden variable in a | | | method. For example, a property | | | name is often used as the | | | parameter name in a set method | | | for the property. In this case, | | | you need to reference the hidden | | | property name in the method in | | | order to set a new value to it. A | | | hidden static variable can be | | | accessed simply by using the | | | ClassName.StaticVariable | | | reference. A hidden instance | | | variable can be accessed by using | | | the keyword **this**, as shown in | | | Figure 3.15(a). | | | | | | ##### Figure 3.15. The keyword th | | | is serves as the proxy for the ob | | | ject that invokes the method. | | | | | | The line this.i = i means | | | \"assign the value of parameter i | | | to the data field i of the | | | calling object.\" The keyword | | | this serves as a proxy for the | | | object that invokes the instance | | | method setI, as shown in Figure | | | 3.15(b). The line Foo.k = k means | | | that the value in parameter k is | | | assigned to the static data field | | | k of the class, which is shared | | | by all the objects of the class. | | | | | | The keyword this can also be used | | | inside a constructor to invoke | | | another constructor of the same | | | class. For example, you can | | | redefine the Circle class as | | | follows: | | | | | | ![](media/image8.png) | | | | | | The line this(1.0) invokes the | | | constructor with a double value | | | argument in the class. | | | | | | Tip | | | | | | -- ---------------------------- | | | --------------------------------- | | | --------------------------------- | | | --------------------------------- | | | --------------------------------- | | | --------------------------------- | | | --------------------------------- | | | --------------------------------- | | | --------------------------------- | | | ----------------------- | | | If a class has multiple cons | | | tructors, it is better to impleme | | | nt them using this(arg-list) as m | | | uch as possible. In general, a co | | | nstructor with no or fewer argume | | | nts can invoke the constructor wi | | | th more arguments using this(arg- | | | list). This often simplifies codi | | | ng and makes the class easier to | | | read and to maintain. | | | -- ---------------------------- | | | --------------------------------- | | | --------------------------------- | | | --------------------------------- | | | --------------------------------- | | | --------------------------------- | | | --------------------------------- | | | --------------------------------- | | | --------------------------------- | | | ----------------------- | | | | | | Note | | | | | | -- ---------------------------- | | | --------------------------------- | | | --------------------------------- | | | ---------------- | | | Java requires that the this( | | | arg-list) statement appear first | | | in the constructor before any oth | | | er statements. | | | -- ---------------------------- | | | --------------------------------- | | | --------------------------------- | | | ---------------- | +-----------------------------------+-----------------------------------+ ### 3.5. Using Classes from the Java Library #### 3.5.1. The Date Class You can use the no-arg constructor in the Date class to create an instance for the current date and time, its getDate() method to return the current date, and its toString method to return the date and time as string. Import java.util.Date; public class TimeClas { public static void main(String args\[\]) { Date date= new Date(); System.out.println(date.getDate()); System.out.println(date.toString()); } } For example, the above code displays the output like this: Today is: 21 Tue Nov 21 03:03:00 PST 2017 ### 3.6. Static Variables, Constants, and Methods The data field radius in the circle class in Listing 3.1 is known as an *instance variable*. An instance variable is tied to a specific instance of the class; it is not shared among objects of the same class. For example, suppose that you create the following objects: Circle circle1 = new Circle(); Circle circle2 = new Circle(5); The radius in circle1 is independent of the radius in circle2, and is stored in a different memory location. Changes made to circle1\'s radius do not affect circle2\'s radius, and vice versa. If you want all the instances of a class to share data, use *static variables*. Static variables store values for the variables in a common memory location. Because of this common location, all objects of the same class are affected if one object changes the value of a static variable. Java supports static methods as well as static variables. *Static methods* can be called without creating an instance of the class. Let us modify the Circle class by adding a static variable numberOfObjects to count the number of circle objects created. When the first object of this class is created, numberOfObjects is 1. When the second object is created, numberOfObjects becomes 2. The UML of the new circle class is shown in Figure 3.10. The Circle class defines the instance variable radius and the static variable numberOfObjects, the instance methods getRadius, setRadius, and getArea, and the static method getNumberOfObjects. (Note that static variables and functions are underlined in the UML class diagram.) ##### Figure 3.10. Instance variables belong to the instances and have memory storage independent of one another. Static variables are shared by all the instances of the same class. To declare a static variable or a static method, put the modifier static in the variable or method declaration. The static variable numberOfObjects and the static method getNumberOfObjects() can be declared as follows: Static int numberOfObjects; Static int getNumberObjects() { Return numberOfObjects; } Constants in a class are shared by all objects of the class. Thus, constants should be declared by final static. For example, the constant PI in the Math class is defined as: final static double PI = 3.14159265358979323846; The new circle class, named Circle2, is declared in Listing 3.3. ##### Listing 3.3. Circle2.java +-----------------------------------------------------------------------+ | 1 public class Circle2 { | | | | 2 /\*\* The radius of the circle \*/ | | | | 3 double radius; | | | | 4 | | | | 5 /\*\* The number of the objects created \*/ | | | | 6 static int numberOfObjects = 0; | | | | 7 | | | | 8 /\*\* Construct a circle with radius 1 \*/ | | | | 9 Circle2() { | | | | 10 radius = 1.0; | | | | 11 numberOfObjects++; | | | | 12 } | | | | 13 | | | | 14 /\*\* Construct a circle with a specified radius \*/ | | | | 15 Circle2(double newRadius) { | | | | 16 radius = newRadius; | | | | 17 numberOfObjects++; | | | | 18 } | | | | 19 | | | | 20 /\*\* Return numberOfObjects \*/ | | | | 21 static int getNumberOfObjects() { | | | | 22 return numberOfObjects; | | | | 23 } | | | | 24 | | | | 25 /\*\* Return the area of this circle \*/ | | | | 26 double getArea() { | | | | 27 return radius \* radius \* Math.PI; | | | | 28 } | | | | 29 } | +-----------------------------------------------------------------------+ Method getNumberOfObjects() in Circle2 is a static method including the main method. Instance methods (e.g., getArea()) and instance data (e.g., radius) belong to instances and can only be used after the instances are created. They are accessed via a reference variable. Static methods (e.g., getNumberOfObjects()) and static data (e.g., numberOfObjects) can be accessed from a reference variable or from their class name. The program in Listing 3.4 demonstrates how to use instance and static variables and methods, and illustrates the effects of using them. ##### Listing 3.4. TestCircle2.java +-----------------------------------------------------------------------+ | 1 public class TestCircle2 { | | | | 2 /\*\* Main method \*/ | | | | 3 public static void main(String\[\] args) { | | | | 4 // Create c1 | | | | 5 Circle2 c1 = new Circle2(); | | | | 6 | | | | 7 // Display c1 BEFORE c2 is created | | | | 8 System.out.println(\"Before creating c2\"); | | | | 9 System.out.println(\"c1 is : radius (\" + c1.radius + | | | | 10 \") and number of Circle objects (\" + | | | | 11 c1.numberOfObjects + \")\"); | | | | 12 | | | | 13 // Create c2 | | | | 14 Circle2 c2 = new Circle2(5); | | | | 15 | | | | 16 // Change the radius in c1 | | | | 17 c1.radius = 9; | | | | 18 | | | | 19 // Display c1 and c2 AFTER c2 was created | | | | 20 System.out.println(\"\\nAfter creating c2 and modifying \" + | | | | 21 \"c1\'s radius to 9\"); | | | | 22 System.out.println(\"c1 is : radius (\" + c1.radius + | | | | 23 \") and number of Circle objects (\" + | | | | 24 c1.numberOfObjects + \")\"); | | | | 25 System.out.println(\"c2 is : radius (\" + c2.radius + | | | | 26 \") and number of Circle objects (\" + | | | | 27 c2.numberOfObjects +\")\"); | | | | 28 } | | | | 29 } | +-----------------------------------------------------------------------+ The main method creates two circles, c1 and c2 (lines 5, 14). The instance variable radius in c1 is modified to become 9 (line 17). This change does not affect the instance variable radius in c2, since these two instance variables are independent. The static variable numberOfObjects becomes 1 after c1 is created (line 5), and it becomes 2 after c2 is created (line 14). Note that PI is a constant defined in Math and Math.PI access the constant. c1.numberOfObjects and c2.numberOfObjects could be replaced by Circle2.numberOfObjects. This improves readability. You can also replace Circle2.numberOfObjects by Circle2.getNumberOfObjects(). Tip -- ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------ Use ClassName.methodName(arguments) to invoke a static method and ClassName.staticVariable. This improves readability because the user can easily recognize the static method and data in the class. -- ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------ Note +-----------------------------------+-----------------------------------+ | | You can use a new JDK 1.5 feature | | | to directly import static | | | variables and methods from a | | | class. The imported data and | | | methods can be referenced or | | | called without specifying a | | | class. For example, you can use | | | PI (instead of Math.PI), and | | | random() (instead of | | | Math.random()), if you have the | | | following import statement in the | | | class: | | | | | | import static java.lang.Math.\*; | +-----------------------------------+-----------------------------------+ Caution -- -------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- Static variables and methods can be used from instance or static methods in the class. However, instance variables and methods can only be used from instance methods, not from static methods, since static variables and methods belong to the class as a whole and not to particular objects. Thus the code given below would be wrong. -- -------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- public class Foo { int i = 5; static int k = 2; public static void main(String\[\] args) { int j = i;// Wrong because i is an instance variable m1();// Wrong because m1() is an instance method } public void m1() { // Correct since instance and static variables and methods // can be used in an instance method i = i + k + m2(i, k); } public static int m2(int i, int j) { return (int)(Math.pow(i, j)); } } Tip -- -------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- How do you decide whether a variable or method should be an instance one or a static one? A variable or method that is dependent on a specific instance of the class should be an instance variable or method. A variable or method that is not dependent on a specific instance of the class should be a static variable or method. For example, every circle has its own radius. Radius is dependent on a specific circle. Therefore, radius is an instance variable of the Circle class. Since the getArea method is dependent on a specific circle, it is an instance method. None of the methods in the Math class, such as random, pow, sin, and cos, is dependent on a specific instance. Therefore, these methods are static methods. The main method is static, and can be invoked directly from a class. -- -------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- Caution -- ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------ It is a common design error to declare an instance method that should have been declared static. For example, the following method factorial(int n) should be declared static, because it is independent of any specific instance. -- ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------ **3.7. Visibility of Modifiers** Java provides several modifiers that control access to data fields, methods, and classes. This section introduces the **public**, **private**, and default modifiers. - public makes classes, methods, and data fields accessible from any class. - private makes methods and data fields accessible only from within its own class. - If public or private is not used, then by default the classes, methods, and data fields are accessible by any class in the same package. This is known as *package-private or package-access*. Figure 3.11illustrates how a public, default, and private data field or method in class C1 can be accessed from a class C2 in the same package, and from a class C3 in a different package. ![](media/image10.png) **Figure 3.11. The private modifier restricts access to its defining a class, the default modifier restricts access to a package, and the public modifier enables unrestricted access.** If a class is not declared public, it can only be accessed within the same package, as shown in Figure 3.12. **Figure 3.12. A non-public class has package-access.** A visibility modifier specifies how data fields and methods in a class can be accessed from the outside of the class. There is no restriction on accessing data fields and methods from inside the class. As shown in Figure 3.13(b), an object foo of the Foo class cannot access its private members, because foo is in the Test class. As shown in [Figure 3.13(a)](mk:@MSITStore:C:%5CUsers%5CAlex%5CDesktop%5CIntroduction%20to%20Java%20Programming(BETTER).chm::/7.7.%20Visibility%20Modifiers.htm#ch07fig14), an object foo of the Foo class can access its private members, because foo is declared inside its own class. ![](media/image12.png) **Figure 3.13. An object can access its private members if it is declared in its own class.** ### 3.8. Passing Objects to Methods Like passing an array, passing an object is actually passing the reference of the object. The following code passes the myCircle object as an argument to the printCircle method: public class TestPassObject { public static void main(String\[\] args) { Circle3 myCircle = new Circle3(5.0); Print Circle(myCircle); } public static void printCircle(Circle3 c) { System.out.println(\"The area of the circle of radius \" \+ c.getRadius() + \" is \" + c.getArea()); } } Java uses exactly one mode of passing arguments: pass-by-value. In the preceding code, the value of myCircle is passed to the print Circle method. This value is a reference to a Circle3 object. Let us demonstrate the difference between passing a primitive type value and passing a reference value with the program in Listing 3.5. ##### Listing 3.5. TestPassObject.java +-----------------------------------------------------------------------+ | 1 public classTestPassObject { | | | | 2 /\*\* Main method \*/ | | | | 3 public static void main(String\[\] args) { | | | | 4 // Create a Circle object with radius 1 | | | | 5 Circle3 myCircle = new Circle3(1); | | | | 6 | | | | 7 // Print areas for radius 1, 2, 3, 4, and 5. | | | | 8 int n = 5; | | | | 9 printAreas(myCircle, n); | | | | 10 | | | | 11 // See myCircle.radius and times | | | | 12 System.out.println(\"\\n\" + \"Radius is \" + | | myCircle.getRadius()); | | | | 13 System.out.println(\"n is \" + n); | | | | 14 } | | | | 15 | | | | 16 /\*\* Print a table of areas for radius \*/ | | | | 17 public static void printAreas(Circle3 c, int times) { | | | | 18 System.out.println(\"Radius \\t\\tArea\"); | | | | 19 while (times \>= 1) { | | | | 20 System.out.println(c.getRadius() + \"\\t\\t\" + c.getArea()); | | | | 21 c.setRadius(c.getRadius() + 1); | | | | 22 times------; | | | | 23 } | | | | 24 } | | | | 25 } | +-----------------------------------------------------------------------+ The program passes a Circle3 object myCircle and an integer value from n to invoke printAreas(myCircle, n) (line 19) which prints a table of areas for radii 1, 2, 3, 4, and 5. When passing an argument of a primitive data type, the value of the argument is passed. In this case, the value of n (5) is passed to times. Inside the printAreas method, the content of times is changed; this does not affect the content of n. When passing an argument of a reference type, the reference of the object is passed. In this case, c contains a reference for the object that is also referenced via myCircle. Therefore, changing the properties of the object through c inside the printAreas method has the same effect as doing so outside the method through the variable myCircle. **3.9. The Scope of Variables** Local variables are declared and used inside a method locally. This section discusses the scope rules of all the variables in the context of a class. Instance and static variables in a class are referred to as the class\'s variables or data fields. A variable defined inside a method is referred to as a local variable. The scope of a class\'s variables is the entire class, regardless of where the variables are declared. A class\'s variables and methods can be declared in any order in the class, as shown in Figure 3.14(a). The exception is when a data field is initialized based on a reference to another data field. In such cases, the other data field must be declared first, as shown in Figure 3.14(b). **Figure 3.14. Members of a class can be declared in any order, with one exception.** You can declare a class\'s variable only once, but you can declare the same variable name in a method many times in different non-nesting blocks. If a local variable has the same name as a class\'s variable, the local variable takes precedence and the class\'s variable with the same name is hidden. For example, in the following program, x is defined as an instance variable and as a local variable in the method. class Foo { intx = 0;// instance variable int y = 0; Foo() { } void p() { intx = 1;// local variable System.out.println(\"x = \" + x); System.out.println(\"y = \" + y); } } What is the printout for f.p(), where f is an instance of Foo? The printout for f.p() is 1 for x and 0 for y. Here is why: - x is declared as a data field with the initial value of 0 in the class, but is also defined in the method p() with an initial value of 1. The latter x is referenced in the System.out.println statement. - y is declared outside the method p(), but is accessible inside it. Tip -- ------------------------------------------------------------------------------------------------------------------------------------------------------------------------ As demonstrated in the example, it is easy to make mistakes. To avoid confusion, do not declare the same variable name twice in a class, except for method parameters. -- ------------------------------------------------------------------------------------------------------------------------------------------------------------------------

Use Quizgecko on...
Browser
Browser