iOS 上的 JavaScriptCore:VM 垃圾收集器不会自动清空

JavaScriptCore on iOS: VM Garbage Collector not automatically emtpying

我正在与 JavaScriptCore 合作进行 iOS Swift 项目。 我现在想使用 XCodes 内部 XCTest 框架测试我的应用程序。 现在,当我调用与 JavaScriptCore class 一起使用的方法时,它会生成一个 JSContext + JSVM,将我需要的东西交给我,仅此而已。但是当我在一个循环中调用这个方法时,JS 垃圾收集器似乎没有清空,而是保持所有调用打开,这导致测试冻结在 VM 的大约 6k 持久分配:JS 垃圾收集器大约 364MB。即使我写了一个手动 deinitiliaze 方法,这仍然会发生。

下面是我测试中的代码class:

let vc = TestTwoViewController()
let hexValues = ["00", "01", "02", "03", "04", "05", "06", "07", "08", "09", "0A", "0B", "0C", "0D", "0E", "0F", "10", "11", "12", "13", "14", "15", "16", "17", "18", "19", "1A", "1B", "1C", "1D", "1E", "1F", "20", "21", "22", "23", "24", "25", "26", "27", "28", "29", "2A", "2B", "2C", "2D", "2E", "2F", "30", "31", "32", "33", "34", "35", "36", "37", "38", "39", "3A", "3B", "3C", "3D", "3E", "3F", "40", "41", "42", "43", "44", "45", "46", "47", "48", "49", "4A", "4B", "4C", "4D", "4E", "4F", "50", "51", "52", "53", "54", "55", "56", "57", "58", "59", "5A", "5B", "5C", "5D", "5E", "5F", "60", "61", "62", "63", "64", "65", "66", "67", "68", "69", "6A", "6B", "6C", "6D", "6E", "6F", "70", "71", "72", "73", "74", "75", "76", "77", "78", "79", "7A", "7B", "7C", "7D", "7E", "7F", "80", "81", "82", "83", "84", "85", "86", "87", "88", "89", "8A", "8B", "8C", "8D", "8E", "8F", "90", "91", "92", "93", "94", "95", "96", "97", "98", "99", "9A", "9B", "9C", "9D", "9E", "9F", "A0", "A1", "A2", "A3", "A4", "A5", "A6", "A7", "A8", "A9", "AA", "AB", "AC", "AD", "AE", "AF", "B0", "B1", "B2", "B3", "B4", "B5", "B6", "B7", "B8", "B9", "BA", "BB", "BC", "BD", "BE", "BF", "C0", "C1", "C2", "C3", "C4", "C5", "C6", "C7", "C8", "C9", "CA", "CB", "CC", "CD", "CE", "CF", "D0", "D1", "D2", "D3", "D4", "D5", "D6", "D7", "D8", "D9", "DA", "DB", "DC", "DD", "DE", "DF", "E0", "E1", "E2", "E3", "E4", "E5", "E6", "E7", "E8", "E9", "EA", "EB", "EC", "ED", "EE", "EF", "F0", "F1", "F2", "F3", "F4", "F5", "F6", "F7", "F8", "F9", "FA", "FB", "FC", "FD", "FE", "FF"]
for i in 1...2 {
    var timestamps = [String]()
    println("Starting loop \(i)")
    for n in hexValues {
        println("current hex value: \(n)")
        let value = vc.checkForValidHexValue(n)
        var start = NSDate.timeIntervalSinceReferenceDate()
        vc.workWithJSBridge(value.description)
        var intervall = NSDate.timeIntervalSinceReferenceDate() - start
        timestamps.append(intervall.description)
    }
    vc.workWithJSBridge(nil)
    println("Finished loop \(i)") 

调用:

func workWithJSBridge(value: String?) -> String {
    if (value != nil) {
        jsi = JSBridge(methName: method, val: value)
        return jsi!.inputValue!
    } else {
        jsi = nil
        return "Deinitilized JS Bridge"
    }
}

然后进入我的 JavaScript 文件:

var methodName: String?
var inputValue: String?
var jsSource: String?
var jsvm: JSVirtualMachine?
var ctx: JSContext?
var test: JSValue?

init(methName: String?, val: String?) {

    let homeDir: String? = NSBundle.mainBundle().resourcePath
    let jsFileName: String? = "/RefScenLibJS.js"
    let jsFilePath = homeDir!+jsFileName!
    var error: NSError?
    jsSource = String(contentsOfFile: jsFilePath, encoding: NSUTF8StringEncoding, error: &error)

    methodName = methName
    inputValue = val
    callMethod(methodName!)
}

deinit {
    println("Manually deinitialize JSBridge")
}

public func callMethod(methodName: String) {
    if methodName == "getValueForInput" {
        createJSEnvironment()
        checkForValue()
    } else if methodName == "getCalcValue" {
        createJSEnvironment()
        calcTestValue()
    } else {
        println("No method found with name: \(methodName)")
    }
}

func createJSEnvironment() {
    // create javascript virtual machine and context and evaluate javascript
    jsvm = JSVirtualMachine()
    ctx = JSContext(virtualMachine: jsvm)
    ctx!.exceptionHandler = { ctx, exception in println("JS Error: \(exception)") }
    ctx!.evaluateScript(jsSource)
}

func calcTestValue() {
    // get function(s)
    let getValueForInputFunction = ctx!.objectForKeyedSubscript(testTwo)
    // use function(s)
    if (!getValueForInputFunction.isUndefined() && inputValue != nil) {
        // call function with parameter
        let valFromJS = getValueForInputFunction.callWithArguments([inputValue!])
        inputValue = valFromJS.description
    } else {
        println("Function not found")
    }
}

这是 Instruments 的图像,显​​示了分配情况:

我希望我能说清楚我的问题出在哪里,我希望有人知道我该如何摆脱它...

提前致谢, 最大值

所以这是我自己的问题,因为我每次从测试中调用该函数时都在初始化一个新的 JavaScript VM。我通过在我的 JSBridge class 开始时初始化一个 VM 来解决这个问题,瞧,每个调用都得到了完美的处理。但似乎仍然无法从 JavaScriptCore 中手动取消初始化 JavaScript 虚拟机。我把这个(在我陷入答案之前)作为一个错误推送给了 WebKit 团队,也许有人会调查这个?! 这是 WebKit Bug Tracker 的 link:https://bugs.webkit.org/show_bug.cgi?id=145433