`
phyeas
  • 浏览: 164642 次
  • 性别: Icon_minigender_1
  • 来自: 珠海
社区版块
存档分类
最新评论

基于文本比较的搜索 - C语言实现(有注释)

阅读更多

周末在家把思路理了一边,先是用python实现了一下,但性能不太理想(100k/s),考虑到可能是由于动态语言的效率本身比较慢的原因,于是将算法改成c语言实现,最终的结果是:1.8M/s(硬件环境:Intel Core Duo 1.73G, 内存2G)。对于这个结果来说,我还是不太满意,比较现在动辄都是上G的数据。这样的效率太慢了,下面放上代码,各位讨论下是否还有优化的余地或者这个算法本身比较慢,或者这个方案是不可行的?

以下代码在Ubuntu9.04下编译并运行通过,测试数据是从je上随便搞了几篇文章。 gcc版本:4.3.3

 

 

#include <stdio.h>

#include <string.h>

#include <sys/types.h>   

#include <dirent.h>   

#include <sys/stat.h> 
#include <time.h> 
#include <stdlib.h> 

#define STEP 10

int count = 0;//文档个数
char* str = NULL;//一个大的字符串,存储所有文档的内容
int* ends;//文档的结束点集合
int ends_len = 0, ends_mem_len = 10;//文档结束点的内存参数(当前长度,内存长度)
int str_len = 0, str_mem_len = 10, str_unicode_len=0;//字符串的内存参数(字符串长度,字符串内存长度, 字符串unicode长度:即一个汉字占一个长度时的长度)
struct id_map{//一个文档在内存中的映射位置
	int id;//文档id
	int start;//字符串中的开始位置
	int end;//字符串中的结束位置
};
struct id_map * idmaps=NULL;//文档在内存中的映射地址
int idmaps_len = 0, idmaps_mem_len=0;//文档映射参数
//添加一个文档映射参数
void addIdMap(struct id_map map){
	if(idmaps==NULL){//如果数组还没有建立,就建立一个数组来进行存储
		idmaps = (struct id_map *)malloc(sizeof(struct id_map)*10);
	}
	//如果当前的文档数已经到达了上一次建立的内存长度,则扩展内存,步长为10
	if(idmaps_len==idmaps_mem_len){
		idmaps_mem_len += STEP;
		idmaps = (struct id_map *)realloc(idmaps, sizeof(struct id_map)*idmaps_mem_len);
		if(idmaps==NULL){
			printf("内存不足");
			return;
		}
	}
	*(idmaps+idmaps_len) = map;
	idmaps_len++;
}

//读取一个文本文件
char* readTextFile(char* path){
	char ch;//当前的字符
	FILE *fp;//文件指针
	int result;
	fp = fopen(path, "rb");
	if(fp!=NULL){//如果文档读取成功
		if(str==NULL){
			//初始化str,ends的内存。这两个的增长步长均为10
			ends = (int *)malloc( sizeof(int) * 10);
			str = (char *)malloc(10);
		}
		if(!str){
			printf("内存不足");
			fclose(fp);
			return NULL;
		}
		int unicode_ = 0;
		while((ch=fgetc(fp))!=EOF){//读取文件,一直读到最后,将内容放到str中。
			if(str_len == str_mem_len){
				str_mem_len += STEP;
				str = (char *)realloc(str, str_mem_len);
				if(str == NULL){
					printf("内存不足");
					fclose(fp);
					return NULL;
				}
			}
			if(unicode_ == 0){//如果上一个字符不是Unicode字符,则判断如果当前字符为unicode字符,则进入unicode计数。
				if(ch>=0 && ch<127){
					str_unicode_len++;
				}else{
					unicode_ = 1;
				}
			}else if(unicode_ == 1){
				unicode_ =2;
			}else if(unicode_ == 2){//按照utf-8编码进行计算,每个汉字占三个字符。
				unicode_ = 0;
				str_unicode_len++;
			}
			*(str+str_len)=ch;
			str_len++;
		}
		//记录结束点
		if(ends_len == ends_mem_len){
			ends_mem_len += STEP;
			ends = (int *)realloc(ends,  sizeof(int) * ends_mem_len);
			if(ends == NULL){
				printf("内存不足");
				fclose(fp);
				return NULL;
			}
		}
		//printf("---%d,%d,%d\n", ends_len,ends_mem_len,str_unicode_len);		
		//*(ends+ends_len) = str_unicode_len;
		*(ends+ends_len) = str_unicode_len;
		ends_len++;
		str = (char *)realloc(str, str_len);
		//*(str+len)='\0';
		fclose(fp);
		return str;
	}
	return NULL;
}

//读入一个文件夹内的所有文件
int init_search_dir(char *path)

{

	DIR *dir;   

      	struct dirent *s_dir;   

      	struct  stat file_stat;	

	char currfile[1024]={0};   

	int len = strlen(path);

	printf("%s\n",path);

	if( (dir=opendir(path)) == NULL)

	{   

		printf("opendir(path) error.\n");   

		return -1;   

	}

      	while((s_dir=readdir(dir))!=NULL)   

	{   

          	if((strcmp(s_dir->d_name,".")==0)||(strcmp(s_dir->d_name,"..")==0))   

			continue;

          	sprintf(currfile,"%s%s",path,s_dir->d_name);   

          	stat(currfile,&file_stat);   

          	if(S_ISDIR(file_stat.st_mode)){//如果是文件夹,则递归读取

              		init_search_dir(currfile);   

          	}else{

              		printf("%-32s\tOK",currfile);
			//设置一个文档与 str的映射,并读取文档的内容
			struct id_map map;
			map.id=atoi(s_dir->d_name);
			map.start = str_unicode_len;
			readTextFile(currfile);
			map.end = str_unicode_len;
			addIdMap(map);
			printf("\t%d\n", str_unicode_len);
		}

		count++;

      	}   

      	closedir(dir);
	ends = (int *)realloc(ends, sizeof(int) * ends_len);

	return 0;

}

//计算一个utf-8字符串的长度(汉字占一个长度)
int utf8_str_len(char* utf8_str){
	int length = 0, unicode_ = 0, i=0;
	for(;i<strlen(utf8_str);i++){
		if(unicode_ == 0){
			if(utf8_str[i]>=0 && utf8_str[i]<127){
				length++;
			}else{
				unicode_ = 1;
			}
		}else if(unicode_ == 1){
			unicode_ =2;
		}else if(unicode_ == 2){
			unicode_ = 0;
			length++;
		}
	}
	return length;
}

//查找该结束点是否存在(2分查找)
int find_ends(int num){
	if(num>ends[ends_len-1]||num<ends[0]){
		return -1;
	}
	int end = ends_len;
	int start = 0;
	int index=ends_len / 2;
	while(1){
		if(ends[index]==num){
			return index;
		}
		if(start == end || index == start || index == end){
			return -1;
		}
		if(ends[index] > num){
			end  = index;
		}else{
			start = index;
		}
		index = start + ((end-start) / 2);
	}
}

//主要函数。搜索所有文档中所有存在于该字符串相似的文档,算法出处及JAVA实现参见:http://www.blogjava.net/phyeas/archive/2009/02/15/254743.html
void search(char* key){
	int key_len = utf8_str_len(key);//计算key的长度
	int i=0, j=0, j_ = 0, i_ = 0;
	//char barr[key_len][str_unicode_len];
	char* barr[key_len];//
	//char narr[key_len][str_unicode_len];
	char* narr[key_len];
	//char darr[key_len][str_unicode_len];
	char* darr[key_len];
	//一个按照最大匹配度排序的文档序列。最大匹配度不可能大于key的长度+1,所以声明一个key_len+1长度的数组进行保存即可。数据格式类似:[[],[2,3],[5],[]]
	int* max_id_maps[key_len + 1];//该数组的第n个下标表示最大匹配度为n的文档有哪些
	int max_id_maps_lens[key_len + 1], max_id_maps_mem_lens[key_len + 1];
	int key_ascii_len = strlen(key);
	
	struct timeval tpstart,tpend;
	float  timeuse; 
	gettimeofday(&tpstart,NULL);
	//初始化三个数组。i_,j_表示当前的坐标,i,j表示当前左右的字符串中的字符位置
	for(i_=key_len-1, i=key_ascii_len-1;i>=0 && i_>=0;i--,i_--){
		barr[i_] = (char*) malloc(str_unicode_len);//动态申请内存是为了解决c语言函数内声明数组的长度有限制
		narr[i_] = (char*) malloc(str_unicode_len);
		darr[i_] = (char*) malloc(str_unicode_len);
		int is_left_ascii = key[i]<0 || key[i] >= 127 ? 0 : 1;
		for(j=str_len-1, j_=str_unicode_len-1;j>=0&&j_>=0;j--,j_--){
			int is_right_ascii = str[j] < 0 || str[j] >= 127 ? 0 : 1;
			barr[i_][j_] = 0;
			if(!is_left_ascii || !is_right_ascii){
				if(!is_left_ascii && !is_right_ascii){
					int k = 2, eq=1;
					for(;k>=0;k--){
						if(i-k >= 0 && j-k>=0 && key[i-k] != str[j-k]){
							eq = 0;
							break;
						}
					}
					barr[i_][j_] = eq;
				}else{
					barr[i_][j_] = 0;
				}
			}else{
				barr[i_][j_] = str[j] == key[i] || tolower(str[j]) == tolower(key[i]) ? 1 : 0;
			}

			darr[i_][j_] = 0;
			narr[i_][j_] = 0;
			int indexOfEnds = find_ends(j_);
			int n_right = 0, n_down = 0, n_rightdown = 0, d_right = 0, d_down = 0, d_rightdown = 0;
			if(indexOfEnds == -1 && j_!=str_unicode_len - 1){
				n_right = narr[i_][j_ + 1];
				d_right = darr[i_][j_ + 1];
			}
			if(i_!=key_len -1){
				n_down = narr[i_ + 1][j_];
				d_down = darr[i_ + 1][j_];
			}
			if(indexOfEnds == -1 && j_!=str_unicode_len - 1 && i_!=key_len -1){
				n_rightdown = narr[i_ + 1][j_ + 1];
				d_rightdown = darr[i_ + 1][j_ + 1];
			}
			n_rightdown += barr[i_][j_];
			narr[i_][j_] = n_right > n_down ? (n_right > n_rightdown ?  n_right : n_rightdown) : (n_down > n_rightdown ? n_down : n_rightdown);
			if(barr[i_][j_]){
				darr[i_][j_] = d_rightdown + 1;
			}else if(n_right >= n_down){
				darr[i_][j_] = d_right;
			}else{
				darr[i_][j_] = d_down + 1;
			}

			
			if(!is_right_ascii){
				j-=2;
			}
			//printf("%d\t", narr[i_][j_]);
		}
		//printf("\n");
		//max_id_maps[i] = (int *)malloc(sizeof(int)*10);
		max_id_maps_mem_lens[i_] = 0;
		max_id_maps_lens[i_] = 0;
		
		if(!is_left_ascii){
			i-=2;
		}
	}

	//max_id_maps[key_len] = (int *)malloc(sizeof(int)*10);
	max_id_maps_mem_lens[key_len] = 0;
	max_id_maps_lens[key_len] = 0;
	int k=0;
	//计算最大匹配度和最优匹配路径长度。并将其放到如到max_id_maps中
	for(k=0;k<idmaps_len;k++){
		int end=idmaps[k].end, j=idmaps[k].start, end_i = key_len, max_ = 0, min_ = -1;
		while(j<end){
			int temp_end_i = -1;
			for(i=0;i<end_i;i++){
				if(barr[i][j]){
					if(temp_end_i==-1){
						temp_end_i = i;
					}
					if(narr[i][j] > max_){
						max_ = narr[i][j];
					}
					if(min_ == -1 || darr[i][j] < min_){
						min_ = darr[i][j];
					}
				}
			}
			if(temp_end_i != -1){
				end_i = temp_end_i;
			}
			j++;
		}
		if(max_ != 0){
			if(max_id_maps_mem_lens[max_] == 0){
				max_id_maps[max_] = (int *)malloc(sizeof(int)*10);
				max_id_maps_mem_lens[max_] = 10;
			}else if(max_id_maps_mem_lens[max_] == max_id_maps_lens[max_]){
				max_id_maps_mem_lens[max_] += STEP;
				max_id_maps[max_] = (int *)realloc(max_id_maps[max_], sizeof(int)*max_id_maps_mem_lens[max_]);
			}
			*(max_id_maps[max_] + max_id_maps_lens[max_]) = idmaps[k].id;
			max_id_maps_lens[max_]++;
		}
	}
	//-----------------计时,计算性能
	gettimeofday(&tpend,NULL); 
	timeuse=1000000*(tpend.tv_sec-tpstart.tv_sec)+tpend.tv_usec-tpstart.tv_usec; 
	timeuse/=1000000; 
	printf("Used Time:%f\n",timeuse); 
	for(i=0;i<=key_len;i++){
		printf("%d -- ",i);
		for(j=0;j<max_id_maps_lens[i];j++){
			printf("%d\t", max_id_maps[i][j]);
		}
		printf("\n");
	}
	//--------------计时结束
	//释放在这个函数中申请的动态内存。
	for(i=0;i<=key_len;i++){
		if(max_id_maps_mem_lens[i]>0){
			//printf("%d,",max_id_maps_mem_lens[i]);
			free(max_id_maps[i]);
		}
		if(i!=key_len){
			free(barr[i]);
			free(narr[i]);
			free(darr[i]);
		}
	}
	//testPrint(&narr, key_len, str_unicode_len);
}
//释放程序中申请的动态内存
void freeMemory(){
	free(ends);
	free(idmaps);
	free(str);
}


int main(){	
	init_search_dir("/home/phyeas/test/");
	search("Java云计算");
	//search("BCXCADFESBABCACA");
	//init_search_dir("/home/phyeas/test/test2/");
	//int i=0;
	freeMemory();
	return 0;
}
 
分享到:
评论
20 楼 deepfuture 2010-03-07  
为什么不用perl,这是perl的强项
19 楼 nicewqx 2009-09-19  
谢谢楼主的这篇帖子,学到了好多.
18 楼 whaosoft 2009-09-16  
谢谢,受教啦
17 楼 phyeas 2009-09-07  
zhwang 写道
看了楼主的博客,你的这种方法不可行,太暴力,最基本的I/O瓶颈就突破不了

给你算笔账,假设你服器上有6块硬盘做raid0,吞吐量按400M/s算,你的文档库有20G文档(已经够少了吧),那么检索一遍的时间是50s,你认为用户会有50s的耐心等待检索结果吗,而且代价是服务器满I/O负载

如果要是1000个用户并发检索呢?哦,的确可以将检索词排队,下次检索一次取出,但每次检索的时间都是50s,这就说明最差情况下用户的等待时间是100s

可以将检索分布式,但代价是为了保证用户能在1s内检索到数据,数据需要分布到50台机器上,20G数据分配到50台机器上,你自己想一下吧


呵呵,是有点很暴力的感觉,只是我想说,这里面没有I/O的事。所有的操作都是在内存中完成。
PS:我已经放弃这种算法了。确实不可行。
16 楼 zhwang 2009-09-07  
看了楼主的博客,你的这种方法不可行,太暴力,最基本的I/O瓶颈就突破不了

给你算笔账,假设你服器上有6块硬盘做raid0,吞吐量按400M/s算,你的文档库有20G文档(已经够少了吧),那么检索一遍的时间是50s,你认为用户会有50s的耐心等待检索结果吗,而且代价是服务器满I/O负载

如果要是1000个用户并发检索呢?哦,的确可以将检索词排队,下次检索一次取出,但每次检索的时间都是50s,这就说明最差情况下用户的等待时间是100s

可以将检索分布式,但代价是为了保证用户能在1s内检索到数据,数据需要分布到50台机器上,20G数据分配到50台机器上,你自己想一下吧

15 楼 phyeas 2009-09-01  
Joo 写道
能不能麻烦楼主把你程序的应用场景交代一下?上来就贴代码,云里雾里

http://phyeas.iteye.com/blog/452733
14 楼 Joo 2009-08-31  
能不能麻烦楼主把你程序的应用场景交代一下?上来就贴代码,云里雾里
13 楼 phyeas 2009-08-31  
Magicloud 写道
diff3不是有個專門算法的么,我記得還看過論文。不知是否可以給樓主參考。

不用了,我想过了,用diff的算法怎么也不比不上分词索引。不过能提供文档的话还是谢谢
12 楼 Magicloud 2009-08-31  
diff3不是有個專門算法的么,我記得還看過論文。不知是否可以給樓主參考。
11 楼 phoenixup 2009-08-31  
这个要MARK一下,最近在做海量数据报文解析~~用python的确有性能不足~
10 楼 phyeas 2009-08-26  
yysolo 写道
从优化的角度:
1. 开一个大的buffer, 然后用fread
2. 多线程,一个读,多个处理。降低IO的时间。如果是目录,必须多线程
3. 在算法中降低分配内存的次数,使用1中的buffer,基本做到zero-copy
4. GCC -O2
5. 换文件格式,换文件系统
....
表层的优化完成后,通过工具找热点

看了下算法并不耗时,LZ多了解一下C,以LZ的机器,单线程60M/s还是有可能的。
1,2,3弄好了,提高IO的效率,考虑换块磁盘或者换架构。最后再考虑5

LZ可以下一个Beyond compare体会专业级的diff工具的性能

前期加载文件那部分不在计算耗时的范围内,仅计算search函数内的耗时。所以你说的这些好像都用不上,不过还是谢谢你的回复
PS:Beyond compare体验过了,它是单个文件进行比较,虽然有文件夹比较,但文件夹比较只是表面(修改日期,创建日期,长度)比较,并不比较文件内容,然后当你确实要比较内容时还要进行一次点击
9 楼 yysolo 2009-08-25  
从优化的角度:
1. 开一个大的buffer, 然后用fread
2. 多线程,一个读,多个处理。降低IO的时间。如果是目录,必须多线程
3. 在算法中降低分配内存的次数,使用1中的buffer,基本做到zero-copy
4. GCC -O2
5. 换文件格式,换文件系统
....
表层的优化完成后,通过工具找热点

看了下算法并不耗时,LZ多了解一下C,以LZ的机器,单线程60M/s还是有可能的。
1,2,3弄好了,提高IO的效率,考虑换块磁盘或者换架构。最后再考虑5

LZ可以下一个Beyond compare体会专业级的diff工具的性能
8 楼 phyeas 2009-08-25  
呵呵,这个还不算正式的代码,只是为了测试这个算法的性能。谢谢LS
7 楼 cyberblue 2009-08-25  
而且ASCII码在127之後还有这堆东西,楼主小心

6 楼 cyberblue 2009-08-25  
while((ch=fgetc(fp))!=EOF){//读取文件,一直读到最后,将内容放到str中。   
            if(str_len == str_mem_len){   
                str_mem_len += STEP;   
                str = (char *)realloc(str, str_mem_len);   
                if(str == NULL){   
                    printf("内存不足");   
                    fclose(fp);   
                    return NULL;   
                }   
            }   


建议直接用low-level IO里的read,一个字节一个字节地处理肯定不大好
5 楼 phyeas 2009-08-24  
night_stalker 写道
根据 lz 的 blog,找到第一篇“基于文本比较的搜索1”,再找到来源是“文本比较算法剖析”,作者竟然说不打算介绍整个思路 …… 幸好评论中有人给出了来源 An Algorithm for Differrential File Comparison,再搜,终于明白了 ……

http://www.cs.dartmouth.edu/~doug/diff.ps

原来实现的不是 grep,是 diff …… 再问下和 diff 比较起来,效率怎么样 ……

刚在网上搜了一下diff的原理,不知道这篇(http://www.avatar.se/molbioinfo2001/dynprog/dynamic.html)讲的是不是,如果是的话我认为原理上时一样的。只是diff要求出的是最优匹配路径,我写的程序只求了最大匹配度。空间复杂度应该都是一样的,需要一个m*n的数组存放计算结果。
4 楼 night_stalker 2009-08-24  
根据 lz 的 blog,找到第一篇“基于文本比较的搜索1”,再找到来源是“文本比较算法剖析”,作者竟然说不打算介绍整个思路 …… 幸好评论中有人给出了来源 An Algorithm for Differrential File Comparison,再搜,终于明白了 ……

http://www.cs.dartmouth.edu/~doug/diff.ps

原来实现的不是 grep,是 diff …… 再问下和 diff 比较起来,效率怎么样 ……
3 楼 Magicloud 2009-08-24  
你的意思是若干份文档,根据关键字找到那文档?
考虑匹配度为全匹配的话,就是 `grep keyword *.doc`?
2 楼 phyeas 2009-08-24  
night_stalker 写道
grep 也是 C 写的,这个和 grep 比哪个快? ……

python -c "print('hi\njava')" | grep 'java'


这个好像和grep要实现的不是同一个功能。我想做的是根据关键字的匹配度搜索文档,grep做的好像是根据表达式搜索文档。不知道我说的对不对,请果果指教
1 楼 night_stalker 2009-08-24  
grep 也是 C 写的,这个和 grep 比哪个快? ……

python -c "print('hi\njava')" | grep 'java'

相关推荐

    C语言开发----c语言黑白棋ai游戏源码.rar

    3. **用户界面**:虽然C语言本身并不直接支持图形用户界面(GUI),但可以使用如ncurses库来创建基于文本的界面,或者通过调用其他库如SDL、Allegro等实现图形化界面。 4. **文件操作**:可能包含读取和保存游戏...

    C语言开发----c语言通讯录管理系统源码.rar

    本篇文章将深入探讨一个基于C语言开发的通讯录管理系统源码,帮助读者理解其设计原理和实现细节。 通讯录管理系统是一个典型的文本界面应用,它通常包括添加联系人、删除联系人、查找联系人和显示所有联系人等功能...

    语法分析器-C语言

    本文将深入探讨由标题和描述所提及的“语法分析器-C语言”实验程序,这是一个基于C语言实现的简单语法分析器。 #### 实验程序概览 该程序通过定义一系列函数来处理特定的语法结构,包括`F()`、`S()`、`T()`、`G()`...

    ACLlib库 ---- C语言图形库

    ACLlib库是一个基于C语言的图形库,专为教学目的设计,由浙江大学开发并开源。这个库提供了丰富的功能,使得开发者能够更容易地在C语言环境下进行图形编程。下面将详细介绍ACLlib库的主要特点、功能以及如何使用。 ...

    精选_基于C语言实现的24游戏-高级语言源程序注释部分的处理-单项选择题标准化考试系统_源码打包

    在本资源中,我们关注的是一个使用C语言编写的24点游戏,以及与之相关的高级语言源程序注释处理和单项选择题标准化考试系统的实现。C语言是一种强大的、广泛应用于系统开发、软件工程和后端编程的低级编程语言。它的...

    基于c语言实现Linux屏幕取词翻译源码.7z

    基于c语言实现Linux屏幕取词翻译源码.7z基于c语言实现Linux屏幕取词翻译源码.7z基于c语言实现Linux屏幕取词翻译源码.7z基于c语言实现Linux屏幕取词翻译源码.7z基于c语言实现Linux屏幕取词翻译源码.7z基于c语言实现...

    C语言实现小游戏-挑战2048-附详细注释.zip

    在本项目中,我们关注的是一个使用C语言实现的小游戏——2048。2048是一款基于数字合并的益智游戏,玩家通过上下左右移动屏幕上的数字方块,当相同数字的方块相遇时会合并成它们的和,目标是创造出2048这个数字。...

    Linux作业基于C语言编写的文本编辑器系统源码+超详细注释.zip

    Linux作业基于C语言编写的文本编辑器系统源码+超详细注释.zipLinux作业基于C语言编写的文本编辑器系统源码+超详细注释.zipLinux作业基于C语言编写的文本编辑器系统源码+超详细注释.zipLinux作业基于C语言编写的文本...

    贪食蛇纯C语言版(源代码带注释)

    5. **屏幕输出与更新**:C语言中的`printf`函数用于输出文本到控制台,实现游戏画面的绘制。游戏过程中,屏幕需要不断更新以反映蛇的移动和吃掉食物后的情况。这涉及到对已打印内容的清除和新内容的覆盖,通常通过...

    C语言实现LZ77压缩算法

    C语言是一种通用的、面向过程的编程语言,常用于系统级编程和嵌入式开发,包括实现各种压缩算法。 C语言实现LZ77压缩算法的关键步骤包括以下几点: 1. **滑动窗口**:LZ77算法基于一个固定大小的滑动窗口,通常是...

    LDA原始论文C语言代码及注释vs2013工程

    通过理解并实践这个LDA的C语言实现,不仅可以掌握LDA模型的基本原理,还能深入学习C语言编程技巧,尤其是在处理大规模文本数据时的性能优化。此外,这个代码资源也可作为进一步研究其他主题模型或者开发自定义NLP...

    基于C语言实现dos界面简单的机房收费管理系统.zip

    DOS(Disk Operating System)是一个早期的操作系统,其用户界面主要是基于文本的命令行。在DOS环境中,程序通常通过接收用户输入的命令来执行相应的操作。因此,开发一个DOS界面的系统,需要我们编写处理键盘输入和...

    基于C语言实现学生考勤系统

    《基于C语言实现学生考勤系统》 C语言是一种强大的编程语言,被广泛应用于系统开发、软件构建以及各种嵌入式系统中。本项目“基于C语言实现学生考勤系统”是一个典型的应用实例,旨在利用C语言的强大功能,构建一个...

    C语言 实现 车辆罚单管理系统

    【车辆罚单管理系统】是一个基于C语言开发的应用程序,主要用于管理交通违章罚款的相关信息。C语言作为基础的编程语言,以其高效、灵活和跨平台的特性,被广泛应用于各种系统开发,包括小型的命令行应用如本例中的...

    C语言实现控制台学生信息管理系统.zip

    在本项目中,"C语言实现控制台学生信息管理系统.zip" 是一个使用C语言编写的命令行应用程序,用于管理学生信息。这个系统可能包括添加、删除、查询和更新学生记录等基本功能,是学习C语言面向结构化编程和数据管理的...

    huffman树的c语言实现

    - 接着,使用优先队列(通常是基于最小堆实现)来合并两个频率最小的节点,形成一个新的内部节点。新节点的频率是其子节点的频率之和,并将新节点插入到队列中。 - 这个过程不断重复,直到队列中只剩下一个节点,...

    C语言-- 图书管理系统.zip

    【标题】"C语言-- 图书管理系统.zip" 是一个基于C语言实现的图书管理系统的源代码包。这个项目旨在提供一个简单的平台,用于图书馆或个人书籍的管理,包括书籍的添加、删除、查询和借阅等功能。通过学习和分析这个...

    维基尼亚1-4.rar(维基尼亚算法的实现,C语言实现)

    C语言实现维基尼亚算法通常涉及以下几个关键步骤: 1. **输入处理**:程序需要获取用户输入的明文、密文和密钥。在"维基尼亚1-4.cpp"文件中,这部分可能包含了`fgets()`函数用于读取用户输入。 2. **预处理**:将...

    小超市商品管理系统C语言.rar

    本项目是一款基于C语言实现的小超市商品管理系统,旨在帮助小型超市进行日常的商品库存管理、销售记录以及简单的数据分析。通过学习和理解这个系统,我们可以深入理解C语言在实际项目开发中的应用,同时也可以掌握...

    记事本源程序(C语言)++注释

    本文档将对一个基于C语言实现的简易记事本程序进行分析。该程序不仅包括了基本的文件操作功能(如新建、加载、删除等),还通过自定义函数实现了屏幕光标的移动及位置获取等功能。接下来,我们将深入探讨该程序的...

Global site tag (gtag.js) - Google Analytics