Android 에서 AIDL 상세 설명

전재 환영 합 니 다. 전재 출처 를 밝 혀 주세요.http://blog.csdn.net/l664675249/article/details/50649676
소개 하 다.
Android 인터페이스 Definition Language (AIDL), Android 인터페이스 정의 언어.시스템 의 프로 세 스 간 에 메모 리 를 공유 할 수 없 기 때문에 서로 다른 프로 세 스 간 에 데이터 통신 Interprocess communication (IPC) 을 하 는 메커니즘 을 제공 해 야 합 니 다.AIDL 이 바로 이 문 제 를 해결 한 것 이다.본 고 를 읽 으 려 면 Service 에 관 한 지식 을 알 아야 합 니 다. Service 에 관 한 설명 은 참고 하 시기 바 랍 니 다.http://blog.csdn.net/l664675249/article/details/48899323
. aidl 파일 만 들 기
aidl 은 자바 문법 으로 작 성 된 것 으로 접미사 가. aidl 인 파일 입 니 다.
  • 모든 aidl 파일 은 인 터 페 이 스 를 정의 해 야 합 니 다. 이 인터페이스 에서 설명 하 는 방법
  • aidl 에 static 속성 (field)
  • 이 있 으 면 안 됩 니 다.
  • adl 은 기본 적 인 데이터 형식 을 지원 합 니 다. 추가 데이터 형식 을 사용 해 야 할 때 import 를 들 어 와 야 합 니 다. 이 파일 과 같은 가방 에 있 더 라 도.

  • 예시
    // IRemoteService.aidl
    package com.example.android;
    
    // Declare any non-default types here with import statements
    
    /** Example service interface */
    interface IRemoteService {
        /** Request the process ID of this service, to do evil things with it. */
        int getPid();
    
        /** Demonstrates some basic types that you can use as parameters * and return values in AIDL. */
        void basicTypes(int anInt, long aLong, boolean aBoolean, float aFloat,
                double aDouble, String aString);
    }

    adl 파일 을 src / 디 렉 터 리 에 저장 합 니 다. build 프로젝트 를 만 들 때 SDK 도 구 는 gen / 디 렉 터 리 에서. aidl 파일 이름과 같은. java 파일 을 생 성 합 니 다.
    실현 인터페이스
    생 성 된 IRemote Service. java 는 다음 과 같 습 니 다.
    public interface IRemoteService extends android.os.IInterface {
    /** Local-side IPC implementation stub class. */
    public static abstract class Stub extends android.os.Binder implements com.spark.meizi.IRemoteService {
    private static final java.lang.String DESCRIPTOR = "com.spark.meizi.IRemoteService";
    /** Construct the stub at attach it to the interface. */
    public Stub()
    {
    this.attachInterface(this, DESCRIPTOR);
    }
    /** * Cast an IBinder object into an com.spark.meizi.IRemoteService interface, * generating a proxy if needed. */
    public static com.spark.meizi.IRemoteService asInterface(android.os.IBinder obj)
    {
    if ((obj==null)) {
    return null;
    }
    android.os.IInterface iin = obj.queryLocalInterface(DESCRIPTOR);
    if (((iin!=null)&&(iin instanceof com.spark.meizi.IRemoteService))) {
    return ((com.spark.meizi.IRemoteService)iin);
    }
    return new com.spark.meizi.IRemoteService.Stub.Proxy(obj);
    }
    @Override public android.os.IBinder asBinder()
    {
    return this;
    }
    @Override public boolean onTransact(int code, android.os.Parcel data, android.os.Parcel reply, int flags) throws android.os.RemoteException
    {
    switch (code)
    {
    case INTERFACE_TRANSACTION:
    {
    reply.writeString(DESCRIPTOR);
    return true;
    }
    case TRANSACTION_basicTypes:
    {
    data.enforceInterface(DESCRIPTOR);
    int _arg0;
    _arg0 = data.readInt();
    long _arg1;
    _arg1 = data.readLong();
    boolean _arg2;
    _arg2 = (0!=data.readInt());
    float _arg3;
    _arg3 = data.readFloat();
    double _arg4;
    _arg4 = data.readDouble();
    java.lang.String _arg5;
    _arg5 = data.readString();
    this.basicTypes(_arg0, _arg1, _arg2, _arg3, _arg4, _arg5);
    reply.writeNoException();
    return true;
    }
    }
    return super.onTransact(code, data, reply, flags);
    }
    private static class Proxy implements com.spark.meizi.IRemoteService {
    private android.os.IBinder mRemote;
    Proxy(android.os.IBinder remote)
    {
    mRemote = remote;
    }
    @Override public android.os.IBinder asBinder()
    {
    return mRemote;
    }
    public java.lang.String getInterfaceDescriptor()
    {
    return DESCRIPTOR;
    }
    /** * Demonstrates some basic types that you can use as parameters * and return values in AIDL. */
    @Override public void basicTypes(int anInt, long aLong, boolean aBoolean, float aFloat, double aDouble, java.lang.String aString) throws android.os.RemoteException
    {
    android.os.Parcel _data = android.os.Parcel.obtain();
    android.os.Parcel _reply = android.os.Parcel.obtain();
    try {
    _data.writeInterfaceToken(DESCRIPTOR);
    _data.writeInt(anInt);
    _data.writeLong(aLong);
    _data.writeInt(((aBoolean)?(1):(0)));
    _data.writeFloat(aFloat);
    _data.writeDouble(aDouble);
    _data.writeString(aString);
    mRemote.transact(Stub.TRANSACTION_basicTypes, _data, _reply, 0);
    _reply.readException();
    }
    finally {
    _reply.recycle();
    _data.recycle();
    }
    }
    }
    static final int TRANSACTION_basicTypes = (android.os.IBinder.FIRST_CALL_TRANSACTION + 0);
    }
    /** * Demonstrates some basic types that you can use as parameters * and return values in AIDL. */
    public void basicTypes(int anInt, long aLong, boolean aBoolean, float aFloat, double aDouble, java.lang.String aString) throws android.os.RemoteException;
    }

    이 파일 에 내부 클래스 Stub 가 있 습 니 다. 이것 은 부모 인터페이스의 추상 적 인 실현 이 고 adl 의 모든 방법 을 설명 합 니 다.에 이 드 에서 생 성 된 인 터 페 이 스 를 실현 하기 위해 서 는 Stub 를 계승 하고 에 이 드 에서 계승 하 는 방법 이 필요 합 니 다.다음은 익명 류 를 사용 한 예 입 니 다.
    private final IRemoteService.Stub mBinder = new IRemoteService.Stub() {
        public int getPid(){
            return Process.myPid();
        }
        public void basicTypes(int anInt, long aLong, boolean aBoolean,
            float aFloat, double aDouble, String aString) {
            // Does nothing
        }
    };

    이렇게 하면 mBinder 는 Stub 의 인 스 턴 스 입 니 다. 다음 단 계 는 client 에서 어떻게 사용 하 는 지, service 단 과 상호작용 을 하 는 것 입 니 다.주:
  • 요청 이 주 스 레 드 에서 실 행 될 것 이 라 고 보장 할 수 없 기 때문에 구축 부터 사용 까지 Service 스 레 드 의 안전성 을 고려 해 야 합 니 다
  • 기본 적 인 상황 에서 요청 은 동기 화 되 므 로 메 인 스 레 드 에서 요청 하지 마 십시오
  • 모든 이상 은 요청 자 에 게 되 돌아 오지 않 습 니 다 (Caller)
  • Client 에서 인터페이스 사용 하기
    서 비 스 를 실현 한 후에 클 라 이언 트 에 노출 시 켜 클 라 이언 트 가 연결 할 수 있 도록 해 야 합 니 다.Service 를 계승 하고 onBind () 방법 을 실현 하여 Stub 를 실현 한 인 스 턴 스 를 되 돌려 줍 니 다.다음은 IRemoteService 를 Client 에 노출 시 킨 예 입 니 다.
    public class RemoteService extends Service {
        @Override
        public void onCreate() {
            super.onCreate();
        }
    
        @Override
        public IBinder onBind(Intent intent) {
            // Return the interface
            return mBinder;
        }
    
        private final IRemoteService.Stub mBinder = new IRemoteService.Stub() {
            public int getPid(){
                return Process.myPid();
            }
            public void basicTypes(int anInt, long aLong, boolean aBoolean,
                float aFloat, double aDouble, String aString) {
                // Does nothing
            }
        };
    }
    

    현재 client (예 를 들 어 하나의 Activity) 는 bindService () 를 호출 하여 이 서 비 스 를 연결 할 수 있 습 니 다. onServiceConnected () 를 통 해 서비스 에서 onBind () 가 돌아 오 는 mBinder 를 받 고 마지막 으로 YourServiceInterface. stub. asInterface (service) 를 사용 하여 돌아 오 는 mBinder 를 YourServiceInterface 형식 으로 변환 할 수 있 습 니 다.예 는 다음 과 같다.
    IRemoteService mIRemoteService;
    private ServiceConnection mConnection = new ServiceConnection() {
        // Called when the connection with the service is established
        public void onServiceConnected(ComponentName className, IBinder service) {
            // Following the example above for an AIDL interface,
            // this gets an instance of the IRemoteInterface, which we can use to call on the service
            mIRemoteService = IRemoteService.Stub.asInterface(service);
        }
    
        // Called when the connection with the service disconnects unexpectedly
        public void onServiceDisconnected(ComponentName className) {
            Log.e(TAG, "Service has unexpectedly disconnected");
            mIRemoteService = null;
        }
    };

    주: 만약 Service 와 Client 가 두 개의 다른 Application 에 있다 면, Client 의 Application src / 디 렉 터 리 에 도 대응 하 는. aidl 파일 이 있어 야 합 니 다.
    클 라 이언 트 의 예
    public static class Binding extends Activity {
        /** The primary interface we will be calling on the service. */
        IRemoteService mService = null;
        /** Another interface we use on the service. */
        ISecondary mSecondaryService = null;
    
        Button mKillButton;
        TextView mCallbackText;
    
        private boolean mIsBound;
    
        /** * Standard initialization of this activity. Set up the UI, then wait * for the user to poke it before doing anything. */
        @Override
        protected void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
    
            setContentView(R.layout.remote_service_binding);
    
            // Watch for button clicks.
            Button button = (Button)findViewById(R.id.bind);
            button.setOnClickListener(mBindListener);
            button = (Button)findViewById(R.id.unbind);
            button.setOnClickListener(mUnbindListener);
            mKillButton = (Button)findViewById(R.id.kill);
            mKillButton.setOnClickListener(mKillListener);
            mKillButton.setEnabled(false);
    
            mCallbackText = (TextView)findViewById(R.id.callback);
            mCallbackText.setText("Not attached.");
        }
    
        /** * Class for interacting with the main interface of the service. */
        private ServiceConnection mConnection = new ServiceConnection() {
            public void onServiceConnected(ComponentName className,
                    IBinder service) {
                // This is called when the connection with the service has been
                // established, giving us the service object we can use to
                // interact with the service. We are communicating with our
                // service through an IDL interface, so get a client-side
                // representation of that from the raw service object.
                mService = IRemoteService.Stub.asInterface(service);
                mKillButton.setEnabled(true);
                mCallbackText.setText("Attached.");
    
                // We want to monitor the service for as long as we are
                // connected to it.
                try {
                    mService.registerCallback(mCallback);
                } catch (RemoteException e) {
                    // In this case the service has crashed before we could even
                    // do anything with it; we can count on soon being
                    // disconnected (and then reconnected if it can be restarted)
                    // so there is no need to do anything here.
                }
    
                // As part of the sample, tell the user what happened.
                Toast.makeText(Binding.this, R.string.remote_service_connected,
                        Toast.LENGTH_SHORT).show();
            }
    
            public void onServiceDisconnected(ComponentName className) {
                // This is called when the connection with the service has been
                // unexpectedly disconnected -- that is, its process crashed.
                mService = null;
                mKillButton.setEnabled(false);
                mCallbackText.setText("Disconnected.");
    
                // As part of the sample, tell the user what happened.
                Toast.makeText(Binding.this, R.string.remote_service_disconnected,
                        Toast.LENGTH_SHORT).show();
            }
        };
    
        /** * Class for interacting with the secondary interface of the service. */
        private ServiceConnection mSecondaryConnection = new ServiceConnection() {
            public void onServiceConnected(ComponentName className,
                    IBinder service) {
                // Connecting to a secondary interface is the same as any
                // other interface.
                mSecondaryService = ISecondary.Stub.asInterface(service);
                mKillButton.setEnabled(true);
            }
    
            public void onServiceDisconnected(ComponentName className) {
                mSecondaryService = null;
                mKillButton.setEnabled(false);
            }
        };
    
        private OnClickListener mBindListener = new OnClickListener() {
            public void onClick(View v) {
                // Establish a couple connections with the service, binding
                // by interface names. This allows other applications to be
                // installed that replace the remote service by implementing
                // the same interface.
                Intent intent = new Intent(Binding.this, RemoteService.class);
                intent.setAction(IRemoteService.class.getName());
                bindService(intent, mConnection, Context.BIND_AUTO_CREATE);
                intent.setAction(ISecondary.class.getName());
                bindService(intent, mSecondaryConnection, Context.BIND_AUTO_CREATE);
                mIsBound = true;
                mCallbackText.setText("Binding.");
            }
        };
    
        private OnClickListener mUnbindListener = new OnClickListener() {
            public void onClick(View v) {
                if (mIsBound) {
                    // If we have received the service, and hence registered with
                    // it, then now is the time to unregister.
                    if (mService != null) {
                        try {
                            mService.unregisterCallback(mCallback);
                        } catch (RemoteException e) {
                            // There is nothing special we need to do if the service
                            // has crashed.
                        }
                    }
    
                    // Detach our existing connection.
                    unbindService(mConnection);
                    unbindService(mSecondaryConnection);
                    mKillButton.setEnabled(false);
                    mIsBound = false;
                    mCallbackText.setText("Unbinding.");
                }
            }
        };
    
        private OnClickListener mKillListener = new OnClickListener() {
            public void onClick(View v) {
                // To kill the process hosting our service, we need to know its
                // PID. Conveniently our service has a call that will return
                // to us that information.
                if (mSecondaryService != null) {
                    try {
                        int pid = mSecondaryService.getPid();
                        // Note that, though this API allows us to request to
                        // kill any process based on its PID, the kernel will
                        // still impose standard restrictions on which PIDs you
                        // are actually able to kill. Typically this means only
                        // the process running your application and any additional
                        // processes created by that app as shown here; packages
                        // sharing a common UID will also be able to kill each
                        // other's processes.
                        Process.killProcess(pid);
                        mCallbackText.setText("Killed service process.");
                    } catch (RemoteException ex) {
                        // Recover gracefully from the process hosting the
                        // server dying.
                        // Just for purposes of the sample, put up a notification.
                        Toast.makeText(Binding.this,
                                R.string.remote_call_failed,
                                Toast.LENGTH_SHORT).show();
                    }
                }
            }
        };
    
        // ----------------------------------------------------------------------
        // Code showing how to deal with callbacks.
        // ----------------------------------------------------------------------
    
        /** * This implementation is used to receive callbacks from the remote * service. */
        private IRemoteServiceCallback mCallback = new IRemoteServiceCallback.Stub() {
            /** * This is called by the remote service regularly to tell us about * new values. Note that IPC calls are dispatched through a thread * pool running in each process, so the code executing here will * NOT be running in our main thread like most other things -- so, * to update the UI, we need to use a Handler to hop over there. */
            public void valueChanged(int value) {
                mHandler.sendMessage(mHandler.obtainMessage(BUMP_MSG, value, 0));
            }
        };
    
        private static final int BUMP_MSG = 1;
    
        private Handler mHandler = new Handler() {
            @Override public void handleMessage(Message msg) {
                switch (msg.what) {
                    case BUMP_MSG:
                        mCallbackText.setText("Received from service: " + msg.arg1);
                        break;
                    default:
                        super.handleMessage(msg);
                }
            }
    
        };
    }

    전재 환영 합 니 다. 전재 출처 를 밝 혀 주세요.http://blog.csdn.net/l664675249/article/details/50649676

    좋은 웹페이지 즐겨찾기