在Backbone模型中定义实例变量的正确方法是什么?

时间:2012-05-24 21:04:05

标签: javascript backbone.js coffeescript

我想知道如何在Backbone模型中定义实例变量。这就是我目前的做法:

class GeneSet extends Backbone.Model
    initialize: (parsedGenes)->
        @set parsedGenes: parsedGenes
        @set geneNames: (gene.gene_name for gene in @get("parsedGenes"))
        @set geneIds: ("gene_#{id}" for id in [1..@get("parsedGenes").length])
        @set columnNames: @getColumnNames()
        @set columnGroups: @getColumnGroups()
        @set geneExpressions: @getGeneExpressions()
        @set groupedGeneExpressions: @getGroupedGeneExpressions()
        @set extent: @getExtent()

    clusterColor: ->
        d3.scale.category20()()

    getGeneNameById: (geneId)->
        @get("geneNames")[@get("geneIds").indexOf(geneId)]

    getColumnGroups: ->
        _.uniq((@get("columnNames")).map((columnName)->
            columnName.split("_")[0]
        ))

    getExtent: ->
        expressions = _.flatten(@get("geneExpressions").map (geneExpression)->
            geneExpression.map (item)->
                item.y
        )
        d3.extent(expressions)

    getColumnNames: ->
        Object.keys(@get("parsedGenes")[0]).filter (columnName) ->
            !columnName.match(/cluster/) && isNumber(parsedGenes[1][columnName])


    getGeneExpressions: ->
        @get("parsedGenes").map (gene) =>
            @get("columnNames").map (columnName) -> 
                x: columnName
                y: +gene[columnName] # make numeric

这似乎有点多余@set columnGroups: @getColumnGroups()并且必须使用@get("...")获取每个变量似乎有点冗长(我希望我能做@variableName)。我的问题是,这是使用模型和实例变量的正确方法还是我做错了?另外,这样做有什么不同吗?:

    class GeneSet extends Backbone.Model
        initialize: (parsedGenes)->
            @parsedGenes = parsedGenes
            @geneNames = (gene.gene_name for gene in @parsedGenes)
            @geneIds = ("gene_#{id}" for id in [1..@parsedGenes.length])
            @clusters = (gene.cluster for gene in @parsedGenes)
            @descriptions = (gene.description for gene in @parsedGenes)
            @columnNames = @getColumnNames()
            @columnGroups = @getColumnGroups()
            @geneExpressions = @getGeneExpressions()
            @groupedGeneExpressions = @getGroupedGeneExpressions()
            @extent = @getExtent()

然后,从仅仅做@model.columnNames

的观点来看

1 个答案:

答案 0 :(得分:3)

这是我在基础模型超类中所做的。这几乎与主干设计无关,但我也讨厌使用getset,我想要所有属性访问背后的真实方法。所以我做了一些元编程并生成了命名的get / set方法。所以不必这样做:

model.get("name")
model.set("name", "Tom")

我可以做到

model.name()
model.name("Tom")

这是我的基本代码,可以自动为任意attributes个对象执行此操作。

addConvenienceMethods = ->
  for prop, value of this.attributes
    ((prop) ->
      #Define a setter/getter function
      this[prop] = (newValue...) ->
        if newValue.length
          obj = {}
          obj[prop] = newValue[0]
          this.set obj
          return this
        else
          return this.get prop
      #Use the newly-defined setter function to store the default value
      this[prop](value)
    ).call(this, prop)

########## Base Model Superclass ##########
class exports.Model extends Backbone.Model
  initialize: ->
    addConvenienceMethods.call this

请注意,这是在一个足以支持set("key", "value")的骨干版本上编写的。如果我要更新它,我可能会使用该变体。请注意,由于set flavor返回对象,因此它们是可链接的:model.name("John").email("john@example.com").role("admin")