首页 > 解决方案 > 有没有办法按字符串中的数字对字符串列表进行排序?

问题描述

有没有办法对类似的东西进行排序:

List<String> hi = ['1hi', '2hi','5hi', '3hi', '4hi'];

到这个?

['1hi', '2hi','3hi', '4hi', '5hi']

标签: flutterdart

解决方案


仅调用List<String>.sort()本身就会进行字典排序。也就是说,您的字符串将按字符代码顺序排序,并且'10'将在'2'. 这通常不是预期的。

如果您的数字具有前导0s 以确保所有数字具有相同的位数,则字典排序将起作用。但是,如果位数是可变的,则需要解析数字的值以进行排序。更通用的方法是提供一个回调来.sort()告诉它如何确定两个项目的相对顺序。

幸运的是,package:collection有一个compareNatural功能可以为您做到这一点:

import 'package:collection/collection.dart';

List<String> hi = ['1hi', '2hi','5hi', '3hi', '4hi'];
hi.sort(compareNatural);

如果你的情况有点复杂并且compareNatural没有做你想做的事情,更通用的方法是让.sort()回调自己解析,例如通过正则表达式:

/// Returns the integer prefix from a string.
///
/// Returns null if no integer prefix is found.
int parseIntPrefix(String s) {
  var re = RegExp(r'(-?[0-9]+).*');
  var match = re.firstMatch(s);
  if (match == null) {
    return null;
  }
  return int.parse(match.group(1));
}

int compareIntPrefixes(String a, String b) {
  var aValue = parseIntPrefix(a);
  var bValue = parseIntPrefix(b);
  if (aValue != null && bValue != null) {
    return aValue - bValue;
  }

  if (aValue == null && bValue == null) {
    // If neither string has an integer prefix, sort the strings lexically.
    return a.compareTo(b);
  }

  // Sort strings with integer prefixes before strings without.
  if (aValue == null) {
    return 1;
  } else {
    return -1;
  }
}

void main() {
  List<String> hi = ['1hi', '2hi','5hi', '3hi', '4hi'];
  hi.sort(compareIntPrefixes);
}

推荐阅读