数据结构与算法之排序: Leetcode 922. 按奇偶排序数组 II (Typescript版)

按奇偶排序数组 II

描述

  • 给定一个非负整数数组 nums, nums 中一半整数是 奇数 ,一半整数是 偶数 。

  • 对数组进行排序,以便当 numsi 为奇数时,i 也是 奇数 ;当 numsi 为偶数时, i 也是 偶数 。

  • 你可以返回 任何满足上述条件的数组作为答案 。

示例 1

复制代码
输入:nums = [4,2,5,7]
输出:[4,5,2,7]
解释:[4,7,2,5],[2,5,4,7],[2,7,4,5] 也会被接受。

示例 2

复制代码
输入:nums = [2,3]
输出:[2,3]

提示

  • 2 <= nums.length <= 2 * 1 0 4 10^4 104
  • nums.length 是偶数
  • nums 中一半是偶数
  • 0 <= numsi <= 1000

进阶

  • 可以不使用额外空间解决问题吗?

算法实现

1 )原生sort + 一般方法

js 复制代码
function sortArrayByParityII(nums: number[]): number[] {
    // 进行升序排序
    nums.sort((a, b) => a - b);
    // 声明一个空数组用来存储奇偶排序后的数组
    const res: number[] = [];
    // 记录奇数、偶数位下标
    let odd: number = 1;
    let even: number = 0;
    // 对数组进行遍历
    nums.forEach(item => {
        if (item % 2 === 1) {
            res[odd] = item;
            odd += 2;
        } else {
            res[even] = item;
            even += 2;
        }
    })
    return res;
};
  • 基于题目,我们需要知道,一半整数是奇数,另一半整数是偶数,可知,整个数组长度是偶数(数学基础)
  • 先整体排好序了,再基于原数组分别处理位置的问题,这里多使用了 res 的空间

2 )两次遍历 + & 运算判断奇偶对位存放

ts 复制代码
function sortArrayByParityII(nums: number[]): number[] {
    const ans: number[] = new Array(nums.length);
    let i: number = 0; // 默认i以0开始
    for (const x of nums) {
        if (!(x & 1)) {
            ans[i] = x;
            i += 2;
        }
    }
    i = 1; // 这里 i 从1开始
    for (const x of nums) {
        if (x & 1) {
            ans[i] = x;
            i += 2;
        }
    }
    return ans;
};
  • 这是官方提供方案,题目没有要求数组完全有序
  • 注意,"&"的运算方法:
    • 两个数值的个位分别相与,同时为1才得1,只要一个为0就为0。
    • 可以利用 & 的运算特性,来判断一个数是否为偶数。
    • 需要注意条件是 非负整数
  • 但是,这里仍然开辟了 新的 ans 数组空间

3 )双指针,奇偶交换

js 复制代码
function sortArrayByParityII(nums: number[]): number[] {
    const n: number = nums.length;
    let j: number = 1; // j奇数从1开始
    // i 索引从0开始,偶数,每次递增2,仍为偶数
    for (let i: number = 0; i < n; i += 2) {
        // i是偶数, 如果当前 nums[i] 是奇数, 注意,题目要求 nums[i] 最终也是偶数
        if (nums[i] & 1) {
            // 如果 nums[j]是奇数, j递增2,j仍旧奇数
            while (nums[j] & 1) {
                j += 2;
            }
            // 跳出while时, nums[j]为偶数了,交换 nums[i](当前奇数, 应为偶数) 和 nums[j](当前偶数, 应为奇数)
            [nums[i], nums[j]] = [nums[j], nums[i]];
        }
    }   
    return nums;
};
  • 这也是官方提供的一个解法,符合题意要求
  • 题目没有要求数组完全有序
  • 做题的步骤和思路都写在上面注释上了
相关推荐
八解毒剂16 分钟前
查找-从二分查找到二叉排序树
数据结构·c++·算法
程序猿追1 小时前
画个饼,给数据点颜色看看——在 HarmonyOS 模拟器上手搓一个饼图/环形图组件
深度学习·算法·harmonyos
net3m331 小时前
mymalloc函数里增加memset来初始化数据 全为0,能解决一些奇怪的问题,
算法
计算机安禾1 小时前
【算法分析与设计】第43篇:空间复杂度类与Savitch定理
java·服务器·网络·数据库·算法
圣保罗的大教堂1 小时前
leetcode 3751. 范围内总波动值 I 中等
leetcode
8Qi82 小时前
LeetCode 416:分割等和子集 —— (0-1背包)
java·算法·leetcode·动态规划·背包问题·01背包
智者知已应修善业2 小时前
【51单片机数码管驱动2位显示0-99按键3短按+1长按+10按键4短按-1长按清零,按键不影响数码管显示】2023-8-16
c++·经验分享·笔记·算法·51单片机
rou2 小时前
Stream Response
算法
_Evan_Yao2 小时前
如何高效刷LeetCode?大一版“从零开始”
算法·leetcode·职场和发展
吴可可1232 小时前
Win7 64位是CAD2014+C#开发最佳系统
算法