问题原因

升级Xcode 16 之后,真机运行APP,发现会有Crash,崩溃堆栈线上Crash 在 YYCache 之中。如下图所示
Xcode16 编译运行YYCache iOS18 sqlite3_finalize 闪退问题解决方案-LMLPHP
Xcode16 编译运行YYCache iOS18 sqlite3_finalize 闪退问题解决方案-LMLPHP
崩溃堆栈如下:

* thread #1, queue = 'com.apple.main-thread', stop reason = signal SIGABRT
    frame #0: 0x00000001d93911d4 libsystem_kernel.dylib`__pthread_kill + 8
    frame #1: 0x00000002114d6ef8 libsystem_pthread.dylib`pthread_kill + 268
    frame #2: 0x0000000190d83ad8 libsystem_c.dylib`abort + 128
    frame #3: 0x0000000190d82e08 libsystem_c.dylib`__assert_rtn + 284
    frame #4: 0x000000010500f7d8 libRPAC.dylib`isBulkReadStatement.cold.3 + 100
    frame #5: 0x0000000104f80c88 libRPAC.dylib`isBulkReadStatement + 360
    frame #6: 0x0000000105005254 libRPAC.dylib`interposed_sqlite3_finalize + 152
  * frame #7: 0x0000000111d7adc0 CsdnPlus.debug.dylib`-[YYKVStorage _dbClose](self=0x000000013448ccc0, _cmd="_dbClose") at YYKVStorage.m:138:21
    frame #8: 0x0000000111d7ee84 CsdnPlus.debug.dylib`-[YYKVStorage dealloc](self=0x000000013448ccc0, _cmd="dealloc") at YYKVStorage.m:731:5
    frame #9: 0x0000000111d77d8c CsdnPlus.debug.dylib`-[YYDiskCache _appWillBeTerminated](self=0x0000000134f91200, _cmd="_appWillBeTerminated") at YYDiskCache.m:155:9
    frame #10: 0x0000000188fc32f4 CoreFoundation`__CFNOTIFICATIONCENTER_IS_CALLING_OUT_TO_AN_OBSERVER__ + 148
    frame #11: 0x0000000188fc3210 CoreFoundation`___CFXRegistrationPost_block_invoke + 88
    frame #12: 0x0000000188fc3158 CoreFoundation`_CFXRegistrationPost + 436
    frame #13: 0x0000000188fc170c CoreFoundation`_CFXNotificationPost + 728
    frame #14: 0x0000000187be4ea4 Foundation`-[NSNotificationCenter postNotificationName:object:userInfo:] + 92
    frame #15: 0x000000018c897df0 UIKitCore`-[UIApplication _terminateWithStatus:] + 232
    frame #16: 0x000000018b8d6f88 UIKitCore`-[_UISceneLifecycleMultiplexer _evalTransitionToSettings:fromSettings:forceExit:withTransitionStore:] + 112
    frame #17: 0x000000018c0b78f4 UIKitCore`-[_UISceneLifecycleMultiplexer forceExitWithTransitionContext:scene:] + 164
    frame #18: 0x000000018c89354c UIKitCore`-[UIApplication workspaceShouldExit:withTransitionContext:] + 164
    frame #19: 0x00000001a25d1f68 FrontBoardServices`__63-[FBSWorkspaceScenesClient willTerminateWithTransitionContext:]_block_invoke_2 + 72
    frame #20: 0x00000001a2591c44 FrontBoardServices`-[FBSWorkspace _calloutQueue_executeCalloutFromSource:withBlock:] + 168
    frame #21: 0x00000001a25d1f08 FrontBoardServices`__63-[FBSWorkspaceScenesClient willTerminateWithTransitionContext:]_block_invoke + 128
    frame #22: 0x000000010563a71c libdispatch.dylib`_dispatch_client_callout + 20
    frame #23: 0x000000010563e13c libdispatch.dylib`_dispatch_block_invoke_direct + 296
    frame #24: 0x00000001a2593300 FrontBoardServices`__FBSSERIALQUEUE_IS_CALLING_OUT_TO_A_BLOCK__ + 52
    frame #25: 0x00000001a2593280 FrontBoardServices`-[FBSMainRunLoopSerialQueue _targetQueue_performNextIfPossible] + 240
    frame #26: 0x00000001a2593158 FrontBoardServices`-[FBSMainRunLoopSerialQueue _performNextFromRunLoopSource] + 28
    frame #27: 0x0000000188fc8328 CoreFoundation`__CFRUNLOOP_IS_CALLING_OUT_TO_A_SOURCE0_PERFORM_FUNCTION__ + 28
    frame #28: 0x0000000188fc82bc CoreFoundation`__CFRunLoopDoSource0 + 176
    frame #29: 0x0000000188fc5dc0 CoreFoundation`__CFRunLoopDoSources0 + 244
    frame #30: 0x0000000188fc4fbc CoreFoundation`__CFRunLoopRun + 840
    frame #31: 0x0000000188fc4830 CoreFoundation`CFRunLoopRunSpecific + 588
    frame #32: 0x00000001d4fa41c4 GraphicsServices`GSEventRunModal + 164
    frame #33: 0x000000018bb2aeb0 UIKitCore`-[UIApplication _run] + 816
    frame #34: 0x000000018bbd95b4 UIKitCore`UIApplicationMain + 340
    frame #35: 0x000000010fae959c CsdnPlus.debug.dylib`main(argc=3, argv=0x000000016b2e7438) at main.m:14:16
    frame #36: 0x00000001ae9b2ec8 dyld`start + 2724

解决方案一

@implementation YYKVStorage

- (BOOL)_dbClose {
   ...
    // 原代码
    if (_dbStmtCache) CFRelease(_dbStmtCache);

    // 替换为
    if (_dbStmtCache) {              
        CFIndex size = CFDictionaryGetCount(_dbStmtCache);
        CFTypeRef *valuesRef = (CFTypeRef *)malloc(size * sizeof(CFTypeRef));
        CFDictionaryGetKeysAndValues(_dbStmtCache, NULL, (const void **)valuesRef);
        const sqlite3_stmt **stmts = (const sqlite3_stmt **)valuesRef;
        for (CFIndex i = 0; i < size; i ++) {
            sqlite3_stmt *stmt = stmts[i];
            sqlite3_finalize(stmt);
        }
        free(valuesRef);
        CFRelease(_dbStmtCache);
    }
    ...
}

解决方案二

static void _finalizeStatement(const void *key, const void *value, void *context) {
    sqlite3_finalize((sqlite3_stmt *)value);
}

- (BOOL)_dbClose {
    ···
    
   // 原代码
    if (_dbStmtCache) CFRelease(_dbStmtCache);
   // 替换为
    if (_dbStmtCache) {              
        CFDictionaryApplyFunction(_dbStmtCache, _finalizeStatement, NULL);
        CFRelease(_dbStmtCache);
    }
    
    ···
}
10-24 12:01