- Java 双重比较
- java - 比较器与 Apache BeanComparator
- Objective-C 完成 block 导致额外的方法调用?
- database - RESTful URI 是否应该公开数据库主键?
我一直在努力效仿这个例子:http://blog.softeq.com/2012/12/using-pre-populated-sqlite-database-in.html
此示例旨在向您展示如何打开存储在 Assets 文件夹中的预填充数据库。
已经创建了与此主题相关的线程,我已经尝试了提出的建议,但它并没有帮助我弄清楚为什么程序没有打开数据库文件。我究竟做错了什么?这是我在 LogCat 的前几行中得到的:
01-13 15:54:33.171: E/Trace(30747): error opening trace file: No such file or directory (2)
01-13 15:54:33.171: D/ActivityThread(30747): setTargetHeapUtilization:0.25
01-13 15:54:33.171: D/ActivityThread(30747): setTargetHeapIdealFree:8388608
01-13 15:54:33.171: D/ActivityThread(30747): setTargetHeapConcurrentStart:2097152
01-13 15:54:33.251: D/AbsListView(30747): Get MotionRecognitionManager
01-13 15:54:33.261: E/SQLiteLog(30747): (14) cannot open file at line 30245 of [00bb9c9ce4]
01-13 15:54:33.261: E/SQLiteLog(30747): (14) os_unix.c:30245: (2) open(/data/data/com.example.prepopdb/databases/yourdb.db) -
01-13 15:54:33.261: E/SQLiteDatabase(30747): Failed to open database '/data/data/com.example.prepopdb/databases/yourdb.db'.
01-13 15:54:33.261: E/SQLiteDatabase(30747): android.database.sqlite.SQLiteCantOpenDatabaseException: unknown error (code 14): Could not open database
01-13 15:54:33.261: E/SQLiteDatabase(30747): at android.database.sqlite.SQLiteConnection.nativeOpen(Native Method)
01-13 15:54:33.261: E/SQLiteDatabase(30747): at android.database.sqlite.SQLiteConnection.open(SQLiteConnection.java:278)
01-13 15:54:33.261: E/SQLiteDatabase(30747): at android.database.sqlite.SQLiteConnection.open(SQLiteConnection.java:217)
01-13 15:54:33.261: E/SQLiteDatabase(30747): at android.database.sqlite.SQLiteConnectionPool.openConnectionLocked(SQLiteConnectionPool.java:464)
01-13 15:54:33.261: E/SQLiteDatabase(30747): at android.database.sqlite.SQLiteConnectionPool.open(SQLiteConnectionPool.java:186)
01-13 15:54:33.261: E/SQLiteDatabase(30747): at android.database.sqlite.SQLiteConnectionPool.open(SQLiteConnectionPool.java:178)
01-13 15:54:33.261: E/SQLiteDatabase(30747): at android.database.sqlite.SQLiteDatabase.openInner(SQLiteDatabase.java:804)
01-13 15:54:33.261: E/SQLiteDatabase(30747): at android.database.sqlite.SQLiteDatabase.open(SQLiteDatabase.java:789)
01-13 15:54:33.261: E/SQLiteDatabase(30747): at android.database.sqlite.SQLiteDatabase.openDatabase(SQLiteDatabase.java:694)
01-13 15:54:33.261: E/SQLiteDatabase(30747): at android.database.sqlite.SQLiteDatabase.openDatabase(SQLiteDatabase.java:669)
01-13 15:54:33.261: E/SQLiteDatabase(30747): at com.example.prepopdb.ExternalDbOpenHelper.checkDataBase(ExternalDbOpenHelper.java:62)
01-13 15:54:33.261: E/SQLiteDatabase(30747): at com.example.prepopdb.ExternalDbOpenHelper.createDataBase(ExternalDbOpenHelper.java:43)
01-13 15:54:33.261: E/SQLiteDatabase(30747): at com.example.prepopdb.ExternalDbOpenHelper.openDataBase(ExternalDbOpenHelper.java:97)
01-13 15:54:33.261: E/SQLiteDatabase(30747): at com.example.prepopdb.ExternalDbOpenHelper.<init>(ExternalDbOpenHelper.java:37)
01-13 15:54:33.261: E/SQLiteDatabase(30747): at com.example.prepopdb.PrepopSqliteDbActivity.onCreate(PrepopSqliteDbActivity.java:35)
我使用上面链接中提到的 SQLite 数据库浏览器创建了测试数据库,除了我的数据库扩展名是“.db”而不是“.sqlite3”。 (虽然我也尝试过使用“.sqlite3”,但有类似/相同的错误)。
这是我的 PrepopSqliteDbActivity 的代码:
package com.example.prepopdb;
import com.example.prepopdb.ExternalDbOpenHelper;
import java.util.ArrayList;
import android.app.ListActivity;
import android.database.Cursor;
import android.database.sqlite.SQLiteDatabase;
import android.os.Bundle;
import android.view.View;
import android.widget.AdapterView;
import android.widget.AdapterView.OnItemClickListener;
import android.widget.ArrayAdapter;
import android.widget.ListView;
import android.widget.TextView;
import android.widget.Toast;
public class PrepopSqliteDbActivity extends ListActivity {
private static final String DB_NAME = "yourdb.db";
// Database field names
private static final String TABLE_NAME = "friends";
private static final String FRIEND_ID = "_id";
private static final String FRIEND_NAME = "name";
private SQLiteDatabase database;
private ListView listView;
private ArrayList<String> friends;
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
ExternalDbOpenHelper dbOpenHelper = new ExternalDbOpenHelper(this,
DB_NAME);
database = dbOpenHelper.openDataBase(); // Database is open
/*SOME OTHER CODE HERE*/
}
这是我的 ExternalDbOpenHelper 的代码:
package com.example.prepopdb;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import android.content.Context;
import android.database.SQLException;
import android.database.sqlite.SQLiteDatabase;
import android.database.sqlite.SQLiteOpenHelper;
import android.os.Environment;
import android.util.Log;
public class ExternalDbOpenHelper extends SQLiteOpenHelper {
// Path to the device folder with database
public static String DB_PATH;
// Database filename
public static String DB_NAME;
public SQLiteDatabase database;
public final Context context;
public SQLiteDatabase getDb() {
return database;
}
public ExternalDbOpenHelper(Context context, String databaseName) {
super(context, databaseName, null, 1);
this.context = context;
// Write the full path to the databases of your application
String packageName = context.getPackageName();
DB_PATH = String.format("%s/data/%s/databases/", Environment.getDataDirectory(), packageName);
DB_NAME = databaseName;
openDataBase();
}
// Create a database if its not yet created
public void createDataBase() {
boolean dbExist = checkDataBase();
if (!dbExist) {
this.getReadableDatabase();
try {
copyDataBase();
} catch (IOException e) {
Log.e(this.getClass().toString(), "Copying error!");
throw new Error("Error copying database!");
}
} else {
Log.i(this.getClass().toString(), "Database already exists");
}
}
//Performing a database existence check
private boolean checkDataBase(){
SQLiteDatabase checkDb = null;
try {
String path = DB_PATH + DB_NAME;
checkDb = SQLiteDatabase.openDatabase(path, null, SQLiteDatabase.OPEN_READONLY);
} catch (SQLException e){
Log.e(this.getClass().toString(), "Error while checking db");
}
if (checkDb != null){
checkDb.close();
}
return checkDb !=null;
}
private void copyDataBase() throws IOException {
//Open a stream for reading, located in the assets
InputStream externalDbStream = context.getAssets().open(DB_NAME);
//Path to the created empty database on your Android device
String outFileName = DB_PATH + DB_NAME;
//Create a stream for writing the database byte by byte
OutputStream localDbStream = new FileOutputStream(outFileName);
//Copy the database
byte[] buffer = new byte[1024];
int bytesRead;
while((bytesRead = externalDbStream.read(buffer)) > 0){
localDbStream.write(buffer, 0, bytesRead);
}
//Close the streams
localDbStream.close();
externalDbStream.close();
}
SQLiteDatabase openDataBase() throws SQLException {
String path = DB_PATH + DB_NAME;
if (database == null) {
createDataBase();
database = SQLiteDatabase.openDatabase(path, null,
SQLiteDatabase.OPEN_READWRITE);
//database = getWritableDatabase();
}
return database;
}
@Override
public synchronized void close(){
if (database != null){
database.close();
}
super.close();
}
@Override
public void onCreate(SQLiteDatabase db) {
// TODO Auto-generated method stub
}
@Override
public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {
// TODO Auto-generated method stub
}
}
更新
谢谢卡洛斯。我同意对路径进行硬编码不是一个好主意,因为它可能因设备而异。我按照您的建议进行了更改,但是我仍然遇到相同的错误。我已验证 yourdb.db 文件具有所有级别的读/写权限。
我在之前的评论中提到了这一点,但是否会因为我使用的是非 root 手机而发生这种情况?将“yourdb.db”的权限从 Asssets 文件夹复制到/data/data/com.example.prepopdb/databases/然后我的程序试图打开它后,它的权限可能会改变吗?我知道如何使用 adb 更改权限,但如何在程序中执行此操作?
这是我的 Logcat 所说的内容:
01-14 12:18:02.240: E/Trace(18167): error opening trace file: No such file or directory (2)
01-14 12:18:02.240: D/ActivityThread(18167): setTargetHeapUtilization:0.25
01-14 12:18:02.240: D/ActivityThread(18167): setTargetHeapIdealFree:8388608
01-14 12:18:02.240: D/ActivityThread(18167): setTargetHeapConcurrentStart:2097152
01-14 12:18:02.360: D/AbsListView(18167): Get MotionRecognitionManager
01-14 12:18:02.370: E/SQLiteLog(18167): (14) cannot open file at line 30245 of [00bb9c9ce4]
01-14 12:18:02.370: E/SQLiteLog(18167): (14) os_unix.c:30245: (2) open(/data/data/com.example.prepopdb/databases/yourdb.db) -
01-14 12:18:02.370: E/SQLiteDatabase(18167): Failed to open database '/data/data/com.example.prepopdb/databases/yourdb.db'.
01-14 12:18:02.370: E/SQLiteDatabase(18167): android.database.sqlite.SQLiteCantOpenDatabaseException: unknown error (code 14): Could not open database
01-14 12:18:02.370: E/SQLiteDatabase(18167): at android.database.sqlite.SQLiteConnection.nativeOpen(Native Method)
01-14 12:18:02.370: E/SQLiteDatabase(18167): at android.database.sqlite.SQLiteConnection.open(SQLiteConnection.java:278)
01-14 12:18:02.370: E/SQLiteDatabase(18167): at android.database.sqlite.SQLiteConnection.open(SQLiteConnection.java:217)
01-14 12:18:02.370: E/SQLiteDatabase(18167): at android.database.sqlite.SQLiteConnectionPool.openConnectionLocked(SQLiteConnectionPool.java:464)
01-14 12:18:02.370: E/SQLiteDatabase(18167): at android.database.sqlite.SQLiteConnectionPool.open(SQLiteConnectionPool.java:186)
01-14 12:18:02.370: E/SQLiteDatabase(18167): at android.database.sqlite.SQLiteConnectionPool.open(SQLiteConnectionPool.java:178)
01-14 12:18:02.370: E/SQLiteDatabase(18167): at android.database.sqlite.SQLiteDatabase.openInner(SQLiteDatabase.java:804)
01-14 12:18:02.370: E/SQLiteDatabase(18167): at android.database.sqlite.SQLiteDatabase.open(SQLiteDatabase.java:789)
01-14 12:18:02.370: E/SQLiteDatabase(18167): at android.database.sqlite.SQLiteDatabase.openDatabase(SQLiteDatabase.java:694)
01-14 12:18:02.370: E/SQLiteDatabase(18167): at android.database.sqlite.SQLiteDatabase.openDatabase(SQLiteDatabase.java:669)
01-14 12:18:02.370: E/SQLiteDatabase(18167): at com.example.prepopdb.ExternalDbOpenHelper.checkDataBase(ExternalDbOpenHelper.java:63)
01-14 12:18:02.370: E/SQLiteDatabase(18167): at com.example.prepopdb.ExternalDbOpenHelper.createDataBase(ExternalDbOpenHelper.java:45)
01-14 12:18:02.370: E/SQLiteDatabase(18167): at com.example.prepopdb.ExternalDbOpenHelper.openDataBase(ExternalDbOpenHelper.java:97)
01-14 12:18:02.370: E/SQLiteDatabase(18167): at com.example.prepopdb.ExternalDbOpenHelper.<init>(ExternalDbOpenHelper.java:39)
01-14 12:18:02.370: E/SQLiteDatabase(18167): at com.example.prepopdb.PrepopSqliteDbActivity.onCreate(PrepopSqliteDbActivity.java:35)
01-14 12:18:02.370: E/SQLiteDatabase(18167): at android.app.Activity.performCreate(Activity.java:5048)
01-14 12:18:02.370: E/SQLiteDatabase(18167): at android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1094)
01-14 12:18:02.370: E/SQLiteDatabase(18167): at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2052)
01-14 12:18:02.370: E/SQLiteDatabase(18167): at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2113)
01-14 12:18:02.370: E/SQLiteDatabase(18167): at android.app.ActivityThread.access$700(ActivityThread.java:139)
01-14 12:18:02.370: E/SQLiteDatabase(18167): at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1224)
01-14 12:18:02.370: E/SQLiteDatabase(18167): at android.os.Handler.dispatchMessage(Handler.java:99)
01-14 12:18:02.370: E/SQLiteDatabase(18167): at android.os.Looper.loop(Looper.java:137)
01-14 12:18:02.370: E/SQLiteDatabase(18167): at android.app.ActivityThread.main(ActivityThread.java:4918)
01-14 12:18:02.370: E/SQLiteDatabase(18167): at java.lang.reflect.Method.invokeNative(Native Method)
01-14 12:18:02.370: E/SQLiteDatabase(18167): at java.lang.reflect.Method.invoke(Method.java:511)
01-14 12:18:02.370: E/SQLiteDatabase(18167): at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:1004)
01-14 12:18:02.370: E/SQLiteDatabase(18167): at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:771)
01-14 12:18:02.370: E/SQLiteDatabase(18167): at dalvik.system.NativeStart.main(Native Method)
01-14 12:18:02.370: E/class com.example.prepopdb.ExternalDbOpenHelper(18167): Error while checking db
01-14 12:18:02.480: D/libEGL(18167): loaded /system/lib/egl/libEGL_adreno200.so
01-14 12:18:02.500: D/libEGL(18167): loaded /system/lib/egl/libGLESv1_CM_adreno200.so
01-14 12:18:02.500: D/libEGL(18167): loaded /system/lib/egl/libGLESv2_adreno200.so
01-14 12:18:02.540: I/Adreno200-EGLSUB(18167): <ConfigWindowMatch:2087>: Format RGBA_8888.
01-14 12:18:02.560: E/(18167): <s3dReadConfigFile:75>: Can't open file for reading
01-14 12:18:02.560: E/(18167): <s3dReadConfigFile:75>: Can't open file for reading
01-14 12:18:02.560: D/OpenGLRenderer(18167): Enabling debug mode 0
01-14 12:18:21.380: W/IInputConnectionWrapper(18167): beginBatchEdit on inactive InputConnection
01-14 12:18:21.380: W/IInputConnectionWrapper(18167): endBatchEdit on inactive InputConnection
最佳答案
您对数据文件夹的假设太多了。您宁愿让系统为您提供正确的路径。您可以使用 context.getDatabasePath();
获取数据库路径,您将所需的名称传递给文件(无论它是否存在)。实际上你应该那样做
为此,您可以在任何地方使用 String outFileName = DB_PATH + DB_NAME;
你应该改变它
String outFileName =myContext.getDatabasePath(DB_NAME).getPath() ;
这是您文件的真正有效位置。所以我们的助手会像这样
public class ExternalDbOpenHelper extends SQLiteOpenHelper {
// Path to the device folder with database
public static String DB_PATH;
// Database filename
public static String DB_NAME;
public SQLiteDatabase database;
public final Context context;
public SQLiteDatabase getDb() {
return database;
}
public ExternalDbOpenHelper(Context context, String databaseName) {
super(context, databaseName, null, 1);
this.context = context;
DB_NAME = databaseName;
DB_PATH =context.getDatabasePath(DB_NAME).getPath() ;
openDataBase();
}
// Create a database if its not yet created
public void createDataBase() {
boolean dbExist = checkDataBase();
if (!dbExist) {
this.getReadableDatabase();
try {
copyDataBase();
} catch (IOException e) {
Log.e(this.getClass().toString(), "Copying error!");
throw new Error("Error copying database!");
}
} else {
Log.i(this.getClass().toString(), "Database already exists");
}
}
//Performing a database existence check
private boolean checkDataBase(){
SQLiteDatabase checkDb = null;
try {
checkDb = SQLiteDatabase.openDatabase(DB_PATH, null, SQLiteDatabase.OPEN_READONLY);
} catch (SQLException e){
Log.e(this.getClass().toString(), "Error while checking db");
}
if (checkDb != null){
checkDb.close();
}
return checkDb !=null;
}
private void copyDataBase() throws IOException {
//Open a stream for reading, located in the assets
InputStream externalDbStream = context.getAssets().open(DB_NAME);
//Create a stream for writing the database byte by byte
OutputStream localDbStream = new FileOutputStream(DB_PATH);
//Copy the database
byte[] buffer = new byte[1024];
int bytesRead;
while((bytesRead = externalDbStream.read(buffer)) > 0){
localDbStream.write(buffer, 0, bytesRead);
}
//FLUSH THE OUT STREAM
localDbStream.flush();
//Close the streams
localDbStream.close();
externalDbStream.close();
}
SQLiteDatabase openDataBase() throws SQLException {
if (database == null) {
createDataBase();
database = SQLiteDatabase.openDatabase(DB_PATH, null,
SQLiteDatabase.OPEN_READWRITE);
//database = getWritableDatabase();
}
return database;
}
@Override
public synchronized void close(){
if (database != null){
database.close();
}
super.close();
}
@Override
public void onCreate(SQLiteDatabase db) {
// TODO Auto-generated method stub
}
@Override
public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {
// TODO Auto-generated method stub
}
}
有了这个它应该可以工作
更新
您可以考虑一些注意事项:
查看系统为您的数据库返回的路径
DB_PATH =context.getDatabasePath(DB_NAME).getPath() ;
Log.i ("myApp", DB_PATH);
检查模拟器中的代码,看看您是否遇到同样的问题。在那里你可以手动检查文件是否实际创建,如解释的那样 here
添加 permissions写入和读取 SD 卡(READ_EXTERNAL_STORAGE 和 WRITE_EXTERNAL_STORAGE)。但我真的不相信你需要这个
关于android - 将位于 Assets 文件夹中的预填充数据库放入 data/data/mypackage/databases,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/21101515/
我在使用NetBeans 6.8时遇到以下问题。我通过项目属性->库->编译选项卡->添加JAR /文件夹添加带有jar的文件夹。在下一个窗口中,我选择文件夹,然后选择“复制到库文件夹”。但是,我仍然
我的网站有一个域别名。我想知道如何将 domainA.ext 的请求重定向到 https://domainA.ext/folderA和对 domainB.ext 的请求到 http://domainB
我应该在 Eclipse 中构建的 Android 项目中创建自己的自定义菜单文件夹吗?例如,我想创建一种出现在所有 Activity 中的标题。我知道菜单应该在 res/menu 文件夹中的 XML
我正在使用 VS2008 和 .net 3.5。我在我的解决方案中创建了一个类库(Myproject.Controllers)。在这个类下,我添加了一个 Controllers 文件夹。在文件夹中我添
我有一个包含生成后步骤的 Visual Studio 2012 扩展项目,我想在其中将 .dll 和 .AddIn 文件复制到当前用户的 Visual Studio 2012 AddIns 文件夹中。
我在专有的 linux 发行版中有一些自动下载。 他们去临时暂存盘。我想在它们完成后将它们 move 到主 RAID 阵列。我能看到的最好方法是检查磁盘上的文件夹,看看内容是否在最后一分钟发生了变化。
我目前正在使用 SVN 对我的软件项目进行版本控制。在一个正在进行的项目中,我有主干,用于客户的共同功能和规范以及分支,用于客户特定的。 有没有办法在每次执行此类操作时标记一些不应合并到分支中的文
这个问题在这里已经有了答案: How to exclude a directory in find . command (45 个回答) 8 年前关闭。 如何删除文件夹中的所有内容并排除特定文件夹和文
如何在特定目录中创建具有当前日期和时间的文件夹或文件? DateTimeFormatter f = DateTimeFormatter.ofPattern("uuuuMMdd HHmmss") ; L
有没有办法在系统文件资源管理器的左侧“文件夹”栏中打开文件或文件夹?如果没有这个,我必须打开文件资源管理器并一直导航到该文件夹所在的位置才能操作文件,这确实很不方便。对于大多数带有这样导航栏的工具
预期:我使用 go get 安装包,它在 src 文件夹中创建了所有必要的文件夹,但它们只出现在 pkg/mod 文件夹中,我不能使用它们。 现实:它说它正在下载,完成,然后什么都没有。 一切都在 W
说 foo.zip包含: a b c |- c1.exe |- c2.dll |- c3.dll 哪里a, b, c是文件夹。 如果我 Expand-Archive .\foo.zip -Destin
不久前我正在删除 var 文件夹中 Magento 的缓存。我可能是错的,但我认为我犯了一个错误,而不是删除 var/cache 中的所有内容,而是意外删除了 var 中的所有内容。 Magento
我在 svn 存储库的单独文件夹中有一些代码项目。 现在我在删除文件时遇到一些问题:大多数时候一切顺利,但有时当我从磁盘删除文件或文件夹时, checkin 过程会出现各种错误。 所以我想知道:在sv
有没有什么方法可以用很少的R命令行自动删除所有文件或文件夹?我知道 unlink() 或 file.remove() 函数,但对于这些函数,您需要定义一个字符向量,其中包含您想要的文件的所有名称删除。
用于在文件夹中查找不符合Get-Childitem的LastWriteTime过滤器日期范围标准的文件的powershell命令是什么? 因此,请检查目录中是否包含不包含在01/10/2012(十月1
我正在为我工作的公司内部使用的应用程序之一编写 NSIS 安装程序,安装过程工作正常,所有 REG 键都已创建,文件夹和服务也没有问题,该应用程序使用。出于某种我无法理解的原因,卸载过程不起作用。
我有一个 Excel 文件,并且在同一文件夹中还有一个包含我想要包含的 CSV 文件的文件夹。使用“来自文件夹”查询,第一步将给出以下查询: = Folder.Files("D:\OneDrive\D
我在docker中玩ScyllaDB。为了使ScyllaDB在docker生产设置中最有效地运行,它需要一个XFS格式的磁盘。 您知道如何在Linux和MacO中创建XFS容器卷,磁盘文件吗? 谢谢
我应该编写一个函数,其中包含之前每次与该数字相乘的乘积 基本上是这样的: > productFromLeftToRight [2,3,4,5] [120,60,20,5] 我应该使用高阶函数,例如折叠
我是一名优秀的程序员,十分优秀!