android 中如何实现关机、重启?

android 中如何实现关机、重启?
手机已经ROOT,程序也得到了root权限,但通过Runtime.getRuntime().exec("reboot");并未重启成功,手机中也确实找到了reboot文件,在终端模拟器上用reboot命令也是可以重启的。
那么,应该怎么做才能实现重启呢?还有关机。

补充:

我用Runtime.getRuntime().exec("su -c \"reboot\"");并没有成功重启,但用Runtime.getRuntime().exec("su -c \"/system/bin/reboot\"");成功重启了。这是为什么?还有一个问题就是这样的重启方法好像和系统自身的重启不一样,系统自身的重启时会有一个关机动画,但这个重启时是屏幕变暗后直接重启,我怕这样的重启是不安全的。

关于关机,我看了一下,手机(已ROOT)中并没有找到shutdown,那么关机要怎么做呢?自己做一个shutdown文件?

Rebooter.java

import java.io.DataOutputStream;
import java.io.IOException;

final class Rebooter extends Thread
{
    public static final String CMD_REBOOT = "reboot";
    public static final String CMD_SHUTDOWN = "reboot -p";

    private final String strEnter = "\n";
    private final String cmd_su = "su";
    private final String cmd_exit = "exit";

    private String cmd = "";

    public Rebooter()
    {
    }

    public void reboot(String command)
    {
        if (!command.equals(CMD_REBOOT) && !command.equals(CMD_SHUTDOWN))
            return;

        cmd = command;
        start();
    }

    @Override
    public void run()
    {
        try
        {
            sleep(1000);

            Process localProcess = Runtime.getRuntime().exec(cmd_su);
            DataOutputStream localDataOutputStream = new DataOutputStream(
                    localProcess.getOutputStream());
            localDataOutputStream.writeBytes(cmd + strEnter);
            localDataOutputStream.writeBytes(cmd_exit + strEnter);
            localDataOutputStream.flush();
            localDataOutputStream.close();
            localProcess.waitFor();
            localProcess.destroy();
        }
        catch (InterruptedException e)
        {
            e.printStackTrace();
        }
        catch (IOException e)
        {
            e.printStackTrace();
        }
    }

}

1、手机必需先获得Root权限,并授予应用程序Root权限。
2、手机中有reboot命令,如果没有就得自己另做处理。

关机:

In frameworks/base/services/java/com/android/server/BatteryService.java

Intent intent = new Intent(Intent.ACTION_REQUEST_SHUTDOWN);

  intent.putExtra(Intent.EXTRA_KEY_CONFIRM, false);

  intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);

  mContext.startActivity(intent);

重启:

Intent i = new Intent(Intent.ACTION_REBOOT);
i.putExtra("nowait", 1);
i.putExtra("interval", 1);
i.putExtra("window", 0);
sendBroadcast(i);
我用Runtime.getRuntime().exec("su -c \"reboot\"");并没有成功重启,但用Runtime.getRuntime().exec("su -c \"/system/bin/reboot\"");成功重启了

这是因为/system/bin/不在$PATH里。

主要思路:

1、需要源码才能编译

2、修改项目的Android.mk文件,添加

LOCAL_CERTIFICATE := platform

3、AndroidManifest.xml中添加权限
3.1 manifest标签中添加

 android:sharedUserId="android.uid.system"

3.2 使用权限

 <uses-permission android:name="android.permission.SHUTDOWN"/>

4、java代码

// 创建Intent  
// 如果是要重启,则使用Intent.ACTION_REBOOT  
Intent intent = new Intent(Intent.ACTION_REQUEST_SHUTDOWN);  
// 设置是否需要用户确认,若不需要,可以不设置或设置为false  
intent.putExtra(Intent.EXTRA_KEY_CONFIRM, true);  
// 当作新任务执行  
intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);  
// 启动  
startActivity(intent);  

据说还可以使用Broadcast的方式调用,不过我试了一下,窗口是出来了,但一直停在关机的进度条那。不知道是不是机子的问题。
代码如下:

Intent i = new Intent(Intent.ACTION_REBOOT);  
i.putExtra("nowait", 1);  
i.putExtra("interval", 1);  
i.putExtra("window", 0);  
sendBroadcast(i);  

我经常用到的也就是Sueyexin的方法
不过我在一个人的博客中看到了这个方法,你可以试试

pad中自带有重启命令(reboot),本以为也应该有关机命令啊,可是找来找去也没有诸如shutdown、poweroff等命令。
然后想到busybox中有一个poweroff命令,故又去编译了一个busybox推进去,执行了busybox poweroff之后看了下返回码
是0(即echo $?),然后,然后什么也没发生……
网上流行的方法是在Java端发送要求关机的Intent,如下:
view plaincopy
Intent intent = new Intent(Intent.ACTION_REQUEST_SHUTDOWN);

intent.putExtra(Intent.EXTRA_KEY_CONFIRM, false);

intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);

mContext.startActivity(intent);

需要加上相关的权限:

android:sharedUserId="android.uid.system"
<uses-permission android:name="android.permission.SHUTDOWN"/>
   但要从apk中实现关机的话,那还不如手动关呢,所以自己动手了,写个诸如shutdown的小命令,实现也很简单,

找到系统怎么处理关机的,然后我们把那段代码拿出来就行了。
系统中处理关机部分的代码路径:frameworks/base/core/jni/android_os_Power.cpp,里面有个
android_os_Power_shutdown方法,具体如下:

view plaincopy
static void android_os_Power_shutdown(JNIEnv *env, jobject clazz)  
{  
    sync();  
#ifdef HAVE_ANDROID_OS  
    reboot(RB_POWER_OFF);  
#endif  
}  

看到有个reboot函数,然后再追踪这个reboot(RB_POWER_OFF)函数,看到其最终会调用__reboot这个函数
(汇编实现的),我们只要给它传递“合适”的参数就行了。
代码实现:
1、在Android源码目录中的packages/apps/下创建一个目录,例如:Hello。
2、编写Android.mk文件:

LOCAL_PATH:= $(call my-dir)
include $(CLEAR_VARS)
LOCAL_SRC_FILES:= shutdown.c
LOCAL_MODULE:= shutdown 
LOCAL_PRELINK_MODULE:= false
LOCAL_SHARED_LIBRARIES:= libutils
include $(BUILD_EXECUTABLE)

3、编写shutdown.c源文件,如下:

view plaincopy
#include <stdio.h>  
#include <stdlib.h>  
#include <utils/Log.h>  
#include <sys/reboot.h>  

#ifndef LOG_TAG  
#define LOG_TAG "dxyh"  
#endif  

int main(int argc, char **argv)  
{  
    int retval;  

    if ((retval = __reboot(LINUX_REBOOT_MAGIC1, LINUX_REBOOT_MAGIC2,  
                    RB_POWER_OFF, NULL)) < 0) {  
        LOGE("Try to shutdown the machine failed!");  
        exit(EXIT_FAILURE);  
    }  
    return 0;  
}  

4、mm
5、将生成的shutdown可执行程序推入到pad中的/system/bin下即可。
6、然后就可以和其他命令一样运行了,例如adb shell shutdown。