访问数据库时保持代码模块化

时间:2012-02-26 21:42:23

标签: mysql sql json node.js data-modeling

我一直在为一个非常基本的社交网站工作数据库:您所能做的就是注册一个帐户并发送/接受朋友请求。服务器是用Javascript(NodeJS)编写的

我有一个方法getUser(用户名)访问数据库以获取代表特定用户的JSON对象及其用户名,以及列出特定用户朋友的方法。从listFriends(用户名)函数,我想返回一个用户对象的数组而不仅仅是他们的用户名,理想情况下我想利用我自己的get()函数而不是改变SQL查询listFriends()使用

通过示例可能更容易理解。如果我有三张桌子:

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()方法会更加清晰,这样如果用户对象的结构发生了变化,只需要更改它在一个地方。

3 个答案:

答案 0 :(得分:1)

尽量保持干净,这样的事情应符合您的要求:

// 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
    }
  };
}

我写这篇文章时会想到一些事情:

  1. UserName和UserProfile感觉它们应该只是一个表,但是据我所知,这是实际数据库的简化版本,所以我将它们保留原样。

  2. 我使用了ECMAScript 5中定义的Array.map,它应该在Node中可用。

  3. 每个表的SQL别名都是在每个函数中进行硬编码的,这意味着您还需要a)设置某种常量或者使用约定来保持所有函数中的所有别名一致(随着项目的发展,可能会很难维护)或b)考虑使用可以为你做这些事情的ORM(以及更多!)这将帮助你避免出现如何构建的细节查询,让您担心更重要的事情,比如您实际需要的数据。我不太了解NodeJS在ORM方面可用的内容,但官方NodeJS wiki应该是一个好的开始。

答案 1 :(得分:0)

最好不要破坏listFriends()的现有功能吗?

相反,我的想法是创建一个名为getFriends()的新函数,它调用listFriends(),并为每个返回的用户名调用get()。每个get()的串联结果系列将作为getFriends()的返回值。

答案 2 :(得分:0)

这是一个相当简单的解决方案,可以在2个网络呼叫中完成我想要的操作(而不是像我原本希望的那样1)。

我添加了一个函数getUserArray(usernames),它接受一个用户名数组并从中返回对象。它通过生成如下所示的SQL来实现:“SELECT ... FROM(UsersName n,UsersProfile p)WHERE n.username = p.username和n.username in(?,?,?,...)by field by field (用户名,?,?,?,...)“并将用户名数组作为要转义的值传递并替换为?。

函数getUserArray()和getUser()共享很多共同点,因此代码的常用部分可以很容易地放入常见的辅助函数中。

listFriends()的功能增强如下:在它返回用户名数组之前,它调用getUserArray()将用户名映射到对象。

简单,只有2个网络电话。并且它使listFriends()不会弄乱用户元数据表的结构。

相关问题