我一直在为一个非常基本的社交网站开发数据库:你所能做的就是注册一个帐户并发送/接受好友请求。服务器是用javascript(nodejs)编写的
我有一个方法get user(username),它访问数据库以获得表示给定用户名的特定用户的json对象,还有一个方法列出特定用户的朋友。从listFriends(username)函数中,我希望返回一个用户对象数组,而不仅仅是它们的用户名,理想情况下,我希望使用自己的get()函数,而不是更改listFriends()使用的SQL查询
举个例子可能更容易理解。如果我有三张桌子:
TABLE: UsersName
username (unique) | firstName | lastName |
------------------|-----------|------------|
pjfry | Phillip | Fry |
proff | Professor | Farnsworth |
bender | Bender | Rodriguez |
TABLE: UsersProfile (their profile description)
username (unique) | description |
------------------|------------------------|
pjfry | I went to the future |
proff | I am very old |
bender | Destroy all humans |
TABLE: Friendships (for simplicity assume that if (a,b) is an entry then so is (b,a))
user1 | user2
-----------|---------------
bender | pjfry
pjfry | bender
pjfry | proff
proff | pjfry
以及获取用户对象的函数:
//the callback accepts the user object
function get (username, callback) {
db.query(
'select * from (UsersName n, UsersProfile p) where n.username=p.username and n.username=\'' + username + '\'',
function (rows) {
//for simplicity assume that this call always succeeds
//the user object is altered a bit:
callback({
username: rows[0].username,
name: {
first: rows[0].firstName,
last: rows[0].lastName,
full: rows[0].firstName + ' ' + rows[0].lastName
},
profile: {
description: rows[0].description
}
});
}
}
下面是列出给定用户的朋友的函数
//callback accepts the array of friends
function listFriends(username, callback) {
db.query(
'select user2 from Friendships where user1=\'' + username + '\'',
function(rows) {
//assume this call always succeeds
callback(rows)
}
)
}
这里的问题是
listFriends()
只返回用户名数组,而不是用户对象。如何修改listFriends()
函数,使其通过使用get()函数返回用户对象?这可以通过修改
listFriends()
中的sql语句来完成,但是使用get()方法会更简洁,这样,如果用户对象的结构发生了更改,则只需在一个地方进行更改。 最佳答案
尽量保持干燥,这样的东西应该符合你的要求:
// Disclaimer: I have not run tested this code myself
//the callback accepts the user object
function get(username, callback) {
getFriendsWithConditions('n.username=\'' + username + '\'', function(rows){
// returns only a single object to the input callback
callback(rows[0]);
});
}
//callback accepts the array of friends
function listFriends(username, callback) {
getFriendsWithConditions('n.username in (select user2 from Friendships where user1=\'' + username + '\')', callback);
}
function getFriendsWithConditions(whereCondition, callback) {
db.query(
'select * from (UsersName n, UsersProfile p) where n.username=p.username and (' + whereCondition + ')',
function(rows) {
// Using ECMAScript 5 Array.map
callback(rows.map(rowToUserObject));
}
);
}
function rowToUserObject(row)
{
return {
username: row.username,
name: {
first: row.firstName,
last: row.lastName,
full: row.firstName + ' ' + row.lastName
},
profile: {
description: row.description
}
};
}
在我写这篇文章的时候,我想到了一些事情:
username和userprofile感觉它们应该只是一个表,但据我所知,这是实际数据库的简化版本,所以我将它们原封不动地保留了下来。
我使用了ecmascript 5中定义的Array.map,它应该在node中可用。
每个表的sql别名在每个函数中都是硬编码的,这意味着您还需要a)设置某种常量,或者使用一种约定来保持所有函数中所有别名的一致性(随着项目的增长,维护这些别名可能会很麻烦),或者b)看起来像into使用一个orm可以为您做这些事情(以及更多!)这将帮助您避免为如何构造查询的细节操心,让您担心更重要的事情,比如您实际需要的数据。我对nodejs在orms方面的可用性不是很熟悉,但是官方的NodeJS wiki应该是一个很好的开始。