19

我正在开发一个应用程序,我使用 sqllite3 数据库来存储值。我有 Nexus S 和 Nexus 7 都是无根设备。如何获取我的应用程序的数据库以进行调试。

我试过(1)我试过这里提到的所有方法

adb shell
run-as app.package.name \
cp /data/data/package.name/databases/application.sqlite /sdcard/
exit
adb pull /sdcard/application.sqlite ~/

这说cp没有找到..

(2) http://developer.android.com/tools/help/adb.html#sqlite

adb -s emulator-5554 shell
# sqlite3 /data/data/com.example.google.rss.rssexample/databases/rssitems.db
SQLite version 3.3.12
Enter ".help" for instructions
.... enter commands, then quit...
sqlite> .exit 
4

11 回答 11

36

以下解决方案仅适用于可调试的应用程序。它可能无法在所有设备上正常运行,因为 run-as 命令在某些设备上不起作用,尤其是在 Jelly Bean 上。

  1. ​创建*.bat文件并复制以下脚本​​​</p>

    adb shell run-as [package] chmod 777 /data/data/[package]/databases/

    adb shell run-as [package] chmod 777 /data/data/[package]/databases/[db_file_name]

    adb shell run-as [package] cp /data/data/[package]/databases/[db_file_name] /sdcard/

    adb pull /sdcard/[db_file_name]

  2. ​将 [package] 更改为所需的应用程序包

  3. 将 [db_file_name] 更改为所需的数据库名称 运行 bat 文件,您应该会在 bat 文件所在的文件夹中看到复制的数据库

上述解决方案假设:

  • 您在 Windows 上工作
  • 该设备已连接并在“adb devices”下可见
于 2015-03-25T14:01:48.327 回答
30

您可以使用以下命令将数据库写入外部存储器:

private void writeToSD() throws IOException {
    File sd = Environment.getExternalStorageDirectory();

    if (sd.canWrite()) {
        String currentDBPath = DB_NAME;
        String backupDBPath = "backupname.db";
        File currentDB = new File(DB_PATH, currentDBPath);
        File backupDB = new File(sd, backupDBPath);

        if (currentDB.exists()) {
            FileChannel src = new FileInputStream(currentDB).getChannel();
            FileChannel dst = new FileOutputStream(backupDB).getChannel();
            dst.transferFrom(src, 0, src.size());
            src.close();
            dst.close();
        }
    }
}

DB_NAME我的数据库名称在哪里,DB_PATH定义如下:

if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN_MR1) {
        DB_PATH = context.getFilesDir().getAbsolutePath().replace("files", "databases") + File.separator;
    }
    else {
        DB_PATH = context.getFilesDir().getPath() + context.getPackageName() + "/databases/";
    }

并添加以下权限(感谢@Sathesh 指出这一点):

<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />

每当我有数据库写入时,我都会调用此方法,以便我最新的数据库文件位于外部存储器中,我可以从那里查看和调试。

然后,您可以使用X-Plore 应用程序从 Android 设备上的外部存储器中查看数据库。

于 2013-07-26T14:08:00.587 回答
10

如果你不知道你的应用程序路径,那么你可以使用这个:

public void copyAppDbToExternalStorage() throws IOException {
    File sd = Environment.getExternalStorageDirectory();
    File currentDB = getApplicationContext().getDatabasePath("databaseName"); //databaseName=your current application database name, for example "my_data.db"
    if (sd.canWrite()) {
        File backupDB = new File(sd, "toDatabaseName"); // for example "my_data_backup.db"
        if (currentDB.exists()) {
            FileChannel src = new FileInputStream(currentDB).getChannel();
            FileChannel dst = new FileOutputStream(backupDB).getChannel();
            dst.transferFrom(src, 0, src.size());
            src.close();
            dst.close();
        }
    }
}

或者,如果您需要将数据库复制到公共“下载”文件夹,那么您可以使用它:

public void copyAppDbToDownloadFolder() throws IOException {
    File backupDB = new File(Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_DOWNLOADS), "toDatabaseName"); // for example "my_data_backup.db"
    File currentDB = getApplicationContext().getDatabasePath("databaseName"); //databaseName=your current application database name, for example "my_data.db"
    if (currentDB.exists()) {
        FileChannel src = new FileInputStream(currentDB).getChannel();
        FileChannel dst = new FileOutputStream(backupDB).getChannel();
        dst.transferFrom(src, 0, src.size());
        src.close();
        dst.close();
    }
}

这在我的 Nexus 4 设备上运行良好。

于 2013-12-27T04:37:12.913 回答
10

这是一个非常简单直接的答案:(在Android上测试:unrooted

adb -d shell 
$ run-as my.package.name
$ cp databases/mydatabase.db /sdcard/mydatabase.db
$ exit
$ exit

现在将您的数据库拉到默认的 adb 路径

adb -d pull /sdcard/mydatabase.db

或者,到您的桌面,例如

adb -d pull /sdcard/mydatabase.db C:\Users\user\Desktop

您可能想使用以下命令删除副本:

adb -d shell "rm /sdcard/mydatabase.db"

-d如果有多个模拟器,选项选择默认设备。

于 2016-10-02T04:12:12.613 回答
3

我只需要做这样的事情并且有办法做到这一点,尽管这很痛苦。您需要将该文件作为应用程序的用户帐户进行分类,然后将其通过管道传输到可写位置。这在我运行 4.3.3 的 Nexus 4 上对我有用:

adb shell "run-as org.your.application cat /data/data/org.your.application/your-file > /mnt/sdcard/your-file"
adb pull /mnt/sdcard/your-file
于 2014-05-28T18:21:01.967 回答
3

尝试使用 facebook 开发的 stetho 库,它使您可以通过网络浏览器查看数据库https://facebook.github.io/stetho/

于 2016-02-16T12:11:56.820 回答
0
adb shell "run-as CHR.Droid chmod 666 /data/data/CHR.Droid/files/CHR.db"
adb shell cp /data/data/CHR.Droid/files/CHR.db /sdcard/

In Xamrin.forms I had to replace databases to files

于 2016-06-20T14:54:37.563 回答
0

如果运行后

adb shell "run-as your.package.name"

您收到“run-as: Package 'your.package.name' is unknown”,然后尝试从模拟器获取数据库。见这里:https ://stackoverflow.com/a/44089632/2914140

在 Android Studio 3.0 中,单击视图 > 工具窗口 > 设备文件资源管理器。展开 /data/data/[package-name] 节点。

于 2017-12-25T16:47:14.483 回答
0

我认为这是最简单的方法。将此依赖项添加到您的 gradle 文件中: debugImplementation 'com.amitshekhar.android:debug-db:1.0.0'

在此之后,当您在控制台中启动应用程序时,您会看到以下消息:D/DebugDB: Open http://192.168.1.114:8080 in your browser

还有你的数据库。

于 2018-12-10T15:29:39.227 回答
-2

您无法访问非根设备上的 /data 文件夹

没有办法做到这一点。Android 拥有强大的安全机制。只有您的应用可以访问自己的文件。

于 2013-07-26T14:09:24.830 回答
-2

没有root权限的手机是不可能的。您无法访问 /data 目录,因此您无法像(2)中那样复制数据库或使用 sqlite 应用程序。如果您需要调试,请使用模拟器,或从您的应用程序运行查询来检查数据库。

于 2013-07-26T14:09:29.747 回答