当前位置: 首页 > 面试题库 >

如何在Swift中自动递增变量?

慕皓君
2023-03-14
问题内容

我希望能够以原子方式递增计数器,而我却找不到有关如何执行此操作的参考。

根据评论添加更多信息:

  • 您正在使用GCD吗?不,我没有使用GCD。必须使用队列系统来增加数字似乎过大了。
  • 您理解基本的线程安全?是的,否则我不会问原子增量的问题。
  • 这个变量是局部变量吗?没有。
  • 它是实例级别吗? 是的,它应该是单个实例的一部分。

我想做这样的事情:

 class Counter {
      private var mux Mutex
      private (set) value Int
      func increment (){
          mux.lock()
          value += 1
          mux.unlock()
      }
 }

问题答案:

来自低级并发API:

OSAtomicIncrement和OSAtomicDecrement函数的清单很长,它们使您能够以原子方式递增和递减整数值–线程安全,而不必获取锁(或使用队列)。如果您需要从多个线程中增加全局计数器以进行统计,这些功能将很有用。如果您要做的只是增加一个全局计数器,那么无障碍的OSAtomicIncrement版本就可以了,并且在没有争用时,调用它们很便宜。

这些函数与固定大小的整数一起使用,您可以根据需要选择32位或64位变体:

class Counter {
    private (set) var value : Int32 = 0
    func increment () {
        OSAtomicIncrement32(&value)
    }
}

注意: 正如Erik Aigner所正确注意到的OSAtomicIncrement32那样,从macOS 10.12 / iOS
10.10开始不推荐使用朋友.Xcode8建议改用from的函数<stdatomic.h>。但是这似乎很困难,请比较一下[Swift3:atomic_compare_exchange_strong]和https://openradar.appspot .com /
27161329
,因此以下基于GCD的方法现在似乎是最好的解决方案。)

或者,可以使用GCD队列进行同步。从“并发编程指南”中的调度队列中:

…使用调度队列,您可以将两个任务都添加到串行调度队列中,以确保在任何给定时间只有一个任务修改了资源。这种类型的基于队列的同步比锁更有效,因为在有争议和无争议的情况下,锁总是需要昂贵的内核陷阱,而分派队列主要在应用程序的进程空间中工作,并且仅在绝对必要时才调用内核。

在你的情况下

// Swift 2:
class Counter {
    private var queue = dispatch_queue_create("your.queue.identifier", DISPATCH_QUEUE_SERIAL)
    private (set) var value: Int = 0

    func increment() {
        dispatch_sync(queue) {
            value += 1
        }
    }
}

// Swift 3:
class Counter {
    private var queue = DispatchQueue(label: "your.queue.identifier") 
    private (set) var value: Int = 0

    func increment() {
        queue.sync {
            value += 1
        }
    }
}

See Adding items to Swift array across multiple threads causing issues
(because arrays aren’t thread safe) - how do I get around
that?
or GCD with static
functions of a struct for more sophisticated examples. This thread
What advantage(s) does dispatch_sync have over
@synchronized? is also very interesting.



 类似资料:
  • 问题内容: 就像在MYSQL中一样,我想要一个递增的ID。 问题答案: MongoDB旨在进行水平扩展。在这种情况下,自动递增将导致id冲突。这就是为什么id看起来更像guid / uuid的原因。

  • 问题内容: 我的架构看起来像这样: 我已经在同一数据库中创建了counters集合,并添加了一个带有’entityId’的_id的页面。从这里我不确定如何使用猫鼬来更新该页面并获取递增编号。 没有计数器的架构,我希望它保持这种状态,因为这实际上不是应用程序使用的实体。仅应在模式中使用它来自动递增字段。 问题答案: 这是一个示例,如何在Mongoose中实现自动增量字段:

  • 问题内容: 如何设置从1001开始的MySQL表中“ id”列的初始值? 我想做一个插入 无需为id列指定初始值。 问题答案: 用这个: 或者,如果您还没有添加ID列,也可以添加它

  • 问题内容: 我正在为购物车制作模型,我需要创建一个在下订单时自动递增的字段: 如何使自动递增? 问题答案: 在Django中 1:我们具有名称为“ id”的默认字段,该字段为自动递增。 2:您可以使用 字段定义自动递增 字段。 如果要使用django的默认ID作为增量字段。 数据库设计

  • 问题内容: 我有一个带有主键字段且具有AUTO_INCREMENT的MySQL表。在阅读了这里的其他文章后,我发现人们遇到了同样的问题,答案也各不相同。有些人建议不要使用此功能,其他人则说它不能被“修复”。 我有: 示例:表中的记录数:18.如果我删除记录16、17和18-我希望输入的下一条记录的courseID为16,但是由于最后输入的courseID为18,所以它将为19。 我的SQL知识不是

  • 问题内容: 我有一个使用hibernate和注解的j2ee应用程序。我如何注释我的pojo类中的Id字段以将其设置为自动增量或自动生成。在添加bean时,我是否将该字段留在bean中为null? 问题答案: 并在保留时将其保留为。(如果使用包装器) 在某些情况下,该策略被解析为或,因此你可能需要手动将其设置为(取决于基础数据库)。 似乎+ 指定序列名称对你有用。