Pass a string by reference in Javascript
JavascriptStringPass by-ReferenceJavascript Problem Overview
I want to create a string and pass it by reference such that I can change a single variable and have that propagate to any other object that references it.
Take this example:
function Report(a, b) {
this.ShowMe = function() { alert(a + " of " + b); }
}
var metric = new String("count");
var a = new Report(metric, "a");
var b = new Report(metric, "b");
var c = new Report(metric, "c");
a.ShowMe(); // outputs: "count of a";
b.ShowMe(); // outputs: "count of b";
c.ShowMe(); // outputs: "count of c";
I want to be able to have this happen:
var metric = new String("count");
var a = new Report(metric, "a");
var b = new Report(metric, "b");
var c = new Report(metric, "c");
a.ShowMe(); // outputs: "count of a";
metric = new String("avg");
b.ShowMe(); // outputs: "avg of b";
c.ShowMe(); // outputs: "avg of c";
Why doesn't this work?
The MDC reference on strings says metric is an object.
I've tried this, which is not what I want, but is very close:
var metric = {toString:function(){ return "count";}};
var a = new Report(metric, "a");
var b = new Report(metric, "b");
var c = new Report(metric, "c");
a.ShowMe(); // outputs: "count of a";
metric.toString = function(){ return "avg";}; // notice I had to change the function
b.ShowMe(); // outputs: "avg of b";
c.ShowMe(); // outputs: "avg of c";
alert(String(metric).charAt(1)); // notice I had to use the String constructor
// I want to be able to call this:
// metric.charAt(1)
The important points here:
- I want to be able to use metric like it's a normal string object
- I want each report to reference the same object.
Javascript Solutions
Solution 1 - Javascript
Strings in Javascript are already passed "by reference" -- calling a procedure with a string does not involve copying the string's contents. There are two issues at hand:
- Strings are immutable. In contrast to C++ strings, once a JavaScript string has been created it cannot be modified.
- In JavaScript, variables are not statically assigned slots like in C++. In your code,
metric
is a label which applies to two entirely separate string variables.
Here's one way to achieve what you want, using closures to implement dynamic scoping of metric
:
function Report(a, b) {
this.ShowMe = function() { alert(a() + " of " + b); }
}
var metric = "count";
var metric_fnc = function() { return metric; }
var a = new Report(metric_fnc, "a");
var b = new Report(metric_fnc, "b");
a.ShowMe(); // outputs: "count of a";
metric = "avg";
b.ShowMe(); // outputs: "avg of b";
Solution 2 - Javascript
You can wrap the string in an object and modify the field the string is stored in. This is similar to what you are doing in the last example only without needing to change the functions.
var metric = { str : "count" }
metric.str = "avg";
Now metric.str will contain "avg"
Solution 3 - Javascript
Closure?
var metric = new function() {
var _value = "count";
this.setValue = function(s) { _value = s; };
this.toString = function() { return _value; };
};
// snip ...
a.ShowMe();
metric.setValue("avg");
b.ShowMe();
c.ShowMe();
or making it a little more generic and performant:
function RefString(s) {
this.value = s;
}
RefString.prototype.toString = function() { return this.value; }
RefString.prototype.charAt = String.prototype.charAt;
var metric = new RefString("count");
// snip ...
a.ShowMe();
metric.value = "avg";
b.ShowMe();
c.ShowMe();
If you don't close on the desired string variable, then I suppose the only other way would be to modify the ShowMe function, as in @John Millikin's answer or re-architect the codebase.
Solution 4 - Javascript
If you pass the variable as an object it will work, since objects are passed by reference in Javascript.
http://sirdarckcat.blogspot.com/2007/07/passing-reference-to-javascript.html
function modifyVar(obj,newVal){
obj.value=newVal;
}
var m={value: 1};
alert(x);
modifyVar("x",321);
alert(x);
Solution 5 - Javascript
Jsfiddle: https://jsfiddle.net/ncwhmty7/1/
Both string primitives (String Literal) and String objects are immutable. This means that any changes to the content of a string variable while in the function are completely separate from anything that happens outside the function. There are couple of options to overcome this:
1. Returning the modified function value form the function
function concatenateString(stringA, stringB) {
return stringA + stringB;
}
alert(concatenateString("Hi", " there"));
2. To convert the string variable to a true Object
function modifyVar(stringA, stringB) {
var result = stringA + stringB;
stringA.valueOf = stringA.toSource = stringA.toString = function() {
return result;
};
}
var stringA = Object('HI');
modifyVar(stringA, ' there');
alert(stringA);
Solution 6 - Javascript
If that string is a property of some object you can send the object and stringKey of the property:
this.modificationFunction = function(object, stringKey){
object[stringKey] = object[stringKey] + "bla bla bla";
}
myObject.myStringProperty = "She said: ";
this.modificationFunction(myObject, "myStringProperty");
// myObject.myStringProperty is now "She said: bla bla bla";
Solution 7 - Javascript
In JavaScript, strings are immutable. You can't change the string itself one the Report
instances have a handle to it.
your solution works, but this may be simpler:
function Report(a, b) {
this.showMe = function() { alert(a.str + " of " + b); }
}
var metric = {};
metric.str = "count";
a.Showme();
metric.str = "avg";
b.ShowMe();