Android系統最多見也是初學者最難搞明白的就是Binder了,不少不少的Service就是經過Binder機制來和客戶端通信交互的。因此搞明白Binder的話,在很大程度上就能理解程序運行的流程。android
這是一個用C++寫的binder,一個服務器一惡搞客戶端,代碼以下:bash
server.cpp服務器
1 #include <binder/IServiceManager.h> 2 #include <binder/IBinder.h> 3 #include <binder/Parcel.h> 4 #include <binder/ProcessState.h> 5 #include <binder/IPCThreadState.h> 6 #include <android/log.h> 7 using namespace android; 8 #ifdef LOG_TAG 9 #undef LOG_TAG 10 #endif 11 #define LOG_TAG "testService" 12 13 #define LOGV(...) __android_log_print(ANDROID_LOG_VERBOSE, "ProjectName", __VA_ARGS__) 14 #define LOGD(...) __android_log_print(ANDROID_LOG_DEBUG , "ProjectName", __VA_ARGS__) 15 #define LOGI(...) __android_log_print(ANDROID_LOG_INFO , "ProjectName", __VA_ARGS__) 16 #define LOGW(...) __android_log_print(ANDROID_LOG_WARN , "ProjectName", __VA_ARGS__) 17 #define LOGE(...) __android_log_print(ANDROID_LOG_ERROR , "ProjectName", __VA_ARGS__) 18 19 class MyService : public BBinder 20 { 21 public: 22 MyService() 23 { 24 mydescriptor = String16("media.hello"); 25 n=0; 26 } 27 virtual ~MyService() {} 28 //This function is used when call Parcel::checkInterface(IBinder*) 29 virtual const String16& getInterfaceDescriptor() const 30 { 31 LOGE("this is enter ==========getInterfaceDescriptor"); 32 return mydescriptor; 33 } 34 protected: 35 void show() 36 { 37 LOGE("this is for test show!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!"); 38 LOGE("this is for test show!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!"); 39 LOGE("this is for test show!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!"); 40 LOGE("this is for test show!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!"); 41 LOGE("this is for test show!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!"); 42 } 43 virtual status_t onTransact( uint32_t code,const Parcel& data,Parcel* reply,uint32_t flags = 0) 44 { 45 LOGD("enter MyService onTransact and the code is %d",code); 46 /* 47 if (data.checkInterface(this)) 48 LOGD("checkInterface OK"); 49 else 50 { 51 LOGW("checkInterface failed"); 52 return BBinder::onTransact(code, data, reply, flags); 53 } 54 */ 55 switch (code) 56 { 57 case 1: 58 LOGD("MyService interface 1"); 59 break; 60 case 2: 61 LOGD("MyService interface 2"); 62 cb = data.readStrongBinder(); 63 break; 64 case 3: 65 { 66 LOGD("MyService interface 3, exit"); 67 //No unregister service routine? 68 //It should return to client first and then call exit in another place. 69 exit(0); 70 break; 71 } 72 case 4: 73 {//call cb 74 LOGD("MyService interface 4 before if================"); 75 cb = data.readStrongBinder(); 76 if (cb != NULL) 77 { 78 LOGD("MyService interface 4"); 79 Parcel in, out; 80 in.writeInterfaceToken(String16("android.os.ISetupCallback")); 81 in.writeInt32(n++); //向客戶端發送數據 82 83 in.writeCString("This is a string !"); 84 cb->transact(2, in, &out, 0); 85 show(); 86 } 87 break; 88 } 89 default: 90 return BBinder::onTransact(code, data, reply, flags); 91 } 92 return 0; 93 } 94 private: 95 String16 mydescriptor; 96 sp<IBinder> cb; 97 int n; 98 }; 99 int main() 100 { 101 sp<IServiceManager> sm = defaultServiceManager(); //獲取ServiceManager服務代理 102 status_t ret; 103 //register MyService to ServiceManager 104 MyService* srv = new MyService(); 105 ret = sm->addService(String16("media.hello"), srv); // 註冊服務 106 LOGD("addservice media.hello return %d", ret); 107 //call binder thread pool to start 108 ProcessState::self()->startThreadPool(); 109 IPCThreadState::self()->joinThreadPool(true); //參數默認也是true,進入服務的循環監聽狀態 110 return 0; 111 }
clinet.cpp函數
1 #include <binder/IServiceManager.h> 2 #include <binder/IBinder.h> 3 #include <binder/Parcel.h> 4 #include <binder/ProcessState.h> 5 #include <binder/IPCThreadState.h> 6 #include <private/binder/binder_module.h> 7 #include <android/log.h> 8 9 using namespace android; 10 #ifdef LOG_TAG 11 #undef LOG_TAG 12 #endif 13 #define LOG_TAG "testCallback" 14 15 #define LOGV(...) __android_log_print(ANDROID_LOG_VERBOSE, "ProjectName", __VA_ARGS__) 16 #define LOGD(...) __android_log_print(ANDROID_LOG_DEBUG , "ProjectName", __VA_ARGS__) 17 #define LOGI(...) __android_log_print(ANDROID_LOG_INFO , "ProjectName", __VA_ARGS__) 18 #define LOGW(...) __android_log_print(ANDROID_LOG_WARN , "ProjectName", __VA_ARGS__) 19 #define LOGE(...) __android_log_print(ANDROID_LOG_ERROR , "ProjectName", __VA_ARGS__) 20 21 class MySetupCallback : public BBinder 22 { 23 public: 24 MySetupCallback() 25 { 26 mydescriptor = String16("android.os.ISetupCallback"); 27 } 28 virtual ~MySetupCallback() {} 29 virtual const String16& getInterfaceDescriptor() const 30 { 31 return mydescriptor; 32 } 33 protected: 34 virtual status_t onTransact( uint32_t code, const Parcel& data, Parcel* reply, uint32_t flags = 0) 35 { 36 LOGD("enter MySetupCallback onTransact, code=%u", code); 37 if (data.checkInterface(this)) //檢查 mydescriptor 類描述字符串 38 LOGD("checkInterface OK"); 39 else 40 { 41 LOGW("checkInterface failed"); 42 return -1; 43 } 44 switch (code) //code爲服務器發送的code,根據code實現不一樣的函數 45 { 46 case 1: 47 LOGD("From Server code = %u", code); 48 LOGD("From Server code = %u", code); 49 break; 50 case 2: 51 { 52 LOGD("From Server code = %u", code); 53 LOGD("Frome server data = %d", data.readInt32()); //從服務端接收數據 54 LOGD("Frome server string = %s", data.readCString()); 55 56 break; 57 } 58 default: 59 break; 60 } 61 return 0; 62 } 63 private: 64 String16 mydescriptor; 65 }; 66 67 int main() 68 { 69 sp<IServiceManager> sm = defaultServiceManager(); //獲取ServiceManager服務代理 70 sp<IBinder> b = sm->getService(String16("media.hello")); //查詢服務 71 if (b == NULL) 72 { 73 LOGW("Can't find binder service \"media.hello\""); 74 return -1; 75 } 76 Parcel in1,out1; 77 MySetupCallback *cb = new MySetupCallback(); 78 in1.writeStrongBinder(sp<IBinder>(cb)); 79 int ret = b->transact(4, in1, &out1, 0); //TRANSACTION_registerSetup = 4 80 LOGD("transact(4) return %d", ret); 81 ProcessState::self()->startThreadPool(); 82 IPCThreadState::self()->joinThreadPool(); //參數默認也是true,進入服務的循環監聽狀態 83 return 0; 84 }
Android.mkui
# Copyright 2006 The Android Open Source Project LOCAL_PATH:= $(call my-dir) include $(CLEAR_VARS) LOCAL_SHARED_LIBRARIES := \ libcutils \ libbinder \ libutils \ libhardware LOCAL_SRC_FILES:= client.cpp LOCAL_MODULE_TAGS = eng tests LOCAL_MODULE:= testClient include $(BUILD_EXECUTABLE) include $(CLEAR_VARS) LOCAL_SHARED_LIBRARIES := \ libcutils \ libbinder \ libutils \ libhardware LOCAL_SRC_FILES:=server.cpp LOCAL_MODULE:= testServer LOCAL_MODULE_TAGS = eng tests include $(BUILD_EXECUTABLE)
客戶端運行結果以下:this
以上代碼參考別人寫的作了點修改,有錯誤的地方歡迎指出來,謝謝。spa