欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Android Web3j OOM解決詳解

 更新時(shí)間:2018年07月16日 09:11:05   作者:沉默的范大叔  
這篇文章主要介紹了Android Web3j OOM解決詳解,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧

在Android客戶端使用Web3j創(chuàng)建錢(qián)包、導(dǎo)入錢(qián)包時(shí)都可能會(huì)產(chǎn)生OOM,相關(guān)issue在Github上已經(jīng)有所提及: https://github.com/web3j/web3j/issues/299 。這個(gè)問(wèn)題在Web3j 3.0版本以前是沒(méi)有的,由于新版的Web3j使用spongycastle庫(kù)替換了lambdaworks庫(kù),雖然在效率上提升了速度,但存在Android端的兼容性問(wèn)題。

本項(xiàng)目代碼地址: https://github.com/uncleleonfan/WalletOOM.git

創(chuàng)建錢(qián)包OOM解決

在創(chuàng)建錢(qián)包時(shí),如果創(chuàng)建一個(gè)Full Wallet,則會(huì)導(dǎo)致OOM:

public void onCreateFullWallet(View view) {
  String filePath = Environment.getExternalStorageDirectory().getAbsolutePath() + "/full";
  File file = new File(filePath);
  file.mkdirs();
  try {
    WalletUtils.generateFullNewWalletFile("a12345678", file);
  } catch (NoSuchAlgorithmException e) {
    e.printStackTrace();
  } catch (NoSuchProviderException e) {
    e.printStackTrace();
  } catch (InvalidAlgorithmParameterException e) {
    e.printStackTrace();
  } catch (CipherException e) {
    e.printStackTrace();
  } catch (IOException e) {
    e.printStackTrace();
  }
}

Log如下:

"Caused by: java.lang.OutOfMemoryError: Failed to allocate a 1036 byte allocation with 16777216 free bytes and 48MB until OOM; failed due to fragmentation (required continguous free 16384 bytes for a new buffer where largest contiguous free 8192 bytes)",
"\tat org.spongycastle.util.Arrays.clone(Arrays.java:602)",
"\tat org.spongycastle.crypto.generators.SCrypt.SMix(SCrypt.java:126)",
"\tat org.spongycastle.crypto.generators.SCrypt.MFcrypt(SCrypt.java:87)",
"\tat org.spongycastle.crypto.generators.SCrypt.generate(SCrypt.java:66)",
"\tat org.web3j.crypto.Wallet.generateDerivedScryptKey(Wallet.java:136)",
"\tat org.web3j.crypto.Wallet.create(Wallet.java:74)",
"\tat org.web3j.crypto.Wallet.createStandard(Wallet.java:93)",
"\tat org.web3j.crypto.WalletUtils.generateWalletFile(WalletUtils.java:61)"

generateFullNewWalletFile里面會(huì)調(diào)用createStandard創(chuàng)建錢(qián)包,使用N_STANDARD,P_STANDARD來(lái)配置加密強(qiáng)度,直接影響需使用的內(nèi)存大小,最終導(dǎo)致OOM的發(fā)生。

public static WalletFile createStandard(String password, ECKeyPair ecKeyPair)
    throws CipherException {
  return create(password, ecKeyPair, N_STANDARD, P_STANDARD);
}

解決方法非常簡(jiǎn)單,創(chuàng)建一個(gè)Light Wallet即可:

public void onCreateLightWallet(View view) {
  String filePath = Environment.getExternalStorageDirectory().getAbsolutePath() + "/light";
  File file = new File(filePath);
  file.mkdirs();
  try {
    WalletUtils.generateLightNewWalletFile("a12345678", file);
  } catch (NoSuchAlgorithmException e) {
    e.printStackTrace();
  } catch (NoSuchProviderException e) {
    e.printStackTrace();
  } catch (InvalidAlgorithmParameterException e) {
    e.printStackTrace();
  } catch (CipherException e) {
    e.printStackTrace();
  } catch (IOException e) {
    e.printStackTrace();
  }
}

generateLightNewWalletFile會(huì)調(diào)用createLight來(lái)創(chuàng)建一個(gè)輕錢(qián)包,使用N_LIGHT,P_LIGHT,他們?cè)跀?shù)值上相對(duì)較小,所以不會(huì)OOM。

public static WalletFile createLight(String password, ECKeyPair ecKeyPair)
    throws CipherException {
  return create(password, ecKeyPair, N_LIGHT, P_LIGHT);
}

我們可以對(duì)比一下N_STANDARD和P_STANDARD, N_LIGHT和P_LIGHT的大?。?/p>

private static final int N_LIGHT = 1 << 12;
private static final int P_LIGHT = 6;

private static final int N_STANDARD = 1 << 18;
private static final int P_STANDARD = 1;

導(dǎo)入錢(qián)包OOM解決

當(dāng)我們導(dǎo)入一個(gè)輕錢(qián)包時(shí),不會(huì)產(chǎn)生OOM,但導(dǎo)入不是一個(gè)輕錢(qián)包時(shí),則有可能產(chǎn)生OOM。例如,我們使用Imtoken創(chuàng)建一個(gè)錢(qián)包并導(dǎo)出Keystore,Keystore如下:

{"address":"9a2e2419f3af050d4730f80e7a65b9f8deb5e16f","crypto":{"cipher":"aes-128-ctr","cipherparams":{"iv":"eaccea79c27a91e307f24988186ef21a"},"ciphertext":"a163e532edf2d76beaee5c26fd2c2fab071a9cb37627aa185ac89e223e41ab97","kdf":"scrypt","kdfparams":{"dklen":32,"n":65536,"p":1,"r":8,"salt":"6a847392a029553f4152dea7bb0b6fb0ac9eec29f55e572fe94603182f5ed7f1"},"mac":"3fad2a31e18c611b10df84db9ae368ce2e189b5c35e9f111e40ca4b4bfb02491"},"id":"032c47c2-c7b7-46f8-a3f7-f526580f6f09","version":3}

可以看到,其中n為65536,p為1,而輕錢(qián)包的n為1<<12,即2的12次方,4096,所以這不是一個(gè)輕錢(qián)包。

我們將該Keystore作為一個(gè)json文件push到SD卡中,然后使用Web3j進(jìn)行導(dǎo)入:

public void onImportWallet(View view) {
  try {
    //需提前將assets目錄下的keystore.json文件推送到手機(jī)SD里面
    String filePath = Environment.getExternalStorageDirectory().getAbsolutePath() + "/keystore.json";
    File file = new File(filePath);
    WalletUtils.loadCredentials("a12345678", file);
  } catch (IOException e) {
    e.printStackTrace();
  } catch (CipherException e) {
    e.printStackTrace();
  }
}

發(fā)現(xiàn)同樣會(huì)OOM:

Caused by: java.lang.OutOfMemoryError: Failed to allocate a 1036 byte allocation with 13588800 free bytes and 12MB until OOM; failed due to fragmentation (required continguous free 16384 bytes for a new buffer where largest contiguous free 12288 bytes)
    at org.spongycastle.util.Arrays.clone(Arrays.java:602)
    at org.spongycastle.crypto.generators.SCrypt.SMix(SCrypt.java:126)
    at org.spongycastle.crypto.generators.SCrypt.MFcrypt(SCrypt.java:87)
    at org.spongycastle.crypto.generators.SCrypt.generate(SCrypt.java:66)
    at org.web3j.crypto.Wallet.generateDerivedScryptKey(Wallet.java:136)
    at org.web3j.crypto.Wallet.decrypt(Wallet.java:214)
    at org.web3j.crypto.WalletUtils.loadCredentials(WalletUtils.java:112)

通過(guò)log可以看出來(lái),這里和創(chuàng)建錢(qián)包的OOM是一樣的,都是最后調(diào)用generateDerivedScryptKey后導(dǎo)致:

private static byte[] generateDerivedScryptKey(
    byte[] password, byte[] salt, int n, int r, int p, int dkLen) throws CipherException {
  return SCrypt.generate(password, salt, n, r, p, dkLen);
}

創(chuàng)建錢(qián)包可以創(chuàng)建一個(gè)輕錢(qián)包,導(dǎo)入錢(qián)包總不能讓用戶換一個(gè)輕錢(qián)包來(lái)導(dǎo)入吧。這里,我們只能還是換回lambda庫(kù)來(lái)完成keystore的編解碼, 即我們可以自己寫(xiě)一個(gè)generateDerivedScryptKey方法,將spongycastle的SCrypt換成lambda的SCrypt。我們使用MyWalletUtils和MyWallet共同完成該任務(wù)。

public void onImportWallet(View view) {
  try {
    //需提前將assets目錄下的keystore.json文件推送到手機(jī)SD里面
    String filePath = Environment.getExternalStorageDirectory().getAbsolutePath() + "/keystore.json";
    File file = new File(filePath);
    Credentials credentials = MyWalletUtils.loadCredentials("a12345678", file);
    Log.d(TAG, "address:" + credentials.getAddress());
  } catch (IOException e) {
    e.printStackTrace();
  } catch (CipherException e) {
    e.printStackTrace();
  }
}

public class MyWalletUtils {
  public static Credentials loadCredentials(String password, File source)
      throws IOException, CipherException {
    WalletFile walletFile = objectMapper.readValue(source, WalletFile.class);
    return Credentials.create(MyWallet.decrypt(password, walletFile));
  }
}

public class MyWallet {

  private static final int CURRENT_VERSION = 3;
  private static final String CIPHER = "aes-128-ctr";
  static final String AES_128_CTR = "pbkdf2";
  static final String SCRYPT = "scrypt";

  private static byte[] generateDerivedScryptKey(
      byte[] password, byte[] salt, int n, int r, int p, int dkLen) {
    try {
      return SCrypt.scrypt(password, salt, n, r, p, dkLen);
    } catch (GeneralSecurityException e) {
      e.printStackTrace();
    }
    return null;
  }
}

按照以上方法處理之后,就可以解決OOM,但是用戶等待的時(shí)間會(huì)稍微長(zhǎng)一點(diǎn),另外,最好還是添加一下Android平臺(tái)的libscrpt.so庫(kù),大家可在本項(xiàng)目的jniLibs中找到。

以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。

相關(guān)文章

  • android 二次打包完成apk多渠道打包的方法

    android 二次打包完成apk多渠道打包的方法

    本篇文章主要介紹了android 二次打包完成apk多渠道打包的方法,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧
    2017-10-10
  • Android微信第三方登錄(個(gè)人筆記)

    Android微信第三方登錄(個(gè)人筆記)

    這篇文章主要為大家詳細(xì)介紹了Android微信第三方登錄的具體過(guò)程,個(gè)人筆記分享,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2016-12-12
  • 解決Android TabLayout 在寬屏幕上tab不能平均分配的問(wèn)題

    解決Android TabLayout 在寬屏幕上tab不能平均分配的問(wèn)題

    這篇文章主要介紹了解決Android TabLayout 在寬屏幕上tab不能平均分配的問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧
    2020-08-08
  • Android App中實(shí)現(xiàn)向右滑動(dòng)銷(xiāo)毀功能的要點(diǎn)解析

    Android App中實(shí)現(xiàn)向右滑動(dòng)銷(xiāo)毀功能的要點(diǎn)解析

    這篇文章主要介紹了Android應(yīng)用中實(shí)現(xiàn)向右滑動(dòng)銷(xiāo)毀條目功能的要點(diǎn)解析,有些類(lèi)似于iOS App中的滑動(dòng)頁(yè)面刪除效果,需要的朋友可以參考下
    2016-04-04
  • Flutter手機(jī)權(quán)限檢查與申請(qǐng)實(shí)現(xiàn)方法詳解

    Flutter手機(jī)權(quán)限檢查與申請(qǐng)實(shí)現(xiàn)方法詳解

    使用flutter進(jìn)行app開(kāi)發(fā),一定會(huì)用到手機(jī)的部分權(quán)限,包括通知推送、定位、相冊(cè)、存儲(chǔ)、相機(jī)、麥克風(fēng)等。而權(quán)限的檢查和獲取,最受歡迎的就是通過(guò)permission_handler這個(gè)插件來(lái)實(shí)現(xiàn)
    2022-11-11
  • Android中使用RecyclerView實(shí)現(xiàn)下拉刷新和上拉加載

    Android中使用RecyclerView實(shí)現(xiàn)下拉刷新和上拉加載

    RecyclerView 是Android L版本中新添加的一個(gè)用來(lái)取代ListView的SDK,它的靈活性與可替代性比listview更好。這篇文章主要介紹了Android中使用RecyclerView實(shí)現(xiàn)下拉刷新和上拉加載的相關(guān)資料,需要的朋友可以參考下
    2016-03-03
  • Android 獲取屏幕高度,標(biāo)題高度,狀態(tài)欄高度(實(shí)例代碼)

    Android 獲取屏幕高度,標(biāo)題高度,狀態(tài)欄高度(實(shí)例代碼)

    getWindow().findViewById(Window.ID_ANDROID_CONTENT)這個(gè)方法獲取到的view就是程序不包括標(biāo)題欄的部分,然后就可以知道標(biāo)題欄的高度了
    2013-11-11
  • Android框架學(xué)習(xí)之Volley和Glide詳解

    Android框架學(xué)習(xí)之Volley和Glide詳解

    這篇文章主要給大家介紹了關(guān)于Android框架學(xué)習(xí)之Volley和Glide的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)各位Android開(kāi)發(fā)者們具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2018-05-05
  • Flutter中關(guān)于angle的踩坑記錄

    Flutter中關(guān)于angle的踩坑記錄

    考慮到可能有很多同學(xué)還沒(méi)有接觸 Flutter,下面這篇文章主要給大家介紹了關(guān)于Flutter中關(guān)于angle的踩坑記錄,文中通過(guò)圖文介紹的非常詳細(xì),需要的朋友可以參考下
    2022-06-06
  • Android編程使用Intent傳遞對(duì)象的方法分析

    Android編程使用Intent傳遞對(duì)象的方法分析

    這篇文章主要介紹了Android編程使用Intent傳遞對(duì)象的方法,結(jié)合實(shí)例形式詳細(xì)分析了Android使用Intent實(shí)現(xiàn)傳遞對(duì)象的相關(guān)技巧與注意事項(xiàng),需要的朋友可以參考下
    2016-01-01

最新評(píng)論