shell小技巧(七十)统计排序

计数排序不是基于比较的排序算法,其核心在于将输入的数据值转化为键存储在额外开辟的数组空间中。

 算法描述
找出待排序的数组中最大和最小的元素;
统计数组中每个值为i的元素出现的次数,存入数组C的第i项;
对所有的计数累加(从C中的第一个元素开始,每一项和前一项相加);
反向填充目标数组:将每个元素i放在新数组的第C(i)项,每放一个元素就将C(i)减去1。

代码:

#!/bin/bash
#Counting Sort
arr=(100 99 97 77 79 80 9 12 22 2 32 35 66 55 51 42 38 37 20 70 109 9)
imax=${#arr[@]}
echo $imax
echo ${arr[@]}
echo "--->>"
i=0
ivaluemax=${arr[0]}
ivaluemin=${arr[0]}
while [ $i -lt $imax ]; do
   if [ ${arr[$i]} -gt $ivaluemax ]; then
       ivaluemax=${arr[$i]}
   elif [ ${arr[$i]} -lt $ivaluemin ]; then
       ivaluemin=${arr[$i]}
   fi
  if [ -z karr[${arr[$i]}] ]; then
       karr[${arr[$i]}]=1
  else 
       let karr[${arr[$i]}]=karr[${arr[$i]}]+1
  fi
 let i=i+1
done
j=0
i=0
while [ $j -le $ivaluemax ];do
   if [ -z ${karr[$j]} ] ; then
      let j=j+1;
   else
      while [ ${karr[$j]} -gt 0 ];do
        Rarr[$i]=$j
        let karr[$j]=karr[$j]-1
        let i=i+1
      done
      let j=j+1
   fi 
done

echo ${Rarr[@]}
 

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
好的,这是一个比较复杂的任务,需要用到多个命令和技巧。以下是一个可能的解决方案: ```bash #!/bin/bash # 读入文件名 echo "请输入要处理的文件名:" read filename # 去掉大小写字母及空格之外的所有字符 tr -cd 'a-zA-Z ' < "$filename" > "$filename.processed" # 将所有大写字母转换为小写字母 tr 'A-Z' 'a-z' < "$filename.processed" > "$filename.processed.lowercase" # 统计各个字符出现的次数并计算频率 total=$(wc -m < "$filename.processed.lowercase") # 总字符数 tr -d ' ' < "$filename.processed.lowercase" | grep -o . | sort | uniq -c | awk '{print $2,$1/$total}' > "$filename.freq" # 按照出现频率降序排序并输出到显示器 sort -k 2nr "$filename.freq" | awk '{printf "%s: %.2f%%\n", $1, $2*100}' ``` 解释一下上面的代码: 1. 首先读入文件名,可以通过 `read` 命令实现。 2. 利用 `tr` 命令去掉大小写字母及空格之外的所有字符,`-cd` 参数表示只保留指定字符集,`a-zA-Z` 表示大小写字母,空格表示空格字符。将处理后的结果输出到一个新文件中。 3. 再次利用 `tr` 命令将所有大写字母转换为小写字母,将结果输出到另一个新文件中。 4. 利用一系列命令统计各个字符出现的次数并计算频率: - `wc -m` 命令可以统计文件中的字符数。 - `tr -d ' '` 命令可以去掉空格。 - `grep -o .` 命令可以将每个字符单独提取出来。 - `sort` 命令可以按照字典序排序。 - `uniq -c` 命令可以统计相同字符出现的次数。 - `awk` 命令可以计算每个字符的出现频率,并将结果输出到一个新文件中。 5. 最后利用 `sort` 命令按照出现频率降序排序,并利用 `awk` 命令格式化输出结果到显示器上。 希望这个解决方案能对你有所帮助!
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

乐大师

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值