草庐IT

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

coder 2023-10-25 原文

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

我有一个方法 getUser(username) 访问数据库以获取代表给定用户名的特定用户的 JSON 对象,以及一个列出特定用户的 friend 的方法。从 listFriends(username) 函数中,我想返回用户 对象 的数组,而不仅仅是他们的用户名,理想情况下我想利用我自己的 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() 方法会更简洁,这样如果用户对象的结构发生变化,它只需要更改一个地方。

最佳答案

尽量保持干燥,像这样的东西应该符合您的要求:

// 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. 我使用了 Array.map如 ECMAScript 5 中所定义,它应该在 Node 中可用。

  3. 每个表的 SQL 别名在每个函数中都是硬编码的,这意味着您还需要 a) 设置某种常量或采用约定来使所有别名在所有函数中保持一致(这随着项目的增长,维护起来可能会很痛苦)或 b) 研究使用可以为您做这些事情的 ORM(以及更多!)这将帮助您避免在如何构建查询,让您担心更重要的事情,例如您实际需要的数据。我不是很熟悉 NodeJS 在 ORM 方面的可用功能,但是官方 NodeJS wiki应该是一个好的开始。

关于mysql - 访问数据库时保持代码模块化,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/9457446/

有关mysql - 访问数据库时保持代码模块化的更多相关文章

  1. ruby - 为什么我可以在 Ruby 中使用 Object#send 访问私有(private)/ protected 方法? - 2

    类classAprivatedeffooputs:fooendpublicdefbarputs:barendprivatedefzimputs:zimendprotecteddefdibputs:dibendendA的实例a=A.new测试a.foorescueputs:faila.barrescueputs:faila.zimrescueputs:faila.dibrescueputs:faila.gazrescueputs:fail测试输出failbarfailfailfail.发送测试[:foo,:bar,:zim,:dib,:gaz].each{|m|a.send(m)resc

  2. ruby - 在 Ruby 中使用匿名模块 - 2

    假设我做了一个模块如下:m=Module.newdoclassCendend三个问题:除了对m的引用之外,还有什么方法可以访问C和m中的其他内容?我可以在创建匿名模块后为其命名吗(就像我输入“module...”一样)?如何在使用完匿名模块后将其删除,使其定义的常量不再存在? 最佳答案 三个答案:是的,使用ObjectSpace.此代码使c引用你的类(class)C不引用m:c=nilObjectSpace.each_object{|obj|c=objif(Class===objandobj.name=~/::C$/)}当然这取决于

  3. ruby-on-rails - Ruby net/ldap 模块中的内存泄漏 - 2

    作为我的Rails应用程序的一部分,我编写了一个小导入程序,它从我们的LDAP系统中吸取数据并将其塞入一个用户表中。不幸的是,与LDAP相关的代码在遍历我们的32K用户时泄漏了大量内存,我一直无法弄清楚如何解决这个问题。这个问题似乎在某种程度上与LDAP库有关,因为当我删除对LDAP内容的调用时,内存使用情况会很好地稳定下来。此外,不断增加的对象是Net::BER::BerIdentifiedString和Net::BER::BerIdentifiedArray,它们都是LDAP库的一部分。当我运行导入时,内存使用量最终达到超过1GB的峰值。如果问题存在,我需要找到一些方法来更正我的代

  4. ruby - 如何在 buildr 项目中使用 Ruby 代码? - 2

    如何在buildr项目中使用Ruby?我在很多不同的项目中使用过Ruby、JRuby、Java和Clojure。我目前正在使用我的标准Ruby开发一个模拟应用程序,我想尝试使用Clojure后端(我确实喜欢功能代码)以及JRubygui和测试套件。我还可以看到在未来的不同项目中使用Scala作为后端。我想我要为我的项目尝试一下buildr(http://buildr.apache.org/),但我注意到buildr似乎没有设置为在项目中使用JRuby代码本身!这看起来有点傻,因为该工具旨在统一通用的JVM语言并且是在ruby中构建的。除了将输出的jar包含在一个独特的、仅限ruby​​

  5. ruby - 解析 RDFa、微数据等的最佳方式是什么,使用统一的模式/词汇(例如 schema.org)存储和显示信息 - 2

    我主要使用Ruby来执行此操作,但到目前为止我的攻击计划如下:使用gemsrdf、rdf-rdfa和rdf-microdata或mida来解析给定任何URI的数据。我认为最好映射到像schema.org这样的统一模式,例如使用这个yaml文件,它试图描述数据词汇表和opengraph到schema.org之间的转换:#SchemaXtoschema.orgconversion#data-vocabularyDV:name:namestreet-address:streetAddressregion:addressRegionlocality:addressLocalityphoto:i

  6. ruby-on-rails - Rails 源代码 : initialize hash in a weird way? - 2

    在rails源中:https://github.com/rails/rails/blob/master/activesupport/lib/active_support/lazy_load_hooks.rb可以看到以下内容@load_hooks=Hash.new{|h,k|h[k]=[]}在IRB中,它只是初始化一个空哈希。和做有什么区别@load_hooks=Hash.new 最佳答案 查看rubydocumentationforHashnew→new_hashclicktotogglesourcenew(obj)→new_has

  7. ruby-on-rails - 在混合/模块中覆盖模型的属性访问器 - 2

    我有一个包含模块的模型。我想在模块中覆盖模型的访问器方法。例如:classBlah这显然行不通。有什么想法可以实现吗? 最佳答案 您的代码看起来是正确的。我们正在毫无困难地使用这个确切的模式。如果我没记错的话,Rails使用#method_missing作为属性setter,因此您的模块将优先,阻止ActiveRecord的setter。如果您正在使用ActiveSupport::Concern(参见thisblogpost),那么您的实例方法需要进入一个特殊的模块:classBlah

  8. ruby - 续集在添加关联时访问many_to_many连接表 - 2

    我正在使用Sequel构建一个愿望list系统。我有一个wishlists和itemstable和一个items_wishlists连接表(该名称是续集选择的名称)。items_wishlists表还有一个用于facebookid的额外列(因此我可以存储opengraph操作),这是一个NOTNULL列。我还有Wishlist和Item具有续集many_to_many关联的模型已建立。Wishlist类也有:selectmany_to_many关联的选项设置为select:[:items.*,:items_wishlists__facebook_action_id].有没有一种方法可以

  9. ruby - 当使用::指定模块时,为什么 Ruby 不在更高范围内查找类? - 2

    我刚刚被困在这个问题上一段时间了。以这个基地为例:moduleTopclassTestendmoduleFooendend稍后,我可以通过这样做在Foo中定义扩展Test的类:moduleTopmoduleFooclassSomeTest但是,如果我尝试通过使用::指定模块来最小化缩进:moduleTop::FooclassFailure这失败了:NameError:uninitializedconstantTop::Foo::Test这是一个错误,还是仅仅是Ruby解析变量名的方式的逻辑结果? 最佳答案 Isthisabug,or

  10. ruby-on-rails - 浏览 Ruby 源代码 - 2

    我的主要目标是能够完全理解我正在使用的库/gem。我尝试在Github上从头到尾阅读源代码,但这真的很难。我认为更有趣、更温和的踏脚石就是在使用时阅读每个库/gem方法的源代码。例如,我想知道RubyonRails中的redirect_to方法是如何工作的:如何查找redirect_to方法的源代码?我知道在pry中我可以执行类似show-methodmethod的操作,但我如何才能对Rails框架中的方法执行此操作?您对我如何更好地理解Gem及其API有什么建议吗?仅仅阅读源代码似乎真的很难,尤其是对于框架。谢谢! 最佳答案 Ru

随机推荐