剑指offer_面试题3(一):找出数组中重复的数字

面试题3(一):找出数组中重复的数字

题目:在一个长度为n的数组里的所有数字都在0到n-1的范围内。数组中某些数字是重复的,但不知道有几个数字重复了,

也不知道每个数字重复了几次。请找出数组中任意一个重复的数字。例如,如果输入长度为7的数组{2, 3, 1, 0, 2, 5, 3},

那么对应的输出是重复的数字2或者3。


分析:这道题看上去很简单,但是要注意审题。明白特点后,还要找到针对不同的特殊用例,采用特殊的处理方法。尤其是处理超出范围的特例。


  • 基本的暴力解法
// 暴力解法  
import java.util.Arrays;         // 导入util包

public class repeat_num     {

	public static void main(String[] args)   { 
		int[] arr = {2, 3, 1, 0, 2, 5, 3 };
		int i = 0;
		int flag = 0;  // 系统默认是没有重复的数字,把flag置为0。
		int length_1 = arr.length;       // 注意这里调用length时,不要写成length( );
		Arrays.sort(arr);                // 对数组进行排序
		System.out.println(length_1);

//			for (int i : arr ) {               // 输出排序结果
//				System.out.println(i);
//			}
//		}

     // Arrays类的binarySearch()方法,可以使用二分搜索法来搜索指定的数组,以获得指定对象。该方法返回要搜索元素的索引值。binarySearch()方法提供多种重载形式,用于满足各种类型数组的查找需要。
	for (i = 1; i <= length_1 - 2; i++ ){
			int test = arr[i] ;
		    for (int j = i + 1; j <= length_1 - 1; j++ ){
				if (test == arr[j])  {
					 flag = 1; //   flag为1时有重复。
					break;
				}
		    }
	}
    System.out.println("flag = " + flag);  

    }
}


  • 巧妙降低时间复杂度的剑指offer解法
// 剑指offer提到的算法
public class repeat_num {
	public static boolean find_repeat_num(int[] numbers)    {
		boolean flag = false ;  // 判断有无重复数字
		int t = 0;     // 初始化中间变量t为0
		int m = 0 ;  // 初始化变量m
		int i = 0;     
		int length_1 = numbers.length;     // 初始化变量length_1作为数组numbers的长度
		// 先创建一个for循环,遍历一边
		for (i = 0; i < length_1; i++ )  {
			if (numbers[i] < 0 || numbers[i] > length_1 - 1 )   {   // 出现范围外的数字后,直接认为不含重复数字。
				flag = false;
				return flag ;
			}
		}
		for (i = 0; i < length_1 ; i++ )   {
			m = numbers[i];
			// 判断当前下标,是否等于元素的值
			if (numbers[i] == i) {
				continue;                  // 下标和元素值相等,分析下一个元素
			}

// 不要在for 循环里做这些事,在for循环外遍历一边,它不香吗?不会把复杂的编程n^2。
//			if (numbers[m] < 0|| m < 0 || numbers[m] > length_1 - 1 || m > length_1 - 1)   {   // 出现范围外的数字后,直接认为不含重复数字。
//				flag = false;
//				break;
//			}

			while (  !(numbers[i] == numbers[m])  )   {  // number[i]和number[m]不相等,进行交换
				   t = numbers[i];
				   numbers[i] = numbers[m];
				   numbers[m] = t;  	
				   m = numbers[i];         // 交换完数字后,记得更新m的值。
			}
	
			// 下标和元素值不相等
            // 判断是否进行交换。
		    if (numbers[i] == numbers[m])  {    // number[i]和number[m]相等,则找到一个重复的数字
				flag = true;   // 标志符置1,表示有重复数字
			}

/* if else 语句运行不同的代码段的前后顺序,也会对结果有影响,不能把这句else放在最后,会导致错误的结果。因为本来要多次重复运行的代码段,这里只运行了一次。所以采用上面的while  来运行更恰当  */
//			else 	 {                                   // number[i]和number[m]不相等,进行交换
//				t = numbers[i];
//				numbers[i] = numbers[m];
//				numbers[m] = t;
//			}
		
		}
		return flag;
	}

	public static void main(String[] args) {
		// 小bug,当出现无效测试用例,长度为n的数组中包含0~n-1之外的数字的时候,运行失败。
		int[]  arr = {2, 1,-10, 4, 0, 4, 1};
		boolean flag = false;
		 flag = find_repeat_num(arr);
		System.out.println("flag = " + flag );
	}
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值