gpt4 book ai didi

android - 使用 ContentProvider 时为 "error code 5: database is locked"

转载 作者:IT王子 更新时间:2023-10-29 06:25:04 27 4
gpt4 key购买 nike

我有一个应用程序在启动并绑定(bind)到 Activity 的单独进程中运行带有服务的 Activity 。该服务包含一个处理程序,该处理程序发布一个可运行的对象以在延迟后运行。

我希望每个组件都记录到数据库中,因此我实现了一个处理数据库访问的内容提供程序,并通过扩展的 AsyncTask 子类从服务或 Activity 中调用它。

这一切在模拟器上运行得很好,但是当我在我的手机上调试运行它时,我在我的数据库写入时遇到一个零星的数据库锁定错误:

更新

我对我的数据库处理做了一些更改,错误发生了轻微的变化。

ERROR/Database(15235): Error inserting MY_MESSAGE
ERROR/Database(15235): android.database.sqlite.SQLiteException: error code 5: database is locked
ERROR/Database(15235): at android.database.sqlite.SQLiteStatement.native_execute(Native Method)
ERROR/Database(15235): at android.database.sqlite.SQLiteStatement.execute(SQLiteStatement.java:61)
ERROR/Database(15235): at android.database.sqlite.SQLiteDatabase.insertWithOnConflict(SQLiteDatabase.java:1591)
ERROR/Database(15235): at android.database.sqlite.SQLiteDatabase.insert(SQLiteDatabase.java:1435)
ERROR/Database(15235): at mypackagename.DatabaseHelper.insertLogging(DatabaseHelper.java:190)
ERROR/Database(15235): at mypackagename.ContentProvider.insert(ContentProvider.java:139)
ERROR/Database(15235): at android.content.ContentProvider$Transport.insert(ContentProvider.java:198)
ERROR/Database(15235): at android.content.ContentResolver.insert(ContentResolver.java:604)
ERROR/Database(15235): at mypackagename.Activity$LogToDatabase.doInBackground(Activity.java:642)
ERROR/Database(15235): at mypackagename.Activity$LogToDatabase.doInBackground(Activity.java:1)
ERROR/Database(15235): at android.os.AsyncTask$2.call(AsyncTask.java:185)
ERROR/Database(15235): at java.util.concurrent.FutureTask$Sync.innerRun(FutureTask.java:306)
ERROR/Database(15235): at java.util.concurrent.FutureTask.run(FutureTask.java:138)
ERROR/Database(15235): at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1088)
ERROR/Database(15235): at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:581)
ERROR/Database(15235): at java.lang.Thread.run(Thread.java:1019)

我之前没有详细说明,因为我认为这是不同进程或线程的问题,但现在我认为问题更可能出在调用数据库的代码中。

问题:

1) 为什么我在使用 ContentProvider 时会遇到锁定?
2) 为什么这不会出现在等效的 API 2.3.3 模拟器上?
3) 我的代码都没有捕获到异常是否意味着错误得到了正确处理并且我可以忽略它?
4)我在另一个地方读到有人建议调整忙碌超时。我该怎么做?

具有讽刺意味的是,导致错误的是我的调试日志记录,但我并没有忘记。

如果我不能解决它,我的下一步是将日志记录消息 bundle 在一个列表中,并一次以十个为一组转储出来。

这是错误代码的路径:

Activity :

private void logDatabaseMessage(String status, String message)
{
String[] args = {status, message};
LogToDatabase logTask = new LogToDatabase();
logTask.execute(args);
}

private class LogToDatabase extends AsyncTask<String, Integer, Void>
{
@Override
protected Void doInBackground(final String... args)
{
try
{
SimpleDateFormat dateFormat = new SimpleDateFormat("yyyy-MM-dd'T'HH:mm:ss.SSS");
String dateText = dateFormat.format(new Date());

ContentValues loggingValues = new ContentValues();
loggingValues.put(MyContentProvider.LOGGING_DATETIME, dateText);
loggingValues.put(MyContentProvider.LOGGING_STATUS, args[0]);
loggingValues.put(MyContentProvider.LOGGING_MESSAGE, args[1]);
getContentResolver().insert(MyContentProvider.LOGGING_CONTENT_URI, loggingValues);
}
catch (Exception ex)
{
Log.e(TAG, "LogToDatabase.doInBackground threw exception: " + ex.getMessage());
ex.printStackTrace();
}

return null;
}
}

内容提供者:

@Override
public Uri insert(Uri uri, ContentValues values)
{
Uri _uri = null;
long rowID = 0;

try
{
switch (uriMatcher.match(uri))
{
case LOGGING:
rowID = dbHelper.insertLogging(values);
if (rowID == 0)
throw new SQLException("Failed to insert row into " + uri);

_uri = ContentUris.withAppendedId(LOGGING_CONTENT_URI, rowID);
break;

default: throw new SQLException("Failed to insert row into " + uri);
}

if (rowID != 0)
getContext().getContentResolver().notifyChange(_uri, null);
}
catch (Exception ex)
{
Log.e(TAG, LogPrefix + "insert threw exception: " + ex.getMessage());
ex.printStackTrace();
}

return _uri;
}

数据库助手:

public long insertLogging(ContentValues values)
{
long rowID = 0;

try
{
rowID = db.insert(LOGGING_TABLE, null, values);
}
catch (Exception ex)
{
Log.e(TAG, LogPrefix + "ERROR: Failed to insert into logging table: " + ex.getMessage());
ex.printStackTrace();
}

return rowID;
}

最佳答案

您是否可能使用多个 SQLiteDatabase(或者可能是 SQLiteOpenHelper)实例访问数据库?

您只能与数据库建立一个连接,否则您将遇到您遇到的错误。

SQLiteDatabase 本身是线程安全的,因此您可以并发访问它。

关于android - 使用 ContentProvider 时为 "error code 5: database is locked",我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/8360400/

27 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com