static member variables

来源:互联网 发布:php自动加载机制 编辑:程序博客网 时间:2024/04/30 08:31

In the previous lesson on static member variables, you learned that classes can have member variables that are shared across all objects of that class type. However, what if our static member variables are private? Consider the following example:

1
2
3
4
5
6
7
8
9
10
11
12
13
classSomething
{
private:
    staticint s_nValue;
 
};
 
intSomething::s_nValue = 1; // initializer
 
intmain()
{
    // how do we access Something::s_nValue?
}

In this case, we can’t access Something::s_nValue directly from main(), because it is private. Normally we access private members through public member functions. While we could create a normal public member function to access s_nValue, we’d then need to instantiate an object of the class type to use the function! We can do better. In this case, the answer to the problem is that we can also make member functions static.

Like static member variables, static member functions are not attached to any particular object. Here is the above example with a static member function accessor:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
classSomething
{
private:
    staticint s_nValue;
public:
    staticint GetValue() { returns_nValue; }
};
 
intSomething::s_nValue = 1; // initializer
 
intmain()
{
    std::cout << Something::GetValue() << std::endl;
}

Because static member functions are not attached to a particular object, they can be called directly by using the class name and the scope operator. Like static member variables, they can also be called through objects of the class type, though this is not recommended.

Static member functions have two interesting quirks worth noting. First, because static member functions are not attached to an object, they have no this pointer! This makes sense when you think about it — the this pointer always points to the object that the member function is working on. Static member functions do not work on an object, so the this pointer is not needed.

Second, static member functions can only access static member variables. They can not access non-static member variables. This is because non-static member variables must belong to a class object, and static member functions have no class object to work with!

Here’s another example using static member variables and functions:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
classIDGenerator
{
private:
    staticint s_nNextID;
 
public:
     staticint GetNextID() { returns_nNextID++; }
};
 
// We'll start generating IDs at 1
intIDGenerator::s_nNextID = 1;
 
intmain()
{
    for(inti=0; i < 5; i++)
        cout << "The next ID is: " << IDGenerator::GetNextID() << endl;
 
    return0;
}

This program prints:

The next ID is: 1The next ID is: 2The next ID is: 3The next ID is: 4The next ID is: 5

Note that because all the data and functions in this class are static, we don’t need to instantiate an object of the class to make use of it’s functionality! This class utilizes a static member variable to hold the value of the next ID to be assigned, and provides a static member function to return that ID and increment it.

Be careful when writing classes with all static members like this. Although such “pure static classes” can be useful, they also come with some potential downsides. First, because all of the members belong to the class, and the class is accessible from anywhere in the program, it’s essentially the equivalent of declaring a global variable of the class type. In the section on global variables, you learned that global variables are dangerous because one piece of code can change the value of the global variable and end up breaking another piece of seemingly unrelated code. The same holds true for pure static classes. Second, because all static members are instantiated only once, there is no way to have multiple copies of a pure static class (without cloning the class and renaming it). For example, if you needed two independent IDGenerators, this would not be possible.

0 0