如何从字符串末尾删除标志?

How to remove a flag from the end of a string?

我发现函数 String.characters.count 有一个奇怪的行为,其中行是表情符号标志:

import UIKit

var flag = ""
print(flag.characters.count)
print(flag.unicodeScalars.count)
print(flag.utf16.count)
print(flag.utf8.count)
flag = "000"
print(flag.characters.count)
print(flag.unicodeScalars.count)
print(flag.utf16.count)
print(flag.utf8.count)

我想在 UITextView 中编写和编辑时限制文本的字符串长度。 实际上我的代码是这样的:

var lastRange: NSRange? = nil
var lastText: String? = nil

func textView(textView: UITextView, shouldChangeTextInRange range: NSRange, replacementText string: String) -> Bool {
    if string == "\n" {
        // Execute same code
        return false
    } 
    var text = string.uppercaseString
    if lastText != text || lastRange != nil && (lastRange!.location != range.location || lastRange!.length != range.length) {
        lastRange = range
        lastText = text

        var text = (self.textView.text ?? "" as NSString).stringByReplacingCharactersInRange(range, withString: string)

        // Delete chars if length more kMaxLengthText 
        while text.utf16.count >= kMaxLengthText {
            text.removeAtIndex(text.endIndex.advancedBy(-1))
        }
        // Set position after insert text
        self.textView.selectedRange = NSRange(location: range.location + lastText!.utf16.count, length: 0)
    }
    return false
}

更新 Swift 4 (Xcode 9)

截至 Swift 4(使用 Xcode 9 beta 测试)标志(即区域对 指标)被视为一个单一的字形簇,由 Unicode 9 标准。所以计算标志并删除最后一个 字符(无论它是否是标志)现在就像:

var flags = ""
print(flags.count) // 6

flags.removeLast()
print(flags.count) // 5
print(flags) // 

(Swift 3 及更早版本的旧答案:)

没有错误。 "Regional Indicator" 个字符的序列是 一个 "extended grapheme cluster",这就是为什么

var flag = ""
print(flag.characters.count)

打印 1(比较 Swift countElements() return incorrect value when count flag emoji)。

另一方面,上面的字符串由 12 个 Unicode 标量组成 ( 是 + ),并且它们每个都需要两个 UTF-16 代码点。

要将字符串分隔成 "visible entities",您必须 考虑 "composed character sequences",比较 .

我没有一个优雅的解决方案(也许有人有更好的解决方案)。 但一种选择是将字符串分隔成一个数组 组合字符,必要时从数组中删除元素, 然后再次组合字符串。

示例:

extension String {

    func composedCharacters() -> [String] {
        var result: [String] = []
        enumerateSubstringsInRange(characters.indices, options: .ByComposedCharacterSequences) {
            (subString, _, _, _) in
            if let s = subString { result.append(s) }
        }
        return result
    }
}

var flags = ""
var chars = flags.composedCharacters()
print(chars.count) // 6
chars.removeLast()
flags = chars.joinWithSeparator("")
print(flags) // 

@Martin 对 swift3 的回答,有一些修改

注意:我已将扩展程序转换为函数

func composedCharacters(str:String) -> [String] {
    var result: [String] = []
    str.enumerateSubstrings(in: str.startIndex..<str.endIndex,options: .byComposedCharacterSequences) {
        (subString, _, _, _) in
        if let s = subString { result.append(s) }
    }
    return result
}
var flags = ""
var chars = composedCharacters(str:flags)
print(chars.count) // 6
chars.removeLast()
flags = chars.joined(separator:"")
print(flags) //