草庐IT

javascript - 主干模型的转换/初始化子模型

coder 2024-07-26 原文

我想我有一个非常简单的问题,很难说出来,因此很难找到解决方案。设置:

  • PathCollection 是一个 Backbone.Collection of Paths
  • Path 是一个 Backbone.Model,它包含 NodeCollection(它是一个 Backbone.Collection)和 EdgeCollection(它是一个 Backbone.Collection)。

  • 当我获取 PathCollection
    paths = new PathCollection()
    paths.fetch()
    

    显然,路径被实例化。但是,我缺少可以允许 Path 从属性哈希实例化其子模型的地方。我真的不能使用解析,对吧?基本上,我在模型实例化并设置属性时寻找模型的入口点。我觉得必须有一些约定。

    最佳答案

    所以我写了几个关于使用 parse() 的答案和 set()实例化和填充子模型和子集合(嵌套数据)。但是,我还没有看到一个真正全面的答案来整合我所看到的许多实践中的一些。当我写很多东西时,我倾向于漫无边际,所以我可能会离题一点,但这可能对遇到类似问题的人有用。

    有几种方法可以做到这一点。使用 parse()是一个。操纵set()是另一个。在您的 initialize() 中实例化这些是另一个。在 Path 模型之外做这一切是另一回事(例如 path = new Path(); path.nodes = new NodeCollection(); 等)

    第二个考虑是这个。您是否希望节点和边集合成为模型属性?还是模型属性?

    哦,这么多选择。很多自由,但有时(令我们沮丧)它使确定“正确方式”变得更加困难。

    由于这种情况经常出现,我将写一篇长篇文章并逐一浏览这些内容。所以请耐心等待我继续更新这个答案。

    在模型之外做 - 简单直接

    当您只在特定模型或集合上需要它时,这通常是添加嵌套模型和集合的简单方法。

    path = new PathModel();
    
    path.nodes = new NodeCollection();
    path.edge = new EdgeCollection();
    
    // Continue to set up the nested data URL, etc.
    

    这是最简单的方法,当您处理不需要定义的一次性模型和集合时效果很好。尽管您可以在对其进行任何操作之前构造此对象的某种方法(例如 View 方法)中轻松生成这些模型。

    使用 initialize()每个模型中的子模型/集合

    如果您知道某个模型的每个实例总是有一个子模型或子集合,那么最简单的设置方法是利用 initialize()功能。

    例如,以您的 Path 模型为例:
    Path = Backbone.Model.extend({
        initialize: function() {
            this.nodes = new NodeCollection();
            this.paths = new PathCollection();
    
            // Maybe assign a proper url in relation to this Path model
            // You might even set up a change:id listener to set the url when this
            // model gets an id, assuming it doesn't have one at start.
            this.nodes.url = this.id ? 'path/' + this.id + '/nodes' : undefined;
            this.paths.url = this.id ? 'path/' + this.id + '/paths' : undefined;
        }
    });
    

    现在你的子集合可以像 path.nodes.fetch() 一样被获取它将路由到正确的 URL。十分简单。

    使用 parse()用于实例化和设置子数据

    也许,如果您不想假设每个模型都有一个节点和边集合,它会变得更加棘手。也许只有在 fetch() 时才需要嵌套模型/集合发回这样的数据。这是使用 parse() 的情况可以派上用场。
    parse()的事是它需要任何 json 服务器响应,并且可以在将其传递给模型之前正确命名空间和处理它 set()功能。因此,我们可以检查是否包含模型或集合原始数据并在将响应减少到父模型属性之前对其进行处理。

    例如,也许我们从我们的服务器得到这个响应:
    // Path model JSON example with nested collections JSON arrays
    {
        'name':'orange site',
        'url':'orange.com',
        'nodes':[
            {'id':'1', 'nodeColor':'red'},
            {'id':'2', 'nodeColor':'white'},
            {'id':'3', 'nodeColor':'blue'}
        ],
        'edge':[
            {'id':'1', 'location':'north'},
            {'id':'1', 'location':'south'},
            {'id':'1', 'location':'east'}
        ]
    }
    

    与默认 parse() Backbone 将吞并它,并为您的路径模型属性“节点”和“边缘”分配一个数组()数据(不是集合)。所以我们要确保我们的 parse()适本地处理这个。
    parse: function(response) {
    
        // Check if response includes some nested collection data... our case 'nodes'
        if (_.has(response, 'nodes')){
    
             // Check if this model has a property called nodes
            if (!_.has(this, 'nodes')) {  // It does not...
                // So instantiate a collection and pass in raw data
                this.nodes = new NodeCollection(response.nodes);
            } else {
                // It does, so just reset the collection
                this.nodes.reset(response.nodes);
            }
    
            // Assuming the fetch gets this model id
            this.nodes.url = 'path/' + response.id + '/nodes';  // Set model relative URL
    
            // Delete the nodes so it doesn't clutter our model attributes
            delete response.nodes;
        }
    
        // Same for edge...
    
        return response;
    }
    

    您也可以使用定制的 set()处理你的子数据。经过多次来回,哪个更好,操纵set()或在 parse() 中进行我决定我喜欢使用 parse()更多的。但我愿意接受其他人对此的看法。

    使用 set()处理你的子数据

    虽然 parse()依赖于获取数据或使用选项 parse:true 将数据传递到集合中有些人觉得改换set()功能。同样,我不确定是否有正确的选择,但这是如何工作的。
    set: function(attributes, options) {
        // If we pass in nodes collection JSON array and this model has a nodes attribute
        // Assume we already set it as a collection
        if (_.has(attributes, 'nodes') && this.get("nodes")) {
            this.get('nodes').reset(attributes.nodes);
            delete attributes.nodes;
        } else if (_.has(attributes, 'nodes') && !this.get('nodes')) {
            this.set('nodes', new NodeCollection(attributes.nodes));
            delete attributes.nodes;
        }
    
        return Backbone.Model.prototype.set.call(this, attributes, options);
    }
    

    所以如果我们已经有一个属性并且它是一个集合,我们reset()它。如果我们有一个属性但它不是一个集合,我们就实例化它。在将子数据的 JSON 数组传递给原型(prototype) set() 之前,确保将其正确转换为集合非常重要。 . Backbone,不会将 JSON 数组解释为集合,您只会得到一个直接的数组。

    所以简而言之,你有很多关于如何去做的选择。同样,目前我更喜欢混合使用 initialize()当我知道某些东西总会有那些子模型/系列和 parse()当情况仅需要 fetch() 上可能的嵌套数据时调用。

    关于你的那个问题......(哦,是的,有一个问题)

    您可以允许 Path 以多种方式从散列实例化子模型。我刚刚给了你 4。如果你愿意,你可以使用 parse,如果你知道你会成为 fetch()路径模型或者甚至是一个 pathCollection... pathCollection.fetch({parse:true})有约定吗?也许也许不是。我喜欢根据我认为我将使用模型/集合的上下文使用多种方式的组合。

    我非常愿意讨论其中一些做法以及它们是好是坏。它们只是我在 Stack 上遇到的许多解决方案,并融入了我自己的工作习惯,它们似乎对我来说效果很好。 :-)

    给自己喝杯咖啡,拍拍他的背,那是一篇长文。

    关于javascript - 主干模型的转换/初始化子模型,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/12350218/

    有关javascript - 主干模型的转换/初始化子模型的更多相关文章

    1. ruby-on-rails - 在 Rails 中将文件大小字符串转换为等效千字节 - 2

      我的目标是转换表单输入,例如“100兆字节”或“1GB”,并将其转换为我可以存储在数据库中的文件大小(以千字节为单位)。目前,我有这个:defquota_convert@regex=/([0-9]+)(.*)s/@sizes=%w{kilobytemegabytegigabyte}m=self.quota.match(@regex)if@sizes.include?m[2]eval("self.quota=#{m[1]}.#{m[2]}")endend这有效,但前提是输入是倍数(“gigabytes”,而不是“gigabyte”)并且由于使用了eval看起来疯狂不安全。所以,功能正常,

    2. ruby-on-rails - Rails - 子类化模型的设计模式是什么? - 2

      我有一个模型:classItem项目有一个属性“商店”基于存储的值,我希望Item对象对特定方法具有不同的行为。Rails中是否有针对此的通用设计模式?如果方法中没有大的if-else语句,这是如何干净利落地完成的? 最佳答案 通常通过Single-TableInheritance. 关于ruby-on-rails-Rails-子类化模型的设计模式是什么?,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.co

    3. ruby - 使用 ruby​​ 将 HTML 转换为纯文本并维护结构/格式 - 2

      我想将html转换为纯文本。不过,我不想只删除标签,我想智能地保留尽可能多的格式。为插入换行符标签,检测段落并格式化它们等。输入非常简单,通常是格式良好的html(不是整个文档,只是一堆内容,通常没有anchor或图像)。我可以将几个正则表达式放在一起,让我达到80%,但我认为可能有一些现有的解决方案更智能。 最佳答案 首先,不要尝试为此使用正则表达式。很有可能你会想出一个脆弱/脆弱的解决方案,它会随着HTML的变化而崩溃,或者很难管理和维护。您可以使用Nokogiri快速解析HTML并提取文本:require'nokogiri'h

    4. ruby-on-rails - 未初始化的常量 Psych::Syck (NameError) - 2

      在我的gem中,我需要yaml并且在我的本地计算机上运行良好。但是在将我的gem推送到ruby​​gems.org之后,当我尝试使用我的gem时,我收到一条错误消息=>"uninitializedconstantPsych::Syck(NameError)"谁能帮我解决这个问题?附言RubyVersion=>ruby1.9.2,GemVersion=>1.6.2,Bundlerversion=>1.0.15 最佳答案 经过几个小时的研究,我发现=>“YAML使用未维护的Syck库,而Psych使用现代的LibYAML”因此,为了解决

    5. ruby - 将数组的内容转换为 int - 2

      我需要读入一个包含数字列表的文件。此代码读取文件并将其放入二维数组中。现在我需要获取数组中所有数字的平均值,但我需要将数组的内容更改为int。有什么想法可以将to_i方法放在哪里吗?ClassTerraindefinitializefile_name@input=IO.readlines(file_name)#readinfile@size=@input[0].to_i@land=[@size]x=1whilex 最佳答案 只需将数组映射为整数:@land边注如果你想得到一条线的平均值,你可以这样做:values=@input[x]

    6. ruby-on-rails - Rails - 一个 View 中的多个模型 - 2

      我需要从一个View访问多个模型。以前,我的links_controller仅用于提供以不同方式排序的链接资源。现在我想包括一个部分(我假设)显示按分数排序的顶级用户(@users=User.all.sort_by(&:score))我知道我可以将此代码插入每个链接操作并从View访问它,但这似乎不是“ruby方式”,我将需要在不久的将来访问更多模型。这可能会变得很脏,是否有针对这种情况的任何技术?注意事项:我认为我的应用程序正朝着单一格式和动态页面内容的方向发展,本质上是一个典型的网络应用程序。我知道before_filter但考虑到我希望应用程序进入的方向,这似乎很麻烦。最终从任何

    7. ruby - 将散列转换为嵌套散列 - 2

      这道题是thisquestion的逆题.给定一个散列,每个键都有一个数组,例如{[:a,:b,:c]=>1,[:a,:b,:d]=>2,[:a,:e]=>3,[:f]=>4,}将其转换为嵌套哈希的最佳方法是什么{:a=>{:b=>{:c=>1,:d=>2},:e=>3,},:f=>4,} 最佳答案 这是一个迭代的解决方案,递归的解决方案留给读者作为练习:defconvert(h={})ret={}h.eachdo|k,v|node=retk[0..-2].each{|x|node[x]||={};node=node[x]}node[

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

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

    9. ruby-on-rails - 如何验证非模型(甚至非对象)字段 - 2

      我有一个表单,其中有很多字段取自数组(而不是模型或对象)。我如何验证这些字段的存在?solve_problem_pathdo|f|%>... 最佳答案 创建一个简单的类来包装请求参数并使用ActiveModel::Validations。#definedsomewhere,atthesimplest:require'ostruct'classSolvetrue#youcouldevencheckthesolutionwithavalidatorvalidatedoerrors.add(:base,"WRONG!!!")unlesss

    10. ruby-on-rails - form_for 中不在模型中的自定义字段 - 2

      我想向我的Controller传递一个参数,它是一个简单的复选框,但我不知道如何在模型的form_for中引入它,这是我的观点:{:id=>'go_finance'}do|f|%>Transferirde:para:Entrada:"input",:placeholder=>"Quantofoiganho?"%>Saída:"output",:placeholder=>"Quantofoigasto?"%>Nota:我想做一个额外的复选框,但我该怎么做,模型中没有一个对象,而是一个要检查的对象,以便在Controller中创建一个ifelse,如果没有检查,请帮助我,非常感谢,谢谢

    随机推荐