对文本行按特定字段排序(前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觉得本论题可有练习文本件读取功力,因此尝试实现一下. ...
随机推荐
- HTML实体符号代码速查表
1.特色的 © © © 版权标志 | | 竖线,常用作菜单或导航中的分隔符 · · · 圆点,有时被用来作为菜单分隔符 ↑ ↑ ↑ 上箭头,常用作网页“返回页面顶部”标识 € € € 欧元标识 ² ...
- MyElipes遇到 source not found解决方案(查看.class文件源码一劳永逸的解决方法)
在用Myeclipse 或者是eclipse进行开发时候经常遇到这个问题. File class editor source not found 问题.原因很简单,就是因为这是一个源码包,相应的没有编 ...
- Java--使用多线程下载,断点续传技术原理(RandomAccessFile)
一.基础知识 1.什么是线程?什么是进程?它们之间的关系? 可以参考之前的一篇文章:java核心知识点学习----并发和并行的区别,进程和线程的区别,如何创建线程和线程的四种状态,什么是线程计时器 简 ...
- CentOS 6 安装 MySQL-python
yum install -y mysql-devel python-devel python-setuptools pip install MySQL-python
- UiAutomator自动化测试框架介绍
UiAutomator自动化测试框架介绍 环境搭建 1 必要条件 1.1 JDK 1.2 SDK(API高于15) 1.3 Eclipse 2 ...
- c# DESEncrypt 加密、解密算法
using System; using System.Security.Cryptography; using System.Text; namespace AttendService { publi ...
- Step by Step 配置使用HTTPS的ASP.NET Web应用[转载]
原创地址:http://www.cnblogs.com/jfzhu/p/4064654.html 转载请注明出处 有关HTTPS.SSL以及SSL证书的工作原理,参见 <HTTPS那些事(一)H ...
- 强大的Spring缓存技术(上)
缓存是实际工作中非常常用的一种提高性能的方法, 我们会在许多场景下来使用缓存. 本文通过一个简单的例子进行展开,通过对比我们原来的自定义缓存和 spring 的基于注释的 cache 配置方法,展现了 ...
- C++关于文件的读写(续)
上次,我们探讨了C++关于文件的读操作,这次继续写操作.写,顾名思义,就是把我们自己的东西放到文件中去. 我们举一个简单的例子,就不在乎它的排版问题了. #include<iostream> ...
- LogNet4日志框架使用
.百度一下log4dll下载 .webconfig 里的<configSetions>节点中添加 <section name="log4net" type=&qu ...