字节跳动Java一面面经

  • 知识点
  • 算法题



实在是自己太沉,跳不动啊


时间:2019.08.19

知识点

1、Java集合类(源码!!!HashMap、HashSet、HashTable、ConcurrentHashMap)。
2、线程安全,synchronized。
3、JVM的一系列命令。
4、数据库的索引、引擎、隔离级别、怎么查看死锁,共享锁排他锁,乐观锁。
5、Https为什么安全,流程是什么。
6、线程的状态,Thread的sleep方法,yield方法。
7、Java锁,完全不会系列。

算法题

思路部分源于面试后的查询。

1、一堆数字,其中有一个数字的数量超过了一半,找出这个数字。

思路:(1)遍历,哈希记录,没啥可说的。
(2)排序:找中位数。
(3)快速查找的思路:长度为n,如果所选数字的位置在n/2处,则为中位数,输出即可;如果其位置大于n/2,则在左边继续找,否则在右边继续找。
(4)根据数组的特点,很nice的思路:计数,首先从头开始,记录开始数字并记录次数,下一个数字如果重复则次数加一,不重复数字减一。如果次数为0了,重新记录下一次出现的数字,并记录其次数为1,继续计数。由于有一个数字出现的次数大于一半,遍历到最后的时候,剩下的只能是次数大于一半的那个数字。

2、一个数组,只有一个数字出现奇数次,其余数字出现偶数次,求出现奇数次的数字。

思路:异或,将所有数字进行异或,出现偶数次的数字异或结果为0,出现奇数次的数字x可以看作是偶数次+1次,其偶数次的异或结果为0,0异或剩下的一个x结果为x。

3、输出数字的全排列,比如:
输入:3
输出:
[1, 2, 3]
[1, 3, 2]
[2, 1, 3]
[2, 3, 1]
[3, 1, 2]
[3, 2, 1]

思路:递归,以3为例,可以看作是:第一个数字是1和其他两个数字的全排列+第一个数字是2和其他两个数字的全排列+第一个数字是3和其他两个数字的全排列。
代码:

import java.util.ArrayList;
import java.util.Arrays;

public class Main {
	public ArrayList<int[]> result = new ArrayList<int[]>();
	/*
	 * 数组的全排列
	 */
	public void allPerm(int[] nums, int start, int len) {
		if(start >= len-1) {
			// 也可以在此处直接输出
			// System.out.println(Arrays.toString(nums));
			result.add(nums.clone());  // 此处不能直接添加nums,如果是nums,添加就是nums的引用,可以去掉clone()看一下输出
			return;
		}
		for(int i = start; i < len; i++) {
			swap(nums, start, i);  // 一次将元素放在开始的位置
			allPerm(nums, start+1, len);  // 递归
			swap(nums, start, i);  // 将元素放回原样,进行下一次遍历
		}
	}
	/*
	 * 数组的两个位置的元素进行交换
	 */
	public void swap(int[] nums, int i, int j) {
		int tmp = nums[i];
		nums[i] = nums[j];
		nums[j] = tmp;
	}
	/*
	 * 用整数生成数组调用全排列
	 */
	public void useAllPerm(int n) {
		int[] nums = new int[n];
		for(int i = 1; i <= n; i++) {
			nums[i-1] = i;
		}
		allPerm(nums, 0, n);
		for(int i = 0; i < result.size(); i++) {
			System.out.println(Arrays.toString(result.get(i)));
		}
	}
	public static void main(String[] args) {
		// TODO Auto-generated method stub
		Main test = new Main();
		test.useAllPerm(4);
	}
}