001/* 002 * Licensed to the Apache Software Foundation (ASF) under one or more 003 * contributor license agreements. See the NOTICE file distributed with 004 * this work for additional information regarding copyright ownership. 005 * The ASF licenses this file to You under the Apache License, Version 2.0 006 * (the "License"); you may not use this file except in compliance with 007 * the License. You may obtain a copy of the License at 008 * 009 * http://www.apache.org/licenses/LICENSE-2.0 010 * 011 * Unless required by applicable law or agreed to in writing, software 012 * distributed under the License is distributed on an "AS IS" BASIS, 013 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 014 * See the License for the specific language governing permissions and 015 * limitations under the License. 016 */ 017package org.apache.commons.collections4.list; 018 019import java.util.Collection; 020import java.util.List; 021import java.util.ListIterator; 022 023import org.apache.commons.collections4.Transformer; 024import org.apache.commons.collections4.collection.TransformedCollection; 025import org.apache.commons.collections4.iterators.AbstractListIteratorDecorator; 026 027/** 028 * Decorates another {@code List} to transform objects that are added. 029 * <p> 030 * The add and set methods are affected by this class. 031 * Thus objects must be removed or searched for using their transformed form. 032 * For example, if the transformation converts Strings to Integers, you must 033 * use the Integer form to remove objects. 034 * </p> 035 * <p> 036 * This class is Serializable from Commons Collections 3.1. 037 * </p> 038 * 039 * @param <E> the type of the elements in the list. 040 * @since 3.0 041 */ 042public class TransformedList<E> extends TransformedCollection<E> implements List<E> { 043 044 /** 045 * Inner class Iterator for the TransformedList 046 */ 047 protected class TransformedListIterator extends AbstractListIteratorDecorator<E> { 048 049 /** 050 * Create a new transformed list iterator. 051 * 052 * @param iterator the list iterator to decorate 053 */ 054 protected TransformedListIterator(final ListIterator<E> iterator) { 055 super(iterator); 056 } 057 058 @Override 059 public void add(E object) { 060 object = transform(object); 061 getListIterator().add(object); 062 } 063 064 @Override 065 public void set(final E object) { 066 getListIterator().set(transform(object)); 067 } 068 } 069 070 /** Serialization version */ 071 private static final long serialVersionUID = 1077193035000013141L; 072 073 /** 074 * Factory method to create a transforming list that will transform 075 * existing contents of the specified list. 076 * <p> 077 * If there are any elements already in the list being decorated, they 078 * will be transformed by this method. 079 * Contrast this with {@link #transformingList(List, Transformer)}. 080 * 081 * @param <E> the type of the elements in the list 082 * @param list the list to decorate, must not be null 083 * @param transformer the transformer to use for conversion, must not be null 084 * @return a new transformed List 085 * @throws NullPointerException if list or transformer is null 086 * @since 4.0 087 */ 088 public static <E> TransformedList<E> transformedList(final List<E> list, 089 final Transformer<? super E, ? extends E> transformer) { 090 final TransformedList<E> decorated = new TransformedList<>(list, transformer); 091 if (!list.isEmpty()) { 092 @SuppressWarnings("unchecked") // list is of type E 093 final E[] values = (E[]) list.toArray(); // NOPMD - false positive for generics 094 list.clear(); 095 for (final E value : values) { 096 decorated.decorated().add(transformer.apply(value)); 097 } 098 } 099 return decorated; 100 } 101 102 /** 103 * Factory method to create a transforming list. 104 * <p> 105 * If there are any elements already in the list being decorated, they 106 * are NOT transformed. 107 * Contrast this with {@link #transformedList(List, Transformer)}. 108 * 109 * @param <E> the type of the elements in the list 110 * @param list the list to decorate, must not be null 111 * @param transformer the transformer to use for conversion, must not be null 112 * @return a new transformed list 113 * @throws NullPointerException if list or transformer is null 114 * @since 4.0 115 */ 116 public static <E> TransformedList<E> transformingList(final List<E> list, 117 final Transformer<? super E, ? extends E> transformer) { 118 return new TransformedList<>(list, transformer); 119 } 120 121 /** 122 * Constructor that wraps (not copies). 123 * <p> 124 * If there are any elements already in the list being decorated, they 125 * are NOT transformed. 126 * 127 * @param list the list to decorate, must not be null 128 * @param transformer the transformer to use for conversion, must not be null 129 * @throws NullPointerException if list or transformer is null 130 */ 131 protected TransformedList(final List<E> list, final Transformer<? super E, ? extends E> transformer) { 132 super(list, transformer); 133 } 134 135 @Override 136 public void add(final int index, E object) { 137 object = transform(object); 138 getList().add(index, object); 139 } 140 141 @Override 142 public boolean addAll(final int index, Collection<? extends E> coll) { 143 coll = transform(coll); 144 return getList().addAll(index, coll); 145 } 146 147 @Override 148 public boolean equals(final Object object) { 149 return object == this || decorated().equals(object); 150 } 151 152 @Override 153 public E get(final int index) { 154 return getList().get(index); 155 } 156 157 /** 158 * Gets the decorated list. 159 * 160 * @return the decorated list 161 */ 162 protected List<E> getList() { 163 return (List<E>) decorated(); 164 } 165 166 @Override 167 public int hashCode() { 168 return decorated().hashCode(); 169 } 170 171 @Override 172 public int indexOf(final Object object) { 173 return getList().indexOf(object); 174 } 175 176 @Override 177 public int lastIndexOf(final Object object) { 178 return getList().lastIndexOf(object); 179 } 180 181 @Override 182 public ListIterator<E> listIterator() { 183 return listIterator(0); 184 } 185 186 @Override 187 public ListIterator<E> listIterator(final int i) { 188 return new TransformedListIterator(getList().listIterator(i)); 189 } 190 191 @Override 192 public E remove(final int index) { 193 return getList().remove(index); 194 } 195 196 @Override 197 public E set(final int index, E object) { 198 object = transform(object); 199 return getList().set(index, object); 200 } 201 202 @Override 203 public List<E> subList(final int fromIndex, final int toIndex) { 204 final List<E> sub = getList().subList(fromIndex, toIndex); 205 return new TransformedList<>(sub, transformer); 206 } 207 208}