- Java 双重比较
- java - 比较器与 Apache BeanComparator
- Objective-C 完成 block 导致额外的方法调用?
- database - RESTful URI 是否应该公开数据库主键?
我正在尝试使用本教程在我的 Android 应用程序中使用我自己创建的数据库 http://www.reigndesign.com/blog/using-your-own-sqlite-database-in-android-applications/
我的 Assets 文件夹中有我的数据库已经工作了 5 个小时来解决问题但徒劳无功..我总是不断得到
01-17 04:09:07.111: E/Database(1060): sqlite3_open_v2("/data/data/com.rahul.besttracker/databases/route", &handle, 1, NULL) failed
和
01-17 04:09:07.271: E/AndroidRuntime(1060): java.lang.RuntimeException: Unable to start activity ComponentInfo{com.rahul.besttracker/com.rahul.besttracker.Busdisplay}: android.database.sqlite.SQLiteException: unable to open database file
我的代码
public class Busdisplay extends ListActivity {
TextView source, destination;
String src, dest;
ArrayList<String> mArrayList;
@Override
protected void onCreate(Bundle savedInstanceState) {
// TODO Auto-generated method stub
super.onCreate(savedInstanceState);
init();
Bundle caught = getIntent().getExtras();
src = caught.getString("source");
dest = caught.getString("dest");
DataBaseHelper entry = null;
entry = new DataBaseHelper(Busdisplay.this);
entry.openDataBase();
mArrayList = entry.readEntry(src, dest);
entry.close();
setListAdapter(new ArrayAdapter<String>(this,
android.R.layout.simple_list_item_1, mArrayList));
}
@Override
protected void onListItemClick(ListView l, View v, int position, long id) {
// TODO Auto-generated method stub
super.onListItemClick(l, v, position, id);
}
void init() {
source = (TextView) findViewById(R.id.textView1);
destination = (TextView) findViewById(R.id.textView2);
}
}
我的 DBhelper 类
public class DataBaseHelper extends SQLiteOpenHelper {
// The Android's default system path of your application database.
private static final String DB_PATH = "/data/data/com.rahul.besttracker/databases/";
private static final String DB_NAME = "route.db";
private static final String DB_TABLE1 = "route1";
private static final String DB_TABLE2 = "route2";
private SQLiteDatabase myDataBase;
private final Context myContext;
/**
* Constructor Takes and keeps a reference of the passed context in order to
* access to the application assets and resources.
*
* @param context
*/
public DataBaseHelper(Context context) {
super(context, DB_NAME, null, 1);
this.myContext = context;
}
/**
* Creates a empty database on the system and rewrites it with your own
* database.
* */
public void createDataBase() throws IOException {
boolean dbExist = checkDataBase();
if (dbExist) {
// do nothing - database already exist
} else {
// By calling this method and empty database will be created into
// the default system path
// of your application so we are gonna be able to overwrite that
// database with our database.
this.getReadableDatabase();
try {
copyDataBase();
} catch (IOException e) {
throw new Error("Error copying database");
}
}
}
/**
* Check if the database already exist to avoid re-copying the file each
* time you open the application.
*
* @return true if it exists, false if it doesn't
*/
private boolean checkDataBase() {
SQLiteDatabase checkDB = null;
try {
String myPath = DB_PATH + DB_NAME;
checkDB = SQLiteDatabase.openDatabase(myPath, null,
SQLiteDatabase.OPEN_READONLY);
} catch (SQLiteException e) {
System.out.print("ERROR");
}
if (checkDB != null) {
checkDB.close();
}
return checkDB != null ? true : false;
}
/**
* Copies your database from your local assets-folder to the just created
* empty database in the system folder, from where it can be accessed and
* handled. This is done by transfering bytestream.
* */
private void copyDataBase() throws IOException {
// Open your local db as the input stream
InputStream myInput = myContext.getAssets().open(DB_NAME);
// Path to the just created empty db
String outFileName = DB_PATH + DB_NAME;
// 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();
}
public void openDataBase() throws SQLException {
// Open the database
String myPath = DB_PATH + DB_NAME;
myDataBase = SQLiteDatabase.openDatabase(myPath, null,
SQLiteDatabase.OPEN_READONLY);
}
@Override
public synchronized void close() {
if (myDataBase != null)
myDataBase.close();
super.close();
}
@Override
public void onCreate(SQLiteDatabase db) {
try {
createDataBase();
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
@Override
public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {
db.execSQL("DROP TABLE IF EXISTS " + DB_TABLE1);
db.execSQL("DROP TABLE IF EXISTS " + DB_TABLE2);
onCreate(db);
}
public ArrayList<String> readEntry(String src, String dest) {
// TODO Auto-generated method stub
Cursor c = myDataBase.rawQuery("SELECT route_no from " + DB_TABLE1
+ " WHERE stops LIKE '%" + src + "%,%" + dest + "%';", null);
ArrayList<String> mArrayList = new ArrayList<String>();
int index = c.getColumnIndex("route_no");
for (c.moveToFirst(); !c.isAfterLast(); c.moveToNext()) {
// The Cursor is now set to the right position
mArrayList.add(c.getString(index));
}
return mArrayList;
}
// Add your public helper methods to access and get content from the
// database.
// You could return cursors by doing "return myDataBase.query(....)" so it'd
// be easy
// to you to create adapters for your views.
}
最佳答案
我什至遇到了同样的问题,对此感到非常沮丧,即使我使用了本教程,但它没有用。我搜索了很多关于这个的话题并找到了一个......即使这不起作用......但我做了一个简单的改变。首先,创建两个类 DatabaseHelper 和 CopyAdapter:
public class DatabaseHelper extends SQLiteOpenHelper{
private static String TAG = "TAG";
private static String DB_PATH = "/data/data/PLACE_HERE_YOUR_INTERNAL_PATH/databases/";
private static String DB_NAME = "PLACE_HERE_YOUR_DATABASE_NAME_WITH_EXTENSION";
private SQLiteDatabase mDataBase;
private final Context mContext;
public DatabaseHelper(Context context)
{
super(context, DB_NAME, null, 1);
DB_PATH = "/data/data/" + context.getPackageName() + "/databases/";
this.mContext = context;
}
public void createDataBase() throws IOException
{
boolean mDataBaseExist = checkDataBase();
Log.d(TAG,"create DB in Helper. Data exists?"+mDataBaseExist);
if(!mDataBaseExist)
{
Log.d(TAG,"get Writable in DatabaseHelper");
this.getWritableDatabase();
try
{
Log.d(TAG,"copy Database");
copyDataBase();
}
catch (IOException mIOException)
{Log.d(TAG,"copy not succeed");
throw new Error("ErrorCopyingDataBase");
}
}
}
private boolean checkDataBase()
{
SQLiteDatabase mCheckDataBase = null;
try
{
String myPath = DB_PATH + DB_NAME;
mCheckDataBase = SQLiteDatabase.openDatabase(myPath, null, SQLiteDatabase.NO_LOCALIZED_COLLATORS);
}
catch(SQLiteException mSQLiteException)
{
Log.e(TAG, "DatabaseNotFound " + mSQLiteException.toString());
}
if(mCheckDataBase != null)
{
mCheckDataBase.close();
}
return mCheckDataBase != null;
}
private void copyDataBase() throws IOException
{
Log.d(TAG,"copy");
InputStream mInput = mContext.getResources().getAssets().open(DB_NAME);
String outFileName = DB_PATH + DB_NAME;
Log.d(TAG,"Output:"+outFileName);
File createOutFile = new File(outFileName);
if(!createOutFile.exists()){
createOutFile.mkdir();
}
OutputStream mOutput = new FileOutputStream(outFileName);
byte[] mBuffer = new byte[1024];
int mLength;
while ((mLength = mInput.read(mBuffer))>0)
{
mOutput.write(mBuffer, 0, mLength);
}
mOutput.flush();
mOutput.close();
mInput.close();
}
public boolean openDataBase() throws SQLException
{
String mPath = DB_PATH + DB_NAME;
mDataBase = SQLiteDatabase.openDatabase(mPath, null, SQLiteDatabase.NO_LOCALIZED_COLLATORS);
return mDataBase != null;
}
@Override
public synchronized void close()
{
if(mDataBase != null)
mDataBase.close();
super.close();
}
@Override
public void onCreate(SQLiteDatabase db)
{ }
@Override
public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion)
{
Log.vTAG, "UpgradingDatabase, This will drop current database and will recreate it");
}
}
public class CopyAdapter {
private final Context mContext;
private SQLiteDatabase mDb;
private DatabaseHelper mDbHelper;
private static String TAG = "TAG";
private static String ACCOUNT_TABLE = "account";
public static String ACCOUNT_EXTRADATA = "extraData";
public static String ACCOUNT_ID = "ID";
public static String ACCOUNT_ADDITIONALDATA = "additionalData";
public static String ACCOUNT_DATA = "data";
public CopyAdapter(Context context)
{
this.mContext = context;
mDbHelper = new DatabaseHelper(mContext);
Log.d(TAG,"done");
}
public CopyAdapter createDatabase() throws SQLException
{
try
{
Log.d(TAG,"create database");
mDbHelper.createDataBase();
}
catch (IOException mIOException)
{
Log.e(TAG, mIOException.toString() + " UnableToCreateDatabase");
throw new Error("UnableToCreateDatabase");
}
return this;
}
public CopyAdapter open() throws SQLException
{
try
{
Log.d(TAG,"Open");
mDbHelper.openDataBase();
mDbHelper.close();
mDb = mDbHelper.getWritableDatabase();
}
catch (SQLException mSQLException)
{
Log.e(TAG, mSQLException.toString());
throw mSQLException;
}
return this;
}
public void close()
{
mDbHelper.close();
}
public int countAccountData()
{
Cursor mCoursor = mDb.query(ACCOUNT_TABLE, new String[] {}, null, null, null, null, null);
int mReturnedCount = mCoursor.getCount();
mCoursor.close();
return mReturnedCount;
}
public long insertData(String mExtra, String mAdditionalData, String mData)
{
ContentValues initialValues = new ContentValues();
initialValues.put(ACCOUNT_EXTRADATA, mExtra);
initialValues.put(ACCOUNT_ADDITIONALDATA, mAdditionalData);
initialValues.put(ACCOUNT_DATA, mData);
return mDb.insert(ACCOUNT_TABLE, null, initialValues);
}
public boolean updateData(int mPosition, String mExtra, String mAdditionalData, String mData)
{
ContentValues initialValues = new ContentValues();
initialValues.put(ACCOUNT_EXTRADATA, mExtra);
initialValues.put(ACCOUNT_ADDITIONALDATA, mAdditionalData);
initialValues.put(ACCOUNT_DATA, mData);
return mDb.update(ACCOUNT_TABLE, initialValues, "ID=" + mPosition, null) > 0;
}
public String retriveData(int mPosition)
{
Cursor mCursor = mDb.query(ACCOUNT_TABLE, new String[] {ACCOUNT_DATA}, "ID=" + mPosition, null, null, null, null);
mCursor.moveToFirst();
String mReturn = mCursor.getString(mCursor.getColumnIndex(ACCOUNT_DATA));
mCursor.close();
return mReturn;
}
public String retriveAdditionalData(int mPosition)
{
Cursor mCursor = mDb.query(ACCOUNT_TABLE, new String[] {ACCOUNT_ADDITIONALDATA}, "ID=" + mPosition, null, null, null, null);
mCursor.moveToFirst();
String mReturn = mCursor.getString(mCursor.getColumnIndex(ACCOUNT_ADDITIONALDATA));
mCursor.close();
return mReturn;
}
public boolean deleteAccount(int mPosition)
{
return mDb.delete(ACCOUNT_TABLE, ACCOUNT_ID + "=" + mPosition, null) > 0;
}
}
然后只需在您需要的地方调用:
CopyAdapter mDbHelper = new CopyAdapter(YourActivity.this);
mDbHelper.createDatabase();
并确保你设置了
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
在你的 manifest.xml 中
例如,我阅读了很多变体,将您的数据库文件放在 Assets 文件夹中的另一个文件夹中,或者将您的数据库文件的结尾重命名为 jpg,或者在将其添加到 Assets 之前将您的数据库文件压缩......忘记了那。我能够使我的应用程序正常工作的唯一方法是将数据库文件不带文件夹或重命名或压缩到我的 Assets 文件夹中。只是纯文件。从这一点开始,它工作没有问题。
我不知道我从哪里得到上面的这个例子,我再也找不到这个网站了(但它在 stackoverflow 上),但是这个,有一点解决方法并将纯数据库文件放在 Assets 文件夹中,是解决方案。希望对您有所帮助..
关于android - 为 Android 应用程序复制 Sqlite 数据库,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/14957835/
我正在开发一个 SQLite 数据库。数据库已经填满了,但我想重构它。这是我需要做的一个示例: 我目前有一张 table : CREATE TABLE Cars (ID INTEGER PRIMARY
我正在使用 Mono、SQLite、Dapper 和 Dapper 扩展。我可以从数据库中读取数据,但插入不起作用。我正在使用 sqlite 的 Mono 驱动程序。 错误并不能提供太多信息,至少对我
我有一个使用 SQLite 的 Windows Phone 8 应用程序。该应用程序具有许多数据库功能,并包含一个 sqlite 数据库文件,在运行该应用程序时,该文件将被复制到本地文件夹并进行访问。
为 sqlite 创建索引时有排序顺序。 https://sqlite.org/lang_createindex.html Each column name or expression can be
顾名思义,我怀疑如果有一些引用被删除的表会发生什么,例如表的某些字段的索引。 SQLite是否会自动处理?在执行drop命令之前,数据库所有者是否应注意任何实例? 最佳答案 我认为不需要家政服务。 S
我想知道是否有可能将从计数中获得的整数转换为REAL 类似于以下内容(尽管这不起作用) SELECT CAST (COUNT (ColumnA) AS Count) AS REAL) FROM Tab
我无法在SQLite数据库上执行一些更新。我正在Windows上使用SQLite 3 Shell。 我正在运行以下命令: update resovled_chrom_counts set genus
我知道SQLite中的触发器顺序是不确定的(您不能确定将首先执行哪个触发器),但是表约束和触发器之间的关系又如何呢? 我的意思是,假设我在一个列中有一个UNIQUE(或CHECK)约束,并且在该表上有
我的 CustomTags 表可能有一系列“临时”记录,其中 Tag_ID 为 0,并且 Tag_Number 将有一些五位数的值。 定期,我想清理我的 Sqlite 表以删除这些临时值。 例如,我可
我有A,B,C和D的记录。 我的SQL1 SELECT * FROM main_table order by main_table.date desc limit 2返回A和B。 我的SQL2 SEL
select round(836.0)返回836.0 我如何删除sqlite查询中的尾随零。 836.00应该是836 836.440应该是836.44 最佳答案 如果需要836.44,则需要十进制返
我正在研究RQDA中的文本,并且正在使用Firefox SQLite Manager访问数据库,以便可以更轻松地搜索文件。我创建并填充了虚拟表: CREATE VIRTUAL TABLE texts
我有这样的数据: table1 id | part | price 1 | ox900 | 100 2 | ox980 | 200 和 table2 id | part | price 1
我正在尝试将一些数据插入现有的SQLite表中。该表和数据库是使用相同的API创建的,但是由于某种原因,插入操作无效,并且从不给我任何错误消息。 我正在BlackBerry 9550模拟器上对此进行测
例如,我在名为SALARY的列中插入一个值。如果插入的值大于1000,我想将字符串HIGH插入到RANK列中,否则将插入LOW中。 我可以使用SQLite做到吗? 最佳答案 在插入之前使用触发器,然后
假设我有一个包含三列A,B,C的表t1,其中(A,B)包含唯一键(具有数十万行)。由于90%的查询将采用SELECT C FROM t1 WHERE A =?和B = ?,我想我要为A,B和C提供覆盖
在一个SQLite3数据库中,我有一个表“ projects”,其id字段由以下方式组成: [user id]_[user's project id] 例如,用户ID = 45,这是一些数据: 45_
我了解PRAGMA foreign_key和ON DELETE RESTRICT/NO ACTION的概念,但是我面临的是另一种情况。 我需要删除一个父行,但保持与之关联的子行。例如: CREATE
我的c#应用程序从Web服务1读取文件列表,并将完整的文件名插入table1,然后从第二个Web服务读取list并将它们插入到table2。 这些表具有相同的结构,如下所示: create table
我在以下情况下尝试将Record1的ID更新为Record2的ID: 两个表中的名称相同,并且 在Record2中权重更大。 记录1 | ID | Weight | Name | |----|----
我是一名优秀的程序员,十分优秀!