Android native進程間通信實例-binder結合共用記憶體

来源:https://www.cnblogs.com/songsongman/archive/2019/08/01/11249974.html
-Advertisement-
Play Games

在android源碼的驅動目錄下,一般會有共用記憶體的相關實現源碼,目錄是:kernel\drivers\staging\android\ashmem.c。但是本篇文章不是講解android共用記憶體的功能實現原理,而是講怎麼運用它。 1. 在linux中,不同進程間擁有自己獨立的記憶體空間,32位操作系 ...


  在android源碼的驅動目錄下,一般會有共用記憶體的相關實現源碼,目錄是:kernel\drivers\staging\android\ashmem.c。但是本篇文章不是講解android共用記憶體的功能實現原理,而是講怎麼運用它。

  1. 

  在linux中,不同進程間擁有自己獨立的記憶體空間,32位操作系統中好像一個進程能用的記憶體大小是4G吧。而且一般不同進程間不能夠互相使用各自記憶體的數據。

  當然不同進程間共用數據方法很多,比如之前說的進程間通信binder,socket等等,不過android出了一個共用記憶體的概念,為的是不同進程間能夠共同操作同一塊記憶體數據,比如進程1001往一塊共用記憶體addr裡面寫數據“hello world”,進程1009往這塊共用記憶體addr讀取出“hello world”,也能夠往這塊共用記憶體addr寫數據“hello china”。這就是共同享用了一塊記憶體的基本概念了(說白了就是同耕一塊田)。講的夠仔細了吧,如果不清楚評論區見。

  註意:好像binder傳輸的數據實現也是類似於共用記憶體,讀者可以自行去瞭解。

  

  2.

  先說一下等會寫程式的思路:

  首先想想代碼編譯出兩個可執行文件後如何操作,打開兩個終端,都進入設備adb shell,第一個終端執行進程a,第二個終端執行進程b。在進程a輸入一串數據後,在進程b中可以讀出這段數據(也能夠改寫這段數據,讀者可以自行添加這部分功能)。

  然後再想想實現的方式,

  進程a:1. 創建共用記憶體,設置共用記憶體大小,這時會得到一個fd。2. 獲取共用記憶體地址。3. 先讀取地址數據,然後往地址寫入數據。4. 把fd通過binder發送給需要使用的進程。

  進程b:1. 通過binder讀取到fd。2. 用fd獲取共用記憶體地址。3. 讀取共用記憶體數據,然後往地址寫入數據。

  註意:linux一切皆文件,所以文件描述符fd很重要。  

  

  3. 

  3.1

  捋清思路後,就可以開始寫代碼了(android.mk的編寫可以參考之前的文章),進程a,命名為mysharememory_a代碼如下:

  

#include <fcntl.h>
#include <stdio.h>
#include <string.h>
#include <linux/ashmem.h>
#include <sys/mman.h>
#include <sys/ioctl.h>
#include <stddef.h>
#include <linux/ipc.h>
#include <linux/shm.h>

#include <binder/IServiceManager.h>
#include <binder/IPCThreadState.h>
#include <binder/Parcel.h>
#include <binder/IInterface.h>

#define DEVASHMEM "/dev/ashmem"
#define SHNAME "hellomemory"
#define MAXBUFSIZE 1024
#define TRANSFDCODE 1000
#define WRITEDATACODE 1001

using namespace android;

int  main(int argc, char *argv[])
{
    int fd = open(DEVASHMEM, O_RDWR);
    if(fd < 0)
    {
        return -1;
    }

    int ret = ioctl(fd, ASHMEM_SET_NAME, SHNAME);
    if(ret < 0){
        close(fd);
        return -1;
    }

    char *get_sh_addr_write = NULL;    
    ret = ioctl(fd, ASHMEM_SET_SIZE, MAXBUFSIZE);
    if(ret < 0){
        close(fd);
        return -1;
    }

    get_sh_addr_write = (char*)mmap(NULL, MAXBUFSIZE , PROT_READ | PROT_WRITE, MAP_SHARED, fd, 0);
    if(NULL == get_sh_addr_write)
    {
        return -1;
    }
    sp<IServiceManager> sm = defaultServiceManager();
    sp<IBinder> binder = sm->checkService(String16("mybindertag"));
    Parcel data, reply;

    data.writeDupFileDescriptor(fd);
    binder->transact(TRANSFDCODE, data, &reply);
    
    char input_data[MAXBUFSIZE] = {0};
    while(1)
    {
        printf("read share memory buf is %s\n", get_sh_addr_write);
        printf("please input data to buf :");
        scanf("%s", input_data);
        getchar();

        strcpy(get_sh_addr_write,input_data);
        binder->transact(WRITEDATACODE, data, &reply);
    }
    return ret;
}

  3.2

   mysharememory_b代碼如下:

  

#include <fcntl.h>
#include <stdio.h>
#include <string.h>
#include <linux/ashmem.h>
#include <sys/mman.h>
#include <sys/ioctl.h>
#include <stddef.h>
#include <linux/ipc.h>
#include <linux/shm.h>

#include <binder/IServiceManager.h>
#include <binder/IPCThreadState.h>
#include <binder/Parcel.h>
#include <binder/IInterface.h>

#define DEVASHMEM "/dev/ashmem"
#define SHNAME "hellomemory"
#define MAXBUFSIZE 1024
#define TRANSFDCODE 1000
#define WRITEDATACODE 1001

using namespace android;

int g_sh_fd = 0;
class MyBinderService : public BBinder{
            status_t onTransact(uint32_t code, const Parcel& data, Parcel* reply, uint32_t flags)
            {
                int ret;
                char *get_sh_addr_read = NULL;
                int get_sh_size;
            
                printf("songsong!! **** onTransact ***** code = %d \n",code);
                switch(code)
                {
                    case TRANSFDCODE:
                        g_sh_fd = data.readFileDescriptor();
                        break;

                    case WRITEDATACODE:
                        get_sh_size = ioctl(g_sh_fd, ASHMEM_GET_SIZE,NULL);
                        if(get_sh_size > 0)
                        { 
                            get_sh_addr_read = (char*)mmap(NULL, get_sh_size, PROT_READ | PROT_WRITE, MAP_SHARED, g_sh_fd, 0);
                        }
                        else
                        {
                            printf("mmap failed %d\n", get_sh_size);
                            return -1;     
                        }
                        printf("what is in the share memory: %s\n", get_sh_addr_read);
                        break;
                    default:

                        break;
                }
                return NO_ERROR;
            }
};

int  main(int argc, char *argv[])
{
    defaultServiceManager()->addService(String16("mybindertag"), new MyBinderService());

    sp<ProcessState> proc(ProcessState::self());
    ProcessState::self()->startThreadPool();
    IPCThreadState::self()->joinThreadPool();
    return 0;
}

 

  3.3

  回收關閉部分代碼可選擇添加在mysharememory_b中,如下:

  

    int ret;
    ret = munmap((void*)get_sh_addr_read, get_sh_size);
    if(ret == -1)
    {
        return -1;
    }
    
    ret = close(g_sh_fd);
    if(ret == -1)
    {
        return -1;
    }

 

  

 

  3.4 

  演示截圖:

  

 

 

   4. 為了騙取評論,我不再解釋代碼,心累。不過您可以把代碼直接拷貝去編譯執行,再通過調試去理解代碼的精髓,也是沒問題的。

  


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • indows Eclipse Scala編寫WordCount程式 ...
  • 史上最全存儲引擎、索引使用及SQL優化的實踐 1 MySQL的體繫結構概述 2. 存儲引擎 2.1 存儲引擎概述 2.2 各種存儲引擎特性 2.2.1 InnoDB 2.2.2 MyISAM 3. 優化SQL步驟 3.1 查看SQL執行頻率 3.2 定位低效率執行SQL 3.3 explain分析執 ...
  • 在應用系統開發初期,由於開發資料庫數據比較少,對於查詢SQL語句,複雜視圖的的編寫等體會不出SQL語句各種寫法的性能優劣,但是如果將應用 系統提交實際應用後,隨著資料庫中數據的增加,系統的響應速度就成為目前系統需要解決的最主要的問題之一。系統優化中一個很重要的方面就是SQL語句的優 化。對於海量數據... ...
  • 子查詢:又分為where型子查詢,from型子查詢,exists型子查詢這三類。 where型子查詢:指把內層查詢的結果作為外層查詢的比較條件: 舉個例子: 我們想查出goods_id最大的商品,要求不能用排序: 我們還想查出每個欄目下goods_id最大的商品,要求使用where型子查詢: fro ...
  • 在這篇文章,我們一起瞭解 Redis 使用中非常重要的兩個機制:Reids 持久化和主從複製。 什麼是 Redis 持久化? Redis 作為一個鍵值對記憶體資料庫(NoSQL),數據都存儲在記憶體當中,在處理客戶端請求時,所有操作都在記憶體當中進行,如下所示 這樣做有什麼問題呢?其實,只要稍微有點電腦 ...
  • MySQL INNER JOIN子句介紹 MySQL 子句將一個表中的行與其他表中的行進行匹配,並允許從兩個表中查詢包含列的行記錄。 子句是 語句的可選部分,它出現在 "FROM子句" 之後。 在使用 子句之前,必須指定以下條件: 首先,在FROM子句中指定主表。 其次,表中要連接的主表應該出現在 ...
  • 一、問題 問題1 場景:如果你未來的丈母娘要求你,第1天給她1分錢,第2天給2分錢,第3天給4分錢,以此類推,每天給前一天的2倍,給1個月(按30天)算就行。問:第30天給多少錢,總共給多少錢? 問題1 問題2 場景:如果有兩份工作。 第1份:第1天給你1分錢,第2天給你2分錢,第3天給你4分錢,以 ...
  • 表單組件是個包含表單元素的區域,表單元素允許用戶輸入內容,比如:文本區域,下拉表單,單選框、覆選框等,常見的應用場景有:登陸、註冊、輸入信息等。表單里有兩個重要的組件,一個是Form組件用來做整個表單提交使用的,另一個是TextFormField組件用來做用戶輸入的。 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...