在我使用camera2api构建相机应用程序的过程中,我在尝试保存原始图像时遇到了一个小问题。
我将捕获结果分配给以下代码中的成员。
@Override
public void onCaptureCompleted(CameraCaptureSession session, CaptureRequest request, TotalCaptureResult result) {
super.onCaptureCompleted(session, request, result);
mCaptureResult = result;
Toast.makeText(getApplicationContext(),
"Image Captured",Toast.LENGTH_SHORT).show();
}
};
当我在此位置调试它时,成员被分配而不是null。但是,当它在DngCreator
类中的ImageSaver
处抛出空错误时。
我的听众:
private ImageReader mImageReader;
private ImageReader.OnImageAvailableListener mOnImageAvailableListener =
new ImageReader.OnImageAvailableListener() {
@Override
public void onImageAvailable(ImageReader imageReader) {
mBackgroundHandler.post(new ImageSaver(imageReader.acquireNextImage(),mUiHandler,
mCaptureResult, mCameraCharacteristics));
}
};
private ImageReader mRawImageReader;
private ImageReader.OnImageAvailableListener mOnRawImageAvailableListener =
new ImageReader.OnImageAvailableListener() {
@Override
public void onImageAvailable(ImageReader imageReader) {
mBackgroundHandler.post(new ImageSaver(imageReader.acquireNextImage(),mUiHandler,
mCaptureResult, mCameraCharacteristics));
}
};
我的图片保护程序课程:
private ImageSaver(Image image, Handler handler, CaptureResult captureResult,
CameraCharacteristics cameraCharacteristics) {
mImage = image;
mHandler =handler;
mCaptureResult = captureResult;
mCameraCharacteristics = cameraCharacteristics;
}
@Override
public void run() {
int format = mImage.getFormat();
switch(format){
case ImageFormat.JPEG:
ByteBuffer byteBuffer = mImage.getPlanes()[0].getBuffer();
byte[] bytes = new byte[byteBuffer.remaining()];
byteBuffer.get(bytes);
FileOutputStream fileOutputStream = null;
try {
fileOutputStream = new FileOutputStream(mImageFile);
fileOutputStream.write(bytes);
}
catch (IOException e){
e.printStackTrace();
}
finally {
mImage.close();
if(fileOutputStream != null){
try{
fileOutputStream.close();
}
catch (IOException e){
e.printStackTrace();
}
}
//Message message = mHandler.obtainMessage();
//message.sendToTarget();
}
break;
case ImageFormat.RAW_SENSOR:
DngCreator dngCreator = new DngCreator(mCameraCharacteristics,mCaptureResult);
FileOutputStream rawFileOutputStream = null;
try {
rawFileOutputStream = new FileOutputStream(mRawImageFile);
dngCreator.writeImage(rawFileOutputStream, mImage);
}
catch (IOException e){
e.printStackTrace();
}
finally{
mImage.close();
if(rawFileOutputStream != null){
try {
rawFileOutputStream.close();
}
catch (IOException e){
e.printStackTrace();
}
}
}
break;
}
}
}
现在,当我尝试初始化DngCreator并且堆栈跟踪如下时,它给了我一个错误。
FATAL EXCEPTION: Camera2 Background Thread
Process: com.something.something, PID: 5162
java.lang.IllegalArgumentException: Null argument to DngCreator constructor
at android.hardware.camera2.DngCreator.<init>(DngCreator.java:89)
at com.something.something.ControlCameraActivity$ImageSaver.run(ControlCameraActivity.java:328)
at android.os.Handler.handleCallback(Handler.java:739)
at android.os.Handler.dispatchMessage(Handler.java:95)
at android.os.Looper.loop(Looper.java:148)
at android.os.HandlerThread.run(HandlerThread.java:61)
有人可以帮我解决这个问题吗?如果您需要我的更多信息,请立即告诉我。感谢
答案 0 :(得分:2)
无法保证onCaptureCompleted()在OnImageAvailable()之前发生。实际上,由于RAW图像缓冲区通常比预览图像处理早,并且结果元数据完成,因此onCaptureCompleted()很可能在onImageAvailable()之后发生。
如果首先发生onImageAvailable,你还没有获得捕获结果。
相反,您应该等待图像和捕获结果都完成,然后创建ImageSaver。作为一个选项,让两个回调在一些共享位置保存它们各自的输出(捕获结果和图像),然后检查两者是否都为非空 - 如果是,则调用ImageSaver。那么先运行哪个并不重要,第二个运行的那个会启动ImageSaver。
答案 1 :(得分:1)
首先,当您调用imageReader.acquireNextImage()
时,将其设置在Image对象中,并在使用后执行Image.close。 ImageReader对于垃圾收集器来说非常危险,如果你的ImageReader maxImages中的数字很少,你会很快填充它,因为你没有关闭图像。
其次,关于你的引用,你的savingImage方法在哪里? image不是可解析的对象,因此,如果您将它发送到服务或其他类,那么您可能遇到问题。
最后,正如@Eddy所说,有时onCaptureComplete在你的OnImageAvailable之前没有被调用。所以,你有两个解决方案:
只需在您的应用中添加“避免FC”检查:
if (mImage != null && mCameraCharacteristics != null && mDngResult != null) {
//process your dng
}
或者,您可以创建一个监听器。因此,当您收到onImageAvailable()
时,您可以将请愿书放入自定义列表或Hastable中。
在那里,检查List是否有captureResult
可用,如果你有CaptureResult,接受并处理你的DNG,如果没有实例化将在你的onCaptureComplete
中调用的监听器。在那里你只需要检查监听器是否为空,如果不是,则调用&#34; process&#34;现在你的结果可用了。
public class MyResultList {
private DngListener mDngListener;
private Hashtable<String, CaptureResult> mDngCaptureResults;
public int getSize() {
return mDngCaptureResults.size();
}
public Object getResult() {
if (mDngCaptureResults.size() == 0){
mDngListener = new DngListener() {
@Override
public void onResultAvailable() {
//Now result is not null, so you can get the available item
YourSavingImage(mDngCaptureResults.get(0));
}
}
return null;
} else {
return mDngCaptureResults.get(0);
}
}
public void add(CaptureResult result, String index){
mDngCaptureResults.put(index,result);
}
}
我刚刚提出这个想法,根据你的目标进行修改。这很简单,如果你有很多依赖,也许不会那么好。