为什么java.security.NoSuchProviderException没有这样的提供者:BC?

时间:2010-09-14 18:20:47

标签: java security cryptography jce

jar(bcprov-jdk16-145.jar)已添加到项目中,Security.addProvider(new org.bouncycastle.jce.provider.BouncyCastleProvider())已添加到类中,BouncyCastleProvider.PROVIDER_NAME确实返回“BC”但AesFileIo.writeFile()仍然引发java.security.NoSuchProviderException No such provider: BC。有什么想法吗?

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);
        }
    }
}

5 个答案:

答案 0 :(得分:106)

我对Android sdk不是很熟悉,但android-sdk似乎已经添加到安全性BouncyCastle提供商。

您在PC环境中需要做的只是手动添加

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

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

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

请注意.5它等于已添加的提供商的序号。

答案 1 :(得分:11)

您可以通过编辑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");

答案 2 :(得分:7)

您可以通过使用以下代码编辑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 :(得分:1)

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

<weblogic_jdk_home>\jre\lib\ext

答案 4 :(得分:1)

我的经验是,每次执行时都可以使用提供程序作为这样的字符串,这样很好

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);

我正在使用bcpkix-jdk15on 1.65版