相关文章推荐
打酱油的猴子  ·  二、Perl常量_perl ...·  1 年前    · 
豪爽的猴子  ·  System.Threading.Tasks ...·  1 年前    · 

所谓非对称,就是使用公钥/私钥加密,然后用对应的私钥/公钥解密

1.配对的公钥,私钥:

//公钥(注意包括头(-----BEGIN PUBLIC KEY-----)尾(-----END PUBLIC KEY-----)标志)
-----BEGIN PUBLIC KEY-----
MIGfMA0GCSqGSIb3DQEBAQUAA4GNADCBiQKBgQDbdq5NjtUEkAQ6wucPuhC0aRvSMsaX3GrhkwsLLdWZnVNVpkJRw
yPFq9HJNuntRw7P9Sb3TkwrpN60x62kZ6qV8h1GoG4jIfofuVWPv1VzudAV6kWJWMl3sc
+DtV5q1dy1KQLb6e90cuOynzxVT3j+Fx7ZOzovmdTkEcoRiYWV1QIDAQAB
-----END PUBLIC KEY-----
//私钥(注意包括头(-----BEGIN RSA PRIVATE KEY-----)尾(-----END RSA PRIVATE KEY-----)标志)
-----BEGIN RSA PRIVATE KEY-----
MIICdgIBADANBgkqhkiG9w0BAQEFAASCAmAwggJcAgEAAoGBAKBc0G8WmfydUuX6TnO5M3lSmNYqy8beEgBn24vWk6sWCZ2va+TRIWEew5dxmMXg5+N40a2XOJfyTxCALvzts+/J0q8efi2WIecPecWGNoDB8ZehGfzv3BQD5skk9sf1tL6ztU5tquPFajwmO85WR2pTFfk0CXgPEtLzPxntYfHpAgMBAAECgYB6DSo24qTgt3zYvOHpAhRDSrI4jun5uhtJY8Kdc/uAQ42eDl24EdOt75Q+N59nO/5CCzrLPEU+oJW8oVMh/mjr6gMv7QxhEr1iUrZmvAGTO2kB5sMgWfqPkiHeGddYWawcWGhAwQEJrrgGVldzaslY9FvWcmTjCjypEVvB7l1BgQJBAOJwB2RbRbI+z0d1M6DOawyu+J7JleMAQ42x9KyhiwYc/D0CZX0gkuyqGcwSh+c1CYjjdoRIh/531e1OZyBmRG0CQQC1TG68k+iXuz14+fgpHEttl2E3lQ6VfIzv+oaQHnRmmT92t2Eci70zkDFjVCGXxwQ279EokmP6RWfJG5Z1nV3tAkAZkHpVKzTQLeUq9SFyCzvsv6hUDQA+E56M1cWA4/AVLZqQrL+Wg+HylDW7Y3AyeztrV/rebm3kHdVqKEreTo11AkBPUou40nYXvQKeZbAgPJL79hnA+eSRnxcDAHfTop+HLFHKHV3N4Y38e4BAV1UDVT4Q00iOGc7Id4l7QijIePvZAkEA1OmJfbsaCx3T+4lZahzo8k7NzP5BS6izPVAVk8O3LHo3TggJZMa69Nc+8O82ZXOiOSt7bSONaST54ejjN8yhtg==
-----END RSA PRIVATE KEY-----

2.判断openssl扩展是否安装

 extension_loaded('openssl') or die('PHP加密需要openssl扩展支持');

3.判断公钥和私钥是否可用

//私钥文件路径   
$this->privateKey = file_get_contents(dirname(__FILE__) .'libraries/YCPAY/account/rsa_private_key.pem');
//公钥文件路径   
$this->publicKey = file_get_contents(dirname(__FILE__) . 'libraries/YCPAY/account/rsa_public_key.pem');  
* 生成Resource类型的密钥,如果密钥文件内容被破坏,openssl_pkey_get_private函数返回false
$privateKey = openssl_pkey_get_private($this->privateKey);
* 生成Resource类型的公钥,如果公钥文件内容被破坏,openssl_pkey_get_public函数返回false
$publicKey = openssl_pkey_get_public($this->publicKey);
($privateKey) or die('密钥不可用');
($publicKey) or die('公钥不可用');

4.公钥加密

由于openssl_public_encrypt()加密对加密串有字符限制(117字节),所有需要分段加密(每段32个字符,加密后拼接起来)

    define("BAOFOO_ENCRYPT_LEN", 32);    
    // 公钥加密
    function encryptedByPublicKey($data_content)
        $publicKey = openssl_pkey_get_public($this->publicKey);
        $encrypted = "";
        $totalLen = strlen($data_content);
        $encryptPos = 0;
        while ($encryptPos < $totalLen) {
//            openssl_public_encrypt(substr($data_content, $encryptPos, BAOFOO_ENCRYPT_LEN), $encryptData, $publicKey, OPENSSL_PKCS1_OAEP_PADDING);
            openssl_public_encrypt(substr($data_content, $encryptPos, BAOFOO_ENCRYPT_LEN), $encryptData, $publicKey);
//            $encrypted .= bin2hex($encryptData);
            $encrypted .= base64_encode($encryptData);
            $encryptPos += BAOFOO_ENCRYPT_LEN;
        return $encrypted;

5.私钥解密

由于openssl_private_decrypt()加密对加密串有字符限制(128字节),所有需要分段解密后拼接起来

    // 私钥解密
    function encryptedByPrivateKey($data_content)
        $privateKey = openssl_pkey_get_private($this->privateKey);
        $crypto = '';
        $data_content = base64_decode($data_content);
        foreach (str_split($data_content, 128) as $chunk) {
            openssl_private_decrypt($chunk, $decryptData, $privateKey);
            $crypto .= $decryptData;
        return $crypto;

 6.完整代码如下,仅供参考

define("BAOFOO_ENCRYPT_LEN", 32);
//CI框架
class YcPay extends MY_Controller
    private $privateKey;
    private $publicKey;
    private $merIdNum;
    private $merKey;
    private $url;
    public function __construct()
        parent::__construct();
        //正式环境
        $this->privateKey = file_get_contents(APPPATH . 'libraries/YCPAY/account/rsa_private_key.pem');
        $this->publicKey = file_get_contents(APPPATH . 'libraries/YCPAY/account/rsa_public_key.pem');
        $this->merIdNum = 'dc0020663';
        $this->merKey = '5446042869aaf86d22c5d19189f21956';
        $this->url = 'https://mis.ixianhe.cn/action/exp/';
    // 公钥加密
    function encryptedByPublicKey($data_content)
//        $data_content = base64_encode($data_content);
        $publicKey = openssl_pkey_get_public($this->publicKey);
        $encrypted = "";
        $totalLen = strlen($data_content);
        $encryptPos = 0;
        while ($encryptPos < $totalLen) {
//            openssl_public_encrypt(substr($data_content, $encryptPos, BAOFOO_ENCRYPT_LEN), $encryptData, $publicKey, OPENSSL_PKCS1_OAEP_PADDING);
            openssl_public_encrypt(substr($data_content, $encryptPos, BAOFOO_ENCRYPT_LEN), $encryptData, $publicKey);
//            $encrypted .= bin2hex($encryptData);
            $encrypted .= base64_encode($encryptData);
            $encryptPos += BAOFOO_ENCRYPT_LEN;
        return $encrypted;
    // 私钥解密
    function encryptedByPrivateKey($data_content)
        $privateKey = openssl_pkey_get_private($this->privateKey);
        $crypto = '';
        $data_content = base64_decode($data_content);
        foreach (str_split($data_content, 128) as $chunk) {
            openssl_private_decrypt($chunk, $decryptData, $privateKey);
            $crypto .= $decryptData;
        return $crypto;
    //检测密钥+加密
    public function rsa_pass($params, $api_method, $is_big)
        extension_loaded('openssl') or die('php需要openssl扩展支持');
         * 生成Resource类型的密钥,如果密钥文件内容被破坏,openssl_pkey_get_private函数返回false
        $privateKey = openssl_pkey_get_private($this->privateKey);
         * 生成Resource类型的公钥,如果公钥文件内容被破坏,openssl_pkey_get_public函数返回false
        $publicKey = openssl_pkey_get_public($this->publicKey);
//        ($privateKey) or die('密钥不可用');
//        ($publicKey) or die('公钥不可用');
        if (empty($privateKey)) {
            $param_back = [
                'resCode' => '9999',
                'resMsg' => '密钥不可用',
            $this->log($param_back, 'rsa_pass', $is_big, '错误报文:');
            echo json_encode($param_back, JSON_UNESCAPED_UNICODE);
            exit;
        if (empty($publicKey)) {
            $param_back = [
                'resCode' => '9999',
                'resMsg' => '公钥不可用',
            $this->log($param_back, $api_method, $is_big, '错误报文:');
            echo json_encode($param_back, JSON_UNESCAPED_UNICODE);
            exit;
         * 原数据
        $originalData = json_encode($params, JSON_UNESCAPED_UNICODE | JSON_UNESCAPED_SLASHES);
//        var_dump(strlen($originalData));die;
        //用公钥加密
        $encryptData = $this->encryptedByPublicKey($originalData);
        return $encryptData;
     * 签名 + post请求
     * @param array $params 请求参数
     * @param string $method 第三方方法名
     * @param string $api_method 接口名
     * @param string $is_big 通道别名
     * @param string $bankcardNo 银行卡号
     * @return mixed
    public function sign_post($params, $method, $api_method, $is_big, $bankcardNo = '')
        $this->log($params, $api_method, $is_big, '业务数据:', $bankcardNo);
        //公钥加密(RSA/ECB/PKCS1Padding)
        $encryptData = $this->rsa_pass($params, $api_method, $is_big);
        $merSign = "merIdNum=" . $this->merIdNum . "&encryptData=" . $encryptData . "&merKey=" . $this->merKey;
        //公共参数
        $request_param = [
            'merVersion' => '3.0',
            'merIdNum' => $this->merIdNum,
            'requestMethod' => 'exp.' . $method,
            'merSign' => md5($merSign),
            'encryptData' => $encryptData,
        $requestUrl = $this->url . $method;
        $this->log($requestUrl, $api_method, $is_big, '请求url:');
        $this->log($request_param, $api_method, $is_big, '请求报文:');
        $request_param = json_encode($request_param, JSON_UNESCAPED_UNICODE | JSON_UNESCAPED_SLASHES);
        $res = $this->curl($requestUrl, $request_param);
        $this->log($res, $api_method, $is_big, '响应报文:');
//        $third_array = [
//            'userStatus'=>'E0000',
//            'userMsg'=>'绑卡申请成功',
//            'encryptData'=>'lVa1eNpFP5V6ImaFbk2s2mu7ptY8cHc+ANrI3o38ZTdeR+OfZsOA1BeWD8vDacBl7eFA7GgTA6OM2mzIuwlT4wkgm8IpNshRwtKHwTAri0c4HsZ+2qTfZJjybiB36eJ3CsiPntlae7wsMcNx9hxlDvcs19D77mufEVhTWYQYcHc=',
//            'merIdNum'=>'dc0020450',
//            'merSign'=>'79d52ca53003c3c7b73916dac313c38a',
//        ];
//        $this->log($third_array, $api_method, $is_big, '响应报文:');
        $third_array = json_decode($res, true);
        if ($third_array['userStatus'] == 'E0000' && isset($third_array['encryptData'])) {
            //私钥解密
            $third_res = $this->encryptedByPrivateKey($third_array['encryptData']);
            $this->log($third_res, $api_method, $is_big, '解密报文:');
            //处理返回
            $third_array['data'] = json_decode($third_res);
            //解密报文:{"orderDesc":"交易成功","orderStatus":"SUCCESS","userLinkId":"YCDA9CA_257919_162701706280078"}
        } else {
            //失败响应的两种情况
            //【2021-07-22 15:56:23】响应报文:{"userStatus":"E0000","userMsg":"结算请求已受理","encryptData":"Rfyvdj3HUOtGzMAbP4Jh4EQpd9VXDr7BLQD5fRwz+TOvCwI6EubNv4MdXCWuRN1zm9C17bvt27jx1I67ceVHhFW/vpqNEBG5eTmqJjXP6PJrJwiyf03yrK3EUOXf7wgpdAH5NW3PrywrWLmt/lC2hNvUf02iQ0gXRxN3Ka6X9k4a1QfjSJRXYTXdUlVshDWx+Y1gMKRvLTrOW4IzaJpyYpd/KXI7ahEAYqWCqEruPhRIUV9N+z+lECHIeG0Ml/sC7/AdBTs4oT+LgMNGsxCH5n1RoFj4EhqcVCrrrYW5O900sdrn78xURWo3yvkL2up6vDMV9fmViV2udwAVC+wD6Q==","merIdNum":"dc0020663","merSign":"342e58237f3e68213f23815410585a9b"}
            //【2021-07-22 15:56:23】解密报文:{"orderDesc":"商户手续费金额超过了最大手续费限制","orderStatus":"FAIL","userLinkId":"YCDA9CA_162990_162694058243001"}
            //------------------------
            //【2021-07-23 11:01:03】响应报文:{"userStatus":"E0002","userMsg":"用户余额不足"}
            //这样处理了之后前端才不会出错 
            $third_array['data'] = [
                "orderDesc" => $third_array['userMsg'],
                "orderStatus" => "FAIL",//订单状态 SUCCESS 交易成功  FAIL交易失败  PROCESSING  交易处理中
        $third_json = json_encode($third_array, JSON_UNESCAPED_UNICODE | JSON_UNESCAPED_SLASHES);
        return $third_json;
    //接口post请求
    public function curl($url = '', $data = '')
        $ch = curl_init();
        curl_setopt($ch, CURLOPT_URL, $url);
        if (!empty($data)) {
            curl_setopt($ch, CURLOPT_POST, 1);
            curl_setopt($ch, CURLOPT_POSTFIELDS, $data);
        curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, false);
        curl_setopt($ch, CURLOPT_TIMEOUT, 30); //超时时间30秒
        curl_setopt($ch, CURLOPT_SSL_VERIFYHOST, false);
        curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
        $headers = [
            'Content-type: application/json'
        curl_setopt($ch, CURLOPT_HTTPHEADER, $headers);
        $output = curl_exec($ch);
        curl_close($ch);
        return $output;
     * 日志记录
     * @param $data /内容
     * @param $method /方法名
     * @param $bankcardNo /银行卡号
     * @param string $is_big /通道别名
     * @param string $title /标题
    public function Log($data, $method, $is_big, $title, $bankcardNo = '')
        $str = is_array($data) ? json_encode($data, JSON_UNESCAPED_UNICODE | JSON_UNESCAPED_SLASHES) : $data;
        if (empty($bankcardNo)) {
            $content = '【' . date('Y-m-d H:i:s') . '】' . $title . $str . PHP_EOL;
        } else {
            $content = ' bankcardNo:' . $bankcardNo . '【' . date('Y-m-d H:i:s') . '】' . $title . $str . PHP_EOL;
        $path = APPPATH . "../PAYLOG/$is_big/$method/";
        if (!is_dir($path)) { //判断目录是否存在 不存在就创建
            mkdir($path, 0777, true);
        file_put_contents($path . date("Y-m-d") . '.txt', $content, FILE_APPEND);
        //换行分割
        file_put_contents($path . date("Y-m-d") . '.txt', '------------------------' . PHP_EOL, FILE_APPEND);

对应的私钥加密和公钥加密函数

openssl_private_encrypt($data,$encrypted,$this->privateKey);//私钥加密

openssl_public_decrypt($encrypted,$decrypted,$this->publicKey);//私钥加密的内容通过公钥可用解密出来

 参考博文:https://blog.csdn.net/qq_27517377/article/details/79047021

本文实例讲述了Python3对称加密算法RSA。分享给大家供大家参考,具体如下: python3 可以使用 Crypto.PublicKey.RSArsa 生成公钥、私钥。 其中 python3.6 Crypto 库的安装方式请参考前面一篇《Python3对称加密算法AES、DES3》 rsa 加解密的库使用 pip3 install rsa 就行了 C:\WINDOWS\system32>pip3 install rsa Collecting rsa   Downloading https://files.pythonhosted.org/packages/e1/ae/baedc9 使用 RSA 加密/解密音频文件的 MATLAB 程序 运行代码前,将待加密的音频文件与代码文件放在同一目录下,并将文件“encryption.m”的第一行更改为与待加密音频文件的文件名相匹配 运行程序时请遵循此顺序 1. 运行 code_gen.m 2.运行encryption.m 3.运行decryption.m Wavesurfer 可用于分析和播放音频文件。 它可以从以下链接下载。 https://sourceforge.net/projects/wavesurfer/ php实现AES 128位加密的相关操作技巧是什么?这篇文章主要介绍了PHP实现的AES 128位加密算法,结合实例形式分析了AES 128位加密的相关概念、原理及php实现AES 128位加密的相关操作技巧,注释中包含详细的说明信息,需要的朋友就来一聚教程网参考一下吧!本文实例讲述了PHP实现的AES 128位加密算法。分享给大家供大家参考,具体如下:/*加密算法一般分为两种:对称加密算法和对... 最近遇到一个项目,需要进行RSA的验签,RSA的密钥为2048位即256个字节长。 上游是先对一个文件进行SHA256做hash,得到32字节的摘要,然后进行填充,填充规则为前面两个字节为0x00,0x01,然后填充全FF,接下来是32字节的摘要数据,这样填充之后的总数据长度为256字节,然后使用RSA的私钥对这256字节进行签名,下发下来,下游需要对这个签名进行处理,得到原始摘要,然后跟自己计算的摘要比对,验证通过则合法。 涉及的知识点如下: 一、RSA签名算法的填充方式,常用的有如下三种 $privateKeyFilePath = '-----BEGIN RSA PRIVATE KEY----- MIICdgIBADANBgkqhkiG9w0BAQEFAASCAmAwggJcAgEAAoGBAJDArkki2k3+xhze v5emztqFtUhOzVlSveQ5dOLC4nNlLg8r81I9xBaXB7m4rb6lVjLgUuAXKUJ8y7yH eE02s695cR/ZD0fierrMKMkmTFZIqOVYchn15N8E 需要先了解的openssl系列函数openssl_pkey_get_private 从证书中解析获取私钥,以供使用。成功,返回真实的密钥资源标识符(Resource ID),否则返回falseopenssl_pkey_get_public 从证书中解析获取公钥,以供使用。成功,返回真实的密钥资源标识符(Resource ID),否则返回falseopenssl_private_encrypt($d... PHP 在进入7.x 时代后,默认就不再附带 mcrypt 扩展,mcrypt 将被 openssl_* 一族函数所替代。所以,对于 PHPer 来说,有必要学习一下 PHP 的 OpenSSL 扩展。上一篇文章《PHP中OpenSSL扩展 - 对称加密》 ,介绍了 OpenSSL 扩展中对称加密的使用方法,本文将介绍对称加密的使用方法。PHP 的 OpenSSL 扩展中,对称加密的相关函数有... 由于近期经常被改犇,我的有些犇犇会使用RSA/ECB/PKCS1Padding加密。我的公钥:Modulus: 774950881648176913095865671797928380709006416290316428444812519465243683541845584100145597962906239671085378870011090482547500240987975893282519...