• 基于openssl的aes_cbc加解密操作、基于libxml2的xml文件解析


    基于openssl的aes_cbc加解密操作、基于libxml2的xml文件解析

    基于aes的数据加解密需要提供两个额外的数据:密钥、初始向量

    aes cbc加解密原理

    cbc加密

    在这里插入图片描述

    CBC的加密从左往右看,初始化IV只有在第一个块加密的时候才会用到,而第N个块的加密IV则是用的N-1(N>1)个加密后的二进制数组。

    cbc解密

    在这里插入图片描述
    CBC的解密则也是从左往右看,但是加密时IV在解密时候,只会用于对第一个块进行解密,其他块的解密则是使用上一块的加密二进制作为IV进行解密操作。
    ps:通过加解密图解可知,初始向量IV在加密时针对整个待加密数据都起作用,但在解密时IV向量只针对第一个数据块起作用,因此在解密时,IV向量的重要度较低。

    加解密源码

    切记,在每次调用完加解密函数后,对应发IV向量的值会同步被修改

    #include 
    #include 
    #include 
    #include 
    #include 
     
    #define AESKEY "df98b715d5c6ed2b25817b6f255411a1"	//HEX密钥
    #define AESIV "2841ae97419c2973296a0d4bdfe19a4f"	//HEX初始向量
     
    //将文本形式的HEX串进行转换
    unsigned char* str2hex(char *str)	
     {
        unsigned char *ret = NULL;
        int str_len = strlen(str);
        int i = 0;
        assert((str_len%2) == 0);
        ret = (char *)malloc(str_len/2);
        for (i =0;i < str_len; i = i+2 ) 
        {
            sscanf(str+i,"%2hhx",&ret[i/2]);
        }
        return ret;
    }
     
    int main()
    {
        AES_KEY encryptkey;
        AES_KEY decryptkey;
     
        unsigned char *key;
        unsigned char *stdiv;
     
        key = str2hex(AESKEY);
        stdiv = str2hex(AESIV);
        AES_set_encrypt_key(key,128,&encryptkey);
        AES_set_decrypt_key(key,128,&decryptkey);
     
        unsigned char plain_text [32];
     
        memcpy(plain_text, "AES encrypt in openssl demo", 27);
        memset(plain_text + 27, 0, 5);
        //需要将加密区块长度填充为16字节整数倍,此处使用zero-padding,即末尾全用0填充
        printf("raw_text: ");
        for(int i = 0; i < 32; i++)
        {
            printf("%02X ", plain_text[i]);
        }
        printf("\n" );
     
        unsigned char encrypted_text [32];
        memset(1, 0, 32);
        unsigned char tmpiv[16];
        memcpy(tmpiv, stdiv, 16);
        AES_cbc_encrypt(plain_text, encrypted_text, 32, &encryptkey, tmpiv, AES_ENCRYPT);
        //初始向量这个参数每次使用都会将其改变,所以如果要多次加密且每次使用固定的初始向量,可以先用tmpiv接收
     
        printf("encrypted_text: " );
        for(int i = 0; i < 32; i++)
        {
            printf("%02X ", encrypted_text[i]);
        }
        printf("\n" );
     
        unsigned char decrypted_text [32];
        memset(decrypted_text, 0, 32);
        memcpy(tmpiv, stdiv, 16);
        AES_cbc_encrypt(encrypted_text, decrypted_text, 32, &decryptkey, tmpiv, AES_DECRYPT);
     
        printf("decrypted_text:");
        for(int i = 0; i < 32; i++)
        {
            printf("%02X ", decrypted_text[i]);
        }
        printf("\n");
        return 0;
    }
    
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38
    • 39
    • 40
    • 41
    • 42
    • 43
    • 44
    • 45
    • 46
    • 47
    • 48
    • 49
    • 50
    • 51
    • 52
    • 53
    • 54
    • 55
    • 56
    • 57
    • 58
    • 59
    • 60
    • 61
    • 62
    • 63
    • 64
    • 65
    • 66
    • 67
    • 68
    • 69
    • 70
    • 71
    • 72
    • 73
    • 74
    • 75
    • 76
    • 77

    编译及运行结果

    @ubuntu:~/aes_source$ gcc test.c -o test -lssl -lcrypto
    @ubuntu:~/aes_source$ ./test 
    raw_text: 41 45 53 20 65 6E 63 72 79 70 74 20 69 6E 20 6F 70 65 6E 73 73 6C 20 64 65 6D 6F 00 00 00 00 00 
    encrypted_text: 33 B7 65 49 2C 7E AB 0B 9C 90 2B 3A 0F 64 DF D3 60 A9 B6 7E 4D 9A 6F E0 08 19 EC AC E7 50 86 36 
    decrypted_text:41 45 53 20 65 6E 63 72 79 70 74 20 69 6E 20 6F 70 65 6E 73 73 6C 20 64 65 6D 6F 00 00 00 00 00 
    @ubuntu:~/aes_source$ 
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6

    基于libxml2的xml文件解析

    libxml2 安装及环境配置

    参考link

    xml文件

    <?xml version="1.0" encoding="UTF-8"?>
    <bmp_para>
      <para id="1">
         <width>1920</width>
         <height>1080</height>
         <bit>3</bit>
         <blue>0</blue>
         <green>0</green>
         <red>255</red>     
      </para>
    </bmp_para>
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11

    xml解析源码

    #include 
    #include 
     
    #include 
    #include 
     
    #define DEFAULT_XML_FILE "test.xml"
     
    //解析para字段,提取出width,height,bit,red,green,blue参数
    static int parse_bmp(xmlDocPtr doc, xmlNodePtr cur)
    {
        assert(doc || cur);
        xmlChar *key;
     
        cur = cur->xmlChildrenNode;
        while (cur != NULL) {
    	    //获取width
    	    if ((!xmlStrcmp(cur->name, (const xmlChar *)"width"))) {
    	        key = xmlNodeListGetString(doc, cur->xmlChildrenNode, 1);
    	        printf("width: %s\r\n", key);
    	        xmlFree(key);
    	    }
    	    //获取height
    	    if ((!xmlStrcmp(cur->name, (const xmlChar *)"height"))) {
    	        key = xmlNodeListGetString(doc, cur->xmlChildrenNode, 1);
    	        printf("height: %s\r\n", key);
    	        xmlFree(key);
    	    }
    	    //获取bit
    	    if ((!xmlStrcmp(cur->name, (const xmlChar *)"bit"))) {
    	        key = xmlNodeListGetString(doc, cur->xmlChildrenNode, 1);
    	        printf("bit: %s\r\n", key);
    	        xmlFree(key);
    	    }
    	    //获取 blue
    	    if ((!xmlStrcmp(cur->name, (const xmlChar *)"blue"))) {
    	        key = xmlNodeListGetString(doc, cur->xmlChildrenNode, 1);
    	        printf("blue: %s\r\n", key);
    	        xmlFree(key);
    	    }
    	    //获取 green
    	    if ((!xmlStrcmp(cur->name, (const xmlChar *)"green"))) {
    	        key = xmlNodeListGetString(doc, cur->xmlChildrenNode, 1);
    	        printf("green: %s\r\n", key);
    	        xmlFree(key);
    	    }
    	    //获取 red
    	    if ((!xmlStrcmp(cur->name, (const xmlChar *)"red"))) {
    	        key = xmlNodeListGetString(doc, cur->xmlChildrenNode, 1);
    	        printf("red: %s\r\n", key);
    	        xmlFree(key);
    	    }
    
    	    cur = cur->next;
        }
        return 0;
    }
     
    static int parse_para(const char *file_name)
    {
    	assert(file_name);
    
    	xmlDocPtr doc;   //xml整个文档的树形结构
    	xmlNodePtr cur;  //xml节点
    	xmlChar *id;     //phone id
    
    	//获取树形结构
    	doc = xmlParseFile(file_name);
    	if (doc == NULL) {
    		fprintf(stderr, "Failed to parse xml file:%s\n", file_name);
    		goto FAILED;
    	}
    
    	//获取根节点
    	cur = xmlDocGetRootElement(doc);
    	if (cur == NULL) {
    		fprintf(stderr, "Root is empty.\n");
    		goto FAILED;
    	}
    
    	if ((xmlStrcmp(cur->name, (const xmlChar *)"bmp_para"))) {
    		fprintf(stderr, "The root is not bmp_para.\n");
    		goto FAILED;
    	}
    
    	//遍历处理根节点的每一个子节点
    	cur = cur->xmlChildrenNode;
    	while (cur != NULL) {
    		if ((!xmlStrcmp(cur->name, (const xmlChar *)"para"))) {
    			id = xmlGetProp(cur, "id");
    			printf("id:%s\r\n",id);
    			parse_bmp(doc, cur);
    		}
    		cur = cur->next;
    	}
    	xmlFreeDoc(doc);
    	return 0;
    	FAILED:
    	if (doc) {
    		xmlFreeDoc(doc);
    	}
    	return -1;
    }
     
    int main(int argc, char*argv[])
    {
        char cFileNameRead[64] = {0}; 
    
    	if(argc < 2)
    	{
    		printf("please input like this:\r\n");
    		printf("./testReadXml.bin test.xml \r\n");
    		printf("test.xml --------------- input xml file \r\n");
    		return -1;
    	}
    
        sprintf(cFileNameRead,"%s",argv[1]);
    	printf("cFileNameRead=%s\r\n",cFileNameRead);
    
    	if (parse_para(cFileNameRead) != 0) {
    		fprintf(stderr, "Failed to parse bmp para.\n");
    		return -1;
    	}
    
    	return 0;
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38
    • 39
    • 40
    • 41
    • 42
    • 43
    • 44
    • 45
    • 46
    • 47
    • 48
    • 49
    • 50
    • 51
    • 52
    • 53
    • 54
    • 55
    • 56
    • 57
    • 58
    • 59
    • 60
    • 61
    • 62
    • 63
    • 64
    • 65
    • 66
    • 67
    • 68
    • 69
    • 70
    • 71
    • 72
    • 73
    • 74
    • 75
    • 76
    • 77
    • 78
    • 79
    • 80
    • 81
    • 82
    • 83
    • 84
    • 85
    • 86
    • 87
    • 88
    • 89
    • 90
    • 91
    • 92
    • 93
    • 94
    • 95
    • 96
    • 97
    • 98
    • 99
    • 100
    • 101
    • 102
    • 103
    • 104
    • 105
    • 106
    • 107
    • 108
    • 109
    • 110
    • 111
    • 112
    • 113
    • 114
    • 115
    • 116
    • 117
    • 118
    • 119
    • 120
    • 121
    • 122
    • 123
    • 124
    • 125
    • 126

    编译并运行xml解析结果:

    @ubuntu:~/xml$ gcc test.c -o test.bin -I /mnt/lib2xml/output/include/libxml2/ -L /mnt/lib2xml/output/lib/ -lxml2
    @ubuntu:~/xml$ ./test.bin test.xml 
    cFileNameRead=test.xml
    id:1
    width: 1920
    height: 1080
    bit: 3
    blue: 0
    green: 0
    red: 255
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
  • 相关阅读:
    常用网络编程函数
    西藏2022中国农民丰收节 国稻种芯:让农牧民成为活动主角
    es如何聚合查询同一字段文档数大于1的文档
    计算机毕业设计开题报告怎么写丨计算机毕业设计开题报告模板下载(关注博主领取)
    《Python魔法大冒险》005 魔法挑战:自我介绍机器人
    JDK、JRE 和 JVM 的区别和联系
    java内部类详解
    按键中断实验
    [Linux系统编程]_多线程(四)
    Trinitycore学习之在Linux环境上搭建服务器并测试运行
  • 原文地址:https://blog.csdn.net/weixin_42734533/article/details/127610731