C++实现一维向量旋转算法

在《编程珠玑》一书的第二章提到了n元一维向量旋转算法(又称数组循环移位算法)的五种思路,并且比较了它们在时间和空间性能上的区别和优劣。本文将就这一算法做较为深入的分析。具体如下所示:

一、问题描述

将一个n元一维向量向左旋转i个位置。例如,假设n=8,i=3,向量abcdefgh旋转为向量defghabc。简单的代码使用一个n元的中间向量在n步内可完成该工作。你能否仅使用几十个额外字节的内存空间,在正比于n的时间内完成向量的旋转?

二、解决方案

思路一:将向量x中的前i个元素复制到一个临时数组中,接着将余下的n-i个元素左移i个位置,然后再将前i个元素从临时数组中复制到x中余下的位置。

性能:这种方法使用了i个额外的位置,如果i很大则产生了过大的存储空间的消耗。

C++代码实现如下:

/*************************************************************************
  > File Name: vector_rotate.cpp
  > Author: SongLee
 ************************************************************************/
#include<iostream>
#include<string>
using namespace std; 

int main()
{
  string s = "abcdefghijklmn";
  cout << "The origin is: " << s << endl;
  // 左移个数
  int i;
  cin >> i;
  if(i > s.size())
  {
    i = i%s.size();
  }
  // 将前i个元素临时保存
  string tmp(s, 0, i);
  // 将剩余的左移i个位置
  for(int  j=i; j<s.size(); ++j)
  {
    s[j-i] = s[j];
  }
  s = s.substr(0, s.size()-i) + tmp;
  cout << "The result is: "<< s << endl;
  return 0;
}

思路二:定义一个函数将x向左旋转一个位置(其时间正比于n),然后调用该函数i次。

性能:这种方法虽然空间复杂度为O(1),但产生了过多的运行时间消耗。

C++代码实现如下:

/*************************************************************************
  > File Name: vector_rotate_1.cpp
  > Author: SongLee
 ************************************************************************/
#include<iostream>
#include<string>
using namespace std; 

void rotateOnce(string &s)
{
  char tmp = s[0];
  int i;
  for(i=1; i<s.size(); ++i)
  {
    s[i-1] = s[i];
  }
  s[i-1] = tmp;
} 

int main()
{
  string s = "abcdefghijklmn";
  cout << "The origin is: " << s << endl;
  // 左移个数
  int i;
  cin >> i;
  if(i > s.size())
  {
    i = i%s.size();
  }
  // 调用函数i次
  while(i--)
  {
    rotateOnce(s);
  }
  cout << "The result is: "<< s << endl;
  return 0;
}

思路三:移动x[0]到临时变量t中,然后移动x[i]到x[0]中,x[2i]到x[i],依次类推,直到我们又回到x[0]的位置提取元素,此时改为从临时变量t中提取元素,然后结束该过程(当下标大于n时对n取模或者减去n)。如果该过程没有移动全部的元素,就从x[1]开始再次进行移动,总共移动i和n的最大公约数次。

性能:这种方法非常精巧,像书中所说的一样堪称巧妙的杂技表演。空间复杂度为O(1),时间复杂度为线性时间,满足问题的性能要求,但还不是最佳。

C++代码实现如下:

/*************************************************************************
  > File Name: vector_rotate_2.cpp
  > Author: SongLee
 ************************************************************************/
#include<iostream>
#include<string>
using namespace std; 

// 欧几里德(辗转相除)算法求最大公约数
int gcd(int i, int j)
{
  while(1)
  {
    if(i > j)
    {
      i = i%j;
      if(i == 0)
      {
        return j;
      }
    }
    if(j > i)
    {
      j = j%i;
      if(j == 0)
      {
        return i;
      }
    }
  }
} 

int main()
{
  string s = "abcdefghijklmn";
  cout << "The origin is: "<< s << endl;
  // 左移个数
  int i;
  cin >> i;
  if(i > s.size())
  {
    i = i%s.size();
  }
  // 移动
  char tmp;
  int times = gcd(s.size(), i);
  for(int j=0; j<times; ++j)
  {
    tmp = s[j];
    int pre = j; // 记录上一次的位置
    while(1)
    {
      int t = pre+i;
      if(t >= s.size())
        t = t-s.size();
      if(t == j) // 直到tmp原来的位置j为止
        break;
      s[pre] = s[t];
      pre = t;
    }
    s[pre] = tmp;
  }
  cout << "The result is: "<< s << endl;
  return 0;
}

思路四:旋转向量x实际上就是交换向量ab的两段,得到向量ba,这里a代表x的前i个元素。假设a比b短。将b分割成bl和br,使br的长度和a的长度一样。交换a和br,将ablbr转换成brbla。因为序列a已在它的最终位置了,所以我们可以集中精力交换b的两个部分了。由于这个新问题和原先的问题是一样的,所以我们以递归的方式进行解决。这种方法可以得到优雅的程序,但是需要巧妙的代码,并且要进行一些思考才能看出它的效率足够高。

//实现代码(略)

思路五:(最佳)将这个问题看做是把数组ab转换成ba,同时假定我们拥有一个函数可以将数组中特定部分的元素逆序。从ab开始,首先对a求逆,得到arb,然后对b求逆,得到arbr。最后整体求逆,得到(arbr)r,也就是ba。

reverse(0, i-1)  /*cbadefgh*/
reverse(i, n-1) /*cbahgfed*/
reverse(0, n-1) /*defghabc*/

性能:求逆序的方法在时间和空间上都很高效,而且代码非常简短,很难出错。

C++代码实现如下:

/*************************************************************************
  > File Name: vector_rotate.cpp
  > Author: SongLee
 ************************************************************************/
#include<iostream>
#include<string>
using namespace std; 

void reverse(string &s, int begin, int end)
{
  while(begin < end)
  {
    char tmp = s[begin];
    s[begin] = s[end];
    s[end] = tmp;
    ++begin;
    --end;
  }
} 

int main()
{
  string s = "abcdefghijklmn";
  cout << "The origin is: "<< s << endl; 

  int i;
  cin >> i;
  if(i > s.size())
  {
    i = i%s.size();
  } 

  reverse(s, 0, i-1);
  reverse(s, i, s.size()-1);
  reverse(s, 0, s.size()-1); 

  cout << "The result is: "<< s << endl;
  return 0;
}

三、扩展延伸

如何将向量abc旋转变成cba?

和前面的问题类似,此向量旋转对应着非相邻内存块的交换模型。解法很相似,即利用恒等式:cba = (arbrcr)r

注意:在面试或笔试时,如若出现向量旋转(内存块交换)问题,建议最好使用思路五答题,不仅高效而且简洁。

(0)

相关推荐

  • C++实现DES加密算法实例解析

    本文所述实例是一个实现DES加密算法的程序代码,在C++中,DES加密是比较常用的加密算法了,且应用非常广泛.本CPP类文件可满足你的DES加密需要,代码中附带了丰富的注释,相信对于大家理解DES可以起到很大的帮助. 具体实现代码如下: #include "memory.h" #include "stdio.h" enum {encrypt,decrypt};//ENCRYPT:加密,DECRYPT:解密 void des_run(char out[8],char

  • C++二进制翻转实例分析

    本文实例讲述了C++二进制翻转的方法,将常用的几种解决方法罗列出来供大家比较选择.具体如下: 首先来看看一个相对笨拙的算法: #include <iostream> using namespace std; void printBinary(unsigned char str, int size = 1) { int flag = 0x01; for (int i = 0; i < size; i++) { for (int i = 0; i < 8; i++) { if (str

  • C++实现矩阵原地转置算法

    本文实例描述了C++实现矩阵原地转置算法,是一个非常经典的算法,相信对于学习C++算法的朋友有很大的帮助.具体如下: 一.问题描述 微软面试题:将一个MxN的矩阵存储在一个一维数组中,编程实现矩阵的转置. 要求:空间复杂度为O(1) 二.思路分析 下面以一个4x2的矩阵A={1,2,3,4,5,6,7,8}进行分析,转置过程如下图: 图中右下角的红色数字表示在一维数组中的下标.矩阵的转置其实就是数组中元素的移动,具体的移动过程如下图: 我们发现,这些移动的元素的下标是一个个环,下标1的元素移动到

  • C++实现迷宫算法实例解析

    本文以实例形式描述了C++实现迷宫算法.本例中的迷宫是一个矩形区域,它有一个入口和一个出口.在迷宫的内部包含不能穿越的墙或障碍.障碍物沿着行和列放置,它们与迷宫的矩形边界平行.迷宫的入口在左上角,出口在右下角 本实例迷宫算法的功能主要有: 1.自动生成10*10迷宫图 2.判断是否有迷宫出口,并且画出路线图 具体实现代码如下: # include <iostream> # include <list> # include <sys/timeb.h> # include

  • 基于C++实现的各种内部排序算法汇总

    提起排序算法相信大家都不陌生,或许很多人已经把它们记得滚瓜烂熟,甚至随时可以写出来.是的,这些都是最基本的算法.这里就把各种内部排序算法总结归纳了一下,包括插入排序(直接插入排序,折半插入排序,希尔排序).交换排序(冒泡排序,快速排序).选择排序(简单选择排序,堆排序).2-路归并排序.(另:至于堆排序算法,前面已经有一篇文章针对堆排序的算法实现做了详细的描述) C++实现代码如下: /*******************************************************

  • C++线性时间的排序算法分析

    前面的文章已经介绍了几种排序算法,如插入排序(直接插入排序,折半插入排序,希尔排序).交换排序(冒泡排序,快速排序).选择排序(简单选择排序,堆排序).2-路归并排序(可以参考前一篇文章:各种内部排序算法的实现)等,这些排序算法都有一个共同的特点,就是基于比较. 本文将介绍三种非比较的排序算法:计数排序,基数排序,桶排序.它们将突破比较排序的Ω(nlgn)下界,以线性时间运行. 一.比较排序算法的时间下界 所谓的比较排序是指通过比较来决定元素间的相对次序. "定理:对于含n个元素的一个输入序列,

  • C++实现查找中位数的O(N)算法和Kmin算法

    本文实例讲述了C++实现查找中位数的O(N)算法和Kmin算法,分享给大家供大家参考.具体方法如下: 利用快速排序的partition操作来完成O(N)时间内的中位数的查找算法如下: #include <iostream> #include <cassert> #include <algorithm> #include <iterator> using namespace std; int array[] = {1, 2, 10, 8, 9, 7, 5};

  • c++ 一个二进制串转化为整数的解决方法

    代码如下: 复制代码 代码如下: <SPAN style="FONT-SIZE: 18px"> char* p = "1010110001100"; int n = 0; for(int i=0;i<strlen(p); i++) {  n = n * 2 + (p[i] - '0'); } printf("%d\n", n);</SPAN>

  • C++ 十进制转换为二进制的实例代码

    题目内容:将十进制整数转换成二进制数. 输入描述:输入数据中含有不多于50个的整数n(-231<n<231). 输出描述:对于每个n,以11位的宽度右对齐输入n值,然后输出"-->",再然后输出二进制数.每个整数n的输出,独立占一行. 题目分析:将某个数从十进制转为二进制的具体方法是,该数对2取余,结果要么为1要么为0,此为该数对应二进制的末位:然后该数除以二,得到的商再次对2取余,结果为对应二进制的倒数第二位--以此类推,知道除以2的结果为0. 参考代码: 复制代码

  • 马尔可夫链算法(markov算法)的awk、C++、C语言实现代码

    1. 问题描述 马尔可夫链算法用于生成一段随机的英文,其思想非常简单.首先读入数据,然后将读入的数据分成前缀和后缀两部分,通过前缀来随机获取后缀,籍此产生一段可读的随机英文. 为了说明方便,假设我们有如下一段话:   复制代码 代码如下: Show your flowcharts and conceal your tables and I will be mystified. Show your tables and your flowcharts will be obvious. 假设前缀的长

  • C++实现N个骰子的点数算法

    本文实例讲述了C++实现N个骰子的点数算法,分享给大家供大家参考之用.具体方法如下: 题目要求:把n个骰子仍在地上,所有点数 实现代码如下: #include <iostream> using namespace std; const int g_maxValue = 6; const int number = 6; int array[(number - 1) * g_maxValue + 1]; void probility(int original, int current, int s

随机推荐