不同长度数据项的排序

注:本文改编自windmissing博客,感谢作者整理!

 

题目:

a)给定一个整数数组,其中不同的整数中包含的数字个数可能不同,但是该数组中,所有整数中总的数字数为n。说明如何在O(n)时间内对该数组进行排序

b)给定一个字符串数组,其中不同的串包含的字符个数可能不同,但所有串中总的字符个数为n。说明如何在O(n)时间内对该数组进行排序

(注意此处的顺序是指标准的字母顺序,例如,a < ab < b)

 

a)先用桶排序方法按数字位数排序O(n),再用基数排序的方法分别对每个桶中的元素排序O(n)
b)递归使用计数排序,先依据第一个字母进行排序,首字相同的放在同一组,再对每一组分别使用计数排序的方法比较第二个字母
见到有人用字典树,也是可以的
  1 //8-2-a
  2 #include <iostream>
  3 #include <cmath>
  4 using namespace std;
  5 
  6 int length_A;
  7 void Print(int *A)
  8 {
  9   int i;
 10   for(i = 1; i <= length_A; i++)
 11       cout<<A[i]<<' ';
 12   cout<<endl;
 13 }
 14 
 15 int Digit(int x)
 16 {
 17   int ret = 0;
 18   while(x)
 19   {
 20      ret++;
 21      x = x / 10;
 22   }
 23 
 24   return ret;
 25 }
 26 
 27 
 28 //基数排序调用的稳定排序
 29 void Counting_Sort(int *A, int *B, int k)
 30 {
 31     int i, j;
 32     //将C数组初始化为0,用于计数
 33     int *C = new int[k+1];
 34     for(i = 0; i <= k; i++)
 35         C[i] = 0;
 36     int *D = new int[length_A+1];
 37     for(j = 1; j <= length_A; j++)
 38     {
 39         //D[j]表示第[j]个元素有i位数字
 40         D[j] = Digit(A[j]);
 41         //C[j]表示数字D[j]在数组A中出现的次数
 42         C[D[j]]++;
 43     }
 44     //C[i]表示所以<=i的数字出现过的次数
 45     for(i = 1; i <= k; i++)
 46         C[i] = C[i] + C[i-1];
 47     //初始化B为0,B用于输出排序结果
 48     for(i = 1; i <= length_A; i++)
 49         B[i] = 0;
 50     for(j = length_A; j >= 1; j--)
 51     {
 52         //如果<=D[j]的数字的个数是x,那么排序后A[j]应该出现在第x个位置,即B[x]=A[j]
 53         B[C[D[j]]] = A[j];
 54         C[D[j]]--;
 55     }
 56     delete []C;
 57     delete []D;
 58 }
 59 //基数排序调用的稳定排序
 60 void Stable_Sort(int *A, int *B, int k, int d,int start, int end)
 61 {
 62     int i, j, radix = 10;
 63     //将C数组初始化为0,用于计数
 64     int *C = new int[k+1];
 65     for(i = 0; i <= k; i++)
 66         C[i] = 0;
 67     int *D = new int[length_A+1];
 68     for(j = start; j <= end; j++)
 69     {
 70         //D[j]表示第[j]个元素的第i位数字
 71         D[j] = A[j] % (int)pow(radix*1.0, d) / (int)pow(radix*1.0, d-1);
 72         //C[j]表示数字D[j]在数组A中出现的次数
 73         C[D[j]]++;
 74     }
 75     //C[i]表示所以<=i的数字出现过的次数
 76     for(i = 1; i <= k; i++)
 77         C[i] = C[i] + C[i-1];
 78     //初始化B为0,B用于输出排序结果
 79     for(i = 1; i <= length_A; i++)
 80         B[i] = 0;
 81     for(j = end; j >= start; j--)
 82     {
 83         //如果<=D[j]的数字的个数是x,那么排序后A[j]应该出现在第x个位置,即B[x]=A[j]
 84         B[C[D[j]]+start-1] = A[j];
 85         C[D[j]]--;
 86     }
 87     delete []C;
 88     delete []D;
 89 }
 90 void Radix_Sort(int *A, int *B, int k ,int digit, int start, int end)
 91 {
 92     int i, j;
 93     //依次对每一位进行排序,从低位到高位
 94     for(i = 1; i <= digit; i++)
 95     {
 96         Stable_Sort(A, B, k, i, start, end);
 97         //输入的是A,输出的是B,再次排序时要把输出数据放入输出数据中
 98         for(j = start; j <= end; j++)
 99         A[j] = B[j];
100     }
101 }
102 int main()
103 {
104     cin>>length_A;
105     int i;
106     //产生随机的测试数据
107     int *A = new int[length_A+1];
108     for(i = 1; i <= length_A; i++)
109         A[i] = rand() % (int)pow(10.0, rand()%5+1);
110     Print(A);
111     int *B = new int[length_A+1];
112     //先进行计数排序,把长度相同的数字排在一起
113     Counting_Sort(A, B, 5);
114     for(i = 1; i <= length_A; i++)
115         A[i] = B[i];
116     Print(A);
117     int start, end, digit = -1;
118     for(i = 1; i <= length_A; i++)
119     {
120         if(digit == -1)
121         {
122             digit = Digit(A[i]);
123             start = i;
124             end = i;
125         }
126         else
127         {
128             if(Digit(A[i]) == digit)
129                 end = i;
130             else
131             {
132                 //找到位数相同的一段,从start到end,单独对这一段进行基数排序
133                 Radix_Sort(A, B, 9, digit, start, end);
134                 i--;
135                 digit = -1;
136                 Print(A);
137             }
138         }
139     }
140     delete []A;
141     delete []B;
142 }
  1 #include <iostream>
  2 #include <string>
  3 using namespace std;
  4 
  5 int length_A;
  6 void Print(string *A)
  7 {
  8     int i;
  9     for(i = 1; i <= length_A; i++)
 10         cout<<A[i]<<' ';
 11     cout<<endl;
 12 }
 13 //基数排序调用的稳定排序,A是输入,B是中间输出,C是计数,d表示对第d位字母排序,start和end分别是排序段的起点和终点
 14 void Counting_Sort(string *A, string *B, int *C, int d, int start, int end)
 15 {
 16     if(start == end)
 17         return;
 18     int i, j;
 19     //将C数组初始化为0,用于计数
 20     for(i = 0; i <= 26; i++)
 21         C[i] = 0;
 22     int *D = new int[length_A+1];
 23     for(j = start; j <= end; j++)
 24     {
 25         //D[j]表示第[j]个元素的第i位数字
 26         if(A[j].length() <= d)
 27             D[j] = 0;
 28         else
 29             D[j] = A[j][d] - 'a';
 30         //C[j]表示数字D[j]在数组A中出现的次数
 31         C[D[j]]++;
 32     }
 33     //C[i]表示所以<=i的数字出现过的次数
 34     for(i = 1; i <= 26; i++)
 35         C[i] = C[i] + C[i-1];
 36     //初始化B为0,B用于输出排序结果
 37     for(i = 1; i <= length_A; i++)
 38         B[i] = "";
 39     for(j = end; j >= start; j--)
 40     {
 41         //如果<=D[j]的数字的个数是x,那么排序后A[j]应该出现在第x个位置,即B[x]=A[j]
 42         B[C[D[j]]+start-1] = A[j];
 43         C[D[j]]--;
 44     }
 45     delete []D;
 46     //输出转为输入
 47     for(i = start; i <= end; i++)
 48         A[i] = B[i];
 49     char c = 'A';
 50     int s, e;//进一步的排序以s为起点,e为终点
 51     //对于排序的这一段,对下一个字母递归使用计数排序
 52     for(i = start; i <= end; i++)
 53     {
 54         //如果长度为d,不参与下一步排序
 55         if(A[i][d] == '\0')
 56             continue;
 57         if(c == 'A')
 58         {
 59             s = i;
 60             e = i;
 61             c = A[i][d];
 62         }
 63         else
 64         {
 65             if(A[i][d] == c)
 66             {
 67                 e = i;
 68                 if(e == end)
 69                     //以第d+1位字母为依据,对s-e段进行计数排序
 70                     Counting_Sort(A, B, C, d+1, s, e);
 71             }
 72             else
 73             {
 74                 //以第d+1位字母为依据,对s-e段进行计数排序
 75                 Counting_Sort(A, B, C, d+1, s, e);
 76                 i--;
 77                 c = 'A';
 78             }
 79         }
 80     }
 81 }
 82 int main()
 83 {
 84     int i, j;
 85     cin>>length_A;
 86     string *A = new string[length_A+1];
 87     //构造随机数据
 88     for(i = 1; i <= length_A; i++)
 89     {
 90         int len = rand()%5+1;
 91         for(j = 1; j <= len; j++)
 92             A[i] = A[i] + (char)(rand()%26+'a');
 93     }
 94     Print(A);
 95     string *B = new string[length_A+1];
 96     int *C = new int[26];
 97     //计数排序
 98     Counting_Sort(A, B, C, 0, 1, length_A);
 99     Print(A);
100     delete []A;
101     delete []C;
102     return 0;
103 }

 

转载于:https://www.cnblogs.com/lixiaohui-ambition/archive/2012/09/20/2695802.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/274597.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

浅谈前端埋点监控

大家好&#xff0c;我是若川。持续组织了近一年的源码共读活动&#xff0c;感兴趣的可以 加我微信lxchuan12 参与&#xff0c;每周大家一起学习200行左右的源码&#xff0c;共同进步。同时极力推荐订阅我写的《学习源码整体架构系列》 包含20余篇源码文章。历史面试系列。另外&…

css版式_第2部分:使版式具有响应能力,并为以后的版本奠定基础

css版式The feedback I’ve received over the past week has been amazing, and matches my own excitement about this project. I’ve spent a lot of time researching, writing, and teaching about creating better typography for reading on digital devices over the …

BBS项目--登录

BBS阶段性测试总要求 django登录报错 Error: [WinError 10013] 以一种访问权限不允许的方式做了一个访问套接字的尝试。 原因分析&#xff1a;出现这种情况在Windows中很常见&#xff0c;就是端口被占用 解决措施&#xff1a;这时我们只需改一下端口便可以了 登录前端页面(HTML…

【声明】

我的公众号和朋友圈有时会有一些课程推广广告&#xff0c;微博的收入来源。我接的广告一般来说都是比自己去买会优惠不少&#xff0c;我也会想方设法争取到更多福利&#xff08;优惠&#xff09;。买过的都知道确实优惠。如果有人看到觉得不合适&#xff0c;不想看到&#xff0…

Win7 访问共享时输入正确密码仍然提示密码错误

1、直接按下winr键&#xff0c;输入secpol.msc&#xff0c;打开本地安全策略。 2、找到“安全设置”的“本地策略”的“安全选项” 3、在右边一栏找到“网络安全&#xff1a;LAN管理器身份验证级别”&#xff0c;双击进入 4、在默认状态选项下&#xff0c;英文版应该为"no…

怎么实现页面友好跳转_如何实现软,友好和一致的UI设计

怎么实现页面友好跳转重点 (Top highlight)Design trends are constantly changing, aren’t they? Each month there is a new visual effect or a trick that becomes “设计趋势在不断变化&#xff0c;不是吗&#xff1f; 每个月都有一个新的视觉效果或技巧&#xff0c;成为…

前端趋势 2022

大家好&#xff0c;我是若川。持续组织了近一年的源码共读活动&#xff0c;感兴趣的可以 加我微信lxchuan12 参与&#xff0c;每周大家一起学习200行左右的源码&#xff0c;共同进步。同时极力推荐订阅我写的《学习源码整体架构系列》 包含20余篇源码文章。历史面试系列。另外&…

MySQL Connector/ODBC 5.2.2 发布

MySQL Connector/ODBC 5.2.2 发布&#xff0c;这是一个稳定版本&#xff0c;下载地址&#xff1a; http://dev.mysql.com/downloads/connector/odbc/5.2.html MySQL Connector/ODBC 是 MySQL 官方发布的 ODBC 驱动程序包。转载于:https://www.cnblogs.com/shihao/archive/2012/…

优秀测试管理工具必备九大功能分析

摘要&#xff1a;测试管理工具对测试的重要性毋庸质疑&#xff0c;两位笔者有着多年的测试实战经验&#xff0c;对市面上的一些测试管理工具有过一定的研究&#xff0c;还根据目前比较流行的敏捷开发过程设计了一款测试管理工具。 这篇文章算是对这个设计过程的总结与分享&…

lightroom预设使用_在Lightroom中使用全景图增强照片游戏

lightroom预设使用Everyone here has taken a panorama with an iphone. We’ve spun around in a circle, trying to keep that arrow right on the line, and more than likely ended up with a strange, squiggly, horizontal photo. Every so often you might get lucky an…

第91次TC39会议举行,这还是我认识的JS吗?

大家好&#xff0c;我是若川。持续组织了近一年的源码共读活动&#xff0c;感兴趣的可以 加我微信 ruochuan12 参与&#xff0c;每周大家一起学习200行左右的源码&#xff0c;共同进步。同时极力推荐订阅我写的《学习源码整体架构系列》 包含20余篇源码文章。历史面试系列。另外…

android调节音量——AudioManager的应用

Android中可以通过程序获取系统手机的铃声和音量。同样&#xff0c;也可以设置铃声和音量。Android中给出了AudioManager类来实现音量获取、音量控制。本篇基于 Android API 中的 AudioManager 作讲述&#xff0c;最后给出实例。下面是本篇大纲&#xff1a;1、认识 AudioManage…

静态创意和动态创意_再次发挥创意需要什么?

静态创意和动态创意重点 (Top highlight)According to Oxford dictionary, creativity means “1. Inventiveness. 2. the use of imagination or original ideas to create something.”根据牛津词典&#xff0c;创造力意味着“ 1。 创造力。 2.利用想象力或独创性的思想来创造…

oracle 存储过程 stored procedure 查询一条记录或多条记录

创建基本表 -- Create table create table USER_INFORMATION ( P_ID NUMBER, USER_LOGIN_NAME NVARCHAR2(30) ) 创建包: create or replace package pack_test is type cur_test is ref cursor; end pack_test; / --这个不能少呀&#xff0c;加上这个就可以在…

我写了 ahooks 源码分析系列,收到官方邀请我一起维护,这是一次提 PR 的记录...

大家好&#xff0c;我是若川。持续组织了近一年的源码共读活动&#xff0c;感兴趣的可以 加我微信 ruochuan12 参与&#xff0c;每周大家一起学习200行左右的源码&#xff0c;共同进步。同时极力推荐订阅我写的《学习源码整体架构系列》 包含20余篇源码文章。历史面试系列。另外…

Hdu 4415 Assassin's Creed 【贪心】.cpp

题意&#xff1a; 某A有一个剑 坚韧度为m 他可以用这个剑去攻打别的队伍 杀掉第 i 个队伍需要消耗的坚韧度为 Ai 并可以用得到的剑去打别的队(Bi个) 但是打完别的队这个剑就不能用了 问怎么用最少的坚韧度击败最多的队伍 给出T组样例 每个样例给出n m n表示有n个队 接下来n行给…

ahooks 整体架构篇,大家都能看得懂

大家好&#xff0c;我是若川。持续组织了近一年的源码共读活动&#xff0c;感兴趣的可以 加我微信 ruochuan12 参与&#xff0c;每周大家一起学习200行左右的源码&#xff0c;共同进步。同时极力推荐订阅我写的《学习源码整体架构系列》 包含20余篇源码文章。历史面试系列。另外…

gif动态图gif出处_我喜欢GIF的怪异事物

gif动态图gif出处I was recently reminded that I never wrote down all the weird things I learned about the GIF file format when implementing GIF decoding/playback at work last year. (I was reminded of this because I wrote a line in a corporate blog post draf…

C#字符串学习笔记

前言&#xff1a;记得我们老师说过一句话&#xff0c;对字符串的学习程度就是当别人打你一拳你知道痛的情况&#xff0c;所以字符串的处理我们必须学的差不多&#xff0c;这几篇博客完全是我的学习过程中记录的笔记&#xff0c;在这里分享一下让很多刚开始学习.net编程的人能够…

Git基础教程(必学)

大家好&#xff0c;我是若川。持续组织了近一年的源码共读活动&#xff0c;感兴趣的可以 加我微信 ruochuan12 参与&#xff0c;每周大家一起学习200行左右的源码&#xff0c;共同进步。同时极力推荐订阅我写的《学习源码整体架构系列》 包含20余篇源码文章。历史面试系列。另外…