Chinaunix首页 | 论坛 | 博客
  • 博客访问: 299842
  • 博文数量: 70
  • 博客积分: 1990
  • 博客等级: 上尉
  • 技术积分: 686
  • 用 户 组: 普通用户
  • 注册时间: 2010-05-02 08:52
文章分类

全部博文(70)

文章存档

2015年(2)

2014年(9)

2013年(2)

2012年(20)

2011年(1)

2010年(36)

分类: C/C++

2012-03-31 22:23:03

源码1C和指针上的原始版本

/*

*filename:collocate characters.c

*description:字符重排列

*输入的第一行是一串列标号,串的最后以一个负数结尾。

*这些列标号成对出现,说明需要打印的输入列的范围。

*例如 0 3 10 12 -1表示第0列到第3列,第10列到第12列的内容将被打印出来。

*/

#include

#include

#include

 

#define MAX_COLS  20     //所能处理的最大列数

#define MAX_INPUT 1000   //每个输入行的最大长度

 

int read_column_numbers(int columns[],int max);

void rearrange(char *output, char const *input,

                                                        int n_columns, int const columns[]);

                                                       

int main(void)

{

       int n_columns;                                   //进行处理的列标号

       int columns[MAX_COLS];    //需要处理的列数

       char input[MAX_INPUT];     //容纳输入行的数组

       char output[MAX_INPUT];   //容纳输出行的数组

      

       /*读取该串列标号*/

       n_columns = read_column_numbers(columns,MAX_COLS);

      

       /*读取、处理和打印剩余的输入行*/

       while(gets(input) != NULL)

       {

              printf("Original input : %s\n",input);

              rearrange(output,input,n_columns,columns);

              printf("Rearranged line: %s\n",output);

       }

       return EXIT_SUCCESS;

}

 

/*读取列标号,如果超出规定范围则不予理会*/

 

int read_column_numbers(int columns[],int max)

{

       int num = 0;

       int ch;

      

       /*取得列标号,如果所读取的数小于0则停止*/

      

       while(num < max && scanf("%d",&columns[num]) == 1

                            && columns[num] >= 0)

                                   num+=1;

                                  

      

       /*确认已经读取的标号为偶数个,因为他们是以对的形式出现的*/

      

       if(num % 2 != 0)

       {

                     puts("Last column number is not paired.");

                     exit(EXIT_FAILURE);

       }

      

      

       /*丢弃该行中包含最后一个数字的那部分内容*/

      

       while((ch = getchar()) != EOF && ch != '\n')   //自注:这里是去除最后的一个负数

              ;

             

       return num;

}

 

 

/*处理输入行,将指定列的字符连接在一起,输入行以NU 结尾*/

 

void rearrange(char *output, char const *input,

                            int n_columns, int const columns[])

{

       int col;                                       //columns数组的下标

       int output_col;               //输出列计数器

       int len;                                        //输入行的长度

      

       len = strlen(input);

       output_col = 0;

      

      

       /*处理每对列标号*/

      

       for(col = 0; col < n_columns; col += 2)

       {

              int nchars = columns[col +1] - columns[col] +1;

             

             

              /*如果输入行结束或输出行数组已满,就结束任务*/

             

              if( columns[col] >= len ||

                            output_col == MAX_INPUT - 1)

                            break;

                           

             

              /*如果输出行数组空间不够,只复制可以容纳的数据*/

             

              if( output_col + nchars > MAX_INPUT - 1)

                            nchars = MAX_INPUT - output_col - 1;

                           

              /*复制相关数据*/

              strncpy(output + output_col,input + columns[col],

                                          nchars);

              output_col += nchars;

             

       }

      

       output[output_col] = '\0';

}

 

运行效果:
 

源码2:我的改进版本

/*

*filename:my_collocate characters.c

*description:字符重排列

*输入的第一行是一串列标号,串的最后以一个负数结尾。

*这些列标号成对出现,说明需要打印的输入列的范围。

*例如 0 3 10 12 -1表示第0列到第3列,第10列到第12列的内容将被打印出来。

*/

#include

#include

#include

 

#define MAX_COLS  20     //所能处理的最大列数

#define MAX_INPUT 1000   //每个输入行的最大长度

 

int read_column_numbers(int columns[],int max);

void rearrange(char *output, char const *input,

                                                        int n_columns, int const columns[]);

                                                       

int main(void)

{

       int n_columns;                     //列标号个数

       int columns[MAX_COLS];    //列标号存放数组

       char input[MAX_INPUT];     //输入缓存

       char output[MAX_INPUT];   //输出缓存

      

       /*读取该串列标号*/

       n_columns = read_column_numbers(columns,MAX_COLS);

      

       /*读取、处理和打印剩余的输入行*/

       //while(gets(input) != NULL)    //自注:使用fgets()输入长度无法控制,容易溢出

       while(printf("Please input characters:") && fgets(input,MAX_INPUT,stdin) != NULL)

       {                          //fgets()获取一行时会将换行符'\n'存储于输入数组

 

              /*除去fgets()最后的换行符'\n'*/

              char * tmp = strrchr(input,'\n');

              if(tmp)

                     *tmp = '\0';

                    

              printf("Original input : %s\n",input);

              rearrange(output,input,n_columns,columns);

              printf("Rearranged line: %s\n",output);

       }

       return EXIT_SUCCESS;

}

 

/*读取列标号,如果超出规定范围则不予理会*/

 

int read_column_numbers(int columns[],int max)

{

       int num = 0;

       int ch;

      

       printf("Please input series of column number : ");

      

       /*取得列标号,如果所读取的数小于0则停止*/    

       //while(num < max && scanf("%d",&columns[num]) == 1

       while(num < max && scanf("%d",columns + num) == 1

                            && columns[num] >= 0)

                                   num+=1;              

                                  

#if 0

       /*确认已经读取的标号为偶数个,因为他们是以对的形式出现的*/

      

       if(num % 2 != 0)

       {

                     puts("Last column number is not paired.");

                     exit(EXIT_FAILURE);

       }

#endif    

      

       /*丢弃该行中包含最后一个数字的那部分内容*/

      

       while((ch = getchar()) != EOF && ch != '\n')   //自注:这里是去除最后的一个负数

              ;

             

       return num;

}

 

 

/*处理输入行,将指定列的字符连接在一起,输出行以NU 结尾*/

 

void rearrange(char *output, char const *input,

                            int n_columns, int const columns[])

{

       int col;                                       //columns数组的下标

       int output_col;               //输出列计数器

       int len;                                        //输入行的长度

      

       len = strlen(input);

       output_col = 0;

      

       /*打印参数*/

       printf("n_columns:%d\n",n_columns);

       printf("len:%d\n",len);

       for(col = 0; col < n_columns; col++)

              printf("%d\t",columns[col]);

      

       /*处理每对列标号*/    

             

       for(col = 0; col < n_columns; col += 2)

       {

              int nchars ;

             

             

              /*自注:如果是最后单独的一个列标号,就复制该列标号到数组尾之间的字符,如果不是就按对处理*/

              if(col == (n_columns - 1) && col % 2 == 0)

              {

                     printf("TEST\n");

                     nchars = len - columns[n_columns] +1;                                                       

              }

              else

                     nchars = columns[col +1] - columns[col] +1;

              //printf("nchars=%d\n",nchars);

             

              /*如果输入行结束或输出行数组已满,就结束任务*/

             

              if( columns[col] >= len ||

                            output_col == MAX_INPUT - 1 ||

                            columns[col] < columns[col-1] )

              {

                     printf("exit\n");      

                     break;

              }

             

             

              /*如果输出行数组空间不够,只复制可以容纳的数据*/

             

              if( output_col + nchars > MAX_INPUT - 1)

                            nchars = MAX_INPUT - output_col - 1;

             

             

              /*复制相关数据*/

              strncpy(output + output_col,input + columns[col],

                                   nchars);                             

              output_col += nchars;

       }

             

      

      

       output[output_col] = '\0';

}

运行效果:
 
 
 
阅读(2095) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~