2

我知道这是一份提案草案。我尝试实现一个简单的 DSL 来构建字符串,如下所示:

@_functionBuilder
struct StringBuilder {
    static func buildExpression(_ string: String) -> [String] {
        [string]
    }
    static func buildBlock(_ children: [String]...) -> [String] {
        children.flatMap{ $0 }
    }
}

func s(separator: String = "", @StringBuilder _ makeString: () -> [String]) -> String {
    makeString().joined(separator: separator)
}

let z = s(separator: " ") {
   "this"
   "is"
   "cool"
}

但是,编译器抱怨“'String' 不能转换为 '[String]'”。这使我相信这buildBlock是目前实施的提案的唯一部分。(这是可以理解的,因为在 SwiftUI 中他们正在构建视图的层次结构,所以这就是他们所需要的。)

这是正确的还是我做错了什么?正确的使用方法是buildExpression什么?

ielyamani 的回答展示了如何构建一个有效的字符串构建器,例如我在上面的示例中使用的。但是,这并不能解决实际问题。我不是想建立一个字符串生成器。我正在尝试找出函数构建器。字符串生成器只是一个示例。例如,如果我们希望有一个接受整数的字符串生成器,理论上我们可以执行以下操作:

@_functionBuilder
struct StringBuilder {
    static func buildExpression(_ int: Int) -> [String] {
        ["\(int)"]
    }

    // The rest of it implemented just as above
}

在这种情况下,当编译器遇到 时Int,它会调用buildExpression然后吐出我们的组件类型,在这种情况下是[String]。但正如 Martin R 在对此问题的评论中所说,buildExpression目前尚未实施。

4

2 回答 2

2

我今天遇到了同样的问题,似乎 buildExpression 没有实现。我最终通过使用协议“ComponentProtocol”然后创建“Expression:ComponentProtocol”和“Component:ComponentProtocol”来解决问题。现在对我有用。我希望它会在以后实施。

protocol ComponentProtocol: ExpressibleByIntegerLiteral, ExpressibleByStringLiteral  {
    var value: String { get }
}

struct Expression: ComponentProtocol {
    let _value: String
    var value: String { _value }
    init(_ value: String) { _value = value }
    init(integerLiteral value: Int) { self.init(value) }
    init(stringLiteral  value: String) { self.init(value) }
    init<E: CustomStringConvertible>(_ value: E) {_value = String(describing: value) }
}

struct Component: ComponentProtocol {
    let _values: [String]
    var value: String { _values.joined(separator: ", ") }
    init(integerLiteral value: Int) { self.init(value) }
    init(stringLiteral  value: String) { self.init(value) }
    init<E: CustomStringConvertible>(_ value: E) { _values = [String(describing: value)] }
    init<T: ComponentProtocol>(_ values: T...) { _values = values.map { $0.value } }
    init<T: ComponentProtocol>(_ values: [T]) { _values = values.map { $0.value } }
}

@_functionBuilder struct StringReduceBuilder {
    static func buildBlock<T: ComponentProtocol>(_ components: T ...) -> Component { Component(components) }

    static func buildEither<T: ComponentProtocol>(first: T) -> Component { Component(first.value) }
    static func buildEither<T: ComponentProtocol>(second: T) -> Component { Component(second.value) }
    static func buildOptional<T: ComponentProtocol>(_ component: T?) -> Component? {
        component == nil ? nil : Component(component!.value)
    }
}

func stringsReduce (@StringReduceBuilder block: () -> Component) -> Component {
    return block()
}

let result = stringsReduce {
    Expression(3)
    "one"
    Expression(5)
    Expression("2")
    83
}

let s2 = stringsReduce {
    if .random () { // random value Bool
        Expression(11)
    } else {
        Expression("another one")
    }
}
于 2019-07-21T16:57:32.720 回答
1

由于buildBlock(_:)需要可变数量的字符串数组,这将起作用:

let z = s(separator: " ") {
    ["this"]
    ["is"]
    ["cool"]
}

但这仍然很笨拙。要获取字符串而不是字符串数组,请添加此函数以StringBuilder获取可变数量的字符串:

static func buildBlock(_ strings: String...) -> [String] {
    Array(strings)
}

现在你可以这样做了:

let z = s(separator: " ") {
    "Hello"
    "my"
    "friend!"
}

print(z)    //Hello my friend!
于 2019-06-08T01:38:35.533 回答