Mutator method
In computer science, a mutator method is a method used to control changes to a variable. They are also widely known as setter methods. Often a setter is accompanied by a getter, which returns the value of the private member variable.
The mutator method is most often used in object-oriented programming, in keeping with the principle of encapsulation. According to this principle, member variables of a class are made private to hide and protect them from other code, and can only be modified by a public member function, which takes the desired new value as a parameter, optionally validates it, and modifies the private member variable. Mutator methods can be compared to assignment operator overloading but they typically appear at different levels of the object hierarchy.
Mutator methods may also be used in non-object-oriented environments. In this case, a reference to the variable to be modified is passed to the mutator, along with the new value. In this scenario, the compiler cannot restrict code from bypassing the mutator method and changing the variable directly. The onus falls to the developers to ensure the variable is only modified through the mutator method and not modified directly.
In programming languages that support them, properties offer a convenient alternative without giving up the utility of encapsulation.
In the examples below, a fully implemented mutator method can also validate the input data or take further action such as triggering an event.
Implications
The alternative to defining mutator and accessor methods, or property blocks, is to give the instance variable some visibility other than private and access it directly from outside the objects. Much finer control of access rights can be defined using mutators and accessors. For example, a parameter may be made read-only simply by defining an accessor but not a mutator. The visibility of the two methods may be different; it is often useful for the accessor to be public while the mutator remains protected, package-private or internal.The block where the mutator is defined provides an opportunity for validation or preprocessing of incoming data. If all external access is guaranteed to come through the mutator, then these steps cannot be bypassed. For example, if a date is represented by separate private
year
, month
and day
variables, then incoming dates can be split by the setDate
mutator while for consistency the same private instance variables are accessed by setYear
and setMonth
. In all cases month values outside of 1 - 12 can be rejected by the same code.Accessors conversely allow for synthesis of useful data representations from internal variables while keeping their structure encapsulated and hidden from outside modules. A monetary
getAmount
accessor may build a string from a numeric variable with the number of decimal places defined by a hidden currency
parameter.Modern programming languages often offer the ability to generate the boilerplate for mutators and accessors in a single line—as for example C#'s
public string Name
and Ruby's attr_accessor :name
. In these cases, no code blocks are created for validation, preprocessing or synthesis. These simplified accessors still retain the advantage of encapsulation over simple public instance variables, but it is common that, as system designs progress, the software is maintained and requirements change, the demands on the data become more sophisticated. Many automatic mutators and accessors eventually get replaced by separate blocks of code. The benefit of automatically creating them in the early days of the implementation is that the public interface of the class remains identical whether or not greater sophistication is added, requiring no extensive refactoring if it is. Manipulation of parameters that have mutators and accessors from inside the class where they are defined often requires some additional thought. In the early days of an implementation, when there is little or no additional code in these blocks, it makes no difference if the private instance variable is accessed directly or not. As validation, cross-validation, data integrity checks, preprocessing or other sophistication is added, subtle bugs may appear where some internal access makes use of the newer code while in other places it is bypassed.
Accessor functions can be less efficient than directly fetching or storing data fields due to the extra steps involved, however such functions are often inlined which eliminates the overhead of a function call.
Examples
Assembly
student struct
age dd ?
student ends
.code
student_get_age proc object:DWORD
mov ebx, object
mov eax, student.age
ret
student_get_age endp
student_set_age proc object:DWORD, age:DWORD
mov ebx, object
mov eax, age
mov student.age, eax
ret
student_set_age endp
C
In file student.h:- ifndef _STUDENT_H
- define _STUDENT_H
typedef struct student student;
student *student_new;
void student_delete;
void student_set_age;
int student_get_age;
char *student_get_name;
- endif
In file student.c:
- include
- include
- include "student.h"
student *student_new
void student_delete
void student_set_age
int student_get_age
char *student_get_name
In file main.c:
- include
- include "student.h"
In file Makefile:
all: out.txt; cat $<
out.txt: main;./$< > $@
main: main.o student.o
main.o student.o: student.h
clean: ;$ *.o out.txt main
C++
In file Student.h:- ifndef STUDENT_H
- define STUDENT_H
- include
- endif
In file Student.cpp:
- include "Student.h"
const std::string& Student::name const
void Student::name
C#
This example illustrates the C# idea of properties, which are a special type of class member. Unlike Java, no explicit methods are defined; a public 'property' contains the logic to handle the actions. Note use of the built-in variablevalue
. public class Student
In later C# versions, this example may be abbreviated as follows, without declaring the private variable
name
.public class Student
Using the abbreviated syntax means that the underlying variable is no longer available from inside the class. As a result, the
set
portion of the property must be present for assignment. Access can be restricted with a set
-specific access modifier.public class Student
Common Lisp
In Common Lisp Object System, slot specifications within class definitions may specify any of the:reader
, :writer
and :accessor
options to define reader methods, setter methods and accessor methods. Slots are always directly accessible through their names with the use of with-slots
and slot-value
, and the slot accessor options define specialized methods that use slot-value
.CLOS itself has no notion of properties, although the MetaObject Protocol extension specifies means to access a slot's reader and writer function names, including the ones generated with the
:accessor
option.The following example shows a definition of a student class using these slot options and direct slot access:
; student-name is setf'able
))
;; Example of a calculated property getter
)
))
;; Example of direct slot access within a calculated property setter
self
new-age))
;; The slot accessing options generate methods, thus allowing further method definitions
D
supports a getter and setter function syntax. In version 2 of the language getter and setter class/struct methods should have the@property
attribute.class Student
A
Student
instance can be used like this:auto student = new Student;
student.name = "David"; // same effect as student.name
auto student_name = student.name; // same effect as student.name
Delphi
This is a simple class in Delphi language which illustrates the concept of public property for accessing a private field.interface
type
TStudent = class
strict private
FName: string;
procedure SetName;
public
///
/// Get or set the name of the student.
///
property Name: string read FName write SetName;
end;
//...
implementation
procedure TStudent.SetName;
begin
FName := Value;
end;
end.
Java
In this example of a simple class representing a student with only the name stored, one can see the variable name is private, i.e. only visible from the Student class, and the "setter" and "getter" are public, namely the "getName
" and "setName
" methods.public class Student
JavaScript
In this example constructor-functionStudent
is used to create objects representing a student with only the name stored.function Student
Or :
function Student
Or :
function Student
Student.prototype = ;
Or :
var Student = ;
Or :
function Student
Object.defineProperty;
Actionscript 3.0
package
Objective-C
Using traditional Objective-C 1.0 syntax, with manual reference counting as the one working on GNUstep on Ubuntu 12.04:@interface Student : NSObject
- name;
- setName:name;
@end
@implementation Student
- name
- setName:name
@end
Using newer Objective-C 2.0 syntax as used in Mac OS X 10.6, iOS 4 and Xcode 3.2, generating the same code as described above:
@interface Student : NSObject
@property NSString *name;
@end
@implementation Student
@synthesize name = _name;
@end
And starting with OS X 10.8 and iOS 6, while using Xcode 4.4 and up, syntax can be even simplified:
@interface Student : NSObject
@property NSString *name;
@end
@implementation Student
//Nothing goes here and it's OK.
@end
Perl
package Student;
sub new
sub set_name
sub get_name
1;
Or, using Class::Accessor
package Student;
use base qw;
__PACKAGE__->follow_best_practice;
Student->mk_accessors;
1;
Or, using the Moose Object System:
package Student;
use Moose;
- Moose uses the attribute name as the setter and getter, the reader and writer properties
- allow us to override that and provide our own names, in this case get_name and set_name
1;
PHP
In this example of a simple class representing a student with only the name stored, one can see the variable name is private, i.e. only visible from the Student class, and the "setter" and "getter" is public, namely thegetName
and setName
methods.class Student
Python
This example uses a Python class with one variable, a getter, and a setter.class Student:
# Initializer
def __init__ -> None:
# An instance variable to hold the student's name
self._name = name
# Getter method
@property
def name:
return self._name
# Setter method
@name.setter
def name:
self._name = new_name
>>> bob = Student
>>> bob.name
Bob
>>> bob.name = "Alice"
>>> bob.name
Alice
>>> bob._name = "Charlie" # bypass the setter
>>> bob._name # bypass the getter
Charlie
Racket
In Racket, the object system is a way to organize code that comes in addition to modules and units. As in the rest of the language, the object system has first-class values and lexical scope is used to control access to objects and methods.- lang racket
)
))
; => "Alice"
; => "Bob"
Struct definitions are an alternative way to define new types of values, with mutators being present when explicitly required:
- lang racket
Ruby
In Ruby, individual accessor and mutator methods may be defined, or the metaprogramming constructsattr_reader
or attr_accessor
may be used both to declare a private variable in a class and to provide either read-only or read-write public access to it respectively.Defining individual accessor and mutator methods creates space for pre-processing or validation of the data
class Student
def name
@name
end
def name=
@name=value
end
end
Read-only simple public access to implied
@name
variableclass Student
attr_reader :name
end
Read-write simple public access to implied
@name
variableclass Student
attr_accessor :name
end
Smalltalk
age: aNumber
" Set the receiver age to be aNumber if is greater than 0 and less than 150 "
ifTrue:
Swift
class Student
Visual Basic .NET
This example illustrates the VB.NET idea of properties, which are used in classes. Similar to C#, there is an explicit use of theGet
and Set
methods. Public Class Student
Private _name As String
Public Property Name
Get
Return _name
End Get
Set
_name = value
End Set
End Property
End Class
In VB.NET 2010, Auto Implemented properties can be utilized to create a property without having to use the Get and Set syntax. Note that a hidden variable is created by the compiler, called
_name
, to correspond with the Property name
. Using another variable within the class named _name
would result in an error. Privileged access to the underlying variable is available from within the class.Public Class Student
Public Property name As String
End Class