对文本行按特定字段排序(前N个字符或后N个字符),TCPL 练习5-17
The C programming language 的关于文本行排序的问题有很多种要求的方式,在对每行的字段排序方面,最简单的是例如对前N个字符或者末位N个字符进行排序,更高一点的要求是,对特殊符号标识的字段排序,例如,对逗号前的字段进行排序等,标识符号要尽可能地自定义,这里的程序实现了前者,即依据命令行参数N,对每行的前N或者后N个字符排序,当然,也实现了和-f(忽略大小写),-d(只对空格数字字母排序),-r(逆序)的组合使用,因此,基本完成了如书上所说,可以进行索引排序。但是哪怕是这一个简单的实现,也需要进一步考虑的问题是:能否对文本行中间N个字符进行排序?某些特殊场合可能会提出这样的要求。
代码如下:
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <ctype.h>
/* 编写文本行排序函数,扩充其功能,使得函数能对特定字段排序,本例实现了对前leng个字符和末位leng个字符的排序,+为指令,还需能和-r,-f,-d等组合使用 */
#define MAXLINES 5000 /* 待排序的最大行数 */
char *lineptr[MAXLINES]; /* 指向问本行的指针 */ int readlines(char *lineptr[], int nlines);
void writelines(char *lineptr[], int nlines);
int inpuntcmp(char *, char *); /* 函数inpuntcmp:将忽略除空格字母数字以外的字符比较,如果某行以上三种字符都没有,则判断为最小 */ char *alloc(int n);
void sort(void *lineptr[], int left, int right, int (*comp)(void *, void *));
int numcmp(char *, char *);
int transfercmp(char *s1, char *s2); /* 函数transfercmp:忽略大小写的按字典比较的函数,例如,认为a=A,和numcmp以及strcmp并列的选项 */
int newstrcmp(char *s1, char *s2); /* 重新编函数newstrcmp: 实现对特定字段的一般字符比较,替代原来的strcmp */
static char allocbuf[];
static char *allocp=allocbuf;
int post=,uptolow=,leng=; /* post 为1则逆序排序;uptolow为1,忽略大小写,leng为字段分隔数,如为正,则对前leng个字符排序,为负,对末位leng个字符排序 */ main(int argc, char *argv[])
{
int nlines; /* 读入的输入行数 */
int numeric=,nw=; /* numeric为1则以数值排序 uptolow为1则转换大小写 */ while(argc-->){
if(strcmp(*++argv,"-n")==)
numeric=;
else if(strcmp(*argv,"-r")==)
post=;
else if(strcmp(*argv,"-f")==)
uptolow=;
else if(strcmp(*argv,"-d")==)
nw=; /* nw为1只对字母数字空格排序 */
else if(**argv=='+') { switch ((*++(*argv))-''>= && (**argv)-''<=) {
case :
if(leng==) (int)leng=atof(*argv);
else
printf("error: leng was defined\n");
break;
case :
if(**argv=='-'&&(*++(*argv))-''>= && (**argv)-''<=&& leng==)
(int)leng=atof(--(*argv)); else
printf("error: leng was defined or unknow command after '-'");
break;
default :
printf("error: please enter number after the '+'\n");
} } else printf("error: unkonwn command\n");
}
if((nlines=readlines(lineptr,MAXLINES))>=) {
if(nw==){
sort((void **)lineptr, post?(nlines-):, post?:(nlines-), (int (*)(void*,void*))(inpuntcmp));
writelines(lineptr,nlines);
return ;
} else {
if(uptolow==)
sort((void **)lineptr, post?(nlines-):, post?:(nlines-), (int (*)(void*,void*))(numeric?numcmp:newstrcmp));
else
sort((void **)lineptr, post?(nlines-):, post?:(nlines-), (int (*)(void*,void*))(numeric?numcmp:transfercmp)); writelines(lineptr,nlines);
return ;
}
}
else {
printf("input too big to sort\n");
return ;
} } void sort(void *v[], int left, int right, int (*comp)(void *, void *))
{
if(post==) {
int i, last;
void swap(void *v[], int, int); if(left>=right)
return;
swap(v,left,(left+right)/);
last =left;
for(i=left+;i<=right;i++)
if((*comp)(v[i],v[left])<)
swap(v,++last,i);
swap(v,left,last);
sort(v,left,last-,comp);
sort(v,last+,right,comp);
} else
{
int i, last;
void swap(void *v[], int, int); if(left<=right)
return;
swap(v,left,(left+right)/);
last =left;
for(i=left-;i>=right;i--)
if((*comp)(v[i],v[left])<)
swap(v,--last,i);
swap(v,left,last);
sort(v,left,last+,comp);
sort(v,last-,right,comp);
}
} int length(char *s);
int numcmp(char *s1,char *s2)
{
double v1,v2;
v1=atof(s1);
v2=atof(s2);
if(v1<v2)
return -;
else if (v1>v2)
return ;
else
return ; } void swap(void *v[], int i, int j) {
void *temp; temp =v[i];
v[i]=v[j];
v[j]=temp; }
#define MAXLEN 1000
int getline(char *, int);
int readlines(char *lineptr[], int maxlines) {
int len,nlines;
char *p,line[MAXLEN];
nlines =;
while((len=getline(line,MAXLEN))>)
if (nlines>= maxlines || (p=alloc(len))==)
return -;
else {
line[len-]=;
strcpy(p,line);
lineptr[nlines++]=p;
}
return nlines;
} void writelines(char *lineptr[], int nlines)
{
int i; for(i=; i< nlines; i++)
printf("%s\n",lineptr[i]);
}
int getline(char *s, int lim) {
int i=,c;
for(; i<lim && (c=getchar()) != EOF && c!='\n';++i)
*(s+i)=c;
if(c=='\n') {
*(s+i)=c;
++i;}
*(s+i)=;
return i; }
char *alloc(int n) {
if(allocbuf +-allocp>=n) {
allocp +=n;
return allocp - n;
} else
return ;
} int inpuntcmp(char *s1, char *s2)
{
char c,d,leng2;
int i=,k,len1=,len2=;
if(leng>=) {
while(*s1 != && *s2!= &&leng?(i<leng):) {
while(ispunct(*s1)) {
if(*s1==||len1>=leng)
return -;
++len1;
++s1;
}
while(ispunct(*s2)) {
if(*s2==||len2>=leng)
return ;
++s2;
++len2;
}
if(uptolow==) {
c=tolower(*s1);
d=tolower(*s2);
}
else {
c=*s1;
d=*s2;
}
if(c<d)
return -;
else if (c>d)
return ;
++s1;
++s2;
++len1;
++len2; }
if(len1>=leng && len2<leng)
return -;
if(len2>=leng && len1<leng)
return ;
if(len1>=leng && len2>=leng)
return ;
if(*s1<*s2)
return -;
else if (*s1>*s2)
return ;
else
return ;
} else {
leng2=-leng;
len1=length(s1);
len2=length(s2);
i=((len1-leng2)>)?(len1-leng2):;
k=((len2-leng2)>)?(len2-leng2):; while(s1[i] != && s2[k] !=) {
while(ispunct(s1[i])) {
if(s1[i]==)
return -;
++i; }
while(ispunct(s2[k])) {
if(s2[k]==)
return ;
++k;
}
if(uptolow==) {
c=tolower(s1[i]);
d=tolower(s2[k]);
}
else {
c=s1[i];
d=s2[k];
}
if(c<d)
return -;
else if (c>d)
return ;
++i;
++k; } if(s1[i]<s2[k])
return -;
else if (s1[i]>s2[k])
return ;
else
return ; } }
int transfercmp(char *s1, char *s2)
{
char c,d,leng2;
int i=,k,len1,len2;
if(leng>=) {
while(*s1 != && *s2!= && leng?(i<leng): ) {
c=tolower(*s1);
d=tolower(*s2);
if(c<d)
return -;
else if (c>d)
return ;
++s1;
++s2;
++i; }
if(i<leng) {
if(*s1<*s2)
return -;
else if (*s1>*s2)
return ;
else
return ;
} else
return ;
} else
{
leng2=-leng;
len1=length(s1);
len2=length(s2);
i=((len1-leng2)>)?(len1-leng2):;
k=((len2-leng2)>)?(len2-leng2):;
while(s1[i] != && s2[k] != ) {
c=tolower(s1[i]);
d=tolower(s2[k]);
if(c<d)
return -;
else if (c>d)
return ;
++i;
++k; }
if(s1[i]<s2[k])
return -;
else if (s1[i]>s2[k])
return ;
else
return ; } }
int newstrcmp(char *s1, char *s2) {
char c,d,leng2;
int i=,k,len1,len2;
if(leng>=) {
while(*s1 != && *s2!= && leng?(i<leng): ) {
c=*s1;
d=*s2;
if(c<d)
return -;
else if (c>d)
return ;
++s1;
++s2;
++i; }
if(i<leng) {
if(*s1<*s2)
return -;
else if (*s1>*s2)
return ;
else
return ;
} else
return ;
} else
{
leng2=-leng;
len1=length(s1);
len2=length(s2);
i=((len1-leng2)>)?(len1-leng2):;
k=((len2-leng2)>)?(len2-leng2):; while(s1[i] != && s2[k] != ) {
c=s1[i];
d=s2[k]; if(c<d)
return -;
else if (c>d)
return ;
++i;
++k; }
if(s1[i]<s2[k])
return -;
else if (s1[i]>s2[k])
return ;
else
return ; } }
int length(char *s) {
int i;
for(i=;*s++!=;++i)
;
return i;
}
对文本行按特定字段排序(前N个字符或后N个字符),TCPL 练习5-17的更多相关文章
- mongodb 通过嵌入文档中的字段排序
mongodb中的全部数据: db.testInfo.find({}) .sort({_id:-1}) .limit(100) 查询结果: /* 1 createdAt:2019/10/11 下午5: ...
- C#中List按特定字段排序
有一个类,如Student,有学号.数学成绩.语文成绩, 存在List列表中,要将List按数学成绩排序,怎么办呢? List<Student> scores=GetScores(); s ...
- C# List<object> 按特定字段排序
using System; using System.Collections; using System.Collections.Generic; using System.Linq; using S ...
- List多字段排序,orderBy,ThenBy
List排序问题,orderBy,ThenBy 1.List中一个字段排序 前几天做的项目中,获取的List<T>需要用某个字段来进行排序,困扰了很久.用OrderBy解决了.具体是这样的 ...
- 对文本行进行排序,新增-d(目录排序),只对字母数字空格排序(TCPL 练习5-16)
文本行的排序用到了命令行参数以及多级指针,在要求只对字母数字空格进行排序时,关键的问题点是兼容-f命令参数,也就是排序的同时忽略大小写.由于在之前的练习中,我将忽略大小写的比较方法重新写了一个函数tr ...
- SQL 按特定字段值排序
SQL 按特定字段值排序的代码,有需要的朋友可以参考下. id, name shandong01 name1 shandong02 name2 shandong03 name3 beijing01 n ...
- MySql 去重且指定某字段在前的排序方法
今天遇到一个棘手的数据查找并去重的问题: 情况: 1.取出数据库中的数据: 2.同一字段A,不同情况<值,如A值为:a1,a2>下取出的其他数据可能相同: 3.将2情况下的重复数据< ...
- mysql 字段指定值靠前排序方法,多字段排序排序方法
背景:SEO下选择某查询条件 查询展示信息为装修设计师以及设计师作品.设计师原型设计为:选择某风格 例如:简约,则列表出现拥有简约风格的设计师信息以及该设计师类型为简约的作品(3条靠前记录) 浏览原型 ...
- 读取txt文件将文本行组合成特定格式
有一网友要求从txt文本文件读取一些数据,然后组合为特定格式的数据行.原论题如下,刚才开始的要求描述得不太清楚,后来补充完整了. Insus.NET觉得本论题可有练习文本件读取功力,因此尝试实现一下. ...
随机推荐
- 类的序列化和反序列化(ObjectOutputStream和ObjectInputStream)
1.需要序列化的类 import java.io.Serializable; /** * 必须继承 Serializable 接口才能实现序列化 */ public class Employee im ...
- SQL注入的原理以及危害
SQL注入,就是通过把SQL命令插入到Web表单递交或输入域名或页面请求的查询字符串,最终达到欺骗服务器执行恶意的SQL命令,比如先前的很多影视网站泄露VIP会员密码大多就是通过WEB表单递交查询字符 ...
- idea debug 变慢
在run模式下很快大概30s左右,在debug模式下花了2分钟,debug信息不断输出: Returning cached instance of singleton bean....网上查了一下,说 ...
- MVC5+EF6 Code First 从零开始——第一章
一直在用Database First 突然感觉关系复杂的时候,生成json经常出现死循环,不够灵活. 今天正好是周五,不太忙就想试试code first吧,试了2次,终于有结果. ok废话不多说,先撸 ...
- 分享自制的13套 JQuery Mobile 界面主题(追加4套新款)
15套整合在一起的,其中2套官方+13套自制,款款精致,方便移动开发. 字体默认为微软雅黑. 适配于 JQuery Mobile 1.4.3 下载地址:http://files.cnblogs.com ...
- 在windows下面配置redis集群遇到的一些坑
最近工作不忙,就决定学习一下redis.因为一直在windows下工作,不会linux,没办法就选择在windows下配置redis. windows下配置redis集群的文章有很多,比如:http: ...
- 分享个win平台cocos2d-x创建项目的快捷方式
由于长期使用vs开发,相信使用vs开发的同学们也和我一样,都习惯点击图标然后就新建项目了,不是很适应命令的方式 由于vs2012一般都是win7 win8 win8.1这类系统开发,而这些系统,都是自 ...
- 《深入理解Spark:核心思想与源码分析》——SparkContext的初始化(叔篇)——TaskScheduler的启动
<深入理解Spark:核心思想与源码分析>一书前言的内容请看链接<深入理解SPARK:核心思想与源码分析>一书正式出版上市 <深入理解Spark:核心思想与源码分析> ...
- Excel导入-----导出(包含所选和全部)操作
在做系统的时候,很多时候信息量太大,这时候就需要进行Excel表格信息的导入和导出,今天就来给大家说一下我使用Excel表格信息导入和导出的心得. 1:首先需要在前端显示界面View视图中添加导入Ex ...
- Max Sum
Description Given a sequence a[1],a[2],a[3]......a[n], your job is to calculate the max sum of a sub ...