为 Android 添加 Java 层服务

1.     为什么要写底层核心服务呢? 
         因为底层核心服务是 Android框架里最接近 Linux/Driver的部分。为了充分发挥硬件设备的差异化特性,核心服务是让上层 Java应用程序来使用 Driver/HW Device 特色的重要管道。例如 Media、 Telephone等底层硬件。

       在开机过程中,就可以启动核心服务(汉字输入法服务等),让众多应用程序来共同使用。

由于共用,所以能有效降低 Java应用程序的大小( Size)。

2.     核心服务与 Java 层的 Service有何区别和关系? 
       Android具有两层服务

             --Java层 SDK-based Service

             --C++层的 Code Service

   

 

 

3. 编写自己的核心服务( C++ 层) 
1). 要点 
      核心服务通常在独立的进程( Process )里执行。

      必须提供 IBinder 接口,让应用程序可以进行跨进程的绑定( Binding )和调用。

      因为共用,所以必须确保多线程安全( Thread-safe )。

使用 C++ 来实现,并调用 IServiceManager::addService() 函数添加到系统的 Binder Driver 里。

上层应用程序通过 ServiceManager 获取该服务。

上层应用程序通过 IBinder::transact() 函数来与核心服进行数据交互。

2). 添加服务 
下面详细介绍如何添加一个底层服务到系统中,假设服务名为 AddService ,其用途是对传入的参数加上 1000 ,并返回结果。

服务实现 
      进入 android 源码 的目录 frameworks/base ,在该目录下建立自己的目录,假设为 addservice ,再在这个目录中建立两个子目录 addserver 和 libaddservice , addserver 用于存放服务的启动文件,其最终的生成为可执行文件,在系统启动的时候运行, libaddservice 用于存放服务的实现文件,最终会生成动态链接库,有 addserver 调用。

 

首先,服务的实现文件包括两个文件,   AddService.h 和 AddService.cpp ,

以下是 AddService.h :

#ifndef ANDROID_GUILH_ADD_SERVICE_H

#define ANDROID_GUILH_ADD_SERVICE_H

 

#include <utils/RefBase.h>

#include <binder/IInterface.h>

#include <binder/Parcel.h>

#include <utils/threads.h>

 

namespace android {

        class AddService : public BBinder{// 从 BBinder 派生,实现本地接口

        

                public:

                static int instantiate();

                AddService();

                virtual ~AddService();

                virtual status_t onTransact(uint32_t, const Parcel&, Parcel*, uint32_t);

        };

}; //namespace

#endif

然后是服务的实现文件 AddService.cpp :

#include "AddService.h"

#include <binder/IServiceManager.h>

#include <binder/IPCThreadState.h>

namespace android {

      static struct sigaction oldact;

static pthread_key_t sigbuskey;

// 把自己注册到系统中

int AddService::instantiate() {

LOGE("AddService instantiate");

int r = defaultServiceManager()->addService(

String16("guilh.add"), new AddService());// 这里主要是把 //AddSerice 这个服务添加到 Binder Driver 中服务名为 guilh.add

LOGE("AddService r = %d/n", r);

return r;

}

// 构造函数

AddService::AddService()

{

LOGV("AddService created");

mNextConnId = 1;

pthread_key_create(&sigbuskey, NULL);

}

// 析构函数

AddService::~AddService()

{

pthread_key_delete(sigbuskey);

LOGV("AddService destroyed");

}

// 这个是服务具体的本地实现,功能实现都应该放在这里面,通过传入执行代码( code ) // 的不同来执行不同的操作,上层隐射为不同的 api 。

status_t AddService::onTransact(

uint32_t code, const Parcel& data, Parcel* reply, uint32_t flags){

switch(code) {

case 0: {// 根据 code 的不同执行不同的操作

pid_t pid = data.readInt32();

int num = data.readInt32();

num = num + 1000;

reply->writeInt32(num);

return NO_ERROR;

}

break;

default:

return BBinder::onTransact(code, data, reply, flags);

}

}}; //namespace

 

以下是编译服务的 Android.mk ,和上面的 cpp 放在一起。

LOCAL_PATH:= $(call my-dir)

include $(CLEAR_VARS)

LOCAL_SRC_FILES:= /

AddService.cpp

LOCAL_C_INCLUDES := /

$(JNI_H_INCLUDE)

LOCAL_SHARED_LIBRARIES :=     /

        libcutils             /

        libutils              /

        libbinder             /

        libandroid_runtime

LOCAL_PRELINK_MODULE := false

LOCAL_MODULE := libAdd

include $(BUILD_SHARED_LIBRARY)   #这一行表示编译为动态库

 

在命令行中退出到 android/目录级 加载编译环境 . build/envsetup.sh

然后 lunch。

然后在 cd  /android/frameworks/base/addservice/ libaddservice/ 目录 输入 mm

之后在 out 目录产出 libAdd.so 文件。

在此 完成核心服务第一步。

 

服务进程实现 
                进入到 cd  /android/frameworks/base/addservice/addserver/ 目录

增加一个文件 addserver.cpp ,文件内容如下:

 

#include <sys/types.h>

#include <unistd.h>

#include <grp.h>

#include <binder/IPCThreadState.h>

#include <binder/ProcessState.h>

#include <binder/IServiceManager.h>

#include <utils/Log.h>

#include <private/android_filesystem_config.h>

#include "../libaddservice/AddService.h"

//#include <libadd/AddService.h>

using namespace android;

int main(int argc, char** argv)

{

sp<ProcessState> proc(ProcessState::self());

sp<IServiceManager> sm = defaultServiceManager();//取得 ServiceManager

LOGI("ServiceManager: %p", sm.get());

AddService::instantiate();//把自己添加到 ServiceManager中

ProcessState::self()->startThreadPool();//启动缓冲池

IPCThreadState::self()->joinThreadPool();//这里是把服务添加到 Binder闭合循环进程中

}

以上为底层服务的标准操作。

下面是这个服务 makefile:

include $(CLEAR_VARS)

 

LOCAL_SRC_FILES:= /

        addserver.cpp

 

LOCAL_SHARED_LIBRARIES := /

        libAdd /

        libutils /

        libbinder

LOCAL_MODULE:= addserver

include $(BUILD_EXECUTABLE)//编译为可执行文件

退出后当前目录执行 mm即可在 out目录的 system/bin下产出 addserver可执行文件。

实现服务进程开机自动运行 
进入到 /android/system/core/rootdir/目录中有个 init.rc文件

vi init.rc

在 service中添加

service addservice    /system/bin/addserver    //将 /system/bin/addserver作为一个服务启动,服务的名称为 addservice(这个不重要)。

 

最后退出到 android/目录下执行全编译:

输入 . build/envsetup.sh

Lunch

Make

完成之后

Emulator打开模拟器

打开另一个 shell终端 输入 adb shell    进入模拟器模式      如果 adbshell系统提示没有发现该命令 就在 android/out/host/linux-x86/bin/中输入   ./adb shell  

在输入 ps  查看进程   找到是否有 addserver进程

如果有就成功一半。

 

测试我们的服务 
 

随便在 android/packages/apps 中 建立一个简单的应用程序,

这里可以直接在 eclipse 中建立好工程 拷贝到 android/packages/apps 中,然后为应用添加一个 Android.mk 文件,可以从其他应用中拷贝来修改。

在应用程序中测试服务的代码:

    private void test(){

                try{

                        IBinder binder = ServiceManager.getService("guilh.add");// 取得服务

                        Parcel data = Parcel.obtain();

                        Parcel reply = Parcel.obtain();

                        if(binder == null)

                                Log.d(TAG,"failed to get service");

                        data.writeInt(Process.myPid());// 固定操作

                        data.writeInt(100);// 传入参数

                        binder.transact(0, data, reply, 0);// 执行远程调用

                        Log.d(TAG,"result="+reply.readInt());// 验证结果

                }catch(Exception e){

                         Log.d(TAG,e.toString());

                }

Java层服务顾名思义即为从 Java层提供的服务,它与 C++层的服务不同之处在于其服务进程的运行由系统( ServiceManager)统一维护。在文件 frameworks /base /services /java /com /android /server /SystemServer.java 中我们可以看到以下代码:

AlarmManagerService alarm = new AlarmManagerService (context );

ServiceManager .addService (Context .ALARM_SERVICE , alarm );

 

这个操作在系统启动时完成,由 ServiceManager 负责创建服务进程并运行之。所以我们要做的事就是实现一个 java 服务,并将其添加到这里并由系统运行起来,以下是具体实现步骤:

 

 

实现自己的 java层 service

1.       在目录

frameworks/base/core/java/android/os中增加自己的 AIDL文件用来申明服务:

BelyService.aidl:

package android.os;

interface IBelyService {

        /**

        *

        */

        int calculateSqu(int value);

}

2.       在目录

frameworks/base/services/java/com/android/server增加 service的实现文件:

BelyService.java:

package com.android.server;

import android.content.Context;

import android.os.IBelyService;

public class BelyService extends IBelyService.Stub {

        public BelyService(Context context){

                super();

        }

        public int calculateSqu(int val){

                return val*val;

        }

}

 

3.       将 java服务添加到 ServiceManager中:

BelyService bs = new BelyService(context);

ServiceManager.addService("BelyService", bs);

 

自此,重新编译 Android并运行,我们所创建的服务即可访问,下面是演示如何调用:

同样在 package/apps下任意创建一个应用,调用服务的方法如下:

import android.os.ServiceManager;

import android.os.IBelyService;

IBelyService bs = IBelyService.Stub.asInterface(ServiceManager.getService("BelyService"));

int ret = bs.calculateSqu(9);

 

上面我们使用的是 Android内部的方法来获取服务,当然也可以使用公开的 API: context.getSystemService来获取。

时间: 2024-10-30 17:24:01

为 Android 添加 Java 层服务的相关文章

【IOS-COCOS2D-X 游戏开发之十三】详细讲解在XCODE中利用预编译并通过JNI调用ANDROID的JAVA层代码(COCOS2DX里访问调用ANDROID函数)!

本站文章均为 李华明Himi 原创,转载务必在明显处注明: 转载自[黑米GameDev街区] 原文链接: http://www.himigame.com/android-game/725.html 很多看过之前Himi的[iOS-cocos2d-X 游戏开发之七]整合Cocos2dX的Android项目到Xcode项目中,Xcode编写&编译代码,Android导入打包运行即可! 博文的童鞋都在问我如何在Xcode中能调用Java的函数,比如在Android端需要加入广告.支付.等第三方平台或者

关于Android 添加系统级(java)服务和调用的编写实现说明

http://www.oschina.net/question/234345_40658 为Android添加Java层服务 http://blog.csdn.net/belyxiong/article/details/5876000

[Android Webkit]JNI基础及Java层与C++层的交互

1. JNI 注册 1.1. JNI的基础结构       JAVA == JNI == Native Code      JNI(Java Native Interface)是Java与Native Code(C/C++/...)代码交互的中介Java+JNI构成主程序 JNI+Native Code以动态库的形式供程序调用.      JNI的实现可涉及两个关键类:JNIEnv和JavaVM.两者都可以理解为函数表(Function Pointer Table), 前者是使用Java程序创建

《深入解析Android 5.0系统》——第6章,第6.3节Android Java层的同步机制

6.3 Android Java层的同步机制 深入解析Android 5.0系统 Java语言和C/C++语言不一样,Java语言中提供了同步关键字synchronized来支持线程间的同步操作. 6.3.1 同步关键字synchronized synchronized关键字最常见的用法是保护一段代码,如下所示: class Foo implements Runnable { private String mLock; public void lockedMethod() { ...... sy

Android平台webkit组件java层代码分析

转自ITeye: constGU 本文将围绕android平台webkit组件中,java层的几个主要类进行分析,说明各个类的职责以及类之间的关系.若有错误疏漏之处,望不吝指出. (源码所在目录  ${framework_sourcecode_loc}\core\java\android\webkit ) 一.概述 Android平台webkit组件java层封装了一系列浏览网页相关的功能,对开发者而言,主要功能有以下两点: 1.接收上层请求(eg:loadUrl, goBack, reload

分享一个Android和java调用RESTful Web服务的利器Resting

分享一个Android和java调用RESTful Web服务的利器Resting   当我们调用Web服务,往往是最终目标是取HTTP响应,将其转化为将在应用中呈现的值对象.Resting可以用来实现这一功能.Resting,在Java的一个轻量级的REST框架,可用于调用一个RESTful Web服务,并转换成响应来自客户端应用程序定制的Java对象.由于它的简单,resting是适合Android等手持设备.   resting目标•暴露简单的get(),post(),put()和dele

framework层服务分析

framework层服务分析       最近在做公司内部员工android系统培训,编写了一些PPT,引用了网上诸位大侠的贡献,下面就分析下framework框架层服务.              Android Service是android 系统中的四大组件之一(Activity.Service.BroadcastReceiver.ContentProvider),大家都知道,它有下面两大特点:     1)运行在后台的一种服务程序     2)一般很少和用户交互,没有可视化界面.    

Java Web服务,第1部分: Java Web服务在未来一年内的发展

2006 年将是 Web 服务(特别是 Java Web 服务)发展标志性的一年.新的第三代框架即将撩开面纱,这些框架将为 doc/lit SOAP 提供更好的支持,并能带来潜在的性能提高.同时,第四代 WS-* 标准也最终开始形成一组可互操作的层,对 SOAP 和 WSDL 进行扩展,以支持核心企业需求. 这篇文章是我的 Java Web 系列的第 1 部分,我将讨论以下 Web 服务目前的状态和在 2006 年即将发生的主要变化,并将简单说明新框架和技术如何相关和交互.后续文章将深入讨论其中

Android中Service(后台服务)详解

  这篇文章主要介绍了Android中Service(后台服务)详解,本文讲解了Service的概念.作用.生命周期.启动方式和代码实例等内容,需要的朋友可以参考下 1.概念: (1).Service可以说是一个在后台运行的Activity.它不是一个单独的进程,它只需要应用告诉它要在后台做什么就可以了. (2).它要是实现和用户的交互的话需要通过通知栏或者是通过发送广播,UI去接收显示. (3).它的应用十分广泛,尤其是在框架层,应用更多的是对系统服务的调用. 2.作用: (1).它用于处理一