Android-执行Runtime.getRuntime()时出错.exec() - Environment Null -ffmpeg

时间:2012-08-26 11:42:05

标签: android eclipse ffmpeg runtime.exec ubuntu-11.10

我在ubuntu 11.10上编译了ffmpeg库,并在android上移植了编译文件 编译完成后,我成功地获得了 libffmpeg.so 。它成功加载到android上。

我在ubuntu 11.10 eclipse android模拟器上做这件事。

我创建了一个小测试应用程序,它充当命令提示符,接受来自用户的命令并显示结果。 (测试ffmpeg命令)

当我运行像“ls”,“ls -l”这样的简单命令时,它运行得很好。但是当我只是输入“ cd mnt ”或“ ffmpeg ”作为命令并尝试运行它。我在Logcat中警告

08-26 16:44:52.553: W/System.err(5961): java.io.IOException: Error running exec(). Command: [ffmpeg] Working Directory: null Environment: null
08-26 16:44:52.573: W/System.err(5961):     at java.lang.ProcessManager.exec(ProcessManager.java:211)
08-26 16:44:52.573: W/System.err(5961):     at java.lang.Runtime.exec(Runtime.java:168)
08-26 16:44:52.573: W/System.err(5961):     at java.lang.Runtime.exec(Runtime.java:241)
08-26 16:44:52.583: W/System.err(5961):     at java.lang.Runtime.exec(Runtime.java:184)
08-26 16:44:52.593: W/System.err(5961):     at ch.ffmpeg.reversit.MainActivity.Execute(MainActivity.java:61)
08-26 16:44:52.593: W/System.err(5961):     at ch.ffmpeg.reversit.MainActivity$1.onClick(MainActivity.java:46)
08-26 16:44:52.593: W/System.err(5961):     at android.view.View.performClick(View.java:3480)
08-26 16:44:52.593: W/System.err(5961):     at android.view.View$PerformClick.run(View.java:13983)
08-26 16:44:52.603: W/System.err(5961):     at android.os.Handler.handleCallback(Handler.java:605)
08-26 16:44:52.603: W/System.err(5961):     at android.os.Handler.dispatchMessage(Handler.java:92)
08-26 16:44:52.603: W/System.err(5961):     at android.os.Looper.loop(Looper.java:137)
08-26 16:44:52.614: W/System.err(5961):     at android.app.ActivityThread.main(ActivityThread.java:4340)
08-26 16:44:52.624: W/System.err(5961):     at java.lang.reflect.Method.invokeNative(Native Method)
08-26 16:44:52.624: W/System.err(5961):     at java.lang.reflect.Method.invoke(Method.java:511)
08-26 16:44:52.634: W/System.err(5961):     at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:784)
08-26 16:44:52.634: W/System.err(5961):     at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:551)
08-26 16:44:52.644: W/System.err(5961):     at dalvik.system.NativeStart.main(Native Method)
08-26 16:44:52.644: W/System.err(5961): Caused by: java.io.IOException: Permission denied
08-26 16:44:52.674: W/System.err(5961):     at java.lang.ProcessManager.exec(Native Method)
08-26 16:44:52.674: W/System.err(5961):     at java.lang.ProcessManager.exec(ProcessManager.java:209)
08-26 16:44:52.684: W/System.err(5961):     ... 16 more

这是我的代码:

imports;
public class MainActivity extends Activity {
    String com;
    Process process;
    EditText command;
    Button run;
    RelativeLayout main_layout;

    static {
     System.loadLibrary("ffmpeg");
    }

    /** Called when the activity is first created. */
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

       //find view 
       command=(EditText)findViewById(R.id.command);
       run=(Button)findViewById(R.id.run);

        run.setOnClickListener(new OnClickListener() {
            public void onClick(View v) {
                com=command.getText().toString();
                try {
                    Execute();
                } catch (IOException e) {
                    // TODO Auto-generated catch block
                    e.printStackTrace();
                } catch (InterruptedException e) {
                    // TODO Auto-generated catch block
                    e.printStackTrace();
                }

            }
        }); 

    }

    public void Execute() throws IOException, InterruptedException{
        process=Runtime.getRuntime().exec(com);
        // process = pb.command(com).redirectErrorStream(true).start();

        if(process!=null)
        ShowOutput();
        else
        Toast.makeText(getBaseContext(),"Null Process",Toast.LENGTH_LONG).show();
    }

    public void ShowOutput() throws IOException, InterruptedException{
        String s,text="",errors="";
        BufferedReader stdInput = new BufferedReader(new 
                InputStreamReader(process.getInputStream()));

           BufferedReader stdError = new BufferedReader(new 
                InputStreamReader(process.getErrorStream()));


           TextView output=(TextView)findViewById(R.id.output);
           TextView error=(TextView)findViewById(R.id.error);

            while ((s = stdInput.readLine()) != null) {
                   text+=s.toString()+"\n";
                   System.out.println("Error: "+s);
               }

         output.setText(text);
         text="";
           // read any errors from the attempted command
           System.out.println("Here is the standard error of the command (if any):\n");
           while ((s = stdError.readLine()) != null) {
               text+=s.toString()+"\n";
               System.out.println("Error: "+s);
           }

           error.setText(text);

           error.setMovementMethod(new ScrollingMovementMethod());
           output.setMovementMethod(new ScrollingMovementMethod());

           stdInput.close();
           stdError.close();

           process.waitFor();
           process.getOutputStream().close();
           process.getInputStream().close();
           process.getErrorStream().close(); 
           process.destroy();


    }

}

我甚至尝试process = pb.command(com).redirectErrorStream(true).start();执行。它给了我同样的错误,但这次的环境是[ANDROID_SOCKET_Zygot] bla bla bla ..


编辑1: 我在ubuntu上使用Openjdk

帮助我!!

3 个答案:

答案 0 :(得分:2)

有一个相当大的[android] [ffmpeg]讨论,以及如何...

在非root用户设备上调用ffmpeg的正常方法(即普通用户的应用程序)是使用NDK和C-lang集成,在java中你可以进行普通的方法调用,包装CLI的东西和集合JNI层将传递给ffmpeg可执行文件接口的parms。

android调用step1.android的例子是......

             new FFMpegTask().execute(invoke_lib_path,"ffmpeg", "-y", 
                    "-i", Picture.getPath(), "-i", recordFilePath,
                    "-vcodec", "mpeg4", "-s", siz,
                    "-r", "15", "-b:v", "200k",
                    "-acodec", "copy", "-f", "3gp"
                    ,pathOut);

Step2.c

  

JNIEXPORT void JNICALL Java_com _..._ naRun(        JNIEnv * env,jobject obj,jobjectArray args){int i = 0; int argc = 0; char ** argv = NULL;

     

if(args!= NULL){argc =( env) - > GetArrayLength(env,args);         argv =(char * )malloc(sizeof(char *)* argc);

  for(i=0;i<argc;i++)         {           jstring str =
     

(jstring)(* env) - &gt; GetObjectArrayElement(env,args,i); argv [i] =   (char *)(* env) - &gt; GetStringUTFChars(env,str,NULL); int j = 0;     j = main(argc,argv); }

尝试使用java runtime.exec()类型的CLI调用是我称之为黑客攻击,这将浪费你的时间。

通过在.apk中使用NDK和普通包装,您可以确保更高程度的可靠性和集成之间的集成,例如部署设备上的处理器体系结构和为ffmpeg构建的处理器。

尝试阅读roman10的intro

然后你可能会尝试依赖许多为android设置ffmpeg的人的面包屑..即google“android-ffmpeg”

如果您是root用户并且已经编译了可执行文件,则可以通过获取shell并使用adb CLI来调用它。请注意,这不像使用java作为runtime.exec调用的包装器。

  

adb push ffmpeg / data / local / ffmpeg / ffmpeg

     

./ ffmpeg -codecs

答案 1 :(得分:1)

AFAIK您无法使用cd命令。它是一个bash指令,没有可执行的cd。我猜ffmpeg由于权限不起作用。在adb shell上执行chmod 777 ffmpeg并再试一次

答案 2 :(得分:0)

我可能会有点迟到

java.io.IOException: Error running exec(). Command: [ffmpeg] Working Directory: null Environment: null

清楚地提到找不到ffmpeg文件。在执行任何命令之前,首先检查文件是否存在于上述位置。

boolean ifFileExists = new File(*path_of_ffmpeg_file*).exists();


       if(ifFileExists==false){
// code to write file in phone
    }
method(commandToExecute);