Android中如何修改系统时间(应用程序获得系统权限)

Android 的API中有提供 SystemClock.setCurrentTimeMillis()函数来修改系统时间,可惜无论你怎么调用这个函数都是没用的,无论模拟器还是真机,在logcat中总会得到"Unable to open alarm driver: Permission denied ".这个函数需要root权限或者运行与系统进程中才可以用。

        本来以为就没有办法在应用程序这一层改系统时间了,后来在网上搜了好久,知道这个目的还是可以达到的。

        第一个方法简单点,不过需要在Android系统源码的环境下用make来编译:

        1. 在应用程序的AndroidManifest.xml中的manifest节点中加入android:sharedUserId="android.uid.system"这个属性。

        2. 修改Android.mk文件,加入LOCAL_CERTIFICATE := platform这一行

        3. 使用mm命令来编译,生成的apk就有修改系统时间的权限了。

 

        第二个方法麻烦点,不过不用开虚拟机跑到源码环境下用make来编译:

        1. 同上,加入android:sharedUserId="android.uid.system"这个属性。

        2. 使用eclipse编译出apk文件,但是这个apk文件是不能用的。

        3. 用压缩软件打开apk文件,删掉META-INF目录下的CERT.SF和CERT.RSA两个文件。

        4. 使用目标系统的platform密钥来重新给apk文件签名。这步比较麻烦,首先找到密钥文件,在我的Android源码目录中的位置是"build/target/product/security",下面的platform.pk8和platform.x509.pem两个文件。然后用Android提供的Signapk工具来签名,signapk的源代码是在"build/tools/signapk"下,用法为"

signapk platform.x509.pem platform.pk8 input.apk output.apk",文件名最好使用绝对路径防止找不到,也可以修改源代码直接使用。

        这样最后得到的apk和第一个方法是一样的。

 

        最后解释一下原理,首先加入android:sharedUserId="android.uid.system"这个属性。通过Shared User id,拥有同一个User id的多个APK可以配置成运行在同一个进程中。那么把程序的UID配成android.uid.system,也就是要让程序运行在系统进程中,这样就有权限来修改系统时间了。

        只是加入UID还不够,如果这时候安装APK的话发现无法安装,提示签名不符,原因是程序想要运行在系统进程中还要有目标系统的platform key,就是上面第二个方法提到的platform.pk8和platform.x509.pem两个文件。用这两个key签名后apk才真正可以放入系统进程中。第一个方法中加入LOCAL_CERTIFICATE := platform其实就是用这两个key来签名。

        这也有一个问题,就是这样生成的程序只有在原始的Android系统或者是自己编译的系统中才可以用,因为这样的系统才可以拿到platform.pk8和platform.x509.pem两个文件。要是别家公司做的Android上连安装都安装不了。试试原始的Android中的key来签名,程序在模拟器上运行OK,不过放到G3上安装直接提示"Package ... has no signatures that match those in shared user android.uid.system",这样也是保护了系统的安全。

 

        最最后还说下,这个android:sharedUserId属性不只可以把apk放到系统进程中,也可以配置多个APK运行在一个进程中,这样可以共享数据,应该会很有用的。



adb shell  进入shell终端界面

1、先设置系统的时区配置

cat /data/property/persist.sys.timezone //查看当前时区配置文件
setprop persist.sys.timezone GMT        //
修改属性 
2、开始设置修改当前系统时间

date -s "yyyymmdd.[[[hh]mm]ss]"  System/bin/date -s "yyyymmdd.[[[hh]mm]ss]"

3、查看是否生效

date 

二、用代码实现修改Android系统时间的方法

public void testDate(){
try {
Process process = Runtime.getRuntime().exec("su");
String datetime="20131023.112800"; //
测试的设置的时间【时间格式 yyyyMMdd.HHmmss
DataOutputStream os = new DataOutputStream(process.getOutputStream());
os.writeBytes("setprop persist.sys.timezone GMT\n");
os.writeBytes("/system/bin/date -s "+datetime+"\n");
os.writeBytes("clock -w\n");
os.writeBytes("exit\n");
os.flush();
} catch (IOException e) {
e.printStackTrace();
}
}

 

android默认系统日期、时间、时区更改

 

    android平台,经常会遇到产品需要更改系统默认时区日期时间的问题。android更改时区相对比较容易实现,网上也有很多资料,直接设置一个属性即可,例如设置上海东八区,persist.sys.timezone=Asia/Shanghai。但是如何实现更改默认系统时间呢?

      android中,Linux内核中、rtc时钟,默认的起始日期都是197011日,那么如何把默认日期指到2012-01-01呢?笔者在实践中发现,在RTC驱动中可以很容易实现。在RTC驱动加载的时候,一般都有个probe函数需要先执行,因此在probe函数里下手最直接有效。RTC1970-01-01开始,那当然很容易把默认值设置到2012-01-01,所需要设置的seconds也就是从1970-01-01所差的秒数,以秒为单位。因此,一旦读出来的RTC值小于我们预想的值比如2012-01-011325402913)小,我们就把它设置到这个时间点。

[cpp] view plaincopy

1.  seconds = rtc_read_time();  

2. printk("init PMU/RTC time to  %ld \n", seconds);  

3.    

4. if(seconds <= 1325402913) {  

5.      seconds = 1325402913;/*2012-01-01*/  

6.     ret = rtc_set_time(seconds);  

7.      RTC_DBG("Init Set time: %ld, ret =0x%x\n", seconds, ret);  

8. }  


 
笔者通过结果显示,在 android 平台上是可行的。要设置到哪天几点几分,只要算好具体 1970-01-01 00 00 差的秒数即可,简单明了。这样客户、用户使用起来更方便一些。



Android 通过应用设置系统日期和时间的方法

android 2.3 

android 4.0

测试可行,不过需要ROOT权限.


    import java.io.DataOutputStream;  
    import java.io.File;  
    import java.io.IOException;  
    import java.util.Calendar;  
      
    import android.os.SystemClock;  
      
    public class SystemDateTime {  
          
        static final String TAG = "SystemDateTime";   
          
        public static void setDateTime(int year, int month, int day, int hour, int minute) throws IOException, InterruptedException {  
      
            requestPermission();  
      
            Calendar c = Calendar.getInstance();  
      
            c.set(Calendar.YEAR, year);  
            c.set(Calendar.MONTH, month-1);  
            c.set(Calendar.DAY_OF_MONTH, day);  
            c.set(Calendar.HOUR_OF_DAY, hour);  
            c.set(Calendar.MINUTE, minute);  
              
              
            long when = c.getTimeInMillis();  
      
            if (when / 1000 < Integer.MAX_VALUE) {  
                SystemClock.setCurrentTimeMillis(when);  
            }  
      
            long now = Calendar.getInstance().getTimeInMillis();  
            //Log.d(TAG, "set tm="+when + ", now tm="+now);  
      
            if(now - when > 1000)  
                throw new IOException("failed to set Date.");   
              
        }  
      
        public static void setDate(int year, int month, int day) throws IOException, InterruptedException {  
      
            requestPermission();  
      
            Calendar c = Calendar.getInstance();  
      
            c.set(Calendar.YEAR, year);  
            c.set(Calendar.MONTH, month);  
            c.set(Calendar.DAY_OF_MONTH, day);  
            long when = c.getTimeInMillis();  
      
            if (when / 1000 < Integer.MAX_VALUE) {  
                SystemClock.setCurrentTimeMillis(when);  
            }  
      
            long now = Calendar.getInstance().getTimeInMillis();  
            //Log.d(TAG, "set tm="+when + ", now tm="+now);  
      
            if(now - when > 1000)  
                throw new IOException("failed to set Date.");  
        }  
      
        public static void setTime(int hour, int minute) throws IOException, InterruptedException {  
              
            requestPermission();  
      
            Calendar c = Calendar.getInstance();  
      
            c.set(Calendar.HOUR_OF_DAY, hour);  
            c.set(Calendar.MINUTE, minute);  
            long when = c.getTimeInMillis();  
      
            if (when / 1000 < Integer.MAX_VALUE) {  
                SystemClock.setCurrentTimeMillis(when);  
            }  
      
            long now = Calendar.getInstance().getTimeInMillis();  
            //Log.d(TAG, "set tm="+when + ", now tm="+now);  
      
            if(now - when > 1000)  
                throw new IOException("failed to set Time.");  
        }  
          
        static void requestPermission() throws InterruptedException, IOException {  
            createSuProcess("chmod 666 /dev/alarm").waitFor();  
        }  
          
        static Process createSuProcess() throws IOException  {  
            File rootUser = new File("/system/xbin/ru");  
            if(rootUser.exists()) {  
                return Runtime.getRuntime().exec(rootUser.getAbsolutePath());  
            } else {  
                return Runtime.getRuntime().exec("su");  
            }  
        }  
          
        static Process createSuProcess(String cmd) throws IOException {  
      
            DataOutputStream os = null;  
            Process process = createSuProcess();  
      
            try {  
                os = new DataOutputStream(process.getOutputStream());  
                os.writeBytes(cmd + "\n");  
                os.writeBytes("exit $?\n");  
            } finally {  
                if(os != null) {  
                    try {  
                        os.close();  
                    } catch (IOException e) {  
                    }  
                }  
            }  
      
            return process;  
        }  
    }  



相关文章:

Android中如何修改系统时间(应用程序获得系统权限)

android 命令修改时间或程序修改系统时间

Android 通过应用设置系统日期和时间的方法


  • 0
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 3
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值