share-data-by-datashareextensionability.md 11.6 KB
Newer Older
G
ge-yafang 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16
# 通过DataShareExtensionAbility实现数据共享


## 场景介绍

跨应用访问数据时,可以通过DataShareExtensionAbility拉起数据提供方的应用以实现对数据的访问。

此种方式支持跨应用拉起数据提供方的DataShareExtension,数据提供方的开发者可以在回调中实现灵活的业务逻辑。用于跨应用复杂业务场景。


## 运作机制

数据共享可分为数据的提供方和访问方两部分。

- 数据提供方:[DataShareExtensionAbility](../reference/apis/js-apis-application-dataShareExtensionAbility.md),可以选择性实现数据的增、删、改、查,以及文件打开等功能,并对外共享这些数据。

G
ge-yafang 已提交
17
- 数据访问方:由[createDataShareHelper()](../reference/apis/js-apis-data-dataShare.md#datasharecreatedatasharehelper)方法所创建的工具类,利用工具类,便可以访问提供方提供的这些数据。
G
ge-yafang 已提交
18

19
**图1** 数据共享运作机制  
G
ge-yafang 已提交
20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61
![dataShare](figures/dataShare.jpg)

- DataShareExtensionAbility模块为数据提供方,实现跨应用数据共享的相关业务。

- DataShareHelper模块为数据访问方,提供各种访问数据的接口,包括增删改查等。

- 数据访问方与提供方通过IPC进行通信,数据提供方可以通过数据库实现,也可以通过其他数据存储方式实现。

- ResultSet模块通过共享内存实现,用于存储查询数据得到的结果集,并提供了遍历结果集的方法。


## 实现说明


### 数据提供方应用的开发(仅限系统应用)

[DataShareExtensionAbility](../reference/apis/js-apis-application-dataShareExtensionAbility.md)提供以下API,根据需要重写对应回调方法。

- **onCreate**:DataShare客户端连接DataShareExtensionAbility服务端时,服务端需要在此回调中实现初始化业务逻辑,该方法可以选择性重写。

- **insert**:业务函数,客户端请求插入数据时回调此接口,服务端需要在此回调中实现插入数据功能,该方法可以选择性重写。

- **update**:业务函数,客户端请求更新数据时回调此接口,服务端需要在此回调中实现更新数据功能,该方法可以选择性重写。

- **delete**:业务函数,客户端请求删除数据时回调此接口,服务端需要在此回调中实现删除数据功能,该方法可以选择性重写。

- **query**:业务函数,客户端请求查询数据时回调此接口,服务端需要在此回调中实现查询数据功能,该方法可以选择性重写。

- **batchInsert**:业务函数,客户端请求批量插入数据时回调此接口,服务端需要在此回调中实现批量插入数据的功能,该方法可以选择性重写。

- **normalizeUri**:业务函数,客户端给定的URI转换为服务端使用的URI时回调此接口,该方法可以选择性重写。

- **denormalizeUri**:业务函数,服务端使用的URI转换为客户端传入的初始URI时服务端回调此接口,该方法可以选择性重写。

开发者在实现一个数据共享服务时,需要在DevEco Studio工程中手动新建一个DataShareExtensionAbility,具体步骤如下。

1. 在工程Module对应的ets目录下,右键选择“New > Directory”,新建一个目录并命名为DataShareExtAbility。

2. 在DataShareAbility目录,右键选择“New > TypeScript File”,新建一个TypeScript文件并命名为DataShareExtAbility.ts。

3. 在DataShareExtAbility.ts文件中,导入
`@ohos.application.DataShareExtensionAbility`模块,开发者可根据应用需求选择性重写其业务实现。例如数据提供方只提供插入、删除和查询服务,则可只重写这些接口,并导入对应的基础依赖模块。
62
   
63 64 65 66 67 68 69 70
```ts
import Extension from '@ohos.application.DataShareExtensionAbility';
import rdb from '@ohos.data.relationalStore';
import dataSharePredicates from '@ohos.data.dataSharePredicates';
import relationalStore from '@ohos.data.relationalStore';
import Want from '@ohos.app.ability.Want';
import { BusinessError } from '@ohos.base'
```
G
ge-yafang 已提交
71 72

4. 数据提供方的业务实现由开发者自定义。例如可以通过数据库、读写文件或访问网络等各方式实现数据提供方的数据存储。
73
   
74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125
```ts
const DB_NAME = 'DB00.db';
const TBL_NAME = 'TBL00';
const DDL_TBL_CREATE = "CREATE TABLE IF NOT EXISTS "
  + TBL_NAME
  + ' (id INTEGER PRIMARY KEY AUTOINCREMENT, name TEXT, age INTEGER, isStudent BOOLEAN, Binary BINARY)';

let rdbStore: relationalStore.RdbStore;
let result: string;

export default class DataShareExtAbility extends Extension {
  // 重写onCreate接口
  onCreate(want: Want, callback: Function) {
    result = this.context.cacheDir + '/datashare.txt';
    // 业务实现使用RDB
    rdb.getRdbStore(this.context, {
      name: DB_NAME,
      securityLevel: rdb.SecurityLevel.S1
    }, (err, data) => {
      rdbStore = data;
      rdbStore.executeSql(DDL_TBL_CREATE, [], (err) => {
        console.info(`DataShareExtAbility onCreate, executeSql done err:${err}`);
      });
      if (callback) {
        callback();
      }
    });
  }

  // 重写query接口
  query(uri: string, predicates: dataSharePredicates.DataSharePredicates, columns: Array<string>, callback: Function) {
    if (predicates === null || predicates === undefined) {
      console.info('invalid predicates');
    }
    try {
      rdbStore.query(TBL_NAME, predicates, columns, (err, resultSet) => {
        if (resultSet !== undefined) {
          console.info(`resultSet.rowCount:${resultSet.rowCount}`);
        }
        if (callback !== undefined) {
          callback(err, resultSet);
        }
      });
    } catch (err) {
      let code = (err as BusinessError).code;
      let message = (err as BusinessError).message
      console.error(`Failed to query. Code:${code},message:${message}`);
    }
  }
  // 可根据应用需求,选择性重写各个接口
};
```
G
ge-yafang 已提交
126 127 128 129 130 131 132 133 134 135 136 137 138

5. 在module.json5中定义DataShareExtensionAbility。

     **表1** module.json5对应属性字段
   
   | 属性名称 | 备注说明 | 必填 | 
   | -------- | -------- | -------- |
   | name | Ability名称,对应Ability派生的ExtensionAbility类名。 | 是 | 
   | type | Ability类型,DataShare对应的Ability类型为“dataShare”,表示基于datashare模板开发的。 | 是 | 
   | uri | 通信使用的URI,是客户端链接服务端的唯一标识。 | 是 | 
   | exported | 对其他应用是否可见,设置为true时,才能与其他应用进行通信传输数据。 | 是 | 
   | readPermission | 访问数据时需要的权限,不配置默认不进行读权限校验。 | 否 | 
   | writePermission | 修改数据时需要的权限,不配置默认不进行写权限校验。 | 否 | 
N
fix  
niudongyao 已提交
139
   | metadata   | 增加静默访问所需的额外配置项,包含name和resource字段。<br /> name类型固定为"ohos.extension.dataShare",是配置的唯一标识。 <br /> resource类型固定为"$profile:data_share_config",表示配置文件的名称为data_share_config.json。 | 若Ability启动模式为"singleton",则metadata必填,Ability启动模式可见[abilities对象的内部结构-launchType](../quick-start/module-structure.md#abilities对象的内部结构);其他情况下无需填写。 | 
G
ge-yafang 已提交
140 141

   **module.json5配置样例:**
142
   
G
ge-yafang 已提交
143 144 145
   ```json
   "extensionAbilities": [
     {
146
       "srcEntry": "./ets/DataShareExtAbility/DataShareExtAbility.ts",
G
ge-yafang 已提交
147 148 149 150
       "name": "DataShareExtAbility",
       "icon": "$media:icon",
       "description": "$string:description_datashareextability",
       "type": "dataShare",
H
f  
hanlu 已提交
151
       "uri": "datashareproxy://com.samples.datasharetest.DataShare",
N
add  
niudongyao 已提交
152 153
       "exported": true,
       "metadata": [{"name": "ohos.extension.dataShare", "resource": "$profile:data_share_config"}]
G
ge-yafang 已提交
154 155 156
     }
   ]
   ```
N
add  
niudongyao 已提交
157 158 159
   
   **表2** data_share_config.json对应属性字段

H
f  
hanlu 已提交
160 161 162 163 164
   | 属性名称          | 备注说明                                     | 必填   |
   | ------------- | ---------------------------------------- | ---- |
   | tableConfig   | 配置标签。                                    | 是    |
   | uri           | 指定配置生效的范围,uri支持以下三种格式,优先级为**表配置>库配置>\***,如果同时配置,高优先级会覆盖低优先级 。<br /> 1. "*" : 所有的数据库和表。<br /> 2. "datashareproxy://{bundleName}/{moduleName}/{storeName}" : 指定数据库。<br /> 3. "datashareproxy://{bundleName}/{moduleName}/{storeName}/{tableName}" : 指定表。 | 是    |
   | crossUserMode | 标识数据是否为多用户共享,配置为1则多用户数据共享,配置为2则多用户数据隔离。  | 是    |
N
add  
niudongyao 已提交
165 166 167 168 169 170 171

   **data_share_config.json配置样例**

   ```json
   "tableConfig": [
    {
      "uri": "*",
N
fix  
niudongyao 已提交
172
      "crossUserMode": 1
N
add  
niudongyao 已提交
173 174
    },
    {
H
f  
hanlu 已提交
175
      "uri": "datashareproxy://com.acts.datasharetest/entry/DB00",
N
fix  
niudongyao 已提交
176
      "crossUserMode": 1
N
add  
niudongyao 已提交
177 178
    },
    {
H
f  
hanlu 已提交
179
      "uri": "datashareproxy://com.acts.datasharetest/entry/DB00/TBL00",
N
add  
niudongyao 已提交
180 181 182 183
      "crossUserMode": 2
    }
   ]
   ```
G
ge-yafang 已提交
184 185 186 187 188


### 数据访问方应用的开发

1. 导入基础依赖包。
189
   
190 191 192 193 194 195 196
```ts
import UIAbility from '@ohos.app.ability.UIAbility';
import dataShare from '@ohos.data.dataShare';
import dataSharePredicates from '@ohos.data.dataSharePredicates';
import { ValuesBucket } from '@ohos.data.ValuesBucket'
import window from '@ohos.window';
```
G
ge-yafang 已提交
197 198

2. 定义与数据提供方通信的URI字符串。
199
   
200 201 202 203
```ts
// 作为参数传递的URI,与module.json5中定义的URI的区别是多了一个"/",是因为作为参数传递的URI中,在第二个与第三个"/"中间,存在一个DeviceID的参数
let dseUri = ('datashareproxy://com.samples.datasharetest.DataShare');
```
G
ge-yafang 已提交
204 205

3. 创建工具接口类对象。
206
   
207 208 209 210 211 212 213 214 215 216 217 218 219
```ts
let dsHelper: dataShare.DataShareHelper | undefined = undefined;
let abilityContext: Context;

export default class EntryAbility extends UIAbility {
  onWindowStageCreate(windowStage: window.WindowStage) {
    abilityContext = this.context;
    dataShare.createDataShareHelper(abilityContext, dseUri, (err, data) => {
      dsHelper = data;
    });
  }
}
```
G
ge-yafang 已提交
220 221

4. 获取到接口类对象后,便可利用其提供的接口访问提供方提供的服务,如进行数据的增删改查等。
222
   
223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258
```ts
// 构建一条数据
let key1 = 'name';
let key2 = 'age';
let key3 = 'isStudent';
let key4 = 'Binary';
let valueName1 = 'ZhangSan';
let valueName2 = 'LiSi';
let valueAge1 = 21;
let valueAge2 = 18;
let valueIsStudent1 = false;
let valueIsStudent2 = true;
let valueBinary = new Uint8Array([1, 2, 3]);
let valuesBucket: ValuesBucket = { key1: valueName1, key2: valueAge1, key3: valueIsStudent1, key4: valueBinary };
let updateBucket: ValuesBucket = { key1: valueName2, key2: valueAge2, key3: valueIsStudent2, key4: valueBinary };
let predicates = new dataSharePredicates.DataSharePredicates();
let valArray = ['*'];
if (dsHelper != undefined) {
  // 插入一条数据
  (dsHelper as dataShare.DataShareHelper).insert(dseUri, valuesBucket, (err, data) => {
    console.info(`dsHelper insert result:${data}`);
  });
  // 更新数据
  (dsHelper as dataShare.DataShareHelper).update(dseUri, predicates, updateBucket, (err, data) => {
    console.info(`dsHelper update result:${data}`);
  });
  // 查询数据
  (dsHelper as dataShare.DataShareHelper).query(dseUri, predicates, valArray, (err, data) => {
    console.info(`dsHelper query result:${data}`);
  });
  // 删除指定的数据
  (dsHelper as dataShare.DataShareHelper).delete(dseUri, predicates, (err, data) => {
    console.info(`dsHelper delete result:${data}`);
  });
}
```
259 260 261 262 263

## 相关实例

针对数据共享开发,有以下相关实例可供参考:

264
- [系统应用跨应用数据共享(ArkTS)(Full SDK)(API9)](https://gitee.com/openharmony/applications_app_samples/tree/master/code/SystemFeature/DataManagement/CrossAppDataShare)