总结C语言在嵌入式开发中应用的知识点(文件数据的加密与解密)

<span style="font-family: Arial, Helvetica, sans-serif; background-color: rgb(255, 255, 255);">     好几天没写总结了,因为最近特别特别忙,各种驱动的代码都要我去改去测试,我主要最近主要是在搞驱动中数据,文件加密解密这块,就这块内容,便运用到了C语言最核心的两个知识点---数组和指针。</span>

 加密数据和文件为什么要用到指针和数组?

         首先你要打开一个文件,那么在标准C中,你可以定义一个文件描述符,标准C语言提供了fopen ,fread,fwrite,fseek等等的一些操作文件的接口,我们不妨来看看,这些接口到底是怎么使用的。

 

函数原型:FILE * fopen(const char * path,const char *mode);

返回值:文件顺利打开后,向该流的文件指针就会被返回。如果文件打开失败则返回NULL,并把错误代码存在errno中。

一般而言,打开文件后会做一些文件读取或写入的动作,若打开文件失败,接下来的读写动作也无法顺利进行,所以一般在fopen()后作错误判断及处理。

mode有下列几种形态字符串:

r 以只读方式打开文件,该文件必须存在。

r+ 以可读写方式打开文件,该文件必须存在。

w 打开只写文件,若文件存在则文件长度清为0,即该文件内容会消失。若文件不存在则建立该文件。

等等….

废话不多说,上代码:

#include<stdio.h>
//定义一个文件的路径
#define F_PATH "d:\\myfile\\file.dat"
int main(void)
{
    FILE*fp=NULL;//需要注意,最好初始化为空,因为文件流也是指针
    fp=fopen(F_PATH,"r");//以只读方式打开
    if(NULL==fp)
    {
        return -1;//要返回错误代码
    }
    fclose(fp);//关闭文件描述符
    fp=NULL;//需要指向空,否则会指向原打开文件地址

     return 0;
}

打开方式总结:各种打开方式主要有三个方面的区别:

①打开是否为二进制文件,用“b”标识。

②读写的方式,有以下几种:只读、只写、读写、追加只写、追加读写这几种方式。

③对文件是否必须存在、以及存在时是清空还是追加会有不同的响应。

 

当然,我只是举了一个最简单的例子,如果你有兴趣,慢慢去尝试。

以下是linux系统编程的open 打开文件的操作:

#include<unistd.h>
#include<sys/types.h>
#include<sys/stat.h>
#include<fcntl.h>
int main()
{
   int fd,size;
   char s [ ]=”Linux Programmer!\n”,buffer[80];
   fd=open(“/tmp/temp”,O_WRONLY|O_CREAT);
   write(fd,s,sizeof(s));
close(fd);
   fd=open(“/tmp/temp”,O_RDONLY);
  size=read(fd,buffer,sizeof(buffer));
  close(fd);
  printf(“%s”,buffer);
}

在UNIX系统编程中,打开文件还可以用open这个函数,具体怎么用自己百度!!要养成看函数的习惯,看多了自然就会了,其实这么厚一本,6-7百页,等你会了10个,那6-7百页的东西就都会了,只是函数而已嘛,拿来用就行了,至于会不会用,那要问问你自己C语言学得怎么样,还有对操作系统的理解程度怎么样。

以上说的是打开文件的一些简单操作,当然还有读,写,偏移等等。

函数原型:

size_t fread(void *buffer, size_t size, size_t count, FILE *stream);

buffer

用于接收数据的内存地址

size

要读的每个数据项的字节数,单位是字节

count

要读count个数据项,每个数据项size个字节.

stream

输入流

返回值:实际读取的元素个数。如果返回值与count不相同,则可能文件结尾或发生错误。从ferror和feof获取错误信息或检测是否到达文件结尾。

写:size_t fwrite(const void* buffer, size_t size, size_tcount, FILE* stream);

注意:这个函数以二进制形式对文件进行操作,不局限于文本文件

返回值:返回实际写入的数据块数目

(1)buffer:是一个指针,对fwrite来说,是要获取数据的地址;

(2)size:要写入内容的单字节数;

(3)count:要进行写入size字节的数据项的个数;

(4)stream:目标文件指针

(5)返回实际写入的数据项个数count。

偏移:

int fseek(FILE *stream, long offset, int fromwhere);

函数设置文件指针stream的位置。如果执行成功,stream将指向以fromwhere(偏移起始位置:文件头0(SEEK_SET),当前位置1(SEEK_CUR),文件尾2(SEEK_END))为基准,偏移offset(指针偏移量)个字节的位置。如果执行失败(比如offset超过文件自身大小),则不改变stream指向的位置。

返回值:
成功,返回0,失败返回-1,并设置errno的值,可以用perror()函数输出错误。

 

与此类似的UNIX接口有:open , read , write ,lseek,create ,opendir ,readdir,等等,慢慢学慢慢用,孰能生巧。

那么,为什么说这些知识点用到了加密解密上了,怎么用的?

比如,现在有一串密码.

       I123W598DWADA88DASDAJ

       这串密码需要进行加密,那么我们先用一个char型的数组将它保存起来。

       Charbuffer[] = {‘ I’,1,2,3,’W’,5,9,8,’D’,’W’,’A’,’D’,’A’,8,8,’D’,’A’,’S’,’D’,’A’,’J’} ;

       那如何加密?

我们可以用下面的方法来对这串数组进行简单的加密:

#include <stdio.h>
#define  NR(x)  (sizeof(x)/sizeof(x[0]))

unsigned char buffer[] = {'I',1,2,3,'W',5,9,8,'D','W','A','D','A',8,8,'D','A','S','D','A','J' } ;

//加密
void  Crypt_Data(unsigned char buffer[]);
//解密
void  Uncrypt_Data(unsigned char buffer[]);
int main(void)
{
	static int i = 0 ;
	//将数组加密
	Crypt_Data(buffer);
	//将数组解密
	Uncrypt_Data(buffer);
	//数组输出
	for(i = 0 ; i < NR(buffer) ; i++){
		if(buffer[i] < 60)
			buffer[i] = buffer[i] + '0' ;
		else
			buffer[i] = buffer[i] ;
	}
	printf("%s\n",buffer);
	return 0 ;
}

//加密
void  Crypt_Data(unsigned char buffer[])
{
	static int i = 0 ;
	for(i = 0 ; i < NR(buffer) ; i++)
	{
		buffer[i] << 1 ;   //将数组中每个数据按位左移1位
		buffer[i] += 2 ;   //左移完都加上2
		buffer[i] += 3 ;   //都加上2后都加上3
		buffer[i] >> 4 ;   //整体数据按位左移4位
	}
}

//解密
void  Uncrypt_Data(unsigned char buffer[])
{
	static int i = 0 ;
	for(i = 0 ; i < NR(buffer) ; i++)
	{
		buffer[i] >> 1 ;   //按照加密的格式将数组解密出来
		buffer[i] -= 2 ;
		buffer[i] -= 3 ;
		buffer[i] << 4 ;
	}
}
运行结果:
<img src="http://img.blog.csdn.net/20160107214053180?watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQv/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/Center" alt="" />


好像没和文件有什么关系对吧?哈哈,这就需要开动脑经啦。如果现在数据都在一个文件里,我要将数据读出来,然后进行加密操作存在一个数组里去,接着将原来的文件的数据清空,然后再写到这个文件里去。思路就是这样子:

首先你先创建一个文件写上我说的‘ I’,1,2,3,’W’,5,9,8,’D’,’W’,’A’,’D’,’A’,8,8,’D’,’A’,’S’,’D’,’A’,’J’这串数据

接着你就可以编程进行操作了: 

1 、 fopen :打开文件   

2、fread :从文件里读数据存到一个buffer数组里   

3、加密    

4、fclose :关闭文件描述符  

5、fopen以清空文件的模式接着打开这个文件。

6、lseek:偏移文件指针的位置     

7、fwrite 将加密完的数据写入这个文件

8、加密完成

9、fclose:关闭文件描述符

加密操作大概就是这个样子,解密也是一样的,不用多说了。

同时,以上的在linux下开发也可以用open , read , write , lseek ,close这些接口来代替。

但是UNIX接口对于块数据效率会比较高,对于小的数据效率就低了,所以,在文件不大的情况下(小于一块),建议还是使用标准C的接口。

今天就到这里,往后遇到什么问题,再做总结
     

时间: 2024-09-29 00:26:49

总结C语言在嵌入式开发中应用的知识点(文件数据的加密与解密)的相关文章

【详解】嵌入式开发中固件的烧录方式

版本:v1.2   Crifan Li 摘要 本文主要介绍了嵌入式开发过程中,将固件从PC端下载到开发板中的各种方式,主要包括NFS挂载,Nand Flash和Nor Flash,USB,RS232,网卡NIC等方式. 本文提供多种格式供: 在线阅读 HTML HTMLs PDF CHM TXT RTF WEBHELP 下载(7zip压缩包) HTML HTMLs PDF CHM TXT RTF WEBHELP HTML版本的在线地址为: http://www.crifan.com/files/

详细描述-嵌入式开发中的无法解释的现象

问题描述 嵌入式开发中的无法解释的现象 解决方案 为自己喝彩,求大神指导! 为什么第一种和第二种测试方法得到的结果不一样 解决方案二: 时间怎么测出来的,代码错了吧?

module-Apache Modulue开发中遇到的问题,关于openssl加密,总是报错,错在哪里?

问题描述 Apache Modulue开发中遇到的问题,关于openssl加密,总是报错,错在哪里? char *iaisession_getobmuid(request_rec *r, const char *encid, const unsigned char *enckey, unsigned char *iv, const char *mtiid) { unsigned char *obmuid = (unsigned char *) apr_pcalloc(r->pool, LENGT

驱动开发-求大神指导PCI驱动程序开发中,由于inf文件的问题提示“无法找到需要的软件因此无法安装”

问题描述 求大神指导PCI驱动程序开发中,由于inf文件的问题提示"无法找到需要的软件因此无法安装" 我这里是用的VS2005 driverstudio3.2 和DDk的框架对pci的驱动程序进行开发的 设备ID是对的 但是就是安装不了,找了好久问题但是不知道出在哪里 求大神指导!!!! inf文件如下: ; File Name: PCI9054.inf ; Install information file for PCI9054 Driver ; ; Generated by C D

asp.net开发中怎样去突破文件依赖缓存_实用技巧

在Web项目中可以使用Session,Application等来缓存数据,也可以使用Cache来缓存. 今天我们特别关注的是Cache缓存.Cache位于命名空间System.Web.Caching命名空间下,看到这里我们想到的是它在Web项目中使用. 说明:Cache 类不能在 ASP.NET 应用程序外使用.它是为在 ASP.NET 中用于为 Web 应用程序提供缓存而设计和测试的.在其他类型的应用程序(如控制台应用程序或 Windows 窗体应用程序)中,ASP.NET 缓存可能无法正常工

Android开发中Listview动态加载数据的方法示例

本文实例讲述了Android开发中Listview动态加载数据的方法.分享给大家供大家参考,具体如下: 最近在研究网络数据加载的问题,比如我有几百,甚至上千条数据,这些数据如果一次性全部加载到arraylist,然后再加载到Listview中.我们必然会去单独开线程来做,这样造成的结果就是会出现等待时间很长,用户体验非常不好.我的想法是动态加载数据,第一次加载十条,然后往下面滑动的时候再追加十条,再往下面滑动的时候再去追加,这样大大减少了用户等待的时间,同时给处理数据留下了时间.网上看到了这样一

c语言 单片机-keil4开发中遇到的一个很奇怪的问题。

问题描述 keil4开发中遇到的一个很奇怪的问题. void main(){ // uchar test_i; const uchar timp_button=0xfF; //this value is that the button havn't sticked uchar n_delay; //delay unsigned char j=0; unsigned char i; intend_initiation(); //调用时间设定程序 while(1){ if((hour==alarm_

详解iOS开发中UItableview控件的数据刷新功能的实现_IOS

实现UItableview控件数据刷新一.项目文件结构和plist文件 二.实现效果 1.说明:这是一个英雄展示界面,点击选中行,可以修改改行英雄的名称(完成数据刷新的操作). 运行界面: 点击选中行: 修改数据后自动刷新: 三.代码示例 数据模型部分: YYheros.h文件 复制代码 代码如下: // //  YYheros.h //  10-英雄展示(数据刷新) // //  Created by apple on 14-5-29. //  Copyright (c) 2014年 itca

Android应用开发中实现apk皮肤文件换肤的思路分析

在android的项目开发中,都会遇到后期功能拓展增强与主程序代码变更的现实矛盾,也就是程序的灵活度. 由于linux平台的安全机制,再加上dalvik的特殊机制,各种权限壁垒,使得开发一个灵活多变的程序,变得比较困难,不像pc平台下那么容易. 这里实际上可以借鉴传统软件中扩展程序的方法: 也就是插件的实现. 如目前所有的浏览器,比如我们使用的eclipse,以及很多优秀的软件,都使用了此种方式. 这样轻松实现了软件的功能扩展,而升级功能时只用更新对应插件, 而不是需要更新整个应用,降低了程序的