题目:给你一个有序数组 nums ,请你 原地 删除重复出现的元素,使每个元素 只出现一次 ,返回删除后数组的新长度。
不要使用额外的数组空间,你必须在 原地 修改输入数组 并在使用 O(1) 额外空间的条件下完成。
示例1:
输入:nums = [1,1,2]
输出:2, nums = [1,2]
解释:函数应该返回新的长度 2 ,并且原数组 nums 的前两个元素被修改为 1, 2 。不需要考虑数组中超出新长度后面的元素。
/**
* count记录元素重复数 == 数组元素向前移动的步数
*/
public int removeDuplicates(int[] nums) {
int count = 0;
for (int i = 1; i < nums.length; i++) {
if(nums[i] == nums[i - 1]){
count ++;
}else {
nums[i - count] = nums[i];
}
}
return nums.length - count;
}
给定一个数组 prices ,其中 prices[i] 是一支给定股票第 i 天的价格。
设计一个算法来计算你所能获取的最大利润。你可以尽可能地完成更多的交易(多次买卖一支股票)。
示例:
输入: prices = [7,1,5,3,6,4]
输出: 7
解释: 在第 2 天(股票价格 = 1)的时候买入,在第 3 天(股票价格 = 5)的时候卖出, 这笔交易所能获得利润 = 5-1 = 4 。
随后,在第 4 天(股票价格 = 3)的时候买入,在第 5 天(股票价格 = 6)的时候卖出, 这笔交易所能获得利润 = 6-3 = 3
解法:
public static int maxProfit(int[] prices){
int total = 0;
for (int i = 0; i < prices.length -1; i++) {
//只需要计算低进高出的部分即可
if(prices[i] <= prices[i + 1]){
total = total + prices[i + 1] - prices[i];
}
}
return total;
}
问题:数组中元素向右轮转k个位置,其中k是非负数
案例1:
int[] array = {1,2,3,4,5,6,7,8,9,10};
k 旋转位数
k = 3
输出:[8,9,10,1,2,3,4,5,6,7]
public static int[] xuanZhuan(int[] array,int k){
int[] temp = array.clone();
for (int i = 0; i < array.length; i++) {
int a = (i + k) % array.length;
array[a] = temp[i];
}
return array;
}
给定一个整数数组,判断是否存在重复元素。
如果存在一值在数组中出现至少两次,函数返回 true 。如果数组中每个元素都不相同,则返回 false 。
示例 1:
输入: [1,2,3,1]
输出: true
示例 2:
输入: [1,2,3,4]
输出: false
示例3:
输入: [1,1,1,3,3,4,3,2,4,2]
输出: true
解法1:暴力
public static boolean method(int[] array){
if(array == null){
return true;
}
for (int i = 0; i < array.length; i++) {
for (int j = i + 1; j < array.length; j++) {
if(array[j] == array[i]){
return true;
}
}
}
return false;
}
解法2:
利用set集合特性
set中不能有重复的元素,添加相同元素失败
public static boolean containsDuplicate(int[] nums) {
Set<Integer> set = new HashSet<>();
for (int num : nums) {
if (!set.add(num))
return true;
}
return false;
}
解法3:
先排序后比较前后元素是否一致
public static boolean containsDuplicate(int[] nums) {
Arrays.sort(nums);
for(int i = 0; i < nums.length - 1; i++) {
if (nums[i] == nums[i + 1]) {
return true;
}
}
return false;
}
给定一个非空整数数组,除了某个元素只出现一次以外,其余每个元素均出现两次。找出那个只出现了一次的元素。
示例:
输入: [4,1,2,1,2]
输出: 4
思路:使用异或运算,将所有值进行异或
public int singleNumber(int[] nums) {
int reduce = 0;
for (int num : nums) {
reduce = reduce ^ num;
}
return reduce;
}
给你两个整数数组 nums1 和 nums2 ,请你以数组形式返回两数组的交集。返回结果中每个元素出现的次数,应与元素在两个数组中都出现的次数一致(如果出现次数不一致,则考虑取较小值)。可以不考虑输出结果的顺序。
示例:
输入:nums1 = [1,2,2,1], nums2 = [2,2]
输出:[2,2]
思路:先对两个数组进行排序,后判断
public static int[] intersect(int[] nums1, int[] nums2) {
Arrays.sort(nums1);
Arrays.sort(nums2);
ArrayList<Integer> list = new ArrayList<>();
int i = 0,j = 0;
while (i < nums1.length && j <nums2.length) {
if(nums1[i] < nums2[j]){
i ++;
}else if(nums1[i] > nums2[j]){
j ++;
}else {
list.add(nums1[i]);
i ++;
j ++;
}
}
int[] target = new int[list.size()];
for (int z = 0; z < list.size(); z++) {
target[z] = list.get(z);
}
return target;
}
给定一个由 整数 组成的 非空 数组所表示的非负整数,在该数的基础上加一。
最高位数字存放在数组的首位, 数组中每个元素只存储单个数字。
你可以假设除了整数 0 之外,这个整数不会以零开头。
示例:
输入:digits = [1,2,3]
输出:[1,2,4]
解释:输入数组表示数字 123。
数组所有元素看成是一个数字:123 + 1 = 124
9 + 1 = 10
99 + 1 = 100
235 + 1 = 236
解决:
public static int[] plusOne(int[] digits) {
for (int i = digits.length - 1; i >= 0; i--) {
//末位元素=9 则置0
if (digits[i] == 9) {
digits[i] = 0;
} else {
//元素不等于9 则加1
digits[i]++;
return digits;
}
}
//元素全为9 则扩充一位 首位置1
int[] temp = new int[digits.length + 1];
temp[0] = 1;
return temp;
}
给定一个数组 nums,编写一个函数将所有 0 移动到数组的末尾,同时保持非零元素的相对顺序。
示例:
输入: [0,1,0,3,12]
输出: [1,3,12,0,0]
解法1:
public static void moveZeroes(int[] nums) {
//记录0出现个数
int pos = 0;
for (int i = 0; i < nums.length; i++) {
if(nums[i] == 0){
pos ++;
}else if(pos > 0){
//元素向前移动
nums[i - pos] = nums[i];
}
}
if(pos > 0){
//最后元素置0
for (int i = 1; i <= pos; i++) {
nums[nums.length - i] = 0;
}
}
}
解法2:
public static void moveZeroes2(int[] nums) {
int pos = 0;
//非0元素向前移动
for (int i = 0; i < nums.length; i++) {
if(nums[i] != 0){
nums[pos ++] = nums[i];
}
}
//最后num.length - pos 个的元素置0
while(pos < nums.length){
nums [pos++] = 0;
}
}
给定一个整数数组 nums 和一个整数目标值 target,请你在该数组中找出 和为目标值 target 的那 两个 整数,并返回它们的数组下标。
你可以假设每种输入只会对应一个答案。但是,数组中同一个元素在答案里不能重复出现。
你可以按任意顺序返回答案。
实例1:
输入:nums = [2,7,11,15], target = 9
输出:[0,1]
解释:因为 nums[0] + nums[1] == 9 ,返回 [0, 1] 。
解法1:暴力算法
//暴力算法
public static int[] twoSum(int[] nums, int target){
for (int i = 0; i < nums.length; i++) {
for (int j = i + 1; j < nums.length; j++) {
if(nums[i] + nums[j] == target){
return new int[]{i,j};
}
}
}
return new int[]{-1,-1};
}
解法2:HashMap
思路:
新建一个HashMap存储元素
存储方式:key(存储元素值):nums[i],value(存储数组下标):i
判断条件 m.get(target - nums[i]) != null,其中target - nums[i]得到是与之匹配的第二个元素值,假如能HashMap中找到对应的key则说明存在该两个元素之和等于target,同时也可得到对应元素的数组下标。
public int[] twoSum(int[] nums, int target) {
Map<Integer, Integer> m = new HashMap<>();
for (int i = 0; i < nums.length; i++) {
if (m.get(target - nums[i]) != null) {
return new int[]{m.get(target - nums[i]), i};
}
m.put(nums[i], i);
}
return new int[]{0, 0};
}
请你判断一个 9 x 9 的数独是否有效。只需要 根据以下规则 ,验证已经填入的数字是否有效即可。
数字 1-9 在每一行只能出现一次。
数字 1-9 在每一列只能出现一次。
数字 1-9 在每一个以粗实线分隔的 3x3 宫内只能出现一次。(请参考示例图)
注意:
一个有效的数独(部分已被填充)不一定是可解的。
只需要根据以上规则,验证已经填入的数字是否有效即可。
空白格用 ‘.’ 表示。
示例1:
输入:
char[][] board = {{'8','3','.','.','7','.','.','.','.'}
,{'6','.','.','1','9','5','.','.','.'}
,{'.','9','8','.','.','.','.','6','.'}
,{'8','.','.','.','6','.','.','.','3'}
,{'4','.','.','8','.','3','.','.','1'}
,{'7','.','.','.','2','.','.','.','6'}
,{'.','6','.','.','.','.','2','8','.'}
,{'.','.','.','4','1','9','.','.','5'}
,{'.','.','.','.','8','.','.','7','9'}};
输出:false
解法1:暴力解
思路:
public static boolean isValidSudoku(char[][] board){
//存储行的二维数组
int[][] line = new int[board.length][board[0].length];
//存储列的二维数组
int[][] col = new int[board.length][board[0].length];
//存储单元的二维数组
int[][] box = new int[board.length][board[0].length];
for (int i = 0; i < board.length; i++) {
for (int j = 0; j < board[0].length; j++) {
//board二维字符数组中'.'表示没有元素,所以可以直接跳过
if(board[i][j] == '.')continue;
//数组下标是0-8 所以pos代表的是排序后元素的位置
//board[i][j] - '1'的区间是[0-8](不懂的小伙伴可以查看ASCII表对应值),对应这排序后的数组下标
int pos = board[i][j] - '1';
//首先将9x9的宫格分成9个3x3的单元格,从左到右排序
//boxIndex代表的是从左到右的单元格的位置,boxIndex的范围是[0,8]
int boxIndex = i/3 * 3 + j/3;
//判断二维数组中位置是否已经被占用(二维数组中没有元素的值默认为0)
if((line[i][pos] != 0) || (col[j][pos] != 0) || (box[boxIndex][pos] != 0))return false;
//位置没有被占用,即可设置固定值 1 表示该位置已经被占用
line[i][pos] = col[j][pos] = box[boxIndex][pos] = 1;
}
}
return true;
}
给定一个 n × n 的二维矩阵 matrix 表示一个图像。请你将图像顺时针旋转 90 度。
你必须在 原地 旋转图像,这意味着你需要直接修改输入的二维矩阵。请不要 使用另一个矩阵来旋转图像。
示例1:
输入:matrix = [[1,2,3],[4,5,6],[7,8,9]]
输出:[[7,4,1],[8,5,2],[9,6,3]]
解法1:
思路:首先将二维矩阵上下对称交换,然后按照对角线交换
123 789 741
456 对称交换---> 456 对角线交换---> 852
789 123 963
public static void rotate(int[][] matrix) {
//调换位置 上下
for (int i = 0; i < matrix.length / 2; i++) {
for (int j = 0; j < matrix[0].length; j++) {
int temp = matrix[i][j];
matrix[i][j] = matrix[matrix.length - 1 - i][j];
matrix[matrix.length - 1 - i][j] = temp;
}
}
//调换位置 对角线
for (int i = 0; i < matrix.length; i++) {
for (int j = 0; j < matrix[0].length; j++) {
if(i == j)break;
int temp = matrix[i][j];
matrix[i][j] = matrix[j][i];
matrix[j][i] = temp;
}
}
}