一道java面试题---去除list中的指定元素

 

  一道比较简单的java面试题,一个list中存放string,例如存放人名,而且这个list的size比较大,现在要求去除该list中的名字为姓“张”的所有元素,至今没有想到很好的方法,自己只知道三种方法,求大神给予更好的答案,性能越高越好,谢谢~

 

import java.util.ArrayList;
import java.util.ConcurrentModificationException;
import java.util.Iterator;
import java.util.List;
import java.util.Random;

import org.junit.Before;

public class ModifyList {
	
	public List<String> lists=null;
	
        String[] prefixName = new String[]{"赵","钱","孙","李","张","陈"};
	
	String[] suffixName = new String[]{"子涵","心怡","子墨","梦瑶","浩轩","轩","俊熙","可馨","睿","思彤","思诚","丹"};
	
	@Before
	public void before(){
		lists=new ArrayList<String>();
		Random random=new Random();
		int i=0;
		while(i<50000){
			int prefix=random.nextInt(6);
			int suffix=random.nextInt(12);
			String name=prefixName[prefix]+suffixName[suffix];
			lists.add(name);
			i++;
		}
	    
	}
	
	
	@org.junit.Test(expected=ConcurrentModificationException.class)
	public void remove1(){
		for(String name:lists){
			if(name!=null&name.indexOf("张")>-1){
				lists.remove(name);
			}
		}
		System.out.println(lists);
	}
	
	/**
	 * 该方法适用于去除唯一一个不合适的值,效率最高
	 */
	@org.junit.Test
	public void remove2(){
		for(String name:lists){
			if(name!=null&name.indexOf("张")>-1){
				lists.remove(name);
				break;
			}
		}
		System.out.println(lists);
	}
	
	/**
	 * 该方法没有报错,因为每次都去获取lists.size
	 * 不会抛出ConcurrentModificationException异常
	 * 但如果连续的list中有连续的姓张的在一起,就会出现不删除现象
	 * 当i=0时,删除“张三”,此时“张九”的下标变为0,i=1,此时就不会对“张九”做处理
	 */
	@org.junit.Test
	public void remove3(){
		for(int i=0;i<lists.size();i++){
			if(lists.get(i)!=null&lists.get(i).indexOf("张")>-1){
				lists.remove(i);
			}
		}
		System.out.println(lists);
	}
	
	/**
	 * 正确方法一
	 */
	@org.junit.Test
	public void remove4(){
		long begin=System.currentTimeMillis();
		System.out.println(begin);
		Iterator<String> iters = lists.iterator();
		while(iters.hasNext()){
			String name=iters.next();
			if(name!=null&name.indexOf("张")>-1){
				iters.remove();
			}
		}
		long end=System.currentTimeMillis();
		System.out.println(end);
		System.out.println(end-begin);
		/*System.out.println(lists);*/
	}
	
	/**
	 * 正确方法二
	 */
	@org.junit.Test
	public void remove5(){
		 
		long begin=System.currentTimeMillis();
		System.out.println(begin);
		for(int i=0;i<lists.size();i++){
			if(lists.get(i)!=null&lists.get(i).indexOf("张")>-1){
				lists.remove(i);
				i--;
			}
		}
		long end=System.currentTimeMillis();
		System.out.println(end);
		System.out.println(end-begin);
		//System.out.println(lists);
	}
	
	/**
	 * 正确方法三,效率最高,其实不算去除
	 */
	@org.junit.Test
	public void remove6(){
		List<String> list2=new ArrayList<String>();
		long begin=System.currentTimeMillis();
		System.out.println(begin);
		for(String name:lists){
			if(name!=null&name.indexOf("张")<0){
				list2.add(name);
			}
		}
		long end=System.currentTimeMillis();
		System.out.println(end);
		System.out.println(end-begin);
		//System.out.println(list2);
	}
	
	

}

 

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
java面试题真的很多,下面我来回答一个有关多线程的问题。 在Java实现多线程有两种方式,一种是继承Thread类,另一种是实现Runnable接口。这两种方式有何区别? 继承Thread类的方式是直接定义一个类继承Thread,并重写它的run()方法。然后创建该类的对象,并调用对象的start()方法来启动线程。这种方式简单直接,但因为Java是单继承的,所以如果某个类已经继承了其他类,就不能再直接继承Thread类实现多线程。 实现Runnable接口的方式是定义一个类实现Runnable接口,并实现其唯一的抽象方法run()。然后创建Thread类的对象,将实现了Runnable的对象作为参数传递给Thread类的构造方法。最后调用Thread对象的start()方法来启动线程。这种方式灵活性更大,因为Java允许一个类实现多个接口,所以即使某个类已经继承了其他类,仍然可以通过实现Runnable接口来实现多线程。 另一个区别在于资源共享的问题。继承Thread类的方式,不管是数据还是方法,都是线程自己拥有的,不存在共享的情况。而实现Runnable接口的方式,多个线程可以共享同一个对象的数据和方法,因为多个线程共同操作的是同一个Runnable对象。 总结来说,继承Thread类方式简单直接,但只能通过单继承来实现多线程;实现Runnable接口方式更灵活,可以解决单继承的限制,并且多个线程可以共享同一个Runnable对象的数据和方法。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值