HarmonyOS Next有应用沙箱提前缓存图片的示例代码吗?

阅读 650
avatarAI BotBETA

HarmonyOS Next有应用沙箱提前缓存图片的示例代码,但这里主要提供一个概念性的示例和流程说明,因为具体的代码实现可能会根据开发环境和具体需求有所不同。以下是一个简化的示例说明:

示例说明

假设你需要在HarmonyOS Next应用中提前缓存网络图片到应用沙箱中,以便后续快速加载和显示。以下是一个基本的流程示例:

  1. 前端调用Native接口

    • 在前端代码中,通过调用Native侧暴露的接口(例如saveImageOfInternet)来请求缓存网络图片。
    • 传入的参数包括网络图片的URL、应用的文件路径以及期望保存的文件名等。
  2. Native层处理

    • Native层接收到前端的请求后,使用网络请求库(如libcurl)来下载网络图片。
    • 下载完成后,将图片数据保存到应用沙箱中的指定路径。
    • 返回沙箱中的图片路径给前端。
  3. 前端显示图片

    • 前端接收到Native层返回的图片路径后,可以使用Image组件将该图片显示出来。
    • 如果需要,还可以将图片路径保存到本地数据库中,以便后续直接使用而无需再次下载。

示例代码框架(伪代码)

以下是一个简化的示例代码框架,用于说明如何在HarmonyOS Next中实现上述流程:

// 前端代码(ArkUI)
@Entry
@Component
struct MyComponent {
    @State private internetPicUrl: string = "http://example.com/image.jpg";
    @State private sandBoxPath: string = "";

    build() {
        Column() {
            Button("保存网络图片")
                .onClick(() => {
                    // 调用Native接口保存网络图片到沙箱
                    this.saveImageToSandbox(this.internetPicUrl);
                });
            
            if (this.sandBoxPath) {
                Image(this.sandBoxPath)
                    .width("100%")
                    .height("300px");
            } else {
                Text("图片正在加载...");
            }
        }
    }

    private async saveImageToSandbox(url: string): Promise<void> {
        // 调用Native接口并等待结果
        let result = await testNapi.saveImageOfInternet(url, this.fileDir, "cachedImage.jpg");
        
        // 更新沙箱路径状态
        if (result) {
            this.sandBoxPath = result;
        } else {
            // 处理错误情况
            console.error("图片保存失败");
        }
    }
}

// Native代码(C++或Java等)
// 假设有一个Native接口saveImageOfInternet,用于保存网络图片到沙箱
public class NativeImageSaver {

    public String saveImageOfInternet(String url, String fileDir, String fileName) {
        // 使用网络请求库下载图片(如libcurl)
        // ...(下载逻辑省略)
        
        // 将下载的图片数据保存到沙箱中的指定路径
        String sandBoxPath = fileDir + "/" + fileName;
        // ...(保存逻辑省略)
        
        // 返回沙箱中的图片路径
        return sandBoxPath;
    }
}

注意事项

  • 上述代码是一个简化的示例,用于说明流程。在实际开发中,你需要根据具体的开发环境和需求来实现相应的逻辑。
  • 网络请求和文件操作可能涉及异步处理,因此需要注意线程安全和异步回调的处理。
  • 图片的缓存机制可以根据需要进行优化,例如设置缓存大小、缓存时间等。

希望这个示例能够帮助你理解如何在HarmonyOS Next中实现应用沙箱提前缓存图片的功能。

1 个回答

子页面PageOne中需展示一张较大的网络图片,在父组件的aboutToAppear()中提前发起网络请求,并做判断文件是否存在,已下载的不再重复请求,存储在应用沙箱中。当父页面点击按钮跳转子页面PageOne,此时触发pixMap请求读取应用沙箱中已缓存解码的网络图片并存储在LocalStorage中,通过在子页面的Image中传入被@StorageLink修饰的变量ImageData进行数据刷新,图片送显。

import { fileIo as fs } from '@kit.CoreFileKit';
import { image } from '@kit.ImageKit';
import { common } from '@kit.AbilityKit';
import { httpRequest } from '../utils/NetRequest';

// 获取应用文件路径
let context = getContext(this) as common.UIAbilityContext;
let filesDir = context.filesDir;
let fileUrl = filesDir + '/xxx.png'; // 当使用实际网络地址时,需填入实际地址的后缀。
let para: Record<string, PixelMap | undefined> = { 'imageData': undefined };
let localStorage: LocalStorage = new LocalStorage(para);

@Entry(localStorage)
@Component
struct MainPage {
  @State childNavStack: NavPathStack = new NavPathStack();
  @LocalStorageLink('imageData') imageData: PixelMap | undefined = undefined;

  getPixMap() { // 从应用沙箱里读取文件
    try {
      let file = fs.openSync(fileUrl, fs.OpenMode.READ_WRITE); // 以同步方法打开文件
      const imageSource: image.ImageSource = image.createImageSource(file.fd);
      const options: image.InitializationOptions = {
        'alphaType': 0, // 透明度
        'editable': false, // 是否可编辑
        'pixelFormat': 3, // 像素格式
        'scaleMode': 1, // 缩略值
        'size': { height: 100, width: 100 }
      };
      fs.close(file);
      imageSource.createPixelMap(options).then((pixelMap: PixelMap) => {
        this.imageData = pixelMap;
      });
    } catch (e) {
      console.error('资源加载错误,文件或不存在!');
    }
  }

  aboutToAppear(): void {
    httpRequest(); // 在父组件提前发起网络请求
  }

  build() {
    Navigation(this.childNavStack) {
      Column() {
        Button('push Path to pageOne', { stateEffect: true, type: ButtonType.Capsule })
          .width('80%')
          .height(40)
          .margin({ bottom: '36vp' })
          .onClick(() => {
            if (!localStorage.get('imageData')) { // 非首次点击,不再重复调用getPixMap(),避免每次点击都从沙箱里读取文件。
              this.getPixMap();
            }
            this.childNavStack.pushPath({ name: 'pageOne' });
          })
      }
      .width('100%')
      .height('100%')
      .justifyContent(FlexAlign.End)
    }
    .backgroundColor(Color.Transparent)
    .title('ParentNavigation')
  }
}
//在NetRequest.ets中定义网络请求httpRequest(),通过fs.access()检查文件是否存在,当文件存在时不再重复请求,并写入沙箱中
import { http } from '@kit.NetworkKit';
import { BusinessError } from '@kit.BasicServicesKit';
import { fileIo as fs } from '@kit.CoreFileKit';
import { common } from '@kit.AbilityKit';

// 获取应用文件路径
let context = getContext(this) as common.UIAbilityContext;
let filesDir = context.filesDir;
let fileUrl = filesDir + '/xxx.png'; // 当使用实际网络地址时,需填入实际地址的后缀。

export async function httpRequest() {
  fs.access(fileUrl, fs.AccessModeType.READ).then((res) => { // 检查文件是否存在
    if (!res) { // 如沙箱里不存在地址,重新请求网络图片资源
      http.createHttp()
        .request('https://www.example.com/xxx.png', // 此处请填写一个具体的网络图片地址。
          (error: BusinessError, data: http.HttpResponse) => {
            if (error) {
              // 下载失败时不执行后续逻辑
              return;
            }
            // 处理网络请求返回的数据
            if (http.ResponseCode.OK === data.responseCode) {
              const imageData: ArrayBuffer = data.result as ArrayBuffer;
              // 保存图片到应用沙箱
              readWriteFileWithStream(imageData);
            }
          }
        )
    }
  })
}

// 写入到沙箱
async function readWriteFileWithStream(imageData: ArrayBuffer): Promise<void> {
  let outputStream = fs.createStreamSync(fileUrl, 'w+');
  await outputStream.write(imageData);
  outputStream.closeSync();
}
//在子组件中通过在子页面的Image中传入被@StorageLink修饰的变量ImageData进行数据刷新,图片送显。
@Builder
export function PageOneBuilder(name: string,param: Object) {
  PageOne()
}

@Component
export struct PageOne {
  pageInfo: NavPathStack = new NavPathStack();
  @State name: string = 'pageOne';
  @LocalStorageLink('imageData') imageData: PixelMap | undefined = undefined;

  build() {
    NavDestination() {
      Row() {
        Image(this.imageData) // 正例:此时Image拿到已提前加载好的网络图片,减少了白块出现时长
          .objectFit(ImageFit.Auto)
          .width('100%')
          .height('100%')
      }
      .width('100%')
      .height('100%')
      .justifyContent(FlexAlign.Center)
    }
    .title(this.name)
  }
}

本文参与了 【 HarmonyOS NEXT 技术问答冲榜,等你来战!】欢迎正在阅读的你也加入。

撰写回答
你尚未登录,登录后可以
  • 和开发者交流问题的细节
  • 关注并接收问题和回答的更新提醒
  • 参与内容的编辑和改进,让解决方法与时俱进
推荐问题