(剑指Offer)面试题14:调整数组顺序使奇数位于偶数前面

题目:

输入一个整数数组,实现一个函数来调整该数组中数字的顺序,使得所有的奇数位于数组的前半部分,所有的偶数位于位于数组的后半部分,并保证奇数和奇数,偶数和偶数之间的相对位置不变。

如果去掉约束条件:并保证奇数和奇数,偶数和偶数之间的相对位置不变?

思路:

如果要保证奇数和奇数,偶数和偶数之间的相对位置不变,那么需要开辟新的空间来保存奇数和偶数,方法有两种:

1、开辟一个新数组,遍历第一遍原数组,将奇数依次写入新数组,遍历第二遍原数组,将偶数依次写入新数组,返回新数组;

2、开辟两个数组,遍历一遍原数组,分别将奇数和偶数存入两个新数组,再将奇数数组和偶数数组分别写入原数组,返回原数组;

如果无需保证奇数和奇数,偶数和偶数之间的相对位置不变,那么可以通过前后遍历,奇偶对换的方法来实现:

维护两个指针,分别指向数组的首尾,然后一个向后一个向前,在两个指针相遇之前,如果第一个指针指向偶数,而第二个指针指向奇数,那么就交换这两个数。(无需开辟新空间)

类似题目:

  • 将题目改成将数组分成两部分,所有负数在非负数前面;
  • 将题目改成将数组分成两部分,能被3整除的在不能被3整除的前面;

为了让代码的扩展性更强,可以将这些判断条件单独写成一个函数接口,而主代码框架保持不变。

代码:

#include <iostream>
#include <vector>

using namespace std;

// not changed the relative position
void ReOrderOddEven_1(vector<int> &array){
    if(array.size()<=0)
        return;
    vector<int> odds;
    vector<int> evens;
    unsigned int i;
    for(i=0;i<array.size();i++){
        if((array[i]&0x1)!=0)
            odds.push_back(array[i]);
        else
            evens.push_back(array[i]);
    }
    for(i=0;i<odds.size();i++)
        array[i]=odds[i];
    int k=i;
    for(i=0;i<evens.size();i++)
        array[k++]=evens[i];
}

// changed the relative position
void ReOrderOddEven_2(int *pdata,unsigned int length){
    if(pdata==NULL || length<=0)
        return;
    int *pBegin=pdata;
    int *pEnd=pdata+length-1;

    while(pBegin<pEnd){
        while(pBegin<pEnd && (*pBegin&0x1)!=0)
            pBegin++;
        while(pBegin<pEnd && (*pEnd&0x1)==0)
            pEnd--;
        if(pBegin<pEnd){
            int tmp=*pBegin;
            *pBegin=*pEnd;
            *pEnd=tmp;
        }
    }
}

bool isEven(int n){
    return (n&1)==0;
}

// more expansible
void ReOrderOddEven_3(int *pdata,unsigned int length,bool (*func)(int)){
    if(pdata==NULL || length<=0)
        return;
    int *pBegin=pdata;
    int *pEnd=pdata+length-1;

    while(pBegin<pEnd){
        while(pBegin<pEnd && !func(*pBegin))
            pBegin++;
        while(pBegin<pEnd && func(*pEnd))
            pEnd--;
        if(pBegin<pEnd){
            int tmp=*pBegin;
            *pBegin=*pEnd;
            *pEnd=tmp;
        }
    }
}

int main()
{
    int A[]={1,2,3,4,5,6,7,8,9};
    int length=sizeof(A)/sizeof(A[0]);
    ReOrderOddEven_2(A,length);
    for(int i=0;i<length;i++)
        cout<<A[i]<<" ";
    cout<<endl;

    int B[]={1,2,3,4,5,6,7,8,9};
    vector<int> C(B,B+length);
    ReOrderOddEven_1(C);
    for(int i=0;i<length;i++)
        cout<<C[i]<<" ";
    cout<<endl;

    ReOrderOddEven_3(B,length,isEven);
    for(int i=0;i<length;i++)
        cout<<B[i]<<" ";
    cout<<endl;
    return 0;
}

在线测试OJ:

http://www.nowcoder.com/books/coding-interviews/beb5aa231adc45b2a5dcc5b62c93f593?rp=1

AC代码:

class Solution {
public:
	void reOrderArray(vector<int> &array) {
	  if(array.size()<=0)
            return;
        vector<int> odds,evens;
        unsigned int i;
        for(i=0;i<array.size();i++){
            if((array[i]&0x1)!=0)
               odds.push_back(array[i]);
            if((array[i]&0x1)==0)
               evens.push_back(array[i]);
        }

        for(i=0;i<odds.size();i++)
            array[i]=odds[i];
        int k=i;
        for(i=0;i<evens.size();i++)
            array[k++]=evens[i];
	}
};

  

时间: 2024-10-10 20:08:08

(剑指Offer)面试题14:调整数组顺序使奇数位于偶数前面的相关文章

【剑指offer】Q14:调整数组顺序使奇数位于偶数前面

def isOdd(n): return n & 1 def Reorder(data, cf = isOdd): odd = 0 even = len( data ) - 1 while True: while not isOdd( data[ even ]) : even -= 1 while isOdd( data[ odd ]) : odd += 1 if odd >= even: break data[ even ], data[ odd ] = data[ odd ], data

《剑指Offer》题目:调整数组顺序使奇数位于偶数前面

题目描述:调整数组顺序使奇数位于偶数前 输入一个整数数组,实现一个函数来调整该数组中数字的顺序,使得所有的奇数位于数组的前半部分,所有的偶数位于位于数组的后半部分,并保证奇数和奇数,偶数和偶数之间的相对位置不变. 题目分析: 题目的关键在于在移位以后,奇数和奇数以及偶数和偶数之间的相对位置保持不变. Java代码: public class ReOrderArray { //这种方法不能保证奇数和奇数,偶数和偶数之间的相对位置不变 public static void reOrderArray(

【剑指offer】八,调整数组顺序使奇数位于偶数前面

题目描述 输入一个整数数组,实现一个函数来调整该数组中数字的顺序,使得所有的奇数位于数组的前半部分,所有的偶数位于位于数组的后半部分,并保证奇数和奇数,偶数和偶数之间的相对位置不变. 分析:此题在oj上练习时多加了条件,即调整后的数字之间的顺序是不变的.目前只想到一种方法,后续有其他方法再做添加.新建一个数组,扫描原数组两遍,第一遍取出原数组中所有的奇数,存入新数组中.第二遍找出所有的偶数存入新数组中.代码实现如下: 1 public class Solution { 2 public void

剑指offer_面试题14_调整数组顺序使奇数位于偶数前面(函数指针用法)

题目:输入一个整数数组,实现一个函数来调整该数组中数字的顺序,使得所有奇数位于数组的前半部分,所有偶数位于数组的后半部分. 1.一般想法,不考虑时间复杂度: 每次遍历数组,碰到一个偶数就拿出来,将后面所有数字向前挪动一位,在将该偶数放到最后. 2.利用冒泡排序的思想,两个指针,一前以后,如果前为偶数,后为奇数,就交换. 算法如下: void Reorder_array(int p[],int length) { if(NULL == p || length <= 0) return; int i

面试题 14:调整数组顺序使奇数位于偶数前面

使用两个指针,在数组头尾相对移动: 循环结束条件:头和尾重叠活着头在尾之后 左边指针右移条件:当前数是奇数 右边指针左移条件:当前数是偶数 当且仅当左边指针是偶数,右边指针是奇数,交换两个指针的值 此题注意扩展,函数功能的重用. package offer; import java.util.Arrays; /*面试题 14:调整数组顺序使奇数位于偶数前面 题目:输入一个整数数组,实现一个函数来调整该函数数组中数字的顺序,使得所有奇数位于数组的前半部分,所有的数组位于数组的后半部分.*/ pub

面试题14:调整数组顺序使奇数位于偶数前面

// 面试题14_调整数组顺序使奇数位于偶数前面.cpp : 定义控制台应用程序的入口点. // #include "stdafx.h" #include <iostream> using namespace std; void swap(int *begin,int *end) { int temp; temp=*begin; *begin=*end; *end=temp; } void ReorderOddEven(int *pData,unsigned int len

剑指offer编程题Java实现——面试题14调整数组顺序使奇数位于偶数之前

题目: 输入一个整数数组,实现一个函数来调整该数组中数组的顺序,使得所有的奇数位于数组的前半部分,偶数位于数组的后半部分. 解题思路:数组中维护两个指针,第一个指针初始化时候指向数组头部,第二个指针初始化时候指向数组尾部,第一个指针指向的数字总是偶数,第二个指针指向的数字总是奇数,如果第一个指针在第二个指针之前,则交换两指针指向的元素. 1 package Solution; 2 3 /** 4 * 剑指offer面试题14:调整数组顺序是奇数位于偶数前面 5 * 题目:输入一个整数数组,实现一

剑指offer-面试题14.调整数组顺序使奇数位于偶数的前面

题目:输入一个整数数组,实现一个函数来调整该数组中的数字的顺序, 使得所有的奇数位于数组的前半部门,所有的偶数位于数组的后半部门. 这一题一看其实是很简单的,当然在不考虑效率的情况可以这样考虑 我们将数组从前开始索引只要发现偶数这将偶数取出来并且将偶数后 面的元素全部前移一个然后将偶数插入到最后,这样扫描一遍之后 就将偶数全部放在了后半部门,奇数放在了前半部分.但是效率不好 每次遇到偶数都要移动偶数后面的所有元素再加上遍历数组的时间 则达到了O(n2) 我们可以用两个索引或者指针,一个指向数组的

剑指offer 面试14—调整数组顺序使奇数位于偶数前面

题目: 输入一个整数数组,实现一个函数来调整该数组中数组的顺序,使得所有奇数位于数组的前半部分,所有偶数位于数组的后半部分. 解法一:O(n^2) 从头扫描数组,每碰到一个偶数,拿出这个数字,并把位于这个数字后面的所有数字往前挪动一位.挪完之后在数组的末尾有一个空位,这时把该偶数放入这个空位中. 由于每碰到一个偶数就需要移动O(n)个数字,因此总的时间复杂度是O(n^2). 解法二:O(n) 类似快排,两个指针,一个指向数组头,只向后移动,一个指向数组尾,只向前移动.如果第一个指针指向的数字是偶

【剑指Offer】面试题21. 调整数组顺序使奇数位于偶数前面

题目 输入一个整数数组,实现一个函数来调整该数组中数字的顺序,使得所有奇数位于数组的前半部分,所有偶数位于数组的后半部分. 示例: 输入:nums =?[1,2,3,4] 输出:[1,3,2,4] 注:[3,1,2,4] 也是正确的答案之一. 提示: 1 <= nums.length <= 50000 1 <= nums[i] <= 10000 思路 代码 时间复杂度:O(n) 空间复杂度:O(1) class Solution { public: vector<int>