跳到主要内容

算法基础面试题(二)

数据范围

整数类型

byte:8 位,范围为 -128 到 127

short:16 位,范围为 -32,768 到 32,767 3万

int:32 位,范围为 -2^31 到 2^31-1 21亿, 4bytes,100w int占4M,1亿int 400M.

long:64 位,范围为 -2^63 到 2^63-1

4G内存可以表示340亿bits, 1G内存可以表示85亿bits

100w int占4M, 1亿int 400M.

1亿bytes 100M, 4亿bytes 400M。

1亿bits 12M,4亿bits 48M内存

浮点类型

float:32 位,范围为 IEEE 754 单精度浮点数表示的取值范围。4bytes,100w float占4M,1亿 400M

double:64 位,范围为 IEEE 754 双精度浮点数表示的取值范围。

字符类型

char:16 位 Unicode 字符,范围为 0 到 65,535。 6万

java中理论上一个字符占用两个字节

Unicode只是一个编码规范,目前实际实现的unicode编码只要有三种:UTF-8,UCS-2和UTF-16,三种unicode字符集之间可以按照规范进行转换。

UTF-8最大的一个特点,就是它是一种变长的编码方式。它可以使用1~6个字节表示一个符号,根据不同的符号而变化字节长度

占2个字节的:〇(〇有两个读音 xīng líng,(一) xīng 同“星”;(二) líng 同“零”。)

占3个字节的:基本等同于GBK,含21000多个汉字

占4个字节的:中日韩超大字符集里面的汉字,有5万多个

一个utf8数字占1个字节

一个utf8英文字母占1个字节

布尔类型

boolean:表示逻辑值,只能取 true 或 false。

java.lang.Integer

MIN_VALUE = 0x80000000 -2^31

MAX_VALUE = 0x7fffffff 2^31-1

Integer.toBinaryString(int)方法将一个整数转换成二进制表示的字符串

Integer.parseInt(binaryString, 2) 把一个二进制字符串转换成为整数

Integer.reverse(123) ---按位反转整数

public class Solution {
// you need treat n as an unsigned value
public int reverseBits(int n) {
int rev =0;
for(int i=0; i<32 && n!=0; i++){
rev |=(n&1)<<(31-i);
n >>>=1;
}
return rev;
}
}
public class Solution {
private static final int M1 = 0x55555555; // 01010101010101010101010101010101
private static final int M2 = 0x33333333; // 00110011001100110011001100110011
private static final int M4 = 0x0f0f0f0f; // 00001111000011110000111100001111
private static final int M8 = 0x00ff00ff; // 00000000111111110000000011111111

public int reverseBits(int n) {
//交换不同的位置
n = n >>> 1 & M1 | (n & M1) << 1;
n = n >>> 2 & M2 | (n & M2) << 2;
n = n >>> 4 & M4 | (n & M4) << 4;
n = n >>> 8 & M8 | (n & M8) << 8;
return n >>> 16 | n << 16;
}
}

Integer.bitCount(122) --统计int中1的个数

    public static int bitCount(int i) {
// HD, Figure 5-2
i = i - ((i >>> 1) & 0x55555555);
i = (i & 0x33333333) + ((i >>> 2) & 0x33333333);
i = (i + (i >>> 4)) & 0x0f0f0f0f;
i = i + (i >>> 8);
i = i + (i >>> 16);
return i & 0x3f;
}

Integer.highestOneBit(123) ---最高位的1的值

    public static int highestOneBit(int i) {
// 先把i最高位1以下的所有位都变成1。折半法
i |= (i >> 1);
i |= (i >> 2);
i |= (i >> 4);
i |= (i >> 8);
i |= (i >> 16);
//用移位运算得到最高位的i
return i - (i >>> 1);
}

Integer.lowestOneBit(3) ---最低位的1的值

public static int lowestOneBit(int i) {
// HD, Section 2-1
return i & -i;
}

Integer.numberOfLeadingZeros(3) ---前面0的个数

public static int numberOfLeadingZeros(int i) {
// HD, Figure 5-6
if (i == 0)
return 32;
int n = 1;
//折半法
if (i >>> 16 == 0) { n += 16; i <<= 16; }
if (i >>> 24 == 0) { n += 8; i <<= 8; }
if (i >>> 28 == 0) { n += 4; i <<= 4; }
if (i >>> 30 == 0) { n += 2; i <<= 2; }
//如果前面条件都不满足
n -= i >>> 31;
return n;
}

Integer.numberOfTrailingZeros(3) ---后面0的个数

    public static int numberOfTrailingZeros(int i) {
// HD, Figure 5-14
int y;
if (i == 0) return 32;
int n = 31;
y = i <<16; if (y != 0) { n = n -16; i = y; }
y = i << 8; if (y != 0) { n = n - 8; i = y; }
y = i << 4; if (y != 0) { n = n - 4; i = y; }
y = i << 2; if (y != 0) { n = n - 2; i = y; }
return n - ((i << 1) >>> 31);
}

java.util.HashMap

map.getOrDefault(a,b);

map.remove(key);

map.containsKey(key)

map.values()

java.util.HashSet

把List转成Set

List list = Arrays.asList("apple", "orange", "banana");

Set set = new HashSet<>(list);

List list = Arrays.asList("apple", "orange", "banana");

Set set = new HashSet<>();

set.addAll(list);

java.util.List

list.add("apple"); // 在末尾添加元素

list.add(0, "banana");

list.remove("apple"); // 删除元素"apple"

list.remove(1);

list.clear();

//list转数组:

list.toArray(new int[0])

list.toArray(new int[0][2])

//list to数组字符串:

list.toString() [1, 2]

Array

[]:

int[] array = {1, 2, 3, 4, 5};

int length = array.length;

java.util.Arrays

List wordList = Arrays.asList(s.split("\\s+")); ----数组变成list

升序Arrays.sort

降序:Arrays.sort(arr, (o1, o2) -> o2 - o1); ---注意如果要对sort进行反向排序,要确保arr不是基础类型的数组,必须是对象类型的数组。

如果对int进行排序,两个int的减法运算可能溢出,需要将其转换成long来计算,compare的返回值是1或者-1.

Arrays.sort(points, (a, b)-> (long)a[1]-(long)b[1]>0?1:-1) ------int[][]是可以的,但是int[]不行。

Arrays.equals(str1,str2) ---两个数组相等

System.arraycopy(src, srcPos, dest, destPos, length) ---拷贝数组

byte[] partArray = Arrays.copyOfRange(fullArray, start, end) ---end不包含,长度可以大于原始数组的长度

byte[] mergedArray = Arrays.copyOf(byteArray1, byteArray1.length + byteArray2.length) ---拷贝数组,长度可以大于原始数组的长度

Arrays.fill(dp, max) ---填充数组

Integer[] a = new Integer[]{1,2,35,34,21};

Arrays.toString(a) ----数组转成字符串。[35, 34, 21, 2, 1]

byte转字符:直接用(char)a

byte数组转字符串:byte[] byteArray = {65, 66, 67}; String result = new String(byteArray);

打印数组

Arrays.stream(arr).forEach(System.out::println);

String

char[] str1 = s.toCharArray() ---字符串转成char数组

s.charAt(1)

str.substring(0, 3) ----不包含end

str.length()

int转String:String s = String.valueOf(x);

String.join(" ",d) ----合并集合,变成字符串。只要是Iterable的都可以join。比如List,Set,但是不能是数组

mainString.contains(subString)

"abc".indexOf("a")

"abc".lastIndexOf("a")

StringBuffer

ans.reverse() ---反转字符

ans.toCharArray() ---转换成char数组

修改其中的某个字符

StringBuffer sb = new StringBuffer(curr); sb.setCharAt(j, keys[k]); String next = sb.toString();

java.util.Random

new Random().nextInt(10)

java.util.Collections

List<Integer> intList = new ArrayList<>();
intList.add(12);
intList.add(15);
Collections.swap(intList, 0, 1); //swap list里面的两个元素

反转数组:

Integer[] arr = {1, 2, 3, 4, 5};  
Collections.reverse(Arrays.asList(arr)); //不能反转数组,需要把数组转换成list, 最后arr数组的值被反转
  1. 排序 (sort):

    List<Integer> numbers = new ArrayList<>(List.of(3, 1, 4, 1, 5, 9, 2, 6, 5));
    Collections.sort(numbers);
    System.out.println(numbers); // 输出 [1, 1, 2, 3, 4, 5, 5, 6, 9]
  2. 反转 (reverse):

    List<String> names = new ArrayList<>(List.of("Alice", "Bob", "Charlie", "David"));
    Collections.reverse(names);
    System.out.println(names); // 输出 [David, Charlie, Bob, Alice]
  3. 打乱 (shuffle):

    List<Character> letters = new ArrayList<>(List.of('A', 'B', 'C', 'D', 'E'));
    Collections.shuffle(letters);
    System.out.println(letters); // 输出可能为 [C, A, B, E, D]
  4. 查找最大值和最小值 (maxmin):

    List<Double> prices = new ArrayList<>(List.of(19.99, 29.99, 15.99, 39.99, 9.99));
    double maxPrice = Collections.max(prices);
    double minPrice = Collections.min(prices);
    System.out.println("Max Price: " + maxPrice); // 输出 Max Price: 39.99
    System.out.println("Min Price: " + minPrice); // 输出 Min Price: 9.99
  5. 替换所有元素 (replaceAll):

    List<String> words = new ArrayList<>(List.of("apple", "banana", "cherry"));
    Collections.replaceAll(words, "banana", "orange");
    System.out.println(words); // 输出 [apple, orange, cherry]
  6. 查找元素位置 (binarySearch):在列表中进行二分查找,该列表必须是按升序排序的,如果未找到,则返回一个负值

    List<Integer> sortedNumbers = new ArrayList<>(List.of(1, 2, 3, 4, 5, 6, 7, 8, 9)); //必须是排序数组
    int index = Collections.binarySearch(sortedNumbers, 5);
    System.out.println("Index of 5: " + index); // 输出 Index of 5: 4

Character

Character.isDigit,

Character.isLetterOrDigit,

Character.isLetter,

Character.toLowerCase

java.util.Stack,PriorityQueue

Stack<Integer> stack = new Stack<>();
stack.push(1);
int top = stack.peek();
int poppedElement = stack.pop();
PriorityQueue<Float> minHeap = new PriorityQueue<>(k);

Deque

Deque<Character> stack = new LinkedList<Character>();
Deque<Integer> deque = new ArrayDeque<Integer>();

add/remove源自集合,所以添加到队尾,从队头删除;
offer/poll源自队列(先进先出 => 尾进头出),所以添加到队尾,从队头删除;
push/pop源自栈(先进后出 => 头进头出),所以添加到队头,从队头删除;
offerFirst/offerLast/pollFirst/pollLast源自双端队列(两端都可以进也都可以出),根据字面意思,offerFirst添加到队头,offerLast添加到队尾,pollFirst从队头删除,pollLast从队尾删除。
总结:
add/offer/offerLast添加队尾,三个方法等价;
push/offerFirst添加队头,两个方法等价。
remove/pop/poll/pollFirst删除队头,四个方法等价;
pollLast删除队尾。

ConcurrentLinkedDeque(线程安全)

ConcurrentLinkedDeque<String> deque = new ConcurrentLinkedDeque<>();
deque.add("element1");
deque.addLast("element2"); // 添加到队列的末尾
deque.addFirst("element3"); // 添加到队列的头部
String firstElement = deque.getFirst(); // 获取队列的头部元素,如果不存在则返回null
String lastElement = deque.getLast(); // 获取队列的尾部元素,如果不存在则返回null
deque.removeFirst(); // 删除队列的头部元素
deque.removeLast(); // 删除队列的尾部元素
if (deque.isEmpty()) {
// 队列为空时执行操作
}
for (String element : deque) {
// 遍历队列中的每个元素并执行操作
}

LinkedList, ArrayDeque, Queue

链表

反转链表:

//迭代反转
private void reverseLinkedList(ListNode head) {
// 也可以使用递归反转一个链表
ListNode pre = null;
ListNode cur = head;

while (cur != null) {
ListNode next = cur.next;
cur.next = pre;
pre = cur;
cur = next;
}
}

//迭代反转两个具体的节点
public ListNode[] myReverse(ListNode head, ListNode tail) {
ListNode prev = tail.next;
ListNode cur = head;
while (prev != tail) {
ListNode nex = cur.next;
cur.next = prev;
prev = cur;
cur = nex;
}
return new ListNode[]{tail, head};
}


//递归反转
ListNode reverse(ListNode head) {
if (head.next == null) return head;
ListNode last = reverse(head.next);
head.next.next = head;
head.next = null;
return last;
}

//反转前n个节点
ListNode successor = null; // 后驱节点

// 反转以 head 为起点的 n 个节点,返回新的头结点
ListNode reverseN(ListNode head, int n) {
if (n == 1) {
// 记录第 n + 1 个节点
successor = head.next;
return head;
}
// 以 head.next 为起点,需要反转前 n - 1 个节点
ListNode last = reverseN(head.next, n - 1);
head.next.next = head;
// 让反转之后的 head 节点和后面的节点连起来
head.next = successor;
return last;
}

//反转m-n之间的节点:
ListNode reverseBetween(ListNode head, int m, int n) {
// base case
if (m == 1) {
return reverseN(head, n);
}
// 前进到反转的起点触发 base case---只到m=1
head.next = reverseBetween(head.next, m - 1, n - 1);
return head;
}

点我查看更多精彩内容:www.flydean.com关注公众号加我好友
Loading Comments...