1

可能重复:
对象是空的吗?

 update: (id, data) ->
    toUpdate = @find(id)
    if toUpdate isnt {}
            console.log "hi mom"
            console.log toUpdate
            toUpdate.setProperty(key, value) for own key, value of data

    return toUpdate

 find:(id) ->
    result = record for record in @storage when record.id is id
    return result or {}

鉴于以下摩卡测试

describe '#update', ->
    it 'should return an updated record from a given id and data when the record exists', ->
      boogie = createData()
      archive = new Archive("Dog")
      dog = archive.create(boogie)
      result = archive.update(1, {name:"Chompie", age:1})
      result.name.should.eql "Chompie"
      result.age.should.eql 1
      result.emotion.should.eql dog.emotion

    it 'should return an updated record from a given id and data when the record does not exist', ->
      boogie = createData()
      archive = new Archive("Dog")
      dog = archive.create(boogie)
      result = archive.update(50, {name:"Chompie", age:1})
      result.should.not.exist

结果是

Archive #update should return an updated record from a given id and data when the record exists: hi mom
{ id: 1,
  validationStrategies: {},
  name: 'Boogie',
  age: 2,
  emotion: 'happy' }
  ✓ Archive #update should return an updated record from a given id and data when the record exists: 1ms
    Archive #update should return empty when the record does not exist: hi mom
{}
    ✖ 1 of 13 tests failed:
    1) Archive #update should return empty when the record does not exist:
    TypeError: Object #<Object> has no method 'setProperty'

...令人惊讶,不是吗?

4

1 回答 1

14

CoffeeScript 的is(AKA ==) 只是 JavaScript 的===,而isnt(AKA !=) 只是 JavaScript 的!==. 所以你的条件:

if toUpdate isnt {}

将始终为真toUpdate,因为对象字面{}量永远不会是同一个对象。

但是,如果@find可以返回在常量中可用的已知“空”对象,那么您可以使用isnt

EMPTY = {}

find: ->
    # ...
    EMPTY

然后:

if toUpdate isnt EMPTY
    #...

例如,考虑这个简单的代码:

a = { }
b = { }
console.log("a is b: #{a is b}")
console.log("a isnt b: #{a isnt b}")

这将在您的控制台中为您提供:

a is b: false
a isnt b: true

但是这个:

class C
    EMPTY = { }
    find: -> EMPTY
    check: -> console.log("@find() == EMPTY: #{@find() == EMPTY}")

(new C).check()

会说:

@find() == EMPTY: true

演示:http: //jsfiddle.net/ambiguous/7JGdq/

所以你需要另一种方法来检查是否toUpdate不为空。您可以计算以下属性toUpdate

if (k for own k of toUpdate).length isnt 0

或者您可以使用上面概述的特殊EMTPY常量方法。还有很多其他方法可以检查空对象,Ricardo Tomasi提出了一些建议:

  1. Underscore 提供_.isEmpty了基本上是for带有一些特殊情况处理和短路的循环方法。
  2. 下划线还提供_.values,以便您查看_(toUpdate).values().length. 这在map内部调用,如果可用,这将是本机map函数。
  3. 你甚至可以通过 JSON 使用JSON.stringify(toUpdate) is '{}',这对我来说似乎有点脆弱,而且相当迂回。
  4. 您可以使用Object.keys而不是for循环:Object.keys(toUpdate).length isnt 0keys虽然并非所有地方都支持,但它适用于 Node、最新的非 IE 浏览器和 IE9+。
  5. Sugar 也有Object.isEmpty,jQuery 也有$.isEmptyObject.

短路for循环似乎是检查空虚的最快方法:

(obj) ->
    for k of toUpdate
        return true
    false

这假设您不需要own避免迭代错误的事情。但鉴于这只是一个测试套件,而且空性测试几乎肯定不会成为您代码中的瓶颈,我会选择您拥有的 Underscore、Sugar 或 jQuery 中的任何一个(如果您需要可移植性并且必须处理通常的浏览器废话),Object.keys(x).length如果您知道它将可用,并且(k for own k of toUpdate).length如果您没有库并且必须处理浏览器废话并且不确定这toUpdate将是一个简单的对象。

于 2012-05-07T06:40:57.030 回答