What does this mean? How do I change a parameter? This short tutorial will help you figure out how parameter passing in Java works and will help you avoid some common mistakes.
First let's get the terminology straight. The terms "arguments" and "parameters" are used interchangeably; they mean the same thing. We use the term formal parameters to refer to the parameters in the definition of the method. In the example that follows, x and y are the formal parameters.
We use the term actual parameters to refer to the variables we use in the method call. In the following example, length and width are actual parameters.
// Method definition public int mult(int x, int y) { return x * y; } // Where the method mult is used int length = 10; int width = 5; int area = mult(length, width);Pass-by-value means that when you call a method, a copy of each actual parameter (argument) is passed. You can change that copy inside the method, but this will have no effect on the actual parameter. Unlike many other languages, Java has no mechanism to change the value of an actual parameter. Isn't this very restrictive? Not really. In Java, we can pass a reference to an object (also called a "handle")as a parameter. We can then change something inside the object; we just can't change what object the handle refers to.
public static void tryPrimitives(int i, double f, char c, boolean test) { i += 10; //This is legal, but the new values c = 'z'; //won't be seen outside tryPrimitives. if(test) test = false; else test = true; }If tryPrimitives is called within the following code what will the final print statement produce?
int ii = 1; double ff = 1.0; char cc = 'a'; boolean bb = false; tryPrimitives(ii, ff, cc, bb); System.out.println("ii = " + ii + ", ff = " + ff + ", cc = " + cc + ", bb = " + bb);
What's really
happening here? When Java calls a method, it makes a copy of its actual
parameters and sends the copies to the method where they become the formal
parameters. Then when the method returns, those copies are discarded and
the variables in the main code are the same as before.
Suppose we have defined the following class:
class Record { int num; String name; }Now we can pass the object as a parameter to a method:
public static void tryObject(Record r) { r.num = 100; r.name = "Fred"; }In some other code we can create an object of our new class Record, set its fields, and call the method tryObject.
Record id = new Record(); id.num = 2; id.name = "Barney"; tryObject(id); System.out.println(id.name + " " + id.num);The print statement prints out "Fred 100".
Note that the object's instances variables are changed in this case. Why? The reference to id is the argument to the method, so the method cannot be used to change that reference; i.e., it can't make id reference a different Record. But the method can use the reference to perform any allowed operation on the Record that it already references.
Side Note: It is often not good programming style to change the values of instance variables outside the object. Normally, the object would have a method to set the values of its instance variables.
We cannot however make the object parameter refer to a different object by reassigning the reference or calling new on the reference. For example the following method would not work as expected:
public void createRecord(Record r, int n, String name) { r = new Record(); r.num = n; r.name = name; }We can still encapsulate the initialization of the Record in a method, but we need to return the reference.
public Record createRecord(int n, String name) { Record r = new Record(); r.num = n; r.name = name; return r; }
String str = "This is a string literal.";
Because str is an object we might think that the string it contains might be changed when we pass str as a parameter to a method. Suppose we have the method tryString:
public static void tryString(String s) { s = "a different string"; }When the following code is executed, what does the print statement produce?
tryString(str); System.out.println("str = " + str);
Why is this? Maybe this picture
will help you understand what goes on behind the scenes. It is important
to remember what Java does when it assigns a string literal to an object.
A different String object is created and the reference variable is set
to point to the newly created object. In other words the value of the formal
parameter, s, has changed, but this does not affect the actual
parameter str. In this example, s pointed to the string
object that contains "This is a string literal". After the first statement
of the method executes, s points to the new string, but str
has not changed.
Like other objects, when we pass a string to a method, we can in principle, change things inside the object (although we can't change which string is referenced, as we just saw). However, this capability is not useful with string because strings are "immutable". They cannot be changed because the String class does not provide any methods to modify its local variables.
B: Given the following method:
public void swap(int x, int y) { int temp = x; x = y; y = temp; }what are the values of p and q after the following code executes?
int p = 20; int q = 15; swap(p, q);
We have to use a bit of a trick to write a swap routine that works.
public static int power(int a, int b) { int i; int total = 1; for(i = 0; i < b; i++) total = total * a; return total; }Now when we call the method power, number gets assigned the value that power returns.
int number = 2; int exponent = 4; number = power(number, exponent); System.out.println("New value of number is " + number);Since methods should be designed to be simple and to do one thing, you will often find that returning a value is enough, and that you don't need to change the value of any parameters to the method. Note that this does not help us write a swap method because a method can only return one value, and to write swap we need to change the value of two variables.
public int tryPrimitives(int x, int y) { x = x + 10; y = y + 10; return x; }what are the values of p, q and r after the following code executes?
int p = 1; int q = 2; int r = 5; r = tryPrimitives(p, q);
public static void tryArray(char[] b) { b[0] = 'x'; b[1] = 'y'; b[2] = 'z'; }When the following code is executed, the array a does indeed have the new values in the array.
char[] a = {'a', 'b', 'c'}; tryArray(a); System.out.println("a[0] = " + a[0] + ", a[1] = " + a[1] + ", a[2] =" + a[2]);The print statements produces "a[0] = x, a[1] = y, a[2] = z".
public void differentArray(float[] x) { x = new float[100]; x[0] = 26.9f; }what is the output after the following code is run?
float[] xx = new float[100]; x[0] = 55.8f; differentArray(xx); System.out.println("x[0] = " + x[0]);
That, combined with an understanding of the different between primitives and references and careful tracing will allow you to understand any Java method calls.