Ответ 1
Обновление для Swift 2.1:
Вы можете создать String
из массива символов UTF-16
с
public init(utf16CodeUnits: UnsafePointer<unichar>, count: Int)
инициализатор. Пример:
let str = "H€llo 😄"
// String to UTF16 array:
let utf16array = Array(str.utf16)
print(utf16array)
// Output: [72, 8364, 108, 108, 111, 32, 55357, 56836]
// UTF16 array to string:
let str2 = String(utf16CodeUnits: utf16array, count: utf16array.count)
print(str2)
// H€llo 😄
Предыдущий ответ:
Нет ничего "встроенного" (насколько я знаю), но вы можете использовать структуру UTF16
который предоставляет метод decode()
:
extension String {
init?(utf16chars:[UInt16]) {
var str = ""
var generator = utf16chars.generate()
var utf16 : UTF16 = UTF16()
var done = false
while !done {
let r = utf16.decode(&generator)
switch (r) {
case .EmptyInput:
done = true
case let .Result(val):
str.append(Character(val))
case .Error:
return nil
}
}
self = str
}
}
Пример:
let str = "H€llo 😄"
// String to UTF16 array:
let utf16array = Array(str.utf16)
print(utf16array)
// Output: [72, 8364, 108, 108, 111, 32, 55357, 56836]
// UTF16 array to string:
if let str2 = String(utf16chars: utf16array) {
print(str2)
// Output: H€llo 😄
}
Немного более общий, вы можете определить метод, который создает строку из массива (или любой последовательности) кодовых точек, используя данный кодек:
extension String {
init?<S : SequenceType, C : UnicodeCodecType where S.Generator.Element == C.CodeUnit>
(codeUnits : S, var codec : C) {
var str = ""
var generator = codeUnits.generate()
var done = false
while !done {
let r = codec.decode(&generator)
switch (r) {
case .EmptyInput:
done = true
case let .Result(val):
str.append(Character(val))
case .Error:
return nil
}
}
self = str
}
}
Затем преобразование из UTF16 выполняется как
if let str2a = String(codeUnits: utf16array, codec: UTF16()) {
print(str2a)
}
Вот еще одно возможное решение. Хотя предыдущие методы являются "чистыми Swift", в этом используется Foundation Foundation и автоматическая
мост между NSString
и Swift String
:
extension String {
init?(utf16chars:[UInt16]) {
let data = NSData(bytes: utf16chars, length: utf16chars.count * sizeof(UInt16))
if let ns = NSString(data: data, encoding: NSUTF16LittleEndianStringEncoding) {
self = ns as String
} else {
return nil
}
}
}