2

我有一个带有 ImageView 的 Activity,如果它是空的并被单击,它应该从 Internet 加载图像并将其保存到内部存储中。所以活动看起来像这样:

public class PlaceCreate extends Activity {
Context context;

@Override
public void onCreate(Bundle savedInstanceState) {
  super.onCreate(savedInstanceState);
  setContentView(R.layout.create);
  context = this;

  ImageView img = (ImageView) findViewById(R.id.imageView);
  img.setOnClickListener(new OnClickListener() {
    public void onClick(View v) {
        EditText edit = (EditText)findViewById(R.id.editTextName);
        Download(edit.getText().toString());
    }
  });
}

private boolean Download(String Name)
{
  try
  {
    URL u = new URL("http://example.com/img.png");
    HttpURLConnection c = (HttpURLConnection)u.openConnection();
    c.setRequestMethod("GET");
    c.setDoOutput(true);
    c.connect();
    FileOutputStream f = context.openFileOutput(Name, Context.MODE_PRIVATE);
    InputStream in = c.getInputStream();
    byte[] buffer = new byte[1024];
    int len1 = 0;
    while((len1 = in.read(buffer)) > 0 )
    {
      f.write(buffer, 0, len1);
    }
    f.close();

    FileInputStream is = openFileInput(Name);
    Bitmap bitmap = BitmapFactory.decodeStream(is);
    is.close();
    ImageView img = (ImageView) findViewById(R.id.imageViewPlan);
    img.setImageBitmap(bitmap);
  }
  catch(IOException e)
  {
    return false;
  }
  return true;
}}

问题是我得到 NullPointerException 与openFileOutput. 我发现了几个类似的问题,但所有答案都暗示这null是一个缺失的上下文。因为在这种情况下它是一个活动,它显然有一个上下文(this),它不为空。我试图打电话openFileOutputthis暗示)并通过context会员。任何一种方法都失败了。我还尝试将v.getContext()from onClickintoDownload作为附加参数传递,并且该值不为 null,但也产生了相同的异常。(如果它很重要,则请求的文件的名称是有效的文件名,例如“abc”。)

有人可以对此有所了解吗?

这是堆栈的示例:

04-22 00:45:10.659: W/dalvikvm(330): threadid=1: thread exiting with uncaught exception (group=0x40015560)
04-22 00:45:10.909: E/AndroidRuntime(330): FATAL EXCEPTION: main
04-22 00:45:10.909: E/AndroidRuntime(330): java.lang.NullPointerException
04-22 00:45:10.909: E/AndroidRuntime(330):  at android.app.ContextImpl.openFileOutput(ContextImpl.java:420)
04-22 00:45:10.909: E/AndroidRuntime(330):  at android.content.ContextWrapper.openFileOutput(ContextWrapper.java:158)
04-22 00:45:10.909: E/AndroidRuntime(330):  at com.example.scanner.PlaceCreate.Download(PlaceCreate.java:93)
04-22 00:45:10.909: E/AndroidRuntime(330):  at com.example.scanner.PlaceCreate.access$0(PlaceCreate.java:84)
04-22 00:45:10.909: E/AndroidRuntime(330):  at com.example.scanner.PlaceCreate$3.onClick(PlaceCreate.java:67)
04-22 00:45:10.909: E/AndroidRuntime(330):  at android.view.View.performClick(View.java:2485)
04-22 00:45:10.909: E/AndroidRuntime(330):  at android.view.View$PerformClick.run(View.java:9080)
04-22 00:45:10.909: E/AndroidRuntime(330):  at android.os.Handler.handleCallback(Handler.java:587)
04-22 00:45:10.909: E/AndroidRuntime(330):  at android.os.Handler.dispatchMessage(Handler.java:92)
04-22 00:45:10.909: E/AndroidRuntime(330):  at android.os.Looper.loop(Looper.java:123)
04-22 00:45:10.909: E/AndroidRuntime(330):  at android.app.ActivityThread.main(ActivityThread.java:3683)
04-22 00:45:10.909: E/AndroidRuntime(330):  at java.lang.reflect.Method.invokeNative(Native Method)
04-22 00:45:10.909: E/AndroidRuntime(330):  at java.lang.reflect.Method.invoke(Method.java:507)
04-22 00:45:10.909: E/AndroidRuntime(330):  at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:839)
04-22 00:45:10.909: E/AndroidRuntime(330):  at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:597)
04-22 00:45:10.909: E/AndroidRuntime(330):  at dalvik.system.NativeStart.main(Native Method)

这里还有一些发现。根据堆栈,错误发生在ContextImpl文件内部,我看到parent该方法内部的局部变量为空。这是代码:

public FileOutputStream openFileOutput(String name, int mode)
  throws FileNotFoundException {
  final boolean append = (mode&MODE_APPEND) != 0;
  File f = makeFilename(getFilesDir(), name);
  try {
     FileOutputStream fos = new FileOutputStream(f, append);
     setFilePermissionsFromMode(f.getPath(), mode, 0);
     return fos;
  } catch (FileNotFoundException e) {
 }

 File parent = f.getParentFile();
 parent.mkdir();
 FileUtils.setPermissions(
     parent.getPath(),
     FileUtils.S_IRWXU|FileUtils.S_IRWXG|FileUtils.S_IXOTH,
     -1, -1);
 FileOutputStream fos = new FileOutputStream(f, append);
 setFilePermissionsFromMode(f.getPath(), mode, 0);
 return fos;
}

所以现在的问题是如何确保调用f.getParentFile()不会返回 null,为什么它有时会起作用?

4

5 回答 5

2

问题是我在 openFileOutput 的行上得到 NullPointerException。

您正在NullPointerException获得. _ openFileOutput()这就是为什么您需要为 SO 提供堆栈跟踪,而不仅仅是提供异常名称。行上的异常与行触发的异常不同,尤其是NullPointerException.

根据我对openFileOutput()on的阅读ContextImpl,要么:

  • Namenull, 或
  • 你有一个非常奇怪的context,或者
  • 您正在运行的 Android 版本还有一些其他可能的问题,这些问题在当前的实现中不可见openFileOutput()

我将从context完全删除开始,因为您在其中Activity并且不需要它。然后,放入一些Log语句或设置断点,看看是什么Name

于 2012-04-21T20:17:40.810 回答
1

这被证明是作为 SDK 的一部分安装在这里的 Android 2.3.3 模拟器的一个功能。如果我使用带有 Android 2.3.3 的真实设备,则不会出现此问题。如果我使用 Android 4.0.3 模拟器,代码也可以正常工作。因此,如果发生一些奇怪的事情,在另一个环境中进行测试可以提供一些启示。我希望这可以帮助其他人解决类似的问题。

于 2012-05-05T11:21:53.503 回答
1

我在运行 ICS 的华硕 TF101 上遇到了同样的问题——即一个真正的设备而不是一个模拟器。这很令人烦恼,因为事实证明这个问题与底层文件夹的 Linux 写权限有关。

我一直在android:sharedUserId尝试在清单文件中使用带有标签的共享资源。由于 UID 已更改,该活动不再具有数据文件目录中的写入权限。

所以任何可能改变 uid 的东西都可能导致这个问题。

为了解决我的问题:

  1. sharedUserID从清单中删除标签(可能是可选的);
  2. 从我的设备上手动卸载了该应用程序。
  3. 重新安装了应用程序。

然后它完美地运行。

于 2012-10-04T08:34:53.077 回答
1

通过阅读 openFileOutput() 的代码,我可以看到 f.getParentFile() 可能返回 null 的情况,这在此处的任何其他答案或评论中均未提及。

在行

File f = makeFilename(getFilesDir(), name);

如果dataDir不存在并且 getFilesDir()创建 dataDir 失败,getFilesDir()可以返回 null 。如果发生这种情况,文件只有在路径(例如“目录/文件”)而不仅仅是文件(例如“文件”)f时才会有父级。name

我不知道在什么情况下会丢失 dataDir 以及何时无法创建它,但听起来不太可能由于某种原因无法在 Android 2.3.3 模拟器中创建它。

于 2015-01-23T15:07:07.383 回答
0

看到这个完整的例子

URL url = new URL (strURL);
input = url.openStream();
byte[] buffer = new byte[1500];
OutputStream output = new FileOutputStream ("/sdcard/abc.png");
int bytesRead = 0;
while ((bytesRead = input.read(buffer, 0, buffer.length)) >= 0) .
{
 output.write(buffer, 0, bytesRead);
 }
于 2012-04-21T15:13:56.740 回答