将元素组替换为python3中的特定元素

时间:2019-02-16 14:09:56

标签: python python-3.x list replacewith

我有超过50个元素的列表。这些元素是小写和大写字母,数字,特殊字符。

例如

sample_list = ['1', '0', 'b', 'B', '2', '6', 'a', '7', '9', '5', 'c', 'd', '4', 'A', 'C', 'f', 'D', 'F', '3', 'C', '8', 'A', 'F', 'B', 'A', 'A', 'D'] 

我想用特殊字符交换特定元素。 例如。

replacing `A,B,C and 1 with @
replacing `D,E,F and 2 with &
replacing `G,H,I and 3 with (

等等,我必须用11个选定的特殊字符替换一组特定的元素。就像我用3个特殊字符替换了几个选定的元素一样。

如何有效地做到这一点。

5 个答案:

答案 0 :(得分:6)

您可以按照以下方式使用translate方法进行操作:

sample_list = ['1', '0', 'b', 'B', '2', '6', 'a', '7', '9', '5', 'c', 'd', '4', 'A', 'C', 'f', 'D', 'F', '3', 'C', '8', 'A', 'F', 'B', 'A', 'A', 'D']
t = ''.maketrans('ABC1DEF2GHI3','@@@@&&&&((((')
out = [i.translate(t) for i in sample_list]
print(out)

输出:

['@', '0', 'b', '@', '&', '6', 'a', '7', '9', '5', 'c', 'd', '4', '@', '@', 'f', '&', '&', '(', '@', '8', '@', '&', '@', '@', '@', '&']
maketrans

str方法用于创建转换表,只需以两个等长的str来馈送转换表,其中第一个由键组成,第二个由值组成。 translate方法接受该表并相应替换字符,如果表中没有此类键,则保留原样。

编辑:正如OlivierMelançon所指出的,仅当您要将1个字符替换为1个字符时,才可以使用它。

答案 1 :(得分:0)

您首先需要定义替换规则,例如以dict为例。特别是,由于dict允许 O(1)访问,因此这与您的方法能够获得时间复杂度的速度一样快:这是通过一次遍历来解决的列表。

然后您可以遍历列表,并在每个元素都有替换规则的情况下替换它们。

rules = {
    'A': '@',
    'B': '@',
    'D': '&',
    'E': '&'
}

for i, c in enumerate(sample_list):
    if c in rules:
        sample_list[i] = rules[c]

如果您希望创建一个新列表而不是更改初始列表,则可以使用列表理解

new_list = [rules.get(c, c) for c in sample_list]

示例

rules = {
    'A': '@',
    'B': '@',
    'D': '&',
    'E': '&'
}

sample_list = ['A', 'A', 'B', 'C', 'D', 'E', 'F']

new_list = [rules.get(c, c) for c in sample_list]

print(new_list)

输出

['@', '@', '@', 'C', '&', '&', 'F']

答案 2 :(得分:0)

尝试此操作,并在字典中设置所有映射:

@Id
@GeneratedValue(strategy = GenerationType.AUTO, generator = "ADDRESS_ID")
@SequenceGenerator(name = "ADDRESS_ID", sequenceName = "ADDRESS_ID_SEQ")
@Column(name="ADDRESS_ID")
private Long id;

private String desc;

答案 3 :(得分:0)

您可以使用地图功能:

例如第一次更换:

new_list = map(lambda x: '@' if x in ('A', 'B', 'C') else x, sample_list)

sample_list也可以是一个字符串(每个字符是一个元素,但是map将以任何一种方式返回一个数组)。

在“字符串”列表上,您可以只使用replace方法:

compact_list = ''.join(sample_list)
new_compact_list = compact_list.replace('A', '@').replace('B', '@').replace('C', '@')

当然,如果源列表是可变的,请在其周围编写一个循环...

答案 4 :(得分:0)

从一个简单的函数开始并对其进行优化。

简单将使用3 sets(因为查找集比列表更好)-如果集不需要突变,请改用frozensets。使用一个简单的循环-直截了当:

sample_list = ['1', '0', 'b', 'B', '2', '6', 'a', '7', '9', '5', 'c', 'd', '4', 'A', 'C', 'f', 'D', 'F', '3', 'C', '8', 'A', 'F', 'B', 'A', 'A', 'D'] 

new_list = []
for item in sample_list:
    abc = frozenset("ABC")    # frozensets do not change, they are immuteable
    dfe = frozenset("DEF")    # and fast for lookups. this is less performant
    ghi = frozenset("GHI")    # then using a dict - but it works as well

    if item in abc:
        new_list.append("@")
    elif item in dfe:
        new_list.append("&")
    elif item in ghi:
        new_list.append("(")
    else:
        new_list.append(item)

print(new_list)

输出:

['1', '0', 'b', '@', '2', '6', 'a', '7', '9', '5', 'c', 'd', '4', '@', '@', 'f', '&', '&', '3', '@', '8', '@', '&', '@', '@', '@', '&']

使用字典和list-comp以获得更多的乐趣:

transform = {"A":"@", "B":"@", "C":"@", 
             "D":"&", "E":"&", "F":"&", 
             "G":"(", "H":"(", "I":"("}

new_list = [ transform.get(i,i) for i in sample_list]

输出:

['1', '0', 'b', '@', '2', '6', 'a', '7', '9', '5', 'c', 'd', '4', '@', '@', 'f', '&', '&', '3', '@', '8', '@', '&', '@', '@', '@', '&']

请参阅: