一个实用的从文本文件读取数据进行排序的程序

程序说明

  这是一个十分可靠的程序,这个程序的查错能力非常强悍。程序包含了文件操作,归并排序和字符串输入等多种技术。

  程序的功能是从外部读取一个包括int型数据的文本文件,然后将它保存到内部临时数组,对数组进行排序后,以文本形式输出到指定的文件上。因为是int类型的数据,没有很严重的损失精度的问题。

  正常运行要求:

  包括数据的源文件内不能包括其他任何除数字和空白字符(空格,制表符,换行符)之外的任何字符,源文件最开始必须是数字字符,要保证源文件的数据计数正确。同时保证文件名有效。

完整代码

警告:版权所有,谨供参考!

#include <stdio.h>
#include <stdlib.h>
#include <conio.h>

/*=============================
    制作于:Aug 16, 2016
    by QQ:1729403632
===============================*/

#define ST 64 //字符串大小

void mergesort(int *, int);
void _mergesort(int *, int, int, int *);
void merge(int *, int, int, int, int *);
char * s_gets(char *, int);

int main(int argc, char * argv[]){
    FILE * sor, * dest; //sor源文件 dest目标文件
    int * ptr;//临时数组
    int i, n; //n表示元素个数
    char fname[ST]; //临时存储字符串

    printf("请输入元素个数:");
    while( scanf("%d", &n) != 1 || n <= 0 ){
        printf("输入错误,请重新输入!\n");
        while(getchar() != ‘\n‘)
            continue;
    }
    while(getchar() != ‘\n‘)
            continue;

    ptr = (int *)malloc( (size_t)n * sizeof(int) ); //申请动态数组//////
    if(ptr == NULL){
        fprintf(stderr, "FAIL TO ASK FOR MEMORY SPACE\n");
        exit(EXIT_FAILURE);
    }

    printf("请输入原文件名:");
    if( s_gets(fname, ST) == NULL ){
        fprintf(stderr, "Fail to get a string\n");
        exit(EXIT_FAILURE);
    }

    sor = fopen(fname, "r"); //打开包含数据的源文件
    if(sor == NULL){
        fprintf(stderr, "Fail to open the source file\n");
        exit(EXIT_FAILURE);
    }

    for(i = 0; i < n; i++) //获取数据到动态数组
        if( fscanf(sor, "%d", &ptr[i]) != 1 ){
            fprintf(stderr, "Fail to get the data\n");
            exit(EXIT_FAILURE);
        }

    mergesort(ptr, n); //排序

    printf("请输入要保存数据的文件名:");
    if( s_gets(fname, ST) == NULL ){
        fprintf(stderr, "Fail to get a string\n");
        exit(EXIT_FAILURE);
    }

    dest = fopen(fname, "w"); //打开目标文件
    if(dest == NULL){
        fprintf(stderr, "Fail to open the destination file\n");
        exit(EXIT_FAILURE);
    }

    for(i = 0; i < n; i++){ //输出数据到目标文件
        if( fprintf(dest, "%d\t", ptr[i]) < 0 ){
            fprintf(stderr, "Fail to save the data\n");
            exit(EXIT_FAILURE);
        }
        if( ((i + 1) % 10) == 0){ //如果写满10个就换行
            if( fprintf(dest, "\n") < 0 ){
                fprintf(stderr, "Fail to save the data\n");
                exit(EXIT_FAILURE);
            }
        }
    }

    if( fclose(sor) != 0 ){ //关闭源文件
        fprintf(stderr, "Fail to close the source file\n");
        exit(EXIT_FAILURE);
    }
    if( fclose(dest) != 0 ){ //关闭目标文件
        fprintf(stderr, "Fail to close the destination file\n");
        exit(EXIT_FAILURE);
    }
    free(ptr); //释放内存

    printf("成功完成!\n请按任意键继续^ ^\b\b");

    getch();
    return 0;
}

void mergesort(int * ar, int size){
    if(size > 0){
        int * temp;
        temp = (int *)malloc( (size_t)size * sizeof(int) ); /////
        if(temp == NULL){
            fprintf(stderr, "Fail to ask for MEMORY SPACE\n");
            exit(EXIT_FAILURE);
        }
        _mergesort(ar, 0, size - 1, temp); //归并排序
        free(temp);
    }
}

void _mergesort(int * ar, int start, int end, int * temp){
    if(start < end){
        int mid = (start + end) / 2;
        _mergesort(ar, start, mid, temp);    //左子数组排序
        _mergesort(ar, mid + 1, end, temp);    //右子数组排序
        merge(ar, start, mid, end, temp);    //合并子数组
    }
}

void merge(int * ar, int p, int q, int r, int * temp){
    int i = p, j = q + 1, k = 0;
    while(i <= q && j <= r){
        if(ar[i] < ar[j])
            temp[k++] = ar[i++];
        else
            temp[k++] = ar[j++];
    }
    while(i <= q)    //如果序列[i...q]存在,追加
        temp[k++] = ar[i++];
    while(j <= r)    //如果序列[j...r]存在,追加
        temp[k++] = ar[j++];

    for(k = 0; k <= (r - p); k++)
        ar[p + k] = temp[k];
}

char * s_gets(char * st, int size){
    char * re;
    int i = 0;

    re = fgets(st, size, stdin);
    if(re){
        while(st[i] != ‘\n‘ && st[i] != ‘\0‘) //如果没有到输入字符串结束
            i++;    //递增
        if(st[i] == ‘\n‘) //如果字符串最后一个字符是‘\n‘
            st[i] = ‘\0‘; //把它变成‘\0‘
        else //否则缓冲区内还有一部分超出读取范围的字符没有被读取
            while(getchar() != ‘\n‘) //把这些字符读取完(清空缓冲区)
                continue;
    }

    return re;
}

main.c

运行结果

data.txt:

obj.txt:

时间: 2024-10-06 12:38:16

一个实用的从文本文件读取数据进行排序的程序的相关文章

C++学习笔记之由文本文件读取数据到vector模板建立的二维数组 并存储为新的文本文件

阅读本文可首先参考: C++学习笔记之输入.输出和文件 测试数据: 1 /*读取txt文件到二维数组*/ 2 #include <iostream> 3 #include <fstream> 4 #include <vector> 5 #include <string> 6 7 using namespace std; 8 9 typedef vector< vector<int> > D2array; //二维数组 10 typed

下单快发货慢:一个 JOIN SQL 引起 SqlClient 读取数据慢的奇特问题

最近遇到一个非常奇特的问题,在一个 ASP.NET Core 项目中从 SQL Server 2008 R2 中查询获取 100 条记录竟然耗时 10 多秒,如果是查询本身慢,那到不是什么奇特的问题. 说它非常奇特是因为耗时主要发生在 SqlDataReader 读取数据时 2019-04-04 21:31:58.546 [Information] Executed DbCommand ("2,656"ms) ... 2019-04-04 21:32:10.690 [Debug] A

视图:根据条件读取数据以及排序

1:声明视图存储DataTbale数据. DataView Dv= DataTable.dt; 2:根据视图内列名称,传入条件进行读取数据.Dv.RowFilter = "ID='" + String + "'"; 3:根据视图内某个字段进行排序. Dv.Sort="Id asc" 4:绑定数据 XX.Datasource=Dv. XX.DATABind();  

从Hadoop URL中读取数据

要从Hadoop文件系统中读取文件,一个最简单的方法是使用java.net.URL对象来打开一个数据流,从而从中读取数据.一般的格式如下: 1.      InputStream in = null; 2.  try { 3.       in = new URL("hdfs://host/path").openStream(); 4.       // process in 5.  } finally { 6.       IOUtils.closeStream(in); 7.  }

Hadoop学习笔记0003——从Hadoop URL读取数据

Hadoop学习笔记0003--从Hadoop URL读取数据 从HadoopURL读取数据   要从Hadoop文件系统中读取文件,一个最简单的方法是使用java.net.URL对象来打开一个数据流,从而从中读取数据.一般的格式如下: InputStream in = null; try { in = new URL("hdfs://host/path").openStream(); // process in } finally { IOUtils.closeStream(in);

已知s.txt文件中有一个这样的字符串 请编写程序读取数据内容,把数据排序后写入 ss.txt文件

package cn.idcast5; import java.io.BufferedReader; import java.io.BufferedWriter; import java.io.FileNotFoundException; import java.io.FileReader; import java.io.FileWriter; import java.io.IOException; import java.util.Arrays; /* * 需求:已知s.txt文件中有一个这样

ACCESS(VBA)上的一个小项目 —— 2、读取ACCESS表数据到TreeView和ListView

有人问我能不能做一个程序的时候,我第一反应都说“能”. --这次在ACCESS中,借助TreeView和ListView做了一个数据联动的模型. 简析过程: 1)从网上找了一份TreeView学习教程<三小时学会树控件>,了解了TreeView的建立以及节点的使用方法: 2)把数据表中的某列按一定规则生成的数据再按一种规则解析成TreeView中的树结构(VPPS): 3)通过遍历TreeView中节点的折叠状态,通过上述一定规则生成的数据(唯一性)把数据表中的内容读取到ListView中:

java基础-输入流-读取文本文件中数据至字符串数组

简介:如题 import java.io.FileInputStream; /** * @author czchina * */ public class TestStream { public static void main(String[] args) { // TODO Auto-generated method stub //声明输入流的引用 FileInputStream fls = null; //声明输出流的引用 FileOutputStream fos =null; try{

C++中文件的读取操作,如何读取多行数据,如何一个一个的读取数据

http://blog.csdn.net/sunhero2010/article/details/50980591 练习8.1:编写函数.接受一个istream&参数,返回值类型也是istream&.此函数必须从给定流中读取数据,直至遇到文件结束标识时停止. [cpp] view plain copy #include <iostream> #include <stdexcept> using std::istream; using std::cin; using