I\'ve been using javascript for a while, but have never learned the language past the basics. I am reading John Resig\'s \"Pro Javascript Techniques\" - I\'m coming up with
Your example #1 shows the usage of the prototype property. This property is available to all javascript objects you create and allows you to add properties or functions to the object declaration, so you had a object with 2 properties and later on you added 4 functions (getters and setters).
You should see the prototype property as the way to modify your object specification at runtime, say you have an object called name:
var Name = {
First: "",
Last: ""
};
You can use the prototype to add a function getFullName() later on by simply:
Name.prototype.getFullName = function() { return this.First + " " + this.Last; }
In the example 2 you inline the declaration of these getters and setters in the object declaration so in the end they are the same. Finally on the 3rd example you use the JavaScript object notation you should see JSON.
About your question 2 you can just declare your object as:
var User = {
name: "",
age: 0
};
this will give you the same object without getters and setters.
If you interested on JSON style JavaScript class declaration talks...
http://mahtonu.wordpress.com/2010/04/13/json-style-javascript-object-declaration/
Every time a function() {} is evaluated, it creates a new function object. Therefore, in #1 all of the User objects are sharing the same getName and getAge functions, but in #2 and #3, each object has its own copy of getName and getAge. All of the different getName functions all behave exactly the same, so you can't see any difference in the output.
The {...} shorthand is a constructor. When evaluated, it constructs a new "Object" with the given properties. When you run "new User(...)", it constructs a new "User". You happen to have created an Object with the same behavior as a User, but they are of different types.
Response to comment:
You can't, directly. You could make a function that creates a new object as per #3. For example:
function make_user(name, age) {
return {
name: name,
age: age,
getName: function() { return name; },
getAge: function() { return age; },
};
}
var user = make_user("Joe", "18");
You can access name and age, without using such functions. In javascript you have to use different hacks to keep something private or protected.
This
User.name = "BoB";
User.age = 44;
will produce same output as your example.
There are no constructors as they appear in other languages. Easiest way would be to just define init() function and call it right after you instance the object.
But my biggest tip for you is to look into http://www.prototypejs.org/. It's a javascript library with a lot of cool features which tries to make javascript "more OO*".
Using prototype library you can make classes to behave more like real OOP classes. It also features constructors.
Edit: As for what you asked in your comment:
person = new User();
person.name = "Bob";
person.age = 44;
Question #1
prototype
has the benefit of monkey patching. As the first example shows, the function are added after-the-fact. You can continue this to add or replace any methods you need (though, with fair warning).
Defining objects like #2 is more along the lines of classic OOP. But, then again, monkey patching isn't allowed in all OOP languages.
Question #2
In your 3rd function, you don't even need the get
and set
functions -- name
and age
are public properties (the potential downside to {}
).
var User = {
name: "",
age: 0
};
User.name = 'Bob';
User.age = 44;
console.log("User: " + User.name + ", Age: " + User.age);
When you create an object using {}
(an object literal), {}
is the constructor (varying on browser). But, essentially, no you can't use a constructor in this format.
If you want to do OOP in JavaScript I'd highly suggest looking up closures. I began my learning on the subject with these three web pages:
http://www.dustindiaz.com/javascript-private-public-privileged/
http://www.dustindiaz.com/namespace-your-javascript/
http://blog.morrisjohns.com/javascript_closures_for_dummies
The differences between 1, 2, and 3 are as follows: 1) Is an example of adding new methods to an existing object. 2) Is the same as #1 except some methods are included in the object in the User function. 3) Is an example of defining an object using JSON. The shortcoming is that you cannot use new (at least not with that example) to define new instances of that object. However you do get the benefit of the convenient JSON coding style.
You should definitely read up on JSON if you don't know it yet. JavaScript will make a lot more sense when you understand JSON.
edit If you want to use new in function #3 you can write it as
function User() {
return {
name: "",
age: 0,
setName: function(name) {
this.name = name;
},
setAge: function(age) {
this.age = age;
},
getName: function() {
return this.name;
},
getAge: function() {
return this.age;
}
};
}
Of course all of those functions and properties would then be public. To make them private you need to use closures. For example you could make age and name private with this syntax.
function User() {
var age=0;
var name="";
return {
setName: function(name_) {
name = name_;
},
setAge: function(age_) {
age = age_;
},
getName: function() {
return name;
},
getAge: function() {
return age;
}
};
}