【问题标题】:Why java.security.NoSuchProviderException No such provider: BC?为什么 java.security.NoSuchProviderException 没有这样的提供者:BC?
【发布时间】:2011-04-12 07:45:53
【问题描述】:

jar (bcprov-jdk16-145.jar) 已添加到项目中,Security.addProvider(new org.bouncycastle.jce.provider.BouncyCastleProvider()) 已添加到类中,BouncyCastleProvider.PROVIDER_NAME 确实返回“BC”但 AesFileIo.writeFile() 仍然抛出 @987654324 @。有什么想法吗?

import java.io.FileOutputStream;
import java.io.InputStreamReader;
import java.io.ObjectOutputStream;
import javax.crypto.Cipher;
import javax.crypto.spec.IvParameterSpec;
import javax.crypto.spec.SecretKeySpec;
import org.bouncycastle.jce.provider.BouncyCastleProvider;

public class AesFileIo {

    private static final String AES_ALGORITHM = "AES/CTR/NoPadding";
    private static final String PROVIDER = BouncyCastleProvider.PROVIDER_NAME;
    private static final byte[] AES_KEY_128 = { // Hard coded for now
        78, -90, 42, 70, -5, 20, -114, 103,
        -99, -25, 76, 95, -85, 94, 57, 54};
    private static final byte[] IV = { // Hard coded for now
        -85, -67, -5, 88, 28, 49, 49, 85,
        114, 83, -40, 119, -65, 91, 76, 108};
    private static final SecretKeySpec secretKeySpec =
            new SecretKeySpec(AES_KEY_128, "AES");
    private static final IvParameterSpec ivSpec = new IvParameterSpec(IV);

    public void AesFileIo() {
        Security.addProvider(new org.bouncycastle.jce.provider
                .BouncyCastleProvider());
    }

    public void writeFile(String fileName, String theFile) {
        try {
            Cipher cipher = Cipher.getInstance(AES_ALGORITHM, PROVIDER);
            cipher.init(Cipher.ENCRYPT_MODE, secretKeySpec, ivSpec);
            byte[] encrypted = cipher.doFinal(theFile.getBytes());
            ObjectOutputStream os = new ObjectOutputStream(
                new FileOutputStream(fileName));
            os.write(encrypted);
            os.flush();
            os.close();
        } catch (Exception e) {
            StackTraceElement se = new Exception().getStackTrace()[0];
            System.err.println(se.getFileName() + " " + se.getLineNumber()
                    + " " + e);
        }
    }
}

【问题讨论】:

标签: java security cryptography jce


【解决方案1】:

我对 Android sdk 不是很熟悉,但似乎 android-sdk 附带的 BouncyCastle 提供程序已经添加到安全性中。

在 PC 环境中你要做的就是手动添加,

Security.addProvider(new org.bouncycastle.jce.provider.BouncyCastleProvider());

如果您有权访问 policy 文件,只需添加如下条目:

security.provider.5=org.bouncycastle.jce.provider.BouncyCastleProvider 

注意.5 等于已添加的提供程序的序号。

【讨论】:

  • 是否需要为所有运行此应用程序的 PC 手动添加到 PC 环境中,还是可以通过编程方式完成(bcprov-jdk16-145.jar 已添加到项目中)?
  • 你需要在你的代码中编码Security.addProvider,以确保它被加载,也许你需要将你的jar添加到你的项目中:)
  • 我在代码中添加了private static final String PROVIDER = BouncyCastleProvider.PROVIDER_NAME(解析为“BC”)作为一个字段,我已经将jar添加到了项目中,但继续得到java.security.NoSuchProviderException: No such provider: BC。我将通过包括整个班级来编辑我的原始问题。
  • 知道了。需要Cipher cipher = Cipher.getInstance(AES_ALGORITHM); 而不是Cipher cipher = Cipher.getInstance(AES_ALGORITHM, PROVIDER);。我想 Security.addProvider(new org.bouncycastle.jce.provider.BouncyCastleProvider()); 强制“BC”,而 Cipher.getInstance(AES_ALGORITHM, PROVIDER); 在这种情况下不被理解。
  • 当 BC 1.38 在 'CertStore certStore = cmssignedData.getCertificatesAndCRLs("Collection", "BC");' 行中抛出 "java.security.NoSuchProviderException: no such provider: BC" 时起作用跨度>
【解决方案2】:

您可以通过编辑 java.security 来添加安全提供程序,并使用以下代码创建静态块:

static {
    Security.addProvider(new BouncyCastleProvider());
}

如果您使用的是 maven 项目,则必须在项目的 pom.xml 文件中添加对 BouncyCastleProvider 的依赖项,如下所示。

<dependency>
            <groupId>org.bouncycastle</groupId>
            <artifactId>bcprov-jdk15on</artifactId>
            <version>1.47</version>
</dependency>

如果您使用的是普通的 java 项目,那么您可以从下面给出的链接添加下载 bcprov-jdk15on-147.jar 并编辑您的类路径。

http://www.java2s.com/Code/Jar/b/Downloadbcprovextjdk15on147jar.htm

【讨论】:

    【解决方案3】:

    您可以通过编辑 java.security 添加安全提供程序 通过增加 security.provider.=org.bouncycastle.jce.provider.BouncyCastleProvider

    或在班级顶部添加一行

    Security.addProvider(new BouncyCastleProvider());
    

    您可以在指定算法时使用下面的行来指定提供者

    Cipher cipher = Cipher.getInstance("AES", "SunJCE");
    

    如果您使用其他提供商,例如 Bouncy Castle,那么

    Cipher cipher =  Cipher.getInstance("AES", "BC");
    

    【讨论】:

      【解决方案4】:

      对于那些使用 web 服务器的人,请确保 bcprov-jdk16-145.jar 已安装在您的服务器 lib 中,因为 weblogic 必须将 jar 放入:

      <weblogic_jdk_home>\jre\lib\ext
      

      【讨论】:

        【解决方案5】:

        我的经验是,当我在每次执行中都有这个时,使用提供程序作为这样的字符串很好

        Security.addProvider(new BounctCastleProvider());
        new JcaPEMKeyConverter().setProvider("BC");
        

        但是当我优化并将以下内容放入构造函数时:

           if(bounctCastleProvider == null) {
              bounctCastleProvider = new BouncyCastleProvider();
            }
        
            if(Security.getProvider(bouncyCastleProvider.getName()) == null) {
              Security.addProvider(bouncyCastleProvider);
            }
        

        然后我必须像这样使用提供者,否则我会收到上述错误:

        new JcaPEMKeyConverter().setProvider(bouncyCastleProvider);
        

        我在 1.65 版上使用 bcpkix-jdk15

        【讨论】:

          猜你喜欢
          • 1970-01-01
          • 1970-01-01
          • 2016-08-25
          • 1970-01-01
          • 1970-01-01
          • 1970-01-01
          • 1970-01-01
          • 1970-01-01
          • 2020-04-20
          相关资源
          最近更新 更多