- Java 双重比较
- java - 比较器与 Apache BeanComparator
- Objective-C 完成 block 导致额外的方法调用?
- database - RESTful URI 是否应该公开数据库主键?
在我的一个 Android 应用程序中,我使用了一个预填充的数据库,我在第一次使用时复制了该数据库。到目前为止,这在所有 Android 版本上运行良好,但对于 API 28,它会失败。它确实复制了数据库,之后它可以连接到数据库,但由于某种原因,无法访问任何表。我得到该表不存在的错误。
我在模拟器上下载了数据库,检查了一下,数据库没有问题。我在代码中创建数据库的另一个应用程序运行良好(创建数据库后可以找到表)。
我用来复制数据库的代码:
public void copyDatabase() throws IOException{
InputStream myInput = mContext.getAssets().open(mDbSource);
// Path to the just created empty db
String outFileName = getDbPath() + mDbName;
//Open the empty db as the output stream
OutputStream myOutput = new FileOutputStream(outFileName);
//transfer bytes from the inputfile to the outputfile
byte[] buffer = new byte[1024];
int length;
while ((length = myInput.read(buffer))>0){
myOutput.write(buffer, 0, length);
}
//Close the streams
myOutput.flush();
myOutput.close();
myInput.close();
}
private String getDbPath(){
return "/data/data/"+mContext.getResources().getString(R.string.package_name)+"/databases/";
}
API 28 中是否有任何更改导致此操作因某种原因失败?这可能是什么原因造成的?
最佳答案
API 28 开启的问题是 WAL(预写日志记录)在默认情况下处于开启状态。通常情况下,如果打开数据库(通常在检查数据库是否存在时完成),则会创建两个文件 -wal 和 -shm 文件。
例如在您的代码中,这似乎表明这就是您正在做的事情:-
// Path to the just created empty db
String outFileName = getDbPath() + mDbName;
当现有数据库被副本覆盖时,-wal 和 -shm 文件仍然存在。
当最终尝试将数据库作为 SQLiteDatabase 打开时,会引发数据库与 -wal 和 -shm 文件之间的不匹配。底层的 SQLiteDatabase open 然后决定数据库无法打开,因此,为了提供可用的数据库,删除并重新创建数据库有效地清空数据库,因此出现这种情况。
三种解决方法
我建议 2 或 3 是更好的方法:-
以上内容均在Ship android app with pre populated database中涵盖.
解决方案代码实际上结合了 2 和 3(尽管应该不需要修复 3(因为修复 2 不会将数据库作为 SQLiteDatabase 打开))。
说我相信以下用于检查数据库是否存在的代码(修复 2)比上面答案中的等效代码更好:-
/**
* Check if the database already exists. NOTE will create the databases folder is it doesn't exist
* @return true if it exists, false if it doesn't
*/
public static boolean checkDataBase(Context context, String dbname) {
File db = new File(context.getDatabasePath(dbname).getPath()); //Get the file name of the database
Log.d("DBPATH","DB Path is " + db.getPath()); //TODO remove if publish App
if (db.exists()) return true; // If it exists then return doing nothing
// Get the parent (directory in which the database file would be)
File dbdir = db.getParentFile();
// If the directory does not exits then make the directory (and higher level directories)
if (!dbdir.exists()) {
db.getParentFile().mkdirs();
dbdir.mkdirs();
}
return false;
}
关于java - 在 Oreo 上复制 SQLite 数据库,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/56241576/
在 Android Oreo 中查找应用程序是否从最近列表中被杀死的替代方法是什么? 要求:应用程序终止状态必须保存在本地数据库中。 是否有任何可用的替代方法或任何库来监视最近被终止的应用程序 Act
我们有私有(private)应用商店,所以我们不需要设置目标 sdk 26(oreo) 或更高版本。 那么如果我将 target sdk 25 设置到我的应用程序,那么即使在 OREO 手机中它也会没
我正在实现代码以在 android Oreo 中获取来电号码。为此,我在服务中使用广播接收器。这是我到目前为止完成的代码。 public class CallService extends Servi
DatePickerDialog 的文本颜色在 oreo(谷歌像素)设备中变为白色,如下图所示,这在 oreo 之前的设备中看起来很完美。 抽屉菜单字体也变为白色,如下图所示,这在 oreo 之前的设
将build.gradle依赖更改为后 compile 'com.android.support:design:26.1.0' 如前所述here ,项目构建成功但是当我尝试运行它时会出现类似的错误
我用 startService() 命令启动了 LocationService,但是因为我了解到当应用程序处于后台时 Oreo 不会接受此服务,所以我切换到 作业计划, 我在 lollipop 和 J
我们现在将我们的应用程序从 Android 25 迁移到 compileSdkVersion 26 buildToolsVersion "26.0.0" 我也修改了所有的依赖版本。 我们仍在我们的应用
我正在使用后台服务获取用户位置并上传到服务器。 我使用了一个扩展 Service 类的 LocationTrack 类和扩展 BroadcastReciver 的 AlramReciver。 我在 l
我正在开发一个任务安排应用程序,它可以在特定时间提醒我某项任务。我曾经使用 getSystemService(AlarmManager.class).set(RTC_WAKEUP, millis, p
我有一个 SMS 监听器,它在 Oreo (API 26) 之前的版本中运行良好,但在 API 26 中似乎被忽略。根据文档,SMS_RECEIVED_ACTION 不受隐式广播限制 ( link )
是否可以更改 EditText 自动填充突出显示颜色?默认情况下,它具有可能与应用程序设计不兼容的黄色背景。 最佳答案 根据 Android 文档,要更改突出显示颜色,您必须编辑应用程序主题。您可以将
这个问题在这里已经有了答案: What are the background location limits in Android 8 (Oreo)? (1 个回答) 关闭去年。 所以,简而言之,我
我正在尝试为 API > 26 添加自定义声音通知。下面是代码 NotificationChannel notificationChannel = new NotificationChannel("c
我有一个应用程序,其中有一个名为“Crilee”的字体,它在其他设备上运行良好,但在 Android 的 Oreo 版本中运行不佳。我使用 settypeface() 来设置字体。我能知道为什么某些字
到目前为止,我已经调整了我的代码以使用 ContextCompat.startForegroundService(context, intentService); 来启动我的服务。这样,它也适用于 a
我有一个网络更改接收器类,它从广播接收器扩展,但我不太确定它是否在 android Oreo 中工作,奥利奥是否支持广播接收器,如果它不支持,还有什么方法可以做到 最佳答案 Oreo 支持广播接收器,
我的 Android 应用程序在 Android oreo 版本的白色圆圈内显示启动器图标。 我想显示启动器图标,因为默认情况下它在奥利奥设备上就像正方形。 浏览了各种博客,发现: 1) 移动用户可以
所以我对 Java 和 Android 都很陌生,正在为一个项目开发一个应用程序。在应用程序中,我希望拥有自己的图像库/查看器,专门用于我们将使用应用程序保存的图像。因此,我想使用 ViewPager
我尝试使用以下代码获取浏览器包,我有两个网络浏览器,UC Browser 和 Google Chrome。但这并没有给我返回任何浏览器。这仅发生在 Android Oreo 为 Google Pixe
我正在尝试在 Oreo 中共享一个音频文件。如果文件在设备的内部存储中,它运行正常,但如果文件存在于外部存储中,它会崩溃并给出此异常 - android.os.FileUriExposedExcept
我是一名优秀的程序员,十分优秀!