Const-correctness
Encyclopedia
In computer science
, const-correctness is the form of program correctness that deals with the proper declaration of objects as mutable or immutable
. The term is mostly used in a C
or C++
context, and takes its name from the
The idea of const-ness does not imply that the variable as it is stored in the computer
's memory
is unwriteable. Rather,
In addition, a method
can be declared as
be modified. In C++, a member variable can be declared as
, reference counting
, and data synchronization
. In these cases, the logical meaning (state) of the object is unchanged, but the object is not physically constant since its bitwise representation may change.
To render the syntax for pointers more comprehensible, a rule of thumb
is to read the declaration from right to left. Thus, everything to the left of the star can be identified as the pointee type and everything to the right of the star are the pointer properties. (For instance, in our example above,
References follow similar rules. A declaration of a
Even more complicated declarations can result when using multidimensional arrays and references (or pointers) to pointers; however, some have argued that these are confusing and error-prone and that they therefore should generally be avoided or replaced with higher-level structures.
C/C++ also allows the following syntax:
and C# or in Microsoft
's C++/CLI
or Managed Extensions for C++
. While
This example illustrates:
In the above code, the implicit "
Often the programmer will supply both a
The
.)
The first, which applies only to C++, is the use of
However, any attempt to modify an object that is itself declared
Computer science
Computer science or computing science is the study of the theoretical foundations of information and computation and of practical techniques for their implementation and application in computer systems...
, const-correctness is the form of program correctness that deals with the proper declaration of objects as mutable or immutable
Immutable object
In object-oriented and functional programming, an immutable object is an object whose state cannot be modified after it is created. This is in contrast to a mutable object, which can be modified after it is created...
. The term is mostly used in a C
C (programming language)
C is a general-purpose computer programming language developed between 1969 and 1973 by Dennis Ritchie at the Bell Telephone Laboratories for use with the Unix operating system....
or C++
C++
C++ is a statically typed, free-form, multi-paradigm, compiled, general-purpose programming language. It is regarded as an intermediate-level language, as it comprises a combination of both high-level and low-level language features. It was developed by Bjarne Stroustrup starting in 1979 at Bell...
context, and takes its name from the
const
keyword in those languages.The idea of const-ness does not imply that the variable as it is stored in the computer
Computer
A computer is a programmable machine designed to sequentially and automatically carry out a sequence of arithmetic or logical operations. The particular sequence of operations can be changed readily, allowing the computer to solve more than one kind of problem...
's memory
Computer storage
Computer data storage, often called storage or memory, refers to computer components and recording media that retain digital data. Data storage is one of the core functions and fundamental components of computers....
is unwriteable. Rather,
const
-ness is a compile-time construct that indicates what a programmer should do, not necessarily what they can do. Note however, that in the case of predefined data (such as const char *
string literals), C const
is often unwritable.In addition, a method
Method (computer science)
In object-oriented programming, a method is a subroutine associated with a class. Methods define the behavior to be exhibited by instances of the associated class at program run time...
can be declared as
const
. In this case, the 'this' pointer inside such a method is of const ThisClass* const type rather than of ThisClass* const type. This means that non-const methods for this object cannot be called from inside such a method, nor can member variablesField (computer science)
In computer science, data that has several parts can be divided into fields. Relational databases arrange data as sets of database records, also called rows. Each record consists of several fields; the fields of all records form the columns....
be modified. In C++, a member variable can be declared as
mutable
, indicating that this restriction does not apply to it. In some cases, this can be useful, for example with cachingCache
In computer engineering, a cache is a component that transparently stores data so that future requests for that data can be served faster. The data that is stored within a cache might be values that have been computed earlier or duplicates of original values that are stored elsewhere...
, reference counting
Reference counting
In computer science, reference counting is a technique of storing the number of references, pointers, or handles to a resource such as an object, block of memory, disk space or other resource...
, and data synchronization
Data synchronization
Data synchronization is the process of establishing consistency among data from a source to a target data storage and vice versa and the continuous harmonization of the data over time. It is fundamental to a wide variety of applications, including file synchronization and mobile device...
. In these cases, the logical meaning (state) of the object is unchanged, but the object is not physically constant since its bitwise representation may change.
C++ syntax
In C++, all data types, including those defined by the user, can be declaredconst
, and const-correctness dictates that all objects should be declared as such unless they need to be modified. Such proactive use of const
makes values "easier to understand, track, and reason about," and it thus increases the readability and comprehensibility of code and makes working in teams and maintaining code simpler because it communicates information about a value's intended use.Simple data types
For simple non-pointer data types, applying theconst
qualifier is straightforward. It can go on either side of the type for historical reasons (that is, const char foo = 'a';
is equivalent to char const foo = 'a';
). On some implementations, using const
on both sides of the type (for instance, const char const
) generates a warning but not an error.Pointers and references
For pointer and reference types, the syntax is slightly more subtle. A pointer object can be declared as aconst
pointer or a pointer to a const
object (or both). A const
pointer cannot be reassigned to point to a different object from the one it is initially assigned, but it can be used to modify the object that it points to (called the "pointee"). Reference variables are thus an alternate syntax for const
pointers. A pointer to a const
object, on the other hand, can be reassigned to point to another object of the same type or of a convertible type, but it cannot be used to modify any object. A const
pointer to a const
object can also be declared and can neither be used to modify the pointee nor be reassigned to point to another object. The following code illustrates these subtleties:To render the syntax for pointers more comprehensible, a rule of thumb
Rule of thumb
A rule of thumb is a principle with broad application that is not intended to be strictly accurate or reliable for every situation. It is an easily learned and easily applied procedure for approximately calculating or recalling some value, or for making some determination...
is to read the declaration from right to left. Thus, everything to the left of the star can be identified as the pointee type and everything to the right of the star are the pointer properties. (For instance, in our example above,
int const *
can be read as a mutable pointer that refers to a non-mutable integer, and int * const
can be read as a non-mutable pointer that refers to a mutable integer.)References follow similar rules. A declaration of a
const
reference is redundant since references can never be made to refer to another object:Even more complicated declarations can result when using multidimensional arrays and references (or pointers) to pointers; however, some have argued that these are confusing and error-prone and that they therefore should generally be avoided or replaced with higher-level structures.
C/C++ also allows the following syntax:
Methods
In order to take advantage of the design-by-contract strategy for user-defined types (structs and classes), which can have methods as well as member data, the programmer must tag instance methods asconst
if they don't modify the object's data members. Applying the const
qualifier to instance methods thus is an essential feature for const-correctness, and is not available in many other object-oriented languages such as JavaJava (programming language)
Java is a programming language originally developed by James Gosling at Sun Microsystems and released in 1995 as a core component of Sun Microsystems' Java platform. The language derives much of its syntax from C and C++ but has a simpler object model and fewer low-level facilities...
and C# or in Microsoft
Microsoft
Microsoft Corporation is an American public multinational corporation headquartered in Redmond, Washington, USA that develops, manufactures, licenses, and supports a wide range of products and services predominantly related to computing through its various product divisions...
's C++/CLI
C++/CLI
C++/CLI is Microsoft's language specification intended to supersede Managed Extensions for C++. It is a complete revision that aims to simplify the older Managed C++ syntax . C++/CLI is standardized by Ecma as ECMA-372...
or Managed Extensions for C++
Managed Extensions for C++
Managed Extensions for C++ or just Managed C++ is a now deprecated Microsoft set of deviations from C++, including grammatical and syntactic extensions, keywords and attributes, to bring the C++ syntax and language to the .NET Framework...
. While
const
methods can be called by const
and non-const
objects alike, non-const
methods can only be invoked by non-const
objects. The const modifier on an instance method applies to the object pointed to by the "thisThis (computer science)In many object-oriented programming languages, this is a keyword that is used in instance methods to refer to the object on which they are working. C++ and languages which derive in style from it generally use this...
" pointer, which is an implicit argument passed to all instance methods. Thus having const methods is a way to apply const-correctness to the implicit "this
" pointer argument just like other arguments.This example illustrates:
In the above code, the implicit "
this
" pointer to Set
has the type "C *const
"; whereas the "this
" pointer to Get
has type "const C *const
", indicating that the method cannot modify its object through the "this
" pointer.Often the programmer will supply both a
const
and a non-const
method with the same name (but possibly quite different uses) in a class to accommodate both types of callers. Consider:The
const
-ness of the calling object determines which version of MyArray::Get
will be invoked and thus whether or not the caller is given a reference with which he can manipulate or only observe the private data in the object. The two methods technically have different signatures because their "this" pointers have different types, allowing the compiler to choose the right one. (Returning a const
reference to an int
, instead of merely returning the int
by value, may be overkill in the second method, but the same technique can be used for arbitrary types, as in the Standard Template LibraryStandard Template Library
The Standard Template Library is a C++ software library which later evolved into the C++ Standard Library. It provides four components called algorithms, containers, functors, and iterators. More specifically, the C++ Standard Library is based on the STL published by SGI. Both include some...
.)
Loopholes to const-correctness
There are several loopholes to pure const-correctness in C and C++. They exist primarily for compatibility with existing code.The first, which applies only to C++, is the use of
const_cast
, which allows the programmer to strip the const
qualifier, making any object modifiable. The necessity of stripping the qualifier arises when using existing code and libraries that cannot be modified but which are not const-correct. For instance, consider this code:However, any attempt to modify an object that is itself declared
const
by means of const_cast
results in undefined behavior according to the ISO C++ Standard. In the example above, if ptr
references a global, local, or member variable declared as const
, or an object allocated on the heap via new const int, the code is only correct if LibraryFunc
really does not modify the value pointed to by ptr
.
Another loophole applies both to C and C++. Specifically, the languages dictate that member pointers and references are "shallow" with respect to the const
-ness of their owners — that is, a containing object that is const
has all const
members except that member pointees (and referees) are still mutable. To illustrate, consider this code:
Although the object s
passed to Foo
is constant, which makes all of its members constant, the pointee accessible through s.ptr
is still modifiable, though this may not be desirable from the standpoint of const
-correctness because s
might solely own the pointee. For this reason, some have argued that the default for member pointers and references should be "deep" const
-ness, which could be overridden by a mutable
qualifier when the pointee is not owned by the container, but this strategy would create compatibility issues with existing code. Thus, for historical reasons, this loophole remains open in C and C++.
The latter loophole can be closed by using a class to hide the pointer behind a const-correct interface, but such classes either don't support the usual copy semantics from a const
object (implying that the containing class cannot be copied by the usual semantics either) or allow other loopholes by permitting the stripping of const
-ness through inadvertent or intentional copying.
Finally, several functions in the C standard libraryC standard libraryThe C Standard Library is the standard library for the programming language C, as specified in the ANSI C standard.. It was developed at the same time as the C POSIX library, which is basically a superset of it...
violate const-correctness, as they accept a const
pointer to a character string and return a non-const
pointer to a part of the same string. strtol
and strchr
are among these functions. Some implementations of the C++ standard library, such as Microsoft's try to close this loophole by providing two overloaded versions of some functions: a "const
" version and a "non-const
" version.
Volatile-correctness
The other qualifier in C and C++, volatile
, indicates that an object may be changed by something external to the program at any time and so must be re-read from memory every time it is accessed. The qualifier is most often found in code that manipulates hardwareHardwareHardware is a general term for equipment such as keys, locks, hinges, latches, handles, wire, chains, plumbing supplies, tools, utensils, cutlery and machine parts. Household hardware is typically sold in hardware stores....
directly (such as in embedded systemEmbedded systemAn embedded system is a computer system designed for specific control functions within a larger system. often with real-time computing constraints. It is embedded as part of a complete device often including hardware and mechanical parts. By contrast, a general-purpose computer, such as a personal...
s and device driverDevice driverIn computing, a device driver or software driver is a computer program allowing higher-level computer programs to interact with a hardware device....
s) and in multithreaded applications (though often used incorrectly in that context; see external links at volatile variableVolatile variableIn computer programming, particularly in the C, C++, C#, and Java programming languages, a variable or object declared with the volatile keyword usually has special properties related to optimization and/or threading...
). It can be used in exactly the same manner as const
in declarations of variables, pointers, references, and member functions, and in fact, volatile
is sometimes used to implement a similar design-by-contract strategy which Andrei AlexandrescuAndrei AlexandrescuAndrei Alexandrescu is a Romanian C++ programmer and author. He is particularly known for his pioneering work on policy-based design implemented via template metaprogramming. These ideas are articulated in his book Modern C++ Design and were first implemented in his programming library, Loki. He...
calls volatile
-correctness, though this is far less common than const
-correctness. The volatile
qualifier also can be stripped by const_cast
, and it can be combined with the const
qualifier as in this sample:
Because hardwareRegister
is volatile
, there is no guarantee that it will hold the same value on two successive reads even though the programmer cannot modify it. The semantics here indicate that the register's value is read-only but not necessarily unchanging.
const
and immutable
in D
In Version 2 of the D programming languageD (programming language)The D programming language is an object-oriented, imperative, multi-paradigm, system programming language created by Walter Bright of Digital Mars. It originated as a re-engineering of C++, but even though it is mainly influenced by that language, it is not a variant of C++...
, two keywords relating to const exist. The immutable
keyword denotes data that cannot be modified through any reference. The const
keyword denotes a non-mutable view of mutable data. Unlike C++ const
, D const
and immutable
are "deep" or transitiveTransitive relationIn mathematics, a binary relation R over a set X is transitive if whenever an element a is related to an element b, and b is in turn related to an element c, then a is also related to c....
, and anything reachable through a const
or immutable
object is const
or immutable
respectively.
Example of const vs. immutable in D
Example of transitive or deep const in D
final
in Java
In JavaJava (programming language)Java is a programming language originally developed by James Gosling at Sun Microsystems and released in 1995 as a core component of Sun Microsystems' Java platform. The language derives much of its syntax from C and C++ but has a simpler object model and fewer low-level facilities...
, the qualifier final
states that the affected data member or variable is not assignable, as below:
It must be decidable by the compilers where the variable with the final
marker is initialized, and it must be performed only once, or the class will not compile. Java's final
and C++'s const
keywords have the same meaning when applied with primitive variables.
Considering pointers, a final
reference in Java means something similar to const
pointer in C++. In C++, one can declare a "const
pointer type".
Here, bar
must be initialised at the time of declaration and cannot be changed again, but what it points is modifiable. I.e. *bar = value
is valid. It just can't point to another location. Final reference in Java work the same way except it can be declared uninitialized.
Note: Java doesn't support pointers.
One can also declare a "read-only" pointer in C++.
Here bar
can be modified to point anything, anytime; just that its value cannot be modified through bar
.
There is no equivalent mechanism in Java. Thus there are also no const
methods. Const-correctness cannot be enforced in Java, although by use of interfaces and defining a read-only interface to the class and passing this around, one can ensure that objects can be passed around the system in a way that they cannot be modified. Java collections frameworkJava collections frameworkThe Java collections framework is a set of classes and interfaces that implement commonly reusable collection data structures.Although it is a framework, it works in a manner of a library...
provides a way to create unmodifiable wrapper of a Collection
via Collections.unmodifiableCollection
and similar methods.
Methods in Java can be declared "final
", but that has a completely unrelated meaning - it means that the method cannot be overridden in subclasses.
Interestingly, the Java language specification regards const
as a reserved keyword — i.e., one that cannot be used as variable identifier — but assigns no semantics to it. It is thought that the reservation of the keyword occurred to allow for an extension of the Java language to include C++-style const
methods and pointer to const
type. An enhancement request ticket for implementing const
correctness exists in the Java Community ProcessJava Community ProcessThe Java Community Process or JCP, established in 1998, is a formalized process that allows interested parties to get involved in the definition of future versions and features of the Java platform....
, but was closed in 2005 on the basis that it was impossible to implement in a backwards-compatible fashion.
const
and readonly
in C#
In C#, the qualifier readonly
has the same effect on data members that final
does in Java and the const
does in C++; The const
modifier in C# has an effect similar (yet typed and class-scoped) to that of #define
in C++. (The other, inheritance-inhibiting effect of Java's final
when applied to methods and classes is induced in C# with the aid of a third keyword, sealed
.)
Unlike C++, C# does not permit methods and parameters to be marked as const
. However one may also pass around read-only subclasses, and the .NET Framework.NET FrameworkThe .NET Framework is a software framework that runs primarily on Microsoft Windows. It includes a large library and supports several programming languages which allows language interoperability...
provides some support for converting mutable collections to immutable ones which may be passed as read-only wrappers.
External links
- "Const-Correctness" by Herb SutterHerb SutterHerb Sutter is a prominent C++ expert. He is also a book author and a columnist for Dr. Dobb's Journal. He joined Microsoft in 2002 as a platform evangelist for Visual C++ .NET, rising to lead software architect for C++/CLI. Sutter served as secretary and convener of the ISO C++ standards committee...
- "Constant Optimization?" by Herb Sutter
- "Const-Correctness Tutorial" by Cprogramming.com
- The C++ FAQ Lite: Const correctness by Marshall Cline
- Section "Value substitution" from the free electronic book Thinking in C++ by Bruce EckelBruce EckelBruce Eckel is the author of numerous books and articles about computer programming. He also gives frequent lectures and seminars for computer programmers...
- "Here A Const, There A Const" by Walter BrightWalter BrightWalter Bright is a computer programmer known for being the designer of the D programming language. He was also the main developer of the first C++ compiler that translated directly to object without going via C, Zortech C++ . Before the C++ compiler, he developed the Datalight C compiler, also...
- "Const and Invariant" from D programming language specification, version 2 (experimental)
- Some notes on const correctness in straight C in "C Pointer Qualifiers" by Thomas Stover