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

Posted

技术标签:

【中文标题】iOS 上的 JavaScriptCore:VM 垃圾收集器不会自动清空【英文标题】:JavaScriptCore on iOS: VM Garbage Collector not automatically emtpying 【发布时间】:2015-05-25 18:20:54 【问题描述】:

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

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

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 的图片,显示了分配情况:

我希望我能弄清楚我的问题出在哪里,我希望有人知道我该如何解决它......

提前致谢, 最大

【问题讨论】:

【参考方案1】:

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

【讨论】:

以上是关于iOS 上的 JavaScriptCore:VM 垃圾收集器不会自动清空的主要内容,如果未能解决你的问题,请参考以下文章

iOS9 上的 JavaScriptCore 崩溃

iOS上的javascriptcore框架可以接入网络吗?

iOS js oc相互调用(JavaScriptCore)

转载 iOS js oc相互调用(JavaScriptCore)

JavaScriptCore - 在 iOS 中访问 DOMParser

JavaScript:浅谈iOS与H5的交互-JavaScriptCore框架