0

当我尝试使用此代码拍照时,总是会收到 NullPointerException。它总是在 camera.takePicture 失败。

我试图用谷歌搜索这个问题并没有找到任何东西。任何帮助,将不胜感激。

这是代码

import android.annotation.TargetApi;
import android.app.Activity;
import android.content.pm.PackageManager;
import android.hardware.Camera;
import android.hardware.Camera.CameraInfo;
import android.os.Build;
import android.os.Bundle;
import android.util.Log;
import android.view.View;
import android.widget.Toast;


@TargetApi(Build.VERSION_CODES.GINGERBREAD)
public class CameraService extends Activity {

final static String DEBUG_TAG = "MakePhotoActivity";
  private Camera camera;
  private int cameraId = 0;

  @TargetApi(Build.VERSION_CODES.GINGERBREAD)
@Override
  public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.main);

    // do we have a camera?
    if (!getPackageManager()
        .hasSystemFeature(PackageManager.FEATURE_CAMERA)) {
      Toast.makeText(this, "No camera on this device", Toast.LENGTH_LONG)
          .show();
    } else {
      cameraId = findFrontFacingCamera();
      if (cameraId < 0) {
        Toast.makeText(this, "No front facing camera found.",
            Toast.LENGTH_LONG).show();
      } else {
        camera = Camera.open(cameraId);
      }
    }
  }

  public void onClick(View view) {
    camera.takePicture(null, null,
        new PhotoHandler(getApplicationContext()));
  }

  private int findFrontFacingCamera() {
    int cameraId = -1;
    // Search for the front facing camera
    int numberOfCameras = Camera.getNumberOfCameras();
    for (int i = 0; i < numberOfCameras; i++) {
      CameraInfo info = new CameraInfo();
      Camera.getCameraInfo(i, info);
      if (info.facing == CameraInfo.CAMERA_FACING_FRONT) {
        Log.d(DEBUG_TAG, "Camera found");
        cameraId = i;
        break;
      }
    }
    return cameraId;
  }

  @Override
  protected void onPause() {
    if (camera != null) {
      camera.release();
      camera = null;
    }
    super.onPause();
  }

}

PhotoHandler 的代码

import java.io.File;
import java.io.FileOutputStream;
import java.text.SimpleDateFormat;
import java.util.Date;

import android.content.Context;
import android.hardware.Camera;
import android.hardware.Camera.PictureCallback;
import android.os.Environment;
import android.util.Log;
import android.widget.Toast;

public class PhotoHandler implements PictureCallback{

private final Context context;

  public PhotoHandler(Context context) {
    this.context = context;
  }

  @Override
  public void onPictureTaken(byte[] data, Camera camera) {

    File pictureFileDir = getDir();

    if (!pictureFileDir.exists() && !pictureFileDir.mkdirs()) {

      Log.d(CameraService.DEBUG_TAG, "Can't create directory to save image.");
      Toast.makeText(context, "Can't create directory to save image.",
          Toast.LENGTH_LONG).show();
      return;

    }

    SimpleDateFormat dateFormat = new SimpleDateFormat("yyyymmddhhmmss");
    String date = dateFormat.format(new Date());
    String photoFile = "Picture_" + date + ".jpg";

    String filename = pictureFileDir.getPath() + File.separator + photoFile;

    File pictureFile = new File(filename);

    try {
      FileOutputStream fos = new FileOutputStream(pictureFile);
      fos.write(data);
      fos.close();
      Toast.makeText(context, "New Image saved:" + photoFile,
          Toast.LENGTH_LONG).show();
    } catch (Exception error) {
      Log.d(CameraService.DEBUG_TAG, "File" + filename + "not saved: "
          + error.getMessage());
      Toast.makeText(context, "Image could not be saved.",
          Toast.LENGTH_LONG).show();
    }
  }

  private File getDir() {
    File sdDir = Environment
      .getExternalStoragePublicDirectory(Environment.DIRECTORY_PICTURES);
    return new File(sdDir, "CameraAPIDemo");
  }
}

logcat 输出

04-15 18:24:42.316: E/AndroidRuntime(23331): FATAL EXCEPTION: main
04-15 18:24:42.316: E/AndroidRuntime(23331): java.lang.IllegalStateException: Could not execute method of the activity
04-15 18:24:42.316: E/AndroidRuntime(23331):    at android.view.View$1.onClick(View.java:3599)
04-15 18:24:42.316: E/AndroidRuntime(23331):    at android.view.View.performClick(View.java:4204)
04-15 18:24:42.316: E/AndroidRuntime(23331):    at android.view.View$PerformClick.run(View.java:17355)
04-15 18:24:42.316: E/AndroidRuntime(23331):    at android.os.Handler.handleCallback(Handler.java:725)
04-15 18:24:42.316: E/AndroidRuntime(23331):    at android.os.Handler.dispatchMessage(Handler.java:92)
04-15 18:24:42.316: E/AndroidRuntime(23331):    at android.os.Looper.loop(Looper.java:137)
04-15 18:24:42.316: E/AndroidRuntime(23331):    at android.app.ActivityThread.main(ActivityThread.java:5041)
04-15 18:24:42.316: E/AndroidRuntime(23331):    at java.lang.reflect.Method.invokeNative(Native Method)
04-15 18:24:42.316: E/AndroidRuntime(23331):    at java.lang.reflect.Method.invoke(Method.java:511)
04-15 18:24:42.316: E/AndroidRuntime(23331):    at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:793)
04-15 18:24:42.316: E/AndroidRuntime(23331):    at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:560)
04-15 18:24:42.316: E/AndroidRuntime(23331):    at dalvik.system.NativeStart.main(Native Method)
04-15 18:24:42.316: E/AndroidRuntime(23331): Caused by: java.lang.reflect.InvocationTargetException
04-15 18:24:42.316: E/AndroidRuntime(23331):    at java.lang.reflect.Method.invokeNative(Native Method)
04-15 18:24:42.316: E/AndroidRuntime(23331):    at java.lang.reflect.Method.invoke(Method.java:511)
04-15 18:24:42.316: E/AndroidRuntime(23331):    at android.view.View$1.onClick(View.java:3594)
04-15 18:24:42.316: E/AndroidRuntime(23331):    ... 11 more
04-15 18:24:42.316: E/AndroidRuntime(23331): Caused by: java.lang.RuntimeException: takePicture failed
04-15 18:24:42.316: E/AndroidRuntime(23331):    at android.hardware.Camera.native_takePicture(Native Method)
04-15 18:24:42.316: E/AndroidRuntime(23331):    at android.hardware.Camera.takePicture(Camera.java:1095)
04-15 18:24:42.316: E/AndroidRuntime(23331):    at android.hardware.Camera.takePicture(Camera.java:1040)
04-15 18:24:42.316: E/AndroidRuntime(23331):    at com.egnoita.ignoramus.CameraService.onClick(CameraService.java:50)
04-15 18:24:42.316: E/AndroidRuntime(23331):    ... 14 more
4

7 回答 7

2

我认为camera您的函数中为 null ,onClick()因为没有保证onCreate()创建camera.

修改onClick()如下:

public void onClick(View view) {
    if(camera == null) {
        // Warn user that camera is not available via "Toast" or similar.
    } else {
        camera.takePicture(null, null,
            new PhotoHandler(getApplicationContext()));
    }
}
于 2013-04-15T08:26:10.533 回答
1

我在 2.3 和 4.0 之间遇到了同样的兼容性问题。在这里,我假设您在打电话startPreview()之前打过电话takePicture()

请在下面查看我的代码,它适用于我的 2.3 和 4.0 设备。

PictureCallback mJpegCallback;
SurfaceView surfaceViewDummy;
Camera mCamera;
mCamera = Camera.open();
try {

    // Important here!!! If use below line under 2.3 works,
    // but not work on 4.0! Will always got 
    // java.lang.RuntimeException: takePicture failed
    // Under 4.0 must create a real SurfaceView object on the screen!

    //surfaceViewDummy = new SurfaceView(this);
    surfaceViewDummy = (SurfaceView) findViewById(R.id.surfaceView1);

    mCamera.setPreviewDisplay(surfaceViewDummy.getHolder());
    mCamera.startPreview();
} catch (Exception e) {

    Log.d("", "Start preview error " + e.toString());
}

mJpegCallback = new PictureCallback() {

    public void onPictureTaken(byte[] data, Camera camera) {
        Log.d("", "Got picture data");
        // Save the picture data by yourself
        // ...
    }
};

mCamera.takePicture(null, null, mJpegCallback);
于 2013-04-27T14:27:38.657 回答
0

您的应用程序是否允许使用相机?

<uses-permission android:name="android.permission.CAMERA"/>
于 2013-04-15T08:16:04.537 回答
0

尝试将此添加到您的清单文件中

<uses-feature android:name="android.hardware.camera"/>

这将告诉设备应用程序使用相机有关更多信息 http://developer.android.com/guide/topics/media/camera.html

于 2013-04-16T11:28:41.730 回答
0

我在我的应用程序中使用与您相同的代码,而且我的应用程序也习惯于在行上抛出空指针异常:

camera.takePicture(null, null,new PhotoHandler(getApplicationContext()));

我认为这是由于相机获取空值引起的。为避​​免此异常,我在这行代码的顶部进行了检查。因此 onClick 函数如下所示:

public void onClick(View view) {
  if (camera != null) {
    Log.d("camera state","camera is NOT null");  
  }else{
    Log.d("camera state","camera is null");
    camera = android.hardware.Camera.open(cameraId);
  }
  camera.takePicture(null, null,new PhotoHandler(getApplicationContext()));
}

希望能帮助到你 !

于 2013-12-09T02:30:45.237 回答
0

好久没有发布查询了,但问题是您必须在调用 takePicture API 之前启动预览,即

 public void onClick(View view) {
camera.startPreview(); //Add this line
camera.takePicture(null, null,
    new PhotoHandler(getApplicationContext()));
}

我也使用相同的代码。另请注意,除非您有前置摄像头,否则这不会让您单击图片。如果您只想使用相机硬件拍照,请检查后置摄像头而不是前置摄像头。

于 2014-03-24T12:18:10.267 回答
-1

NullPointerException在通话时说takePicture,所以我猜错误在通话本身而不是在PhotoHandler. 我们需要LogCat验证。

我认为罪魁祸首是:new PhotoHandler(getApplicationContext(). 您应该PhotoHandler在调用之前创建takePicture,并在函数中使用它的值takePicture

PhotoHandler mPhotoHandler = null;
public void onClick (View view) {
    mPhotoHandler = new PhotoHandler(getApplicationContext());
    camera.takePicture(null, null, mPhotoHandler);
}
于 2013-04-15T08:10:39.427 回答