• 设为首页
  • 点击收藏
  • 手机版
    手机扫一扫访问
    迪恩网络手机版
  • 关注官方公众号
    微信扫一扫关注
    迪恩网络公众号

ios - SqlCommand 对象的构造函数和 Dispose 方法在 Mono 中是线程安全的吗?

[复制链接]
菜鸟教程小白 发表于 2022-12-13 09:58:52 | 显示全部楼层 |阅读模式 打印 上一主题 下一主题

以下是我在 iOS xamarin 应用程序中使用的一些数据访问代码的简化表示:

public async Task<List<Foo>> GetAllFoos()
{
    var foos = new List<Foo>();
    using(CommandWrapper command = GetCommandWrapper("SELECT Id, Name, Rank FROM Foos"))//this creates SqliteCommand object and puts it in wrapper object
    {
        using(ReaderWrapper reader = ExecuteReaderAsync(_Connection, command))
        {
            while(reader.Read())
            {
                var foo = ConstructFoo(reader);//construct foo with column values read from reader
                foos.Add(foo);
            }
        }
    }
    return foos;
}

private SemaphoreSlim _commandLockSemaphore = new SemaphoreSlim(1, 1);//only 1 thread at once

private async ExecuteReaderAsync(Mono.Data.Sqlite.SqliteConnection connection, CommandWrapper command)
{
    if(!(await _SemaphoreLock.WaitAsync(_TimeoutTimespan))) throw new Exception("timeout");

    //lock now held

    Mono.Data.Sqlite.SqliteCommand sqliteCommand = command.UnderlyingCommand;

    try
    {
        sqliteCommand.Connection = connection;
        IDataReader rawReader = await sqliteCommand.ExecuteReaderAsync();
    }
    catch(Exception)
    {
        //emergency lock release
        _SemaphoreLock.Release();
        throw;
    }
    return new ReaderWrapper(){DataReader = rawReader, SemaphoreLock = _SemaphoreLock};
}

internal class ReaderWrapper : IDisposable
{
    internal IDataReader DataReader{get; set;}
    internal SemaphoreSlim SemaphoreLock{get; set;}

    //... [read methods]

    public void Dispose()
    {
        DataReader.Dispose();
        SemaphoreLock.Release();
    }
}

由于我们知道 Sqlite 不支持连接上的多个线程,因此我们将 SemaphoreSlim 放在适当的位置以确保连接的使用和任何命令的执行仅发生在单个线程中。 SemaphoreSlim 随后会在阅读器用尽并处置后释放。

但是,当我有多个线程调用 GetAllFoos 时,应用程序会崩溃并显示以下内容:

2015-08-19 14:33:22.296 MyCoolIosApp[8421:5311923] critical: Stacktrace:
2015-08-19 14:33:22.296 MyCoolIosApp[8421:5311923] critical:   at  
2015-08-19 14:33:22.297 MyCoolIosApp[8421:5311923] critical:   at (wrapper managed-to-native) Mono.Data.Sqlite.UnsafeNativeMethods.sqlite3_prepare (intptr,intptr,int,intptr&,intptr&) 
2015-08-19 14:33:22.297 MyCoolIosApp[8421:5311923] critical:   at Mono.Data.Sqlite.SQLite3.Prepare (Mono.Data.Sqlite.SqliteConnection,string,Mono.Data.Sqlite.SqliteStatement,uint,string&) [0x00044] in //Library/Frameworks/Xamarin.iOS.framework/Versions/8.10.4.46/src/mono/mcs/class/Mono.Data.Sqlite/Mono.Data.Sqlite_2.0/SQLite3.cs:268
2015-08-19 14:33:22.298 MyCoolIosApp[8421:5311923] critical:   at Mono.Data.Sqlite.SqliteCommand.BuildNextCommand () [0x00019] in //Library/Frameworks/Xamarin.iOS.framework/Versions/8.10.4.46/src/mono/mcs/class/Mono.Data.Sqlite/Mono.Data.Sqlite_2.0/SQLiteCommand.cs:230
2015-08-19 14:33:22.298 MyCoolIosApp[8421:5311923] critical:   at Mono.Data.Sqlite.SqliteCommand.GetStatement (int) [0x0000b
] in //Library/Frameworks/Xamarin.iOS.framework/Versions/8.10.4.46/src/mono/mcs/class/Mono.Data.Sqlite/Mono.Data.Sqlite_2.0/SQLiteCommand.cs:264
2015-08-19 14:33:22.299 MyCoolIosApp[8421:5311923] critical:   at Mono.Data.Sqlite.SqliteDataReader.NextResult () [0x000cc] in //Library/Frameworks/Xamarin.iOS.framework/Versions/8.10.4.46/src/mono/mcs/class/Mono.Data.Sqlite/Mono.Data.Sqlite_2.0/SQLiteDataReader.cs:914
2015-08-19 14:33:22.299 MyCoolIosApp[8421:5311923] critical:   at Mono.Data.Sqlite.SqliteDataReader..ctor (Mono.Data.Sqlite.SqliteCommand,System.Data.CommandBehavior) [0x00051] in //Library/Frameworks/Xamarin.iOS.framework/Versions/8.10.4.46/src/mono/mcs/class/Mono.Data.Sqlite/Mono.Data.Sqlite_2.0/SQLiteDataReader.cs:89
2015-08-19 14:33:22.300 MyCoolIosApp[8421:5311923] critical:   at Mono.Data.Sqlite.SqliteCommand.ExecuteReader (System.Data.CommandBehavior) [0x00006] in //Library/Frameworks/Xamarin.iOS.framework/Versions/8.10.4.46/src/mono/mcs/class/Mono.Data.Sqlite/Mono.Data.Sqlite_2.0/SQLiteCommand.cs:539
2015-08-19 14:33:22.300 MyCoolIosApp[8421:5311923] critical:   at Mono.Data.Sqlite.SqliteCommand.ExecuteReader () [0x00000] in //Library/Frameworks/Xamarin.iOS.framework/Versions/8.10.4.46/src/mono/mcs/class/Mono.Data.Sqlite/Mono.Data.Sqlite_2.0/SQLiteCommand.cs:551
...

According to various discussions, that error is caused by too many threads on Sqlite on iOS devices. I can corroborate this since it crashes 100% of the time if I have a pair of threads attempt to endlessly call GetAllFoos in a while loop; and also when I rearrange the locks (e.g. add a simple lock statement around construction and disposal of the command), it fixes the issue:

public async Task<List<Foo>> GetAllFoos()
{
    var foos = new List<Foo>();
    lock(_someStaticObject)//why is this needed
    {
        using(CommandWrapper command = GetCommandWrapper("SELECT Id, Name, Rank FROM Foos"))//this creates SqliteCommand object and puts it in wrapper object
        {
            using(ReaderWrapper reader = ExecuteReaderAsync(_Connection, command))
            {
                while(reader.Read())
                {
                    var foo = ConstructFoo(reader);//construct foo with column values read from reader
                    foos.Add(foo);
                }
            }
        }
    }
    return foos;
}

据我所知,SqliteCommand 上的 Dispose 方法(可能还有构造函数)导致了并发问题。

SqliteCommand 对象的构造函数和 Dispose 方法在 Mono 中是线程安全的吗?或者我是否需要将一个或两个都考虑为用于锁定目的的关键部分?



Best Answer-推荐答案


使用 Mono 的好处在于它是开源的,所以让我们回答您的问题“SqliteCommand 对象的构造函数和 Dispose 方法在 Mono 中是线程安全的吗?”

如果我们在这里快速浏览一下:

https://github.com/mono/mono/blob/master/mcs/class/Mono.Data.Sqlite/Mono.Data.Sqlite_2.0/SQLiteCommand.cs#L108

这里:

https://github.com/mono/mono/blob/master/mcs/class/Mono.Data.Sqlite/Mono.Data.Sqlite_2.0/SQLiteCommand.cs#L153

...那么我们可以着重回答“不,Mono 的 SQLiteCommand 的构造函数和 Dispose 都不是线程安全的”。

就此而言,假设任何地方都没有线程安全是正确的,除非它通过文档明确说明。即便如此,有时它也是一个谎言。

(现在微软的东西也是开源的,所以我们可以在 Mono 之外用同样的方式回答这些问题。耶!)

关于ios - SqlCommand 对象的构造函数和 Dispose 方法在 Mono 中是线程安全的吗?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/32104351/

回复

使用道具 举报

懒得打字嘛,点击右侧快捷回复 【右侧内容,后台自定义】
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

关注0

粉丝2

帖子830918

发布主题
阅读排行 更多
广告位

扫描微信二维码

查看手机版网站

随时了解更新最新资讯

139-2527-9053

在线客服(服务时间 9:00~18:00)

在线QQ客服
地址:深圳市南山区西丽大学城创智工业园
电邮:jeky_zhao#qq.com
移动电话:139-2527-9053

Powered by 互联科技 X3.4© 2001-2213 极客世界.|Sitemap