我在我的项目中有一个SQLite数据库,我想添加一个备份功能来将该数据库保存在SD卡中。有没有什么简单的方法可以在使用Xamarin.Forms的安卓和IOS系统上做到这一点?我在老问题(其中一些是非常老的)中搜索,但我找不到一个明确的答案。
发布于 2018-06-04 09:08:37
iOS:
iOS上没有"sdcard“,但你可以将db复制(File.Copy
)到应用程序的Documents
目录,这样就可以通过iTunes应用程序访问它,这样你就可以复制到你的PC/Mac上:
使用此目录存储用户生成的内容。这个目录的内容可以通过文件共享提供给用户;因此,他的目录应该只包含您希望向用户公开的文件。此目录的内容由iTunes和iCloud备份。
因此,假设您的数据库使用的是App Support目录,您应该是:
public string GetDBPath(string dbFileName)
{
// If you are not using App Support directory for your DBs you are doing it wrong on iOS ;-)
var supportDir = NSSearchPath.GetDirectories(NSSearchPathDirectory.ApplicationSupportDirectory, NSSearchPathDomain.User, true)[0];
var dbDir = Path.Combine(supportDir, "database");
if (!Directory.Exists(dbDir))
Directory.CreateDirectory(dbDir);
return Path.Combine(supportDir, dbDir, dbFileName);
}
复制到应用程序的doc目录非常简单,只需:
public void CopyDBToDocs(string dbFileName)
{
var docDir = NSSearchPath.GetDirectories(NSSearchPathDirectory.DocumentDirectory, NSSearchPathDomain.User)[0];
File.Copy(GetDBPath(dbFileName), Path.Combine(docDir, dbFileName));
}
Android:
在Android上,这可能真的很简单,也可能很痛苦,这取决于设备的API以及您正在开发的应用程序的Android API目标。
就像这样简单:
public void CopyDBToSdCard(string dbName)
{
File.Copy(GetDBPath(dbName), Path.Combine(Android.OS.Environment.DirectoryDownloads, dbName);
}
本例中的GetDBPath
实际上使用的是“真实”数据库目录,它是应用程序沙箱中的"FileDir“位置的子目录:
public string GetDBPath(string dbFileName)
{
// FYI: GetDatabasePath("") fails on some APIs &| devices &|emulators so hack it... (Some API 23 devices, but not API 21, 27, 28, ...)
var dbPath = context.GetDatabasePath("ZZZ").AbsolutePath.Replace("/ZZZ", "");
if (!Directory.Exists(dbPath))
Directory.CreateDirectory(dbPath);
return context.GetDatabasePath(dbFileName).AbsolutePath;
}
现在,要实际执行从应用程序到“外部”位置的文件复制,需要应用程序的清单权限:
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"/>
此外,如果你的应用有“目标”,并且运行在API-23 (Android 6.0)或更高版本上,你需要在运行时得到用户的同意。
使用Android.Support.Compat
是最简单的方法。基本上,您需要检查权限是否已被授予,如果没有,则向用户显示为什么需要权限,然后让操作系统要求用户接受/拒绝请求。然后,您将收到这些perm结果的通知。
注意: JamesM为非粗体编码人员提供了表单的Perm插件;-) PermissionsPlugin
void GetExternalPerms()
{
const string writePermission = Manifest.Permission.WriteExternalStorage;
const string readPermission = Manifest.Permission.ReadExternalStorage;
string[] PermissionsLocation =
{
writePermission, readPermission
};
if ((ContextCompat.CheckSelfPermission(this, writePermission) == Permission.Granted) && (ContextCompat.CheckSelfPermission(this, readPermission) == Permission.Granted))
{
return;
}
if (ActivityCompat.ShouldShowRequestPermissionRationale(this, writePermission))
// Displaying a dialog would make sense at this point...
}
ActivityCompat.RequestPermissions(this, PermissionsLocation, 999);
}
然后,用户接受或拒绝该请求的结果将通过OnRequestPermissionsResult
返回
public override void OnRequestPermissionsResult(int requestCode, string[] permissions, Permission[] grantResults)
{
switch (requestCode)
{
case 999: // This is the code that we supply via RequestPermissions
if (grantResults[0] == Permission.Granted)
{
// you have permission, so defer to your DB copy routine now
}
break;
default:
break;
}
}
https://stackoverflow.com/questions/50672131
复制相似问题