ここでの問題は、メモリのプライベート スペースに保存されたファイルをギャラリーで開こうとしていることにあると思います (getCacheDir はアプリケーションに対する相対パスを返し、アプリケーションだけがそのメモリ パスにアクセスできます)。
外部メモリに保存できない場合は、パブリック パスに保存することもできます (ただし、メディア ファイルはすべてのアプリで操作でき、アプリケーションをアンインストールしても、そこに保存された生成済みメディアは消去されません)。
プライベートな内部メモリを使用したい場合は、 ContentProvider を書くことができます
私が言ったことを達成するために使用するコンテンツプロバイダーを投稿するために編集します。これは私のコンテンツプロバイダーです(必要な関連部分を投稿しました):
public class MediaContentProvider extends ContentProvider {
private static final String TAG = "MediaContentProvider";
// name for the provider class
public static final String AUTHORITY = "com.way.srl.HandyWay.contentproviders.media";
private MediaData _mediaData;
// UriMatcher used to match against incoming requests
private UriMatcher _uriMatcher;
@Override
public int delete(Uri uri, String selection, String[] selectionArgs) {
// TODO Auto-generated method stub
return 0;
}
@Override
public String getType(Uri uri) {
// TODO Auto-generated method stub
return null;
}
@Override
public Uri insert(Uri uri, ContentValues values) {
// TODO Auto-generated method stub
return null;
}
@Override
public boolean onCreate() {
uriMatcher = new UriMatcher(UriMatcher.NO_MATCH);
// Add a URI to the matcher which will match against the form
// 'content://com.stephendnicholas.gmailattach.provider/*'
// and return 1 in the case that the incoming Uri matches this pattern
_uriMatcher.addURI(AUTHORITY, "*", 1);
return true;
}
@Override
public Cursor query(Uri uri, String[] projection, String selection, String[] selectionArgs, String sortOrder) {
// TODO Auto-generated method stub
return null;
}
@Override
public int update(Uri uri, ContentValues values, String selection, String[] selectionArgs) {
// TODO Auto-generated method stub
return 0;
}
@Override
public ParcelFileDescriptor openFile(Uri uri, String mode) throws FileNotFoundException {
Log.v(TAG, "Called with uri: '" + uri + "'." + uri.getLastPathSegment());
// Check incoming Uri against the matcher
switch (_uriMatcher.match(uri)) {
// If it returns 1 - then it matches the Uri defined in onCreate
case 1:
// The desired file name is specified by the last segment of the
// path
// E.g.
// 'content://com.stephendnicholas.gmailattach.provider/Test.txt'
// Take this and build the path to the file
// String fileLocation = getContext().getCacheDir() + File.separator + uri.getLastPathSegment();
Integer mediaID = Integer.valueOf(uri.getLastPathSegment());
if (_mediaData == null) {
_mediaData = new MediaData();
}
Media m = _mediaData.get(mediaID);
// Create & return a ParcelFileDescriptor pointing to the file
// Note: I don't care what mode they ask for - they're only getting
// read only
ParcelFileDescriptor pfd = ParcelFileDescriptor.open(new File(m.filePath), ParcelFileDescriptor.MODE_READ_ONLY);
return pfd;
// Otherwise unrecognised Uri
default:
Log.v(TAG, "Unsupported uri: '" + uri + "'.");
throw new FileNotFoundException("Unsupported uri: " + uri.toString());
}
}
次に、マニフェストでコンテンツプロバイダーへの参照が必要です。私の場合は
<provider
android:name=".contentproviders.MediaContentProvider"
android:authorities="com.way.srl.HandyWay.contentproviders.media" >
</provider>
そして、このように使用します
Intent intent = new Intent();
intent.setAction(Intent.ACTION_VIEW);
intent.setDataAndType(Uri.parse("content://" + MediaContentProvider.AUTHORITY + "/" + m.id), "image/jpg");
私の場合、m は sqlite db を指す ID を格納するエンティティであり、データをフェッチしてオブジェクトに再度データを入力するクラスを使用します (_mediaData を使用)。必要に応じてコードを変更するだけです。
このようにして、アプリケーションであなたの問題を正確に解決しました