解析android framework下利用app_process来调用java写的命令及示例

在android SDK的framework/base/cmds目录下了,有不少目录,这些目的最终都是build出一个bin文件,再存放到/system/bin目录下,对于C/CPP写的命令,我们还是比较好理解的,都有一个main函数作为入口,但是在cmds目录下还有一些原生代码是java的,比如input、settings,那么这种类型的命令是怎么实现的呢?

笔者研习了原生的命令实现,写了一个demo,抛砖引玉吧!暂时叫strong吧!我们都知道java写的文件最后都是编译成了class文件,java类里面也有很多接口,在android平台上cmds目录下的各模块的java文件都实现了一个共同的方法,还是叫main(),真是情有独钟啊!当然从技术角度看叫其他名字也是可以的。那我们就简单实现以下这个class吧!如下:

  1. /*
  2. * Copyright (C) 2012 The Android Open Source Project
  3. *
  4. * Licensed under the Apache License, Version 2.0 (the "License");
  5. * you may not use this file except in compliance with the License.
  6. * You may obtain a copy of the License at
  7. *
  8. * http://www.apache.org/licenses/LICENSE-2.0
  9. *
  10. * Unless required by applicable law or agreed to in writing, software
  11. * distributed under the License is distributed on an "AS IS" BASIS,
  12. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13. * See the License for the specific language governing permissions and
  14. * limitations under the License.
  15. */
  16.  
  17. package com.android.commands.strong;
  18.  
  19. import android.app.ActivityManagerNative;
  20. import android.app.IActivityManager;
  21. import android.app.IActivityManager.ContentProviderHolder;
  22. import android.content.IContentProvider;
  23. import android.os.Binder;
  24. import android.os.Bundle;
  25. import android.os.IBinder;
  26. import android.os.RemoteException;
  27. import android.os.UserHandle;
  28.  
  29. public final class strongcmd {
  30. static final String TAG = "strong";
  31.  
  32. static String[] mArgs;
  33. int mNextArg;
  34. static int value = 0;
  35.  
  36. public static void main(String[] args) {
  37. int c;
  38.  
  39. printUsage();
  40. System.err.println("Wellcom strong test function!!");
  41.  
  42. try {
  43. new strongcmd().run();
  44. } catch (Exception e) {
  45. System.err.println("Unable to run settings command");
  46. }
  47. }
  48.  
  49. public void run() {
  50.  
  51. try {
  52. System.err.println("Now strong run() again");
  53. } catch (Exception e) {
  54. System.err.println("Now strong run() Exception");
  55. }
  56.  
  57. }
  58.  
  59. private String nextArg() {
  60. if (mNextArg >= mArgs.length) {
  61. return null;
  62. }
  63. String arg = mArgs[mNextArg];
  64. mNextArg++;
  65. return arg;
  66. }
  67.  
  68. private static void printUsage() {
  69. System.err.println("usage: strong -a -b -h");
  70. System.err.println("'a' is for add");
  71. System.err.println("-h for help");
  72. }
  73. }

/*****************************************************************************************************/
声明:本博内容均由http://blog.csdn.net/edsam49原创,转载请注明出处,谢谢!
/*****************************************************************************************************/

写好Android.mk,编译好这个文件,会生成strong.jar包,包含这个class。那么,又怎么跟命令挂钩呢?先看看Android.mk,如下:

  1. # Copyright 2011 The Android Open Source Project
  2. #
  3. LOCAL_PATH:= $(call my-dir)
  4. include $(CLEAR_VARS)
  5.  
  6. LOCAL_SRC_FILES := $(call all-subdir-java-files)
  7. LOCAL_MODULE := strong
  8. LOCAL_MODULE_TAGS := optional
  9. include $(BUILD_JAVA_LIBRARY)
  10.  
  11. include $(CLEAR_VARS)
  12. LOCAL_MODULE := strong
  13. LOCAL_SRC_FILES := pre_strong
  14. LOCAL_MODULE_CLASS := EXECUTABLES
  15. LOCAL_MODULE_TAGS := optional
  16. include $(BUILD_PREBUILT)

上一部分是BUILD_JAVA_LIBRARY,关键在下面,利用的是BUILD_PREBUILT,添加一个预编译好的应用程序,我们叫pre_strong,它有可执行的权限,看看它的具体实现吧!

  1. # Script to start "strong" on the device
  2. #
  3. base=/system
  4. export CLASSPATH=$base/framework/strong.jar
  5. exec app_process $base/bin com.android.commands.strong.strongcmd "$@"

首先还是设置好这个java lib的路径,如何再调用app_process来执行,主要是把这个类名要给对,app_process其实也是个命令。在app_process里面,还是一样利用JNI技术,在java ENV里面查找传给app_process的class,找到这个class后再去找main函数接口的field,然后再call这个main接口,这样就call到java里面去了。

下面简要看看app_process的关键代码吧!

  1. virtual void onVmCreated(JNIEnv* env)
  2. {
  3. if (mClassName == NULL) {
  4. return; // Zygote. Nothing to do here.
  5. }
  6.  
  7. /*
  8. * This is a little awkward because the JNI FindClass call uses the
  9. * class loader associated with the native method we're executing in.
  10. * If called in onStarted (from RuntimeInit.finishInit because we're
  11. * launching "am", for example), FindClass would see that we're calling
  12. * from a boot class' native method, and so wouldn't look for the class
  13. * we're trying to look up in CLASSPATH. Unfortunately it needs to,
  14. * because the "am" classes are not boot classes.
  15. *
  16. * The easiest fix is to call FindClass here, early on before we start
  17. * executing boot class Java code and thereby deny ourselves access to
  18. * non-boot classes.
  19. */
  20. char* slashClassName = toSlashClassName(mClassName);
  21. mClass = env->FindClass(slashClassName);
  22. if (mClass == NULL) {
  23. ALOGE("ERROR: could not find class '%s'\n", mClassName);
  24. }
  25. free(slashClassName);
  26.  
  27. mClass = reinterpret_cast<jclass>(env->NewGlobalRef(mClass));
  28. }
  29.  
  30. virtual void onStarted()
  31. {
  32. sp<ProcessState> proc = ProcessState::self();
  33. ALOGV("App process: starting thread pool.\n");
  34. proc->startThreadPool();
  35.  
  36. AndroidRuntime* ar = AndroidRuntime::getRuntime();
  37. ar->callMain(mClassName, mClass, mArgC, mArgV);
  38.  
  39. IPCThreadState::self()->stopProcess();
  40. }
  41.  
  42. if (className) {
  43. // Remainder of args get passed to startup class main()
  44. runtime.mClassName = className;
  45. runtime.mArgC = argc - i;
  46. runtime.mArgV = argv + i;
  47. runtime.start("com.android.internal.os.RuntimeInit",
  48. application ? "application" : "tool");
  49. }

Android平台提供的app_process,还是相当不错的,比较实用,利用好app_process还是可以写成很多供我们自己开发、测试、定制一些特殊的程序,给开发带来了很大的便利。

解析android framework下利用app_process来调用java写的命令及示例的更多相关文章

  1. [Android] 解析android framework下利用app_process来调用java写的命令及示例

    reference to :http://bbs.9ria.com/thread-253058-1-1.html 在android SDK的framework/base/cmds目录下了,有不少目录, ...

  2. Android平台下利用zxing实现二维码开发

    Android平台下利用zxing实现二维码开发 现在走在大街小巷都能看到二维码,而且最近由于项目需要,所以研究了下二维码开发的东西,开源的二维码扫描库主要有zxing和zbar,zbar在iPos平 ...

  3. (转载)Android平台下利用zxing实现二维码开发

    Android平台下利用zxing实现二维码开发 现在走在大街小巷都能看到二维码,而且最近由于项目需要,所以研究了下二维码开发的东西,开源的二维码扫描库主要有zxing和zbar,zbar在iPos平 ...

  4. Jmeter用BeanShell Sampler调用java写的jar包进行MD5加密

    [前言] 在工作中,有时候我们请求的参数可能需要加密,比如登录接口中的密码做了加密操作,今天我就给大家介绍一种方法:Jmeter用BeanShell Sampler调用java写的jar包进行MD5加 ...

  5. .NET调用Java写的WebService

    最近遇到一个用.net调用java写的webservice的应用,对方程序员提供了一个后缀为wsdl的文件,这个跟.Net里面生成的wsdl文件差不多,起初没什么概念就查了点资料,知道可以将这个wsd ...

  6. delphi7调用java写的webservice,在调用的时候弹出“wssecurityhandler:request does not contain required security header”

    delphi7调用java编写的webservice问题我用delphi7调用java写的webservice,在调用的时候弹出“wssecurityhandler:request does not ...

  7. 【转】Android平台下利用zxing实现二维码开发

    http://www.cnblogs.com/dolphin0520/p/3355728.html 现在走在大街小巷都能看到二维码,而且最近由于项目需要,所以研究了下二维码开发的东西,开源的二维码扫描 ...

  8. 深入了解android平台的jni---本地多线程调用java代码

    一.jni调用java对象     JNI提供的功能之一是在本地代码中使用Java对象.包括:创建一个java类对象和通过函数传递一个java对象.创建一个java类对象,首先需要得到得到使用Find ...

  9. Android JNI之C/C++层调用JAVA

    转载请声明:原文转自:http://www.cnblogs.com/xiezie/p/5930032.html 从C/C++层调用JAVA层代码步骤: 1. 在JAVA类中创建java方法和本地方法 ...

随机推荐

  1. 金额的计算BigDecimal类

    金额的计算BigDecimal类 double d = 9.84; double d2 = 1.22; //注意需要使用BigDecimal(String val)构造方法 BigDecimal bi ...

  2. c#引用web.config中的ConnectionString

    c#引用web.config中的ConnectionString <connectionStrings>  <add name="JKXTConnectionString& ...

  3. java-创建线程的两种方式

    1. 继承Thread类 定义类继承Thread类. 覆盖run方法. 实例化子类对象,调用start()方法,从而调用run方法. 2.实现Runnable接口 定义类实现Runnable接口. 覆 ...

  4. linux命令:scp

    有时候ftp被禁用了, 就用scp替代; 命令行: scp from to_user@to_ip:dir_to/file_name 执行该命令之后,按照提示输入to_host的登陆密码即可. scp ...

  5. linux 进程线程拓展

    依次参考: 多线程和多进程的区别(小结) Linux内核源代码分析——fork()原理&多进程网络模型 Linux写时拷贝技术(copy-on-write) linux内核 do_fork 函 ...

  6. MVC控制器里面使用dynamic和ExpandoObject

    MVC控制器里面使用dynamic和ExpandoObject 在很多时候,我们在数据库里面定义表字段和实际在页面中展示的内容,往往是不太匹配的,页面数据可能是多个表数据的综合体,因此除了我们在表设计 ...

  7. SpringMVC记住密码功能

    CookieTool (Cookie帮助类): package com.utcsoft.common.cookie; import java.util.HashMap; import java.uti ...

  8. 基于visual Studio2013解决面试题之0602全排列

     题目

  9. Magento给产品添加“new”或者折扣数量标签 magento new label. discount label

    文章最底部有效果图. 给新产品添加“new”的标签.给折扣产品,显示出折扣的数量. 这个可以自己写一段代码加在到模板文件夹下面的catalog/product/list.phtml中. 以下是代码 & ...

  10. VC++ WIN32 sdk实现按钮自绘详解.

    网上找了很多,可只是给出代码,没有详细解释,不便初学者理解.我就抄回冷饭.把这个再拿出来说说. 实例图片:    首先建立一个标准的Win32 Application 工程.选择a simple Wi ...