`
gstarwd
  • 浏览: 1525418 次
  • 性别: Icon_minigender_1
  • 来自: 杭州
社区版块
存档分类
最新评论

数据结构面试大全

阅读更多

1.判断链表是否存在环型链表 问题:判断一个链表是否存在环,例如下面这个链表就存在一个环:
例如 N1->N2->N3->N4->N5->N2 就是一个有环的链表,环的开始结点是 N5 这里有一个比较简单的解法。设置两个指针 p1 p2 。每次循环 p1 向前走一步, p2 向前走两步。直到 p2 碰到 NULL 指针或者两个指针相等结束循环。如果两个指针相等则说明存在环。

struct link 
{
   int data;
    link* next;
};
 
bool IsLoop(link* head)
{
    link* p1=head, *p2 = head;
     if (head ==NULL || head->next ==NULL) 
     {
          return false;
     }
    do{
        p1= p1->next;
        p2 = p2->next->next;
    } while(p2 && p2->next && p1!=p2);     
     if(p1 == p2)
          return true;
     else
          return false;
}

2, 链 表反转

单向链表的反转是一个经常被问 到的一个面试题,也是一个非常基础的问题。比如一个链表是这样的: 1->2->3->4->5 通过反转后成为 5->4->3->2->1 。最容易想到的方法遍历一遍链表,利用一个辅助指针,存储遍历过程中当前指针指向的 下一个元素,然后将当前节点元素的指针反转后,利用已经存储的指针往后面继续遍历。源代码如下:

struct linka {
     int data;
     linka* next;
};
 
void reverse(linka*& head)
{
     if(head ==NULL)
          return;
     linka*pre, *cur, *ne;
     pre=head;
     cur=head->next;
     while(cur)
     {
          ne = cur->next;
          cur->next = pre;
          pre = cur;
          cur = ne;
     }
     head->next = NULL;
     head = pre;
}


还有一种利用递归的方法。这种方法的基本思想是在反转当前节点之前先调用递归函数反 转后续节点。源代码如下。不过这个方法有一个缺点,就是在反转后的最后一个结点会形成一个环,所以必须将函数的返回的节点的 next 域置为 NULL 。因为要改变 head 指针,所以 我用了引用。算法的源代码如下:

linka* reverse(linka* p,linka*& head)
{
     if(p == NULL || p->next == NULL)
     {
          head=p;
          return p;
     }
     else
     {
          linka* tmp = reverse(p->next,head);
          tmp->next = p;
          return p;
     }
}

3, 判 断两个数组中是否存在相同的数字

给定 两个排好序的数组,怎样高效得判断这两个数组中存在相同的数字?
这个问题首先想到的是一个 O(nlogn) 的算法。就是任意挑选一个数组,遍历这个数组的所有元素,遍历过程中,在另一个数组中对第一个数组中的每个元素进行 binary search 。用 C++ 实现代码如下:

bool findcommon(int a[],int size1,int b[],int size2)
{
     int i;
     for(i=0;i<size1;i++)
     {
          int start=0,end=size2-1,mid;
          while(start<=end)
          {
               mid=(start+end)/2;
               if(a[i]==b[mid])
                    return true;
               else if (a[i]<b[mid])
                    end=mid-1;
               else
                    start=mid+1;
          }
     }
     return false;
}


后来发现有一个 O(n) 算法。因 为两个数组都是排好序的。所以只要一次遍历就行了。首先设两个下标,分别初始化为两个数组的起始地址,依次向前推进 。推进的规则是比较两个 数组中的数字,小的那个数组的下标向前推进一步,直到任何一个数组的下标到达数组末尾时,如果这时还没碰到相同的数 字,说明数组中没有相同的数字。

bool findcommon2(int a[], int size1, int b[], int size2)
{
     int i=0,j=0;
     while(i<size1 && j<size2)
     {
          if(a[i]==b[j])
               return true;
          if(a[i]>b[j])
               j++;
          if(a[i]<b[j])
               i++;
     }
     return false;
}

4, 最 大子序列

问题:
给定一整数序列 A1 A2 ... An (可能有负数),求 A1~An 的一个子序列 Ai~Aj ,使得 Ai Aj 的和最大
例如:
整数序列 -2, 11, -4, 13, -5, 2, -5, -3, 12, -9 的最 大子序列的和为 21
对于这个问题,最简单也是最容易想到 的那就是穷举所有子序列的方法。利用三重循环,依次求出所有子序列的和然后取最大的那个。当然算法复杂度会达到 O(n^3) 。显然这种方法不是最优的,下面给出一个算法复杂度为 O(n) 的线性算法实现,算法的来源 于 Programming Pearls 一书。
在给出线性算法之前,先来看一个对穷举算法进行优化的算法,它的算法复杂度为 O(n^2) 。其实这个算法只是对对穷举算法稍微做了一些修改:其实子序列的和我们并不需要每次都重新计算一遍。假设 Sum(i, j) A[i] ... A[j] 的和,那么 Sum(i, j+1) = Sum(i, j) + A[j+1] 。利用这一个递推,我们就可以得到下面这个算法:

int max_sub(int a[],int size)
{
     int i,j,v,max=a[0];
     for(i=0;i<size;i++)
     {
          v=0;
          for(j=i;j<size;j++)
          {
               v=v+a[j];//Sum(i, j+1) = Sum(i, j) + A[j+1]
               if(v>max)
                    max=v;
          }
     }
     return max;
}


那怎样才能达到线性复杂度呢?这里运用动态规划的思想。先看一下源代码实现:

int max_sub2(int a[], int size)
{
     int i,max=0,temp_sum=0;
     for(i=0;i<size;i++)
     {
          temp_sum+=a[i];
          if(temp_sum>max)
               max=temp_sum;
          else if(temp_sum<0)
               temp_sum=0;
     }
     return max;
}


在这一遍扫描数组当中,从左到右记录当前子序列的和 temp_sum ,若这个和不断增加,那么最大子序列的和 max 也不断增加 ( 不断更新 max) 。如果往前扫描中遇到负数,那么当前子序列的和将会减小。此时 temp_sum 将会小于 max ,当然 max 也就不更新。如果 temp_sum 降到 0 时,说明前面已经扫描的那一段就可以抛弃了,这时将 temp_sum 置为 0 。然后, temp_sum 将从后面开始将这个 子段进行分析,若有比当前 max 大的子段,继续更新 max 。这样一趟扫 描结果也就出来了。 5, 找出单向链表的中间结点

这道题和解 判断链表是否存在环 ,我用的是非常 类似的方法,只不过结束循环的条件和函数返回值不一样罢了。设置两个指针 p1 p2 。每次循环 p1 向前走一步, p2 向前走两步。当 p2 到达链表的末尾时, p1 指向的时链表的中间。

link* mid(link* head)
{
       link* p1,*p2;
       p1=p2=head;
       if(head==NULL || head->next==NULL)
              return head;
       do {
              p1=p1->next;
              p2=p2->next->next;
       } while(p2 && p2->next);
       return p1;
}

6, 按 单词反转字符串

并不是简单的字符串反 转,而是按给定字符串里的单词将字符串倒转过来,就是说字符串里面的单词还是保持原来的顺序,这里的每个单词用空格分开。例如:
Here is www.fishksy.com.cn
经过反转后变为:
www.fishksy.com.cn is Here
如果只是简单的将所有字符串翻转的话,可以遍历字符串,将第一个字符和最后一个交换, 第二个和倒数第二个交换,依次循环。其实按照单词反转的话可以在第一遍遍历的基础上,再遍历一遍字符串,对每一个单词再反转一次。这样每个单词又恢复了原 来的顺序。

char* reverse_word(const char* str)
{
     int len = strlen(str);
     char* restr = new char[len+1];
     strcpy(restr,str);
     int i,j;
     for(i=0,j=len-1;i<j;i++,j--)
     {
          char temp=restr[i];
          restr[i]=restr[j];
          restr[j]=temp;
     }
     int k=0;
     while(k<len)
     {
          i=j=k;
          while(restr[j]!='' '' && restr[j]!='''' )
               j++;
          k=j+1;
          j--;
          for(;i<j;i++,j--)
          {
               char temp=restr[i];
               restr[i]=restr[j];
               restr[j]=temp;
          }
     }
     return restr;
}


如果考虑空间和时间的优化的话,当然可以将上面代码里两个字符串交换部分改为异或实 现。
例如将

          char temp=restr[i];
          restr[i]=restr[j];
          restr[j]=temp;


改 为

          restr[i]^=restr[j];
          restr[j]^=restr[i];
          restr[i]^=restr[j];


7,字符串反转


我没
有记错的话是一道
MSN
的笔试题,网上无意中看到的,拿来做了一下。题目是这样的,给定一个字符串,一个这个
字符串的子串,将第一个字符串反转,但保留子串的顺序不变。例如:


输入:
第一个字符串
: "This is fishsky ''s Chinese site: 
http://www.fishsky.com.cn/cn"

子串
: "fishsky"

输出:
 "nc/nc.moc.fishsky.www//:ptth :etis esenihC s''fishsky si 
sihT"

一般的方法是先扫描一边第一个字符串,
然后用
stack
把它反转,同时记录下子串出现的位置。然后再扫描一遍把记录下来的子串再用
stack
反转。我用的方法是用一遍扫描数组的方法。扫描中如果发现子串,就将子串倒过来压入堆栈。


最后再将堆栈里的字符弹出,这样子串又恢复了原来的顺序。源代码如下:
#include <iostream>
#include <cassert>
#include <stack>
using namespace std;
//reverse the string ''s1'' except the substring ''token''.
const char* reverse(const char* s1, const char* token)
{
       assert(s1 && token);
       stack<char> stack1;
       const char* ptoken = token, *head = s1, *rear = s1;
       while (*head != '''')
       {
              while(*head!= '''' && *ptoken == *head)
              {
                 ptoken++;
                 head++;
              }
              if(*ptoken == '''')//contain the token
              {
                 const char* p;
                 for(p=head-1;p>=rear;p--)
                        stack1.push(*p);
                  ptoken = token;
                 rear = head;
              }
              else
              {
                 stack1.push(*rear);
                 head=++rear;
                 ptoken = token;
              }
       }
       char * return_v = new char[strlen(s1)+1];
       int i=0;
       while(!stack1.empty())
       {
              return_v[i++] = stack1.top();
              stack1.pop();
       }
       return_v[i]='''';
       return return_v;
}
int main(int argc, char* argv[])
{
       
       cout<<"This is 
fishsky
 ''s Chinese site: http://www.
fishsky
.com.cn/cn";
       cout<<reverse("This is 
fishsky
''s Chinese site: http://www.
 fishsky
.com.cn/cn","
 fishsky
 ");
       return 0;
}
 
8,
 
删除数组中重复的数字


问题:一个动态长度可变的数字序列,以数字
0
为结束标志,要求将重复的数字用一个数字代
替,例如:


将数组
 1,1,1,2,2,2,2,2,7,7,1,5,5,5,0 
转变成
1,2,7,1,5,0

问题比较简单,要注意的是这个数组是动态的。所以避免麻烦我还是用了
STL

vector

#include <iostream>
#include <vector>
using namespace std;
//remove the duplicated numbers in an intger array, the array was end with 0;
//e.g. 1,1,1,2,2,5,4,4,4,4,1,0 --->1,2,5,4,1,0
void static remove_duplicated(int a[], vector<int>& _st)
{
       _st.push_back(a[0]);
       for(int i=1;_st[_st.size()-1]!=0;i++)
       {
              if(a[i-1]!=a[i])
                 _st.push_back(a[i]);
       }
}

当然如果可以改变原来的数组的话,可以不用
STL
,仅需要指针操作就可以了。下面这个程序将修改原来数组的内容。
void static remove_duplicated2(int a[])
{
       if(a[0]==0 || a==NULL)
              return;
       int insert=1,current=1;
       while(a[current]!=0)
       {
              if(a[current]!=a[current-1])
              {
                 a[insert]=a[current];
                 insert++;
                 current++;
              }
              else
                 current++;
       }
       a[insert]=0;
}
 9,
如
何判断一棵二叉树是否是平衡二叉树


问
题:判断一个二叉排序树是否是平衡二叉树
 

解决方案:


根据平衡二叉树的定义,如
果任意节点的左右子树的深度相差不超过
1
,那这棵树就是平衡二叉树。


首先编写
一个计算二叉树深度的函数,利用递归实现。
template<typename T>
static int Depth(BSTreeNode<T>* pbs)
{
       if (pbs==NULL)
              return 0;
       else
       {
              int ld = Depth(pbs->left);
              int rd = Depth(pbs->right);
              return 1 + (ld >rd ? ld : rd);
       }
}

下面是利用递归判断左右子树的深度是否相差
1
来判断是否是平衡二叉树的函数:
template<typename T>
static bool isBalance(BSTreeNode<T>* pbs)
{
       if (pbs==NULL) 
              return true;
       int dis = Depth(pbs->left) - Depth(pbs->right);
       if (dis>1 || dis<-1 )
              return false;
       else
              return isBalance(pbs->left) && isBalance(pbs->right);
}

10, strstr()
的简单实现
 

strstr(s1,s2)
是一个经常用的函数,他的作用就是在字符串
s1
中寻找字符串
s2
如果找到了就返回指针,否则返回
NULL



下面是这个函数的一个简单实现:
static const char* _strstr(const char* s1, const char* s2)
{
     assert(s2 && s1);
     const char* p=s1, *r=s2;
     while(*p!='''')
     {
          while(*p++==*r++);
          if(*r=='''')
               return p;
          else
          {
               r=s2;
               p=++s1;
          }
     }
     return NULL;
}
11,
半
素数

题目定义了一种叫半素数的数:只要一个数
能被分解成两个素数,那么这个数就是半素数。


Prime Number Definition
An integer greater than one is called a
 prime number if its only positive divisors (factors) are one and 
itself. For instance, 2, 11, 67, 89 are prime numbers but 8, 20, 27 are 
not.

Semi-Prime Number 
Definition
An integer greater than one is called a semi-prime number 
if it can be decompounded to TWO prime numbers. For example, 6 is a 
semi-prime number but 12 is not.

Your task is just to determinate whether a given number is a 
semi-prime number.

Input

There are several test cases in the 
input. Each case contains a single integer N (2 <= N <= 1,000,000)

Output

One line with a single integer for each case. If the 
number is a semi-prime number, then output "Yes", otherwise "No".

Sample Input

3
4
6
12

Sample Output
No
Yes
Yes
No

没什么好说的,解法很简单,解法如下:
#include <iostream>
#include <cmath>
using namespace std;
bool isprime(long test)
{
     int i;
     for(i=2;i<=sqrt((long double)test);i++)
     {
          if(test%i ==0)
               return false;
     }
     return true;
}
bool isSemiPrime(long test)
{
     int i;
     for(i=2;i<=sqrt((long double)test);i++)
     {
          if(test%i ==0)
          {
               int temp = test/i;
               return isprime(i) && isprime(temp);
          }
     }
     return false;
}
int main()
{
     long n;
     while(cin>>n && n !=0)
     {
          if(isSemiPrime(n))
               cout<<"Yes"<<endl;
          else
               cout<<"No"<<endl;
     }
}
12,
淘
汰赛问题


题目:

Our school is planning to hold a 
new exciting computer programming contest. During each round of the 
contest, the competitors will be paired, and compete head-to-head. The 
loser will be eliminated, and the winner will advance to next round. It 
proceeds until there is only one competitor left, who is the champion. 
In a certain round, if the number of the remaining competitors is not 
even, one of them will be chosed randomly to advance to next round 
automatically, and then the others will be paired and fight as usual. 
The contest committee want to know how many rounds is needed to produce 
to champion, then they could prepare enough problems for the contest.

Input 
The input consists of several
 test cases. Each case consists of a single line containing a integer N -
 the number of the competitors in total. 1 <= N <= 2,147,483,647. 
An input with 0(zero) signals the end of the input, which should not be 
processed.

Output 
For
 each test case, output the number of rounds needed in the contest, on a
 single line.

Sample 
Input 
8
16
15
0

Sample Output 
3
4
4

题目比较简单,下面是我给的解法。其实就是计算一个数是
2
的几次方。
#include <iostream>
using namespace std;
long calculate(long test)
{
     long ret = 0;
     bool is2square = true;
     while(test!=1)
     {
          if(test % 2)
               is2square = false;
          test /= 2;
         ret++;
     }
     if(!is2square)
          ret++;
     return ret;
}
 int main()
{
     long n;
     while(cin>>n && n !=0)
     {
          cout<<calculate(n)<<endl;
     }
}
分享到:
评论

相关推荐

    数据结构面试大全 面试 算法

    本资料集“数据结构面试大全”显然是为准备面试者提供了一个全面的资源库,包含了各种知名企业的真实面试题目,以及数据结构的基础知识。 1. **数组**:是最基本的数据结构,提供了随机访问元素的能力。面试中可能...

    数据结构面试大全—数据结构面试一网打尽

    数据结构面试大全旨在帮助求职者全面准备这方面的知识,确保在面试中能够应对自如。 1. **数组**:数组是最基本的数据结构,它允许在固定位置存储相同类型的数据。面试中可能会问到数组的特性,如直接访问、连续...

    java和数据结构面试大全

    "java和数据结构面试大全"这个资源显然旨在帮助求职者全面准备这一领域的面试问题。以下是一些关键的知识点,涵盖了这两个主题: 1. **数据结构**: - **数组**:是最基础的数据结构,提供了固定大小且通过索引...

    数据结构面试大全.rar

    在面试中,数据结构是衡量程序员能力的重要标准,因为它们直接影响到程序的性能、可读性和可维护性。对于任何从事编程工作的人来说,掌握数据结构的基本原理和应用是至关重要的。 首先,我们来看看一些常见的数据...

    数据结构面试大全.pdf

    数据结构面试大全.pdf

    Java常见数据结构面试题(带答案)

    "Java常见数据结构面试题(带答案)" 以下是对Java常见数据结构面试题的知识点总结: 栈和队列 * 栈和队列的共同特点是只允许在端点处插入和删除元素。 * 栈通常采用的两种存储结构是线性存储结构和链表存储结构...

    数据结构面试宝典数据结构面试宝典

    "数据结构面试宝典"旨在帮助求职者掌握这一领域的关键知识,提高面试的成功率。 在数据结构中,常见的类型包括数组、链表、栈、队列、树、图、哈希表等。每种数据结构都有其独特的特点和适用场景。 1. **数组**:...

    数据结构面试题目大全

    本资源“数据结构面试题目大全”旨在帮助求职者全面准备与数据结构相关的笔试和面试问题,确保他们在面对这类挑战时能够游刃有余。 数据结构主要涉及如何在计算机中有效地存储和组织数据,以便进行高效的访问和修改...

    数据结构面试题

    面试中常见的数据结构题目包括但不限于:实现特定数据结构(如自平衡树)、分析时间复杂度和空间复杂度、解决实际问题(如使用数据结构优化算法)。此外,面试官可能还会询问如何使用数据结构优化已有的解决方案,...

    各大公司数据结构面试题集锦

    本资料“各大公司数据结构面试题集锦”汇聚了多个知名公司的面试题,涵盖了C和C++语言,旨在帮助求职者充分准备面试。 首先,我们来了解一下基础的数据结构类型: 1. 数组:是最基本的数据结构,它将元素存储在...

    Java 数据结构 面试用

    在准备Java面试的过程中,数据结构是必不可少的知识领域。数据结构是计算机存储、组织数据的方式,它直接影响到程序的效率和性能。本资料包主要聚焦于Java实现的数据结构,旨在帮助面试者深入理解并掌握这些核心概念...

    数据结构+C++面试题

    数据结构与C++在面试中是两个至关重要的领域,它们是计算机科学的基础,也是软件工程师必备的技能。本文将深入探讨这两个主题中的关键知识点,并结合面试题的形式进行讲解。 首先,我们来关注数据结构。数据结构是...

    刀疤鸭之数据结构面试题

    在给定的文件中,“刀疤鸭之数据结构面试题”涉及了大量的数据结构相关的问题,这些问题覆盖了多种数据结构类型和算法,包括树、图、数组、链表和栈等。 1. 二元查找树转换为双向链表: 二元查找树(BST)是一种...

    典型数据结构面试题

    ### 数据结构面试题详解 #### 1. 单链表插入操作 在单链表中,在节点`p`前插入节点`s`(值为`e`)的操作涉及到指针的重新指向。正确的步骤是先找到`p`节点的前一个节点`q`,然后更新指针,使得`q-&gt;next`指向`s`,再...

    数据结构面试常用经典问题

    这里我们聚焦于“数据结构面试常用经典问题”,通过分析《数据结构与算法经典问题解析:Java语言描述(原书第2版)》这本书中的内容,我们可以深入探讨几个关键知识点。 1. **数组**:数组是最基础的数据结构,它...

    c 数据结构 面试题 算法 面试题

    本资源包聚焦于C语言实现的数据结构面试题,旨在帮助应聘者准备相关面试,提升对数据结构和算法的理解。 一、链表 链表是一种线性数据结构,与数组不同,它在内存中不是连续存储的。每个节点包含数据和指向下一个...

Global site tag (gtag.js) - Google Analytics