The Essence of OOP using Java: Static Members

类别:Java 点击:0 评论:0 推荐:

The Essence of OOP using Java: Static Members
PrefaceThis lesson is one of a series of lessons designed to teach you about the essence of Object-Oriented Programming (OOP) using Java. The first lesson in the group was entitled The Essence of OOP Using Java, Objects, and Encapsulation.  That lesson, and each of the lessons following that one, has provided explanations of certain aspects of the essence of Object-Oriented Programming using Java. The previous lesson was entitled The Essence of OOP using Java, Polymorphism and Interfaces, Part 2.

Necessary and significant aspects

This miniseries will describe and discuss the necessary and significant aspects of OOP using Java. If you have a general understanding of computer programming, you should be able to read and understand the lessons in this miniseries, even if you don't have a strong background in the Java programming language.

Supplementary material

I recommend that you also study the other lessons in my extensive collection of online Java tutorials. You will find those lessons published at Gamelan.com. However, as of the date of this writing, Gamelan doesn't maintain a consolidated index of my Java tutorial lessons, and sometimes they are difficult to locate there.  You will find a consolidated index at Baldwin's Java Programming Tutorials.

PreviewStatic members

There is another aspect of OOP in Java that I have avoided up to this point in the discussion: static variables and static methods.

Tends to complicate ...

I have avoided this topic because, while not particularly difficult, the existence of static members tends to break up the simple structures that I have discussed in previous lessons in this miniseries.

While static members can be useful in some situations, the existence of static members tends to complicate the overall object-oriented structure of Java.

Avoid overuse of static members

Furthermore, the overuse of static members can lead to problems similar to those experienced in languages like C and C++ that support global variables and global functions.

When to use static members

I will discuss the use of static members in this lesson, and will provide some guidelines for their use.

The class named Class

I will also introduce the class named Class and discuss how it enters into the use of static variables and methods.

Instance members vs.. class members

I will describe the differences between instance members and class members with particular emphasis being placed on their accessibility.

Three kinds of objects

From a conceptual viewpoint, there are at least three kinds of objects involved in a Java program:

Ordinary objects Array objects Class objects Ordinary objects

All of the discussion up to this point in the miniseries deals with what I have referred to in the above list as ordinary objects.

These are the objects that you instantiate in you code by applying the new operator to a constructor for a class in order to create a new instance (object) of that class.  (There are also a couple of other ways to create ordinary objects, but I'm not going to get into that at this time.)

Array objects

I haven't discussed array objects thus far in this miniseries. (I plan to discuss them in a subsequent lesson.)

Suffice it for now to say that Array objects are objects whose purpose is to encapsulate a one-dimensional array structure that can contain either primitive values, or references to other objects (including other Array objects).

I will discuss Class objects in this lesson.

Discussion and Sample CodeClass objects

Let me emphasize at the beginning that the following discussion is conceptual in nature.  In this discussion, I will describe how the Java system behaves, not how it is implemented.  In other words, however it is implemented, it behaves as though it is implemented as described below.

The class named Class

There is a class whose name is Class.  The purpose of this class is to encapsulate information about some other class (actually, it can also be used to encapsulate information about primitive types as well as class types).

Here is part of what Sun has to say about this class:

"Instances of the class Class represent classes and interfaces in a running Java application. ...

Class has no public constructor. Instead Class objects are constructed automatically by the Java Virtual Machine as classes are loaded ..."

What does this mean?

As a practical matter, when one or more objects are instantiated from a given class, an extra object of the Class class is also instantiated automatically.  This object contains information about the class from which the objects were instantiated.

(Note that it is also possible to cause a Class object that describes a specific class to be created in the absence of objects of that class, but that is a topic that will be reserved for more advanced lessons.)A real-world analogy

Here is an attempt to describe a real-world analogy.  Remember that a class definition contains the blueprint for objects instantiated from that class.

A certain large construction company is in the business of building condominium projects.  This contractor builds condos of many different sizes, types, and price ranges.  However, each different condo project contains condos of only two or three different types or price ranges.

A library of blueprints

There is a large library of blueprints at the contractor's central office.  This library contains blueprints for all of the different types of condos that the contractor has built or is building.  (This library is analogous to the class libraries available to the Java programmer.)

A subset from the blueprint library

When a condo project begins, the contractor delivers copies of several sets of blueprints to the construction site.  The blueprints delivered to that site describe only the types of condos being constructed on that site.

Condo is analogous to an object

Each condo unit is analogous to an ordinary Java object.

Each set of blueprints delivered to the construction site is roughly analogous to an object of the class named Class.  In other words, each set of blueprints describes one or more condo units constructed from that set of blueprints.

When construction is complete

When the construction project is complete, the contractor delivers a set of blueprints for each type of condo unit to the management firm that has been hired to manage the condo complex.  Each set of blueprints continues to be analogous to an object of the class named Class.  The blueprints remain at the site of the condo units.

RTTI

Thus, information regarding the construction, wiring, plumbing, air conditioning, etc., for each condo unit (object) continues to be available at the site even after the construction has been completed.

(This is somewhat analogous to something called runtime type information and often abbreviated as RTTI.  A Class object contains RTTI for objects instantiated from that class.)What are those analogies again?

In the above scenario, each condo unit is (roughly) analogous to an object instantiated from a specific class (set of blueprints).

Each set of blueprints remaining onsite after construction is complete is roughly analogous to a Class object that describes the characteristics of one or more condo units.

What do you care?

Until you get involved in such advanced topics as reflection and introspection, you don't usually have much involvement or much interest in Class objects.  They are created automatically, and are primarily used by the Java virtual machine during runtime to help it do the things that it needs to do.

An exception to that rule

However, there is one area where you will be interested in the use of these Class objects from early on.  You will be interested whenever variables or methods in the class definition are declared to be static.

Class variables and methods

According to the current jargon, declaring a member variable to be static causes it to be a class variable.

(Note that local variables cannot be declared static.  Only member variables can be declared static.)Similarly, declaring a method to be static causes it to be a class method.

Instance variables and methods

On the other hand, according to the current jargon, not declaring a variable to be static causes it to be an instance variable, and not declaring a method to be static causes it to be an instance method.

In general, we can refer to them as class members and instance members.

What is the difference?

Here are some of the differences between class and instance members insofar as this discussion is concerned.

How many copies of member variables exist?

Every object instantiated from a given class has its own copy of each instance variable defined in the class.

(Instance variables are not shared among objects.)However, every object instantiated from a given class shares the same copy of each class variable defined in the class. (It is as though the class variable belongs to the single Class object and not to the individual objects instantiated from that class.)Access to an instance variable

Every object has its own copy of each instance variable (the object owns the instance variable).  Therefore, the only way that you can access an instance variable is to use that object's reference to send a message to the object requesting access to the variable (even then, you may not be given access, depending on access modifiers).

Why call it an instance variable?

According to the current jargon, an object is an instance of a class.

(I probably told you that somewhere before in this miniseries.  At least, I hope that I did.  After writing eight or ten lessons in a miniseries, I sometimes forget what I told you before.)Each object has its own copy of each non-static variable.  Hence, they are often called instance variables.  (Every instance of the class has one.)

Access to a class variable

You can also send a message to an object requesting access to a class variable that the object shares with other objects instantiated from the same class.  (Again, you may or may not gain access, depending the access modifiers).

Access using the Class object

More importantly, you can also access a class variable without a requirement to go through an object instantiated from the class.  (In fact, a class variable can be accessed in the total absence of objects of that class.)

(Remember, this discussion is conceptual in nature, and may not represent an actual implementation.)Assuming that a class variable is otherwise accessible, you can access the class variable by sending an access request message to the Class object to which the variable belongs.

One way to think of this

To help you keep track of thing in a message-passing sense, you can pretend that there is a global reference variable whose name is the same as the name of a class.

This (hypothetical) reference variable contains a reference to the Class object that owns the class variable.  Using standard Java message-passing syntax, you can access the class variable by joining the name of the reference variable to the name of the class variable with a period.  Example syntax is shown below:

ReferenceVariableName.ClassVariableName

As a result of the hypothetical substitution process that I described above, this is equivalent to the following:

ClassName.ClassVariableName

We will see an example of this in the sample program that I will discuss later.

Be careful with this thought process

While this thought process may be useful when thinking about static variables and methods, I want to point out, that the thought process breaks down very quickly when dealing with Class objects in a deeper sense.

For example, when invoking the getName method on a Class object, an actual reference of type Class is required to access the members of the Class object.  The name of the class will not suffice.

If this discussion of a global reference variable whose name matches the name of the class is confusing to you, just forget it.  Simply remember that you can access class variables by joining the name of the class to the name of the class variable using a period as the joining operator.

Characteristics of class methods

I'm not going to talk very much about instance methods and class methods in this lesson.  However, there are a couple of characteristics of class methods that deserve a brief discussion in this context.

Cannot access instance members

First, the code in a class method has direct access only to other static members of the class.

(A class method does not have direct access to instance variables or instance methods of the class.)This is sort of like saying that a class method has access to the methods and variables belonging to the Class object, but does not have access to the methods and variables belonging to the ordinary objects instantiated from the class described by the Class object. Once again, be careful

Once again, this thinking breaks down very quickly once you get beyond static members.  A Class object also has instance methods, such as getName, which can only be accessed using an actual reference to the Class object.

Now you are probably beginning to understand why I deferred this discussion until after I finished discussing the easy stuff.

No object required

Another important characteristic is that a class method can be accessed without a requirement for an object of the class to exist.

As with class variables, class methods can be accessed by joining the name of the class to the name of the method with a period.

I will illustrate much of this with a sample program named MyClass01.

Discuss in fragments

I will discuss the program in fragments.  You will find a complete listing of the program in Listing 13 near the end of the lesson.

Listing 1 shows the beginning of the class definition.
 

class MyClass01{   static Date v1 = new Date();   Date v2 = new Date(); Listing 1

Two member variables

The code in Listing 1 declares two member variables, named v1 and v2, and initializes each of those variables with a reference to a new object of the Date class.

(When instantiated using the constructor with no arguments, the new Date object encapsulates the current date and time from the system clock.)Note the static keyword

The important thing to note here is the use of the static keyword when declaring the variable named v1.  This causes v1 to be a class variable, exhibiting the characteristics of class variables described earlier.

An instance variable

On the other hand, the variable named v2 is not declared static.  This causes it to be an instance variable, as described above.

The main method is a class method

Listing 2 shows the signature for the main method.
 

  public static void main(                         String[] args){ Listing 2

The important thing to note here is that the main method is declared static.  That causes it to be a class method.

As a result, the main method can be invoked without a requirement for an object of the class to exist.

(Also, the main method has direct access only to other static members.)How a Java application starts running

In fact, that is how the Java Virtual Machine starts an application running.

First the JVM finds the specified file having an extension of .class.  Then it examines that file to see if it has a main method with the correct signature.  If not, an error occurs.

If the JVM finds a main method with the correct signature, it invokes that method without instantiating an object of the class.  That is how the Java Virtual Machine causes a Java application to start running.

A side note regarding applets
For those of you who are familiar with Java applets, you should know that this is not the case for an applet.  You should know that an applet does not use a main method.  When an applet is started, an object of the controlling class is instantiated by the browser, by the appletviewer program, or by whatever program is being used to control the execution of the applet.A poor programming technique

Basically, this entire sample program is coded inside the main method.  As a practical manner, that is a very poor programming technique, but it works well for this example.

Display some text

The code in Listing 3, which is the first executable statement in the main method, causes the words Static variable to appear on the computer screen.  I will come back and discuss the details of this and similar statements later in the lesson.
 

    System.out.println(                     "Static variable"); Listing 3

Display date information

Proceeding down through the code in the main method, the code in Listing 4 causes the current contents of the Date object referred to by the contents of the class variable named v1 to be displayed on the computer screen.
 

    System.out.println(MyClass01.v1); Listing 4

No object required

For the moment, concentrate on the boldface text in the statement in Listing 4.

IMPORTANT:  Because the variable named v1 is a class variable, it's value is accessed by joining the name of the class to the name of the variable with a period.

What was the output?

I will also discuss the remaining portion of statements of this sort later.  For now, just be aware that it caused the output shown in Figure 1 to be displayed on my computer screen when I ran the program.
 

Mon Sep 17 09:52:27 CDT 2001 Figure 1

Displays date and time

Obviously, the date and time displayed will depend on when you run the program.

Pay particular attention to the seconds portion of the time.  I will refer back to this later.

A five-second delay

The code in Listing 5 (still in the main method) causes the main thread of the program to go to sleep for five seconds.  Don't worry about it if you don't understand this code.  The only reason that I included it in the program was to force a five-second delay in the execution of the program.
 

  try{     Thread.currentThread().sleep(5000);   }catch(InterruptedException e){} Listing 5

Instantiate a new object

Having caused the program to sleep for five seconds, the code in Listing 6 instantiates a new object of the class named MyClass01.  The code stores the new object's reference in the reference variable named ref1.
 

    MyClass01 ref1 = new MyClass01(); Listing 6

A new Date object also

Recall from Listing 1 above that the class declares an instance variable named v2 of the type Date.

When the new object is instantiated by the code in Listing 6, a new Date object is also instantiated.  A reference to that object is stored in the instance variable named v2.

(In other words, the new object of the class MyClass01 owns a reference to a new object of the class Date.  That reference is stored in an instance variable named v2 in the new MyClass01 object.)Display the new Date object

The code in Listing 7 causes a textual representation of the new Date object referred to by the reference variable named v2 belonging to the object referred to by the reference variable named ref1, to be displayed on the standard output device.
 

    System.out.println(ref1.v2); Listing 7

Five seconds later

This code caused the date and time shown in Figure 2 to appear on the computer screen when I ran the program:
 

Mon Sep 17 09:52:32 CDT 2001 Figure 2

If you note the time in the above output, you will see that it is five seconds later than the time reflected in the Date object referred to by the class variable named v1.  That time was displayed by the code in Listing 4 earlier.

So, what does this mean?

It means that the Date object referred to by the static reference variable named v1 was created five seconds earlier than the Date object referred to by the instance variable named v2.

When is a class variable created?

I can't tell you precisely when a class variable comes into existence.  All I can say is that the virtual machine brings it into existence as soon as it is needed.

My guess is that it comes into existence at the first mention (in the program) of the class to which it belongs.

When is an instance variable created?

An instance variable doesn't come into existence until the object to which it belongs is created (an instance variable cannot exist until the object to which it belongs exists).

If the instance variable is initialized with a reference to a new object (such as a new Date object in this sample program), that new object comes into existence when the object to which it belongs comes into existence.

A five-second delay

In this program, I purposely inserted a five-second delay between the first mention of the class named MyClass01 in Listing 4, and the instantiation of the object of the class named MyClass01 in Listing 6.

As a result, the Date object referred to by the instance variable named v2 was created about five seconds later than the Date object referred to by the class variable named v1.

This is reflected in the date and time values displayed and discussed earlier.

Accessing class variable via an object

While it is possible to access a class variable using the name of the class joined to the name of the variable, it is also possible to access a class variable using a reference to any object instantiated from the class.

(As mentioned earlier, if two or more objects are instantiated from the same class, they share the same class variable.)The boldface code in Listing 8 uses the reference variable named ref1 to access the class variable named v1, and to cause the contents of the Date object referred to by the class variable to be displayed.
 

    System.out.println(ref1.v1); Listing 8

The output

This caused the date and time shown in Figure 3 to be displayed on my computer screen.
 

Mon Sep 17 09:52:27 CDT 2001 Figure 3

Same date and time

As you have probably already surmised, this is the same date and time shown earlier in Figure 1.  This is because the code in Listing 8 refers to the same class variable as the code in Listing 4.  Nothing has caused the contents of that class variable to change, so both Figure 1 and Figure 3 display the contents of the same Date object.

(Only one class variable exists and it doesn't matter how you access it.  Either way, you gain access to the same Date object whose reference is stored in the class variable.  Thus, the same date and time is shown in both cases.)Another new object

If you examine the code in Listing 13 near the end of the program, you will see that an additional five-second delay is introduced at this point in the program.

Following that delay, the code in Listing 9 instantiates another new object of the class named MyClass01, and stores the object's reference in a new reference variable named ref2.

(The object referred to by ref1 is a different object than the object referred to by ref2.  Each object has its own instance variable named v2, and in this case, each instance variable is initialized to instantiate and refer to a new Date object when the new MyClass01 object is instantiated.)

    MyClass01 ref2 = new MyClass01(); Listing 9

Display the date and time

Then, the code in Listing 10 causes the contents of the Date object referred to by the instance variable named v2 in the second object of the class named MyClass01 to be displayed.
 

    System.out.println(ref2.v2); Listing 10

This caused the output shown in Figure 4 to be displayed on my computer screen when I ran the program (again, you will get different results if you compile and run the program because the date and time shown is the date and time that you run the program).
 

Mon Sep 17 09:52:37 CDT 2001 Figure 4

Five seconds later

As you have probably figured out by now, the time encapsulated in this Date object is five seconds later than the time encapsulated in the Date object displayed in Figure 2.  This is because the program was put to sleep for five seconds between the instantiation of the two objects referred to by ref1 and ref2.

Every object has one

Every object instantiated from a given class has its own copy of each instance variable declared in the class definition.  There is no sharing of instance variables among objects.

Each instance variable comes into existence when the object to which it belongs comes into existence, and ceases to exist when the object to which it belongs ceases to exist.

(If the instance variables are reference variables holding references to other objects, as is the case here, and if there are no other reference variables holding references to those same objects, the secondary objects cease to exist when the primary objects cease to exist.  Technically, the objects may not cease to exist.  Technically they become eligible for garbage collection, which means that the memory that they occupy becomes eligible for reuse.  However, as a practical matter, they cease to exist insofar as the program is concerned.  They are no longer accessible.)Since the two objects referred to by ref1 and ref2 came into existence with a five-second delay, the Date objects belonging to those two object reflect a five-second difference in the time encapsulated in the objects.

Only one copy of class variable exists

Also remember that if a variable is a class variable, only one copy of the variable exists, and all objects instantiated from the class share that one copy.

This is illustrated by the code in Listing 11, which uses the reference to the second object instantiated from the class named MyClass01, to cause the contents of the class variable named v1 to be displayed.
 

    System.out.println(ref2.v1);   }//end main Listing 11

The output produced by the code in Listing 11 is shown in Figure 5.
 

Mon Sep 17 09:52:27 CDT 2001 Figure 5

Same output as before

As you can see, this is the same as the output shown in Figure 1 and Figure 3 earlier.

Accessing the same physical class variable

Since only one class variable named v1 exists, and all objects instantiated from the class named MyClass01 share that single copy, it doesn't matter whether you access the class variable using the name of the class, or access it using a reference to either of the objects instantiated from the class.  In all three cases, you are accessing the same physical class variable.

Since nothing was done to cause the contents of the class variable to change after it came into existence and was initialized, Figures 1, 3, and 5 are simply three different displays of the date and time encapsulated in the same Date object whose reference is stored in the class variable.

Let's revisit System.out.println...

Now, I want to revisit the statement originally shown in Listing 8 and repeated in Listing 12 for viewing convenience.
 

    System.out.println(ref1.v1); Listing 12

Java programmer wanted

I often tell my students that if I were out in industry, interviewing prospective Java programmers, my first question would be to ask the prospective employee to tell me everything that she knows about the statement in Listing 12.

Covers a lot of Java OOP technology

This is not because there is a great demand for the use of this statement in real-world problems.  (In fact, in a GUI-driven software product world, there is probably very little demand for the use of this statement.)  Rather, it is because a lot of Java object-oriented technology is embodied in this single statement.

In that scenario, I would expect to receive a verbal dissertation of fifteen to twenty minutes in length to cover all the important points.

The short version

Let me give you the short version.  There is a class named System. The System class declares three static (class) variables having the following types, names, and modifiers:

public static final PrintStream out public static final InputStream in public static final PrintStream err (Note that these class variables are also declared final, causing them to behave as constants.)Access the out variable without an object

Because out is a class variable, System.out returns the contents of the class variable named out (an object of the System class is not required in order to access a class variable of the System class).

In general, (ignoring the possibility of subclasses and interfaces) because out is a reference variable of type PrintStream, the returned value must either be null (no object reference) or a reference to a valid PrintStream object.

Object of the PrintStream class

When the Java Virtual Machine starts an application running, it instantiates an object of the PrintStream class and connects it to the standard output device.

(By default, the standard output device is typically the computer screen, but it can be redirected at the operating system level to be some other device.  The following discussion assumes that the screen is the standard output device.)Assign object's reference to out variable

When the PrintStream object is instantiated by the virtual machine, the object's reference is assigned to the class variable of the System class named out.

(Because the variable named out is final, the contents of the variable cannot be modified later.)Reference to a PrintStream object

Therefore, the expression System.out returns a reference to the PrintStream object, which is connected to the standard output device.

Many instance methods

An object of the PrintStream class contains many instance methods.  This includes numerous overloaded versions of a method named println.  The signature of one of those overloaded versions of the println method follows:

public void println(Object x)

Textual representation of an object

The purpose of this overloaded version of the println method is to:

Create a textual representation of the object referred to by the incoming parameter of type Object (because Object is a totally generic type, this version of the println method can accept an incoming parameter that is a reference to any type of object) Send that textual representation to the output device In general ...
(In general, a new PrintStream object can be connected to a variety of output devices when it is instantiated.  However, in the special case of the PrintStream object instantiated by the virtual machine when the program starts, whose reference is stored in the class variable named out of the System class, the purpose of the object is to provide a display path to the standard output device.)Our old friend, the toString method

To accomplish this, the code in the println method invokes the toString method on the incoming reference.

(I discussed the toString method in detail in earlier lessons in this miniseries.)The toString method may, or may not, have been overridden in the definition of the class from which the object was instantiated, or in some superclass of the class from which the object was instantiated.

Default version of toString

If not overridden, the default version of the toString method defined in the Object class is used to produce a textual representation of the object.  As we learned in an earlier lesson, that textual representation looks something like the following:

ClassName@HexHashCode

Overridden version of toString method

If the class from which the object was instantiated (or some superclass of that class) contains an overridden version of the toString method, runtime polymorphism kicks in and the overridden version of the method is executed to produce the textual representation of the object.

The Date class overrides toString

In the case of this sample program, the object was instantiated from the Date class.  The Date class does override the toString method.

When the overridden toString method is invoked on a Date object's reference, the String returned by the method looks something like that shown in Figure 6.
 

Mon Sep 17 09:52:27 CDT 2001 Figure 6

You will recall that this is the output that was produced by the code shown in Listing 8 and Listing 12.

More than you ever wanted to know ...

And that is probably more than you ever wanted to know about the expression System.out.println....

It is also probably more than you ever wanted to know about class variables, class methods, instance variables, and instance methods.

Some cautions

Before leaving this topic, I do want to express some cautions.  Basically, I want to suggest that you use static members very sparingly, if at all.  Several good sets of guidelines have been published regarding the use of static members.  As of this writing, you can find one of them, written by Mark L. Fussell, at the following URL.

Guidelines for use of static members

I will try to summarize the general sense of most published guidelines in a few words.

Static variables

To begin with, don't ever use static variables without declaring them final unless you understand exactly why you are declaring them static.

(Static final variables, or constants, are often very appropriate.  See the fields in the Color class for example.)I can only think of a few situations where the use of a non-final static variable might be appropriate. (One appropriate use might be to count the number of objects instantiated from a specific class.  I suspect there are a few other appropriate uses as well.)Static methods

Don't declare methods static if there is any requirement for the method to remember anything from one invocation to the next.

There are many appropriate uses for static methods, but in most cases, the purpose of the method will be to completely perform some action with no requirement to remember anything from that invocation to the next.

The method should probably also be self-contained.  By this I mean that all information that the method needs to do its job should either come from incoming parameters or from final static member variables (constants).  The method probably should not depend on the values stored in non-final static member variables, which are subject to change over time.

(A static method only has access to other static members of the class, so it cannot depend on instance variables defined in the class.)An appropriate example of a static method is the sqrt method of the Math class.  This method computes and "Returns the correctly rounded positive square root of a double" where the double value is provided as a parameter to the method.  Each time the method is invoked, it completes its task and doesn't attempt to save any values from that invocation to the next.  Furthermore, it gets all the information that it needs to do its job from an incoming parameter. SummaryAdded complexity

The existence of static members tends to break up the simple OOP structures that I have discussed in previous lessons in this miniseries.

While static members can be useful in some situations, the existence of static members tends to complicate the overall object-oriented structure of Java.

Furthermore, the overuse of static members can lead to problems similar to those experienced in languages like C and C++ that support global variables and global functions.

The class named Class

I discussed the class named Class and how a conceptual object of type Class exists in memory following a reference to a specific class in the program code.

The Class object represents the referenced class in memory, and contains the static variables and static methods belonging to that class. (It contains some other information as well, such as the name of the superclass.)

Class members and instance members

Class variables and class methods are declared static (declaring a member static in the class definition causes to be called a class member).

Instance variables and instance methods are not declared static.

Each object has its own copy ...

Every object instantiated from a given class has its own copy of each instance variable declared in the class definition.  (Instance variables are not shared among objects.)

Every object instantiated from a given class acts like it has its own copy of every instance method declared in the class definition.  (Although instance methods are actually shared among objects in order to reduce the amount of memory required, they are shared in such a way that they don't appear to be shared.)

Every object shares ...

Every object instantiated from a given class shares the same single copy of each class variable declared in the class definition.  Similarly, every object instantiated from a given class shares the same copy of each class method.

Accessing an instance member

An instance variable or an instance method can only be accessed by using a reference to the object that owns it.  Even then, it may or may not be accessible depending on the access modifier assigned by the programmer.

Accessing a class member

The single shared copy of a class variable or a class method can be accessed in either of two ways:

Via a reference to any object instantiated from the class By simply joining the name of the class to the name of the class variable or the class method Again, the variable or method may or may not be accessible, depending on the access modifiers assigned by the programmer.

When to use class variables

It is very often appropriate to use final static variables, as constants in your programs.  It is rarely, if ever, appropriate to use non-final static variables in your programs.  The use of non-final static variables should definitely be minimized.

When to use static methods

It is often appropriate to use static methods in your programs, provided there is no requirement for the method to remember anything from one invocation to the next.  Static methods should be self-contained.

What's Next?The next lesson in this miniseries will address the special case of Array Objects. Complete Program ListingA complete listing of the sample program is shown in Listing 13.
 

/*File MyClass01.java Copyright 2002, R.G.Baldwin This program illustrates static  members of a class.  Output is:    Static variable Mon Sep 17 09:52:27 CDT 2001 Instance variable Mon Sep 17 09:52:32 CDT 2001 Static variable Mon Sep 17 09:52:27 CDT 2001 Instance variable Mon Sep 17 09:52:37 CDT 2001 Static variable Mon Sep 17 09:52:27 CDT 2001 **************************************/ import java.util.Date; class MyClass01{   static Date v1 = new Date();   Date v2 = new Date();   public static void main(                         String[] args){     //Display static variable     System.out.println(                     "Static variable");     System.out.println(MyClass01.v1);          //Delay for five seconds     try{     Thread.currentThread().sleep(5000);     }catch(InterruptedException e){}          //Instantiate an object and      // display instance variable     MyClass01 ref1 = new MyClass01();     System.out.println();//blank line     System.out.println(                   "Instance variable");     System.out.println(ref1.v2);          //Now, display the static     // variable using object reference     System.out.println(                     "Static variable");     System.out.println(ref1.v1);          System.out.println();//blank line          //Delay for five seconds     try{     Thread.currentThread().sleep(5000);     }catch(InterruptedException e){}          //Instantiate another object      MyClass01 ref2 = new MyClass01();     System.out.println();//blank line     System.out.println(                   "Instance variable");     System.out.println(ref2.v2);          //Now, display the same static     // variable using object reference     System.out.println(                     "Static variable");     System.out.println(ref2.v1);   }//end main }//end class MyClass01 //===================================// Listing 13

PrefaceThis lesson is one of a series of lessons designed to teach you about the essence of Object-Oriented Programming (OOP) using Java. The first lesson in the group was entitled The Essence of OOP Using Java, Objects, and Encapsulation.  That lesson, and each of the lessons following that one, has provided explanations of certain aspects of the essence of Object-Oriented Programming using Java. The previous lesson was entitled The Essence of OOP using Java, Polymorphism and Interfaces, Part 2.

Necessary and significant aspects

This miniseries will describe and discuss the necessary and significant aspects of OOP using Java. If you have a general understanding of computer programming, you should be able to read and understand the lessons in this miniseries, even if you don't have a strong background in the Java programming language.

Supplementary material

I recommend that you also study the other lessons in my extensive collection of online Java tutorials. You will find those lessons published at Gamelan.com. However, as of the date of this writing, Gamelan doesn't maintain a consolidated index of my Java tutorial lessons, and sometimes they are difficult to locate there.  You will find a consolidated index at Baldwin's Java Programming Tutorials.

PreviewStatic members

There is another aspect of OOP in Java that I have avoided up to this point in the discussion: static variables and static methods.

Tends to complicate ...

I have avoided this topic because, while not particularly difficult, the existence of static members tends to break up the simple structures that I have discussed in previous lessons in this miniseries.

While static members can be useful in some situations, the existence of static members tends to complicate the overall object-oriented structure of Java.

Avoid overuse of static members

Furthermore, the overuse of static members can lead to problems similar to those experienced in languages like C and C++ that support global variables and global functions.

When to use static members

I will discuss the use of static members in this lesson, and will provide some guidelines for their use.

The class named Class

I will also introduce the class named Class and discuss how it enters into the use of static variables and methods.

Instance members vs.. class members

I will describe the differences between instance members and class members with particular emphasis being placed on their accessibility.

Three kinds of objects

From a conceptual viewpoint, there are at least three kinds of objects involved in a Java program:

Ordinary objects Array objects Class objects Ordinary objects

All of the discussion up to this point in the miniseries deals with what I have referred to in the above list as ordinary objects.

These are the objects that you instantiate in you code by applying the new operator to a constructor for a class in order to create a new instance (object) of that class.  (There are also a couple of other ways to create ordinary objects, but I'm not going to get into that at this time.)

Array objects

I haven't discussed array objects thus far in this miniseries. (I plan to discuss them in a subsequent lesson.)

Suffice it for now to say that Array objects are objects whose purpose is to encapsulate a one-dimensional array structure that can contain either primitive values, or references to other objects (including other Array objects).

I will discuss Class objects in this lesson.

Discussion and Sample CodeClass objects

Let me emphasize at the beginning that the following discussion is conceptual in nature.  In this discussion, I will describe how the Java system behaves, not how it is implemented.  In other words, however it is implemented, it behaves as though it is implemented as described below.

The class named Class

There is a class whose name is Class.  The purpose of this class is to encapsulate information about some other class (actually, it can also be used to encapsulate information about primitive types as well as class types).

Here is part of what Sun has to say about this class:

"Instances of the class Class represent classes and interfaces in a running Java application. ...

Class has no public constructor. Instead Class objects are constructed automatically by the Java Virtual Machine as classes are loaded ..."

What does this mean?

As a practical matter, when one or more objects are instantiated from a given class, an extra object of the Class class is also instantiated automatically.  This object contains information about the class from which the objects were instantiated.

(Note that it is also possible to cause a Class object that describes a specific class to be created in the absence of objects of that class, but that is a topic that will be reserved for more advanced lessons.)A real-world analogy

Here is an attempt to describe a real-world analogy.  Remember that a class definition contains the blueprint for objects instantiated from that class.

A certain large construction company is in the business of building condominium projects.  This contractor builds condos of many different sizes, types, and price ranges.  However, each different condo project contains condos of only two or three different types or price ranges.

A library of blueprints

There is a large library of blueprints at the contractor's central office.  This library contains blueprints for all of the different types of condos that the contractor has built or is building.  (This library is analogous to the class libraries available to the Java programmer.)

A subset from the blueprint library

When a condo project begins, the contractor delivers copies of several sets of blueprints to the construction site.  The blueprints delivered to that site describe only the types of condos being constructed on that site.

Condo is analogous to an object

Each condo unit is analogous to an ordinary Java object.

Each set of blueprints delivered to the construction site is roughly analogous to an object of the class named Class.  In other words, each set of blueprints describes one or more condo units constructed from that set of blueprints.

When construction is complete

When the construction project is complete, the contractor delivers a set of blueprints for each type of condo unit to the management firm that has been hired to manage the condo complex.  Each set of blueprints continues to be analogous to an object of the class named Class.  The blueprints remain at the site of the condo units.

RTTI

Thus, information regarding the construction, wiring, plumbing, air conditioning, etc., for each condo unit (object) continues to be available at the site even after the construction has been completed.

(This is somewhat analogous to something called runtime type information and often abbreviated as RTTI.  A Class object contains RTTI for objects instantiated from that class.)What are those analogies again?

In the above scenario, each condo unit is (roughly) analogous to an object instantiated from a specific class (set of blueprints).

Each set of blueprints remaining onsite after construction is complete is roughly analogous to a Class object that describes the characteristics of one or more condo units.

What do you care?

Until you get involved in such advanced topics as reflection and introspection, you don't usually have much involvement or much interest in Class objects.  They are created automatically, and are primarily used by the Java virtual machine during runtime to help it do the things that it needs to do.

An exception to that rule

However, there is one area where you will be interested in the use of these Class objects from early on.  You will be interested whenever variables or methods in the class definition are declared to be static.

Class variables and methods

According to the current jargon, declaring a member variable to be static causes it to be a class variable.

(Note that local variables cannot be declared static.  Only member variables can be declared static.)Similarly, declaring a method to be static causes it to be a class method.

Instance variables and methods

On the other hand, according to the current jargon, not declaring a variable to be static causes it to be an instance variable, and not declaring a method to be static causes it to be an instance method.

In general, we can refer to them as class members and instance members.

What is the difference?

Here are some of the differences between class and instance members insofar as this discussion is concerned.

How many copies of member variables exist?

Every object instantiated from a given class has its own copy of each instance variable defined in the class.

(Instance variables are not shared among objects.)However, every object instantiated from a given class shares the same copy of each class variable defined in the class. (It is as though the class variable belongs to the single Class object and not to the individual objects instantiated from that class.)Access to an instance variable

Every object has its own copy of each instance variable (the object owns the instance variable).  Therefore, the only way that you can access an instance variable is to use that object's reference to send a message to the object requesting access to the variable (even then, you may not be given access, depending on access modifiers).

Why call it an instance variable?

According to the current jargon, an object is an instance of a class.

(I probably told you that somewhere before in this miniseries.  At least, I hope that I did.  After writing eight or ten lessons in a miniseries, I sometimes forget what I told you before.)Each object has its own copy of each non-static variable.  Hence, they are often called instance variables.  (Every instance of the class has one.)

Access to a class variable

You can also send a message to an object requesting access to a class variable that the object shares with other objects instantiated from the same class.  (Again, you may or may not gain access, depending the access modifiers).

Access using the Class object

More importantly, you can also access a class variable without a requirement to go through an object instantiated from the class.  (In fact, a class variable can be accessed in the total absence of objects of that class.)

(Remember, this discussion is conceptual in nature, and may not represent an actual implementation.)Assuming that a class variable is otherwise accessible, you can access the class variable by sending an access request message to the Class object to which the variable belongs.

One way to think of this

To help you keep track of thing in a message-passing sense, you can pretend that there is a global reference variable whose name is the same as the name of a class.

This (hypothetical) reference variable contains a reference to the Class object that owns the class variable.  Using standard Java message-passing syntax, you can access the class variable by joining the name of the reference variable to the name of the class variable with a period.  Example syntax is shown below:

ReferenceVariableName.ClassVariableName

As a result of the hypothetical substitution process that I described above, this is equivalent to the following:

ClassName.ClassVariableName

We will see an example of this in the sample program that I will discuss later.

Be careful with this thought process

While this thought process may be useful when thinking about static variables and methods, I want to point out, that the thought process breaks down very quickly when dealing with Class objects in a deeper sense.

For example, when invoking the getName method on a Class object, an actual reference of type Class is required to access the members of the Class object.  The name of the class will not suffice.

If this discussion of a global reference variable whose name matches the name of the class is confusing to you, just forget it.  Simply remember that you can access class variables by joining the name of the class to the name of the class variable using a period as the joining operator.

Characteristics of class methods

I'm not going to talk very much about instance methods and class methods in this lesson.  However, there are a couple of characteristics of class methods that deserve a brief discussion in this context.

Cannot access instance members

First, the code in a class method has direct access only to other static members of the class.

(A class method does not have direct access to instance variables or instance methods of the class.)This is sort of like saying that a class method has access to the methods and variables belonging to the Class object, but does not have access to the methods and variables belonging to the ordinary objects instantiated from the class described by the Class object. Once again, be careful

Once again, this thinking breaks down very quickly once you get beyond static members.  A Class object also has instance methods, such as getName, which can only be accessed using an actual reference to the Class object.

Now you are probably beginning to understand why I deferred this discussion until after I finished discussing the easy stuff.

No object required

Another important characteristic is that a class method can be accessed without a requirement for an object of the class to exist.

As with class variables, class methods can be accessed by joining the name of the class to the name of the method with a period.

I will illustrate much of this with a sample program named MyClass01.

Discuss in fragments

I will discuss the program in fragments.  You will find a complete listing of the program in Listing 13 near the end of the lesson.

Listing 1 shows the beginning of the class definition.
 

class MyClass01{   static Date v1 = new Date();   Date v2 = new Date(); Listing 1

Two member variables

The code in Listing 1 declares two member variables, named v1 and v2, and initializes each of those variables with a reference to a new object of the Date class.

(When instantiated using the constructor with no arguments, the new Date object encapsulates the current date and time from the system clock.)Note the static keyword

The important thing to note here is the use of the static keyword when declaring the variable named v1.  This causes v1 to be a class variable, exhibiting the characteristics of class variables described earlier.

An instance variable

On the other hand, the variable named v2 is not declared static.  This causes it to be an instance variable, as described above.

The main method is a class method

Listing 2 shows the signature for the main method.
 

  public static void main(                         String[] args){ Listing 2

The important thing to note here is that the main method is declared static.  That causes it to be a class method.

As a result, the main method can be invoked without a requirement for an object of the class to exist.

(Also, the main method has direct access only to other static members.)How a Java application starts running

In fact, that is how the Java Virtual Machine starts an application running.

First the JVM finds the specified file having an extension of .class.  Then it examines that file to see if it has a main method with the correct signature.  If not, an error occurs.

If the JVM finds a main method with the correct signature, it invokes that method without instantiating an object of the class.  That is how the Java Virtual Machine causes a Java application to start running.

A side note regarding applets
For those of you who are familiar with Java applets, you should know that this is not the case for an applet.  You should know that an applet does not use a main method.  When an applet is started, an object of the controlling class is instantiated by the browser, by the appletviewer program, or by whatever program is being used to control the execution of the applet.A poor programming technique

Basically, this entire sample program is coded inside the main method.  As a practical manner, that is a very poor programming technique, but it works well for this example.

Display some text

The code in Listing 3, which is the first executable statement in the main method, causes the words Static variable to appear on the computer screen.  I will come back and discuss the details of this and similar statements later in the lesson.
 

    System.out.println(                     "Static variable"); Listing 3

Display date information

Proceeding down through the code in the main method, the code in Listing 4 causes the current contents of the Date object referred to by the contents of the class variable named v1 to be displayed on the computer screen.
 

    System.out.println(MyClass01.v1); Listing 4

No object required

For the moment, concentrate on the boldface text in the statement in Listing 4.

IMPORTANT:  Because the variable named v1 is a class variable, it's value is accessed by joining the name of the class to the name of the variable with a period.

What was the output?

I will also discuss the remaining portion of statements of this sort later.  For now, just be aware that it caused the output shown in Figure 1 to be displayed on my computer screen when I ran the program.
 

Mon Sep 17 09:52:27 CDT 2001 Figure 1

Displays date and time

Obviously, the date and time displayed will depend on when you run the program.

Pay particular attention to the seconds portion of the time.  I will refer back to this later.

A five-second delay

The code in Listing 5 (still in the main method) causes the main thread of the program to go to sleep for five seconds.  Don't worry about it if you don't understand this code.  The only reason that I included it in the program was to force a five-second delay in the execution of the program.
 

  try{     Thread.currentThread().sleep(5000);   }catch(InterruptedException e){} Listing 5

Instantiate a new object

Having caused the program to sleep for five seconds, the code in Listing 6 instantiates a new object of the class named MyClass01.  The code stores the new object's reference in the reference variable named ref1.
 

    MyClass01 ref1 = new MyClass01(); Listing 6

A new Date object also

Recall from Listing 1 above that the class declares an instance variable named v2 of the type Date.

When the new object is instantiated by the code in Listing 6, a new Date object is also instantiated.  A reference to that object is stored in the instance variable named v2.

(In other words, the new object of the class MyClass01 owns a reference to a new object of the class Date.  That reference is stored in an instance variable named v2 in the new MyClass01 object.)Display the new Date object

The code in Listing 7 causes a textual representation of the new Date object referred to by the reference variable named v2 belonging to the object referred to by the reference variable named ref1, to be displayed on the standard output device.
 

    System.out.println(ref1.v2); Listing 7

Five seconds later

This code caused the date and time shown in Figure 2 to appear on the computer screen when I ran the program:
 

Mon Sep 17 09:52:32 CDT 2001 Figure 2

If you note the time in the above output, you will see that it is five seconds later than the time reflected in the Date object referred to by the class variable named v1.  That time was displayed by the code in Listing 4 earlier.

So, what does this mean?

It means that the Date object referred to by the static reference variable named v1 was created five seconds earlier than the Date object referred to by the instance variable named v2.

When is a class variable created?

I can't tell you precisely when a class variable comes into existence.  All I can say is that the virtual machine brings it into existence as soon as it is needed.

My guess is that it comes into existence at the first mention (in the program) of the class to which it belongs.

When is an instance variable created?

An instance variable doesn't come into existence until the object to which it belongs is created (an instance variable cannot exist until the object to which it belongs exists).

If the instance variable is initialized with a reference to a new object (such as a new Date object in this sample program), that new object comes into existence when the object to which it belongs comes into existence.

A five-second delay

In this program, I purposely inserted a five-second delay between the first mention of the class named MyClass01 in Listing 4, and the instantiation of the object of the class named MyClass01 in Listing 6.

As a result, the Date object referred to by the instance variable named v2 was created about five seconds later than the Date object referred to by the class variable named v1.

This is reflected in the date and time values displayed and discussed earlier.

Accessing class variable via an object

While it is possible to access a class variable using the name of the class joined to the name of the variable, it is also possible to access a class variable using a reference to any object instantiated from the class.

(As mentioned earlier, if two or more objects are instantiated from the same class, they share the same class variable.)The boldface code in Listing 8 uses the reference variable named ref1 to access the class variable named v1, and to cause the contents of the Date object referred to by the class variable to be displayed.
 

    System.out.println(ref1.v1); Listing 8

The output

This caused the date and time shown in Figure 3 to be displayed on my computer screen.
 

Mon Sep 17 09:52:27 CDT 2001 Figure 3

Same date and time

As you have probably already surmised, this is the same date and time shown earlier in Figure 1.  This is because the code in Listing 8 refers to the same class variable as the code in Listing 4.  Nothing has caused the contents of that class variable to change, so both Figure 1 and Figure 3 display the contents of the same Date object.

(Only one class variable exists and it doesn't matter how you access it.  Either way, you gain access to the same Date object whose reference is stored in the class variable.  Thus, the same date and time is shown in both cases.)Another new object

If you examine the code in Listing 13 near the end of the program, you will see that an additional five-second delay is introduced at this point in the program.

Following that delay, the code in Listing 9 instantiates another new object of the class named MyClass01, and stores the object's reference in a new reference variable named ref2.

(The object referred to by ref1 is a different object than the object referred to by ref2.  Each object has its own instance variable named v2, and in this case, each instance variable is initialized to instantiate and refer to a new Date object when the new MyClass01 object is instantiated.)

    MyClass01 ref2 = new MyClass01(); Listing 9

Display the date and time

Then, the code in Listing 10 causes the contents of the Date object referred to by the instance variable named v2 in the second object of the class named MyClass01 to be displayed.
 

    System.out.println(ref2.v2); Listing 10

This caused the output shown in Figure 4 to be displayed on my computer screen when I ran the program (again, you will get different results if you compile and run the program because the date and time shown is the date and time that you run the program).
 

Mon Sep 17 09:52:37 CDT 2001 Figure 4

Five seconds later

As you have probably figured out by now, the time encapsulated in this Date object is five seconds later than the time encapsulated in the Date object displayed in Figure 2.  This is because the program was put to sleep for five seconds between the instantiation of the two objects referred to by ref1 and ref2.

Every object has one

Every object instantiated from a given class has its own copy of each instance variable declared in the class definition.  There is no sharing of instance variables among objects.

Each instance variable comes into existence when the object to which it belongs comes into existence, and ceases to exist when the object to which it belongs ceases to exist.

(If the instance variables are reference variables holding references to other objects, as is the case here, and if there are no other reference variables holding references to those same objects, the secondary objects cease to exist when the primary objects cease to exist.  Technically, the objects may not cease to exist.  Technically they become eligible for garbage collection, which means that the memory that they occupy becomes eligible for reuse.  However, as a practical matter, they cease to exist insofar as the program is concerned.  They are no longer accessible.)Since the two objects referred to by ref1 and ref2 came into existence with a five-second delay, the Date objects belonging to those two object reflect a five-second difference in the time encapsulated in the objects.

Only one copy of class variable exists

Also remember that if a variable is a class variable, only one copy of the variable exists, and all objects instantiated from the class share that one copy.

This is illustrated by the code in Listing 11, which uses the reference to the second object instantiated from the class named MyClass01, to cause the contents of the class variable named v1 to be displayed.
 

    System.out.println(ref2.v1);   }//end main Listing 11

The output produced by the code in Listing 11 is shown in Figure 5.
 

Mon Sep 17 09:52:27 CDT 2001 Figure 5

Same output as before

As you can see, this is the same as the output shown in Figure 1 and Figure 3 earlier.

Accessing the same physical class variable

Since only one class variable named v1 exists, and all objects instantiated from the class named MyClass01 share that single copy, it doesn't matter whether you access the class variable using the name of the class, or access it using a reference to either of the objects instantiated from the class.  In all three cases, you are accessing the same physical class variable.

Since nothing was done to cause the contents of the class variable to change after it came into existence and was initialized, Figures 1, 3, and 5 are simply three different displays of the date and time encapsulated in the same Date object whose reference is stored in the class variable.

Let's revisit System.out.println...

Now, I want to revisit the statement originally shown in Listing 8 and repeated in Listing 12 for viewing convenience.
 

    System.out.println(ref1.v1); Listing 12

Java programmer wanted

I often tell my students that if I were out in industry, interviewing prospective Java programmers, my first question would be to ask the prospective employee to tell me everything that she knows about the statement in Listing 12.

Covers a lot of Java OOP technology

This is not because there is a great demand for the use of this statement in real-world problems.  (In fact, in a GUI-driven software product world, there is probably very little demand for the use of this statement.)  Rather, it is because a lot of Java object-oriented technology is embodied in this single statement.

In that scenario, I would expect to receive a verbal dissertation of fifteen to twenty minutes in length to cover all the important points.

The short version

Let me give you the short version.  There is a class named System. The System class declares three static (class) variables having the following types, names, and modifiers:

public static final PrintStream out public static final InputStream in public static final PrintStream err (Note that these class variables are also declared final, causing them to behave as constants.)Access the out variable without an object

Because out is a class variable, System.out returns the contents of the class variable named out (an object of the System class is not required in order to access a class variable of the System class).

In general, (ignoring the possibility of subclasses and interfaces) because out is a reference variable of type PrintStream, the returned value must either be null (no object reference) or a reference to a valid PrintStream object.

Object of the PrintStream class

When the Java Virtual Machine starts an application running, it instantiates an object of the PrintStream class and connects it to the standard output device.

(By default, the standard output device is typically the computer screen, but it can be redirected at the operating system level to be some other device.  The following discussion assumes that the screen is the standard output device.)Assign object's reference to out variable

When the PrintStream object is instantiated by the virtual machine, the object's reference is assigned to the class variable of the System class named out.

(Because the variable named out is final, the contents of the variable cannot be modified later.)Reference to a PrintStream object

Therefore, the expression System.out returns a reference to the PrintStream object, which is connected to the standard output device.

Many instance methods

An object of the PrintStream class contains many instance methods.  This includes numerous overloaded versions of a method named println.  The signature of one of those overloaded versions of the println method follows:

public void println(Object x)

Textual representation of an object

The purpose of this overloaded version of the println method is to:

Create a textual representation of the object referred to by the incoming parameter of type Object (because Object is a totally generic type, this version of the println method can accept an incoming parameter that is a reference to any type of object) Send that textual representation to the output device In general ...
(In general, a new PrintStream object can be connected to a variety of output devices when it is instantiated.  However, in the special case of the PrintStream object instantiated by the virtual machine when the program starts, whose reference is stored in the class variable named out of the System class, the purpose of the object is to provide a display path to the standard output device.)Our old friend, the toString method

To accomplish this, the code in the println method invokes the toString method on the incoming reference.

(I discussed the toString method in detail in earlier lessons in this miniseries.)The toString method may, or may not, have been overridden in the definition of the class from which the object was instantiated, or in some superclass of the class from which the object was instantiated.

Default version of toString

If not overridden, the default version of the toString method defined in the Object class is used to produce a textual representation of the object.  As we learned in an earlier lesson, that textual representation looks something like the following:

ClassName@HexHashCode

Overridden version of toString method

If the class from which the object was instantiated (or some superclass of that class) contains an overridden version of the toString method, runtime polymorphism kicks in and the overridden version of the method is executed to produce the textual representation of the object.

The Date class overrides toString

In the case of this sample program, the object was instantiated from the Date class.  The Date class does override the toString method.

When the overridden toString method is invoked on a Date object's reference, the String returned by the method looks something like that shown in Figure 6.
 

Mon Sep 17 09:52:27 CDT 2001 Figure 6

You will recall that this is the output that was produced by the code shown in Listing 8 and Listing 12.

More than you ever wanted to know ...

And that is probably more than you ever wanted to know about the expression System.out.println....

It is also probably more than you ever wanted to know about class variables, class methods, instance variables, and instance methods.

Some cautions

Before leaving this topic, I do want to express some cautions.  Basically, I want to suggest that you use static members very sparingly, if at all.  Several good sets of guidelines have been published regarding the use of static members.  As of this writing, you can find one of them, written by Mark L. Fussell, at the following URL.

Guidelines for use of static members

I will try to summarize the general sense of most published guidelines in a few words.

Static variables

To begin with, don't ever use static variables without declaring them final unless you understand exactly why you are declaring them static.

(Static final variables, or constants, are often very appropriate.  See the fields in the Color class for example.)I can only think of a few situations where the use of a non-final static variable might be appropriate. (One appropriate use might be to count the number of objects instantiated from a specific class.  I suspect there are a few other appropriate uses as well.)Static methods

Don't declare methods static if there is any requirement for the method to remember anything from one invocation to the next.

There are many appropriate uses for static methods, but in most cases, the purpose of the method will be to completely perform some action with no requirement to remember anything from that invocation to the next.

The method should probably also be self-contained.  By this I mean that all information that the method needs to do its job should either come from incoming parameters or from final static member variables (constants).  The method probably should not depend on the values stored in non-final static member variables, which are subject to change over time.

(A static method only has access to other static members of the class, so it cannot depend on instance variables defined in the class.)An appropriate example of a static method is the sqrt method of the Math class.  This method computes and "Returns the correctly rounded positive square root of a double" where the double value is provided as a parameter to the method.  Each time the method is invoked, it completes its task and doesn't attempt to save any values from that invocation to the next.  Furthermore, it gets all the information that it needs to do its job from an incoming parameter. SummaryAdded complexity

The existence of static members tends to break up the simple OOP structures that I have discussed in previous lessons in this miniseries.

While static members can be useful in some situations, the existence of static members tends to complicate the overall object-oriented structure of Java.

Furthermore, the overuse of static members can lead to problems similar to those experienced in languages like C and C++ that support global variables and global functions.

The class named Class

I discussed the class named Class and how a conceptual object of type Class exists in memory following a reference to a specific class in the program code.

The Class object represents the referenced class in memory, and contains the static variables and static methods belonging to that class. (It contains some other information as well, such as the name of the superclass.)

Class members and instance members

Class variables and class methods are declared static (declaring a member static in the class definition causes to be called a class member).

Instance variables and instance methods are not declared static.

Each object has its own copy ...

Every object instantiated from a given class has its own copy of each instance variable declared in the class definition.  (Instance variables are not shared among objects.)

Every object instantiated from a given class acts like it has its own copy of every instance method declared in the class definition.  (Although instance methods are actually shared among objects in order to reduce the amount of memory required, they are shared in such a way that they don't appear to be shared.)

Every object shares ...

Every object instantiated from a given class shares the same single copy of each class variable declared in the class definition.  Similarly, every object instantiated from a given class shares the same copy of each class method.

Accessing an instance member

An instance variable or an instance method can only be accessed by using a reference to the object that owns it.  Even then, it may or may not be accessible depending on the access modifier assigned by the programmer.

Accessing a class member

The single shared copy of a class variable or a class method can be accessed in either of two ways:

Via a reference to any object instantiated from the class By simply joining the name of the class to the name of the class variable or the class method Again, the variable or method may or may not be accessible, depending on the access modifiers assigned by the programmer.

When to use class variables

It is very often appropriate to use final static variables, as constants in your programs.  It is rarely, if ever, appropriate to use non-final static variables in your programs.  The use of non-final static variables should definitely be minimized.

When to use static methods

It is often appropriate to use static methods in your programs, provided there is no requirement for the method to remember anything from one invocation to the next.  Static methods should be self-contained.

What's Next?The next lesson in this miniseries will address the special case of Array Objects. Complete Program ListingA complete listing of the sample program is shown in Listing 13.
 

/*File MyClass01.java Copyright 2002, R.G.Baldwin This program illustrates static  members of a class.  Output is:    Static variable Mon Sep 17 09:52:27 CDT 2001 Instance variable Mon Sep 17 09:52:32 CDT 2001 Static variable Mon Sep 17 09:52:27 CDT 2001 Instance variable Mon Sep 17 09:52:37 CDT 2001 Static variable Mon Sep 17 09:52:27 CDT 2001 **************************************/ import java.util.Date; class MyClass01{   static Date v1 = new Date();   Date v2 = new Date();   public static void main(                         String[] args){     //Display static variable     System.out.println(                     "Static variable");     System.out.println(MyClass01.v1);          //Delay for five seconds     try{     Thread.currentThread().sleep(5000);     }catch(InterruptedException e){}          //Instantiate an object and      // display instance variable     MyClass01 ref1 = new MyClass01();     System.out.println();//blank line     System.out.println(                   "Instance variable");     System.out.println(ref1.v2);          //Now, display the static     // variable using object reference     System.out.println(                     "Static variable");     System.out.println(ref1.v1);          System.out.println();//blank line          //Delay for five seconds     try{     Thread.currentThread().sleep(5000);     }catch(InterruptedException e){}          //Instantiate another object      MyClass01 ref2 = new MyClass01();     System.out.println();//blank line     System.out.println(                   "Instance variable");     System.out.println(ref2.v2);          //Now, display the same static     // variable using object reference     System.out.println(                     "Static variable");     System.out.println(ref2.v1);   }//end main }//end class MyClass01 //===================================// Listing 13

本文地址:http://com.8s8s.com/it/it9767.htm