出题:给定两个单向链表的头结点,判断其是否有公共节点并确定第一个公共节点的索引;

分析:

  • 由于是单向链表,所以每个节点有且仅有一个后续节点,所以只可能是Y型交叉(每条链表中的某个节点同时指向一个公共节点,并共享后面的所有节点)。首先计 算每条链表的长度,m和n(m>n);
  • 然后设定两个指针A和B分别指向两条链表头(A指向较长链表),让A先走(m-n)步,然后A和B同时前进, 当(A==B)成立的时候A和B当前指向的节点就是第一个公共节点;

解题:

 struct Node {
int v;
Node *next;
}; Node* findFirstCommon(Node *first, Node *second) {
/**
* 首先计算first和second各自的长度
* 当任一链表长度为0时返回NULL
* */
Node *firstT=first, *firstP=NULL;
Node *secondT=second, *secondP=NULL;
int firstL=, secondL=;
while(firstT!=NULL) {
firstL++;
firstP=firstT;
firstT=firstT->next;
}
while(secondT!=NULL) {
secondL++;
secondP=secondT;
secondT=secondT->next;
}
/**
* 判断是否存在公共节点,也就是长度不为0并且最后一个
* 节点相同
* */
if(firstL== || secondL==)
return NULL;
if(firstP!=secondP)
return NULL;
/**
* 同步链条链表的指针,使得firstT和secondT指向的
* 位置距离链表最后一个节点有相同的距离
* */
int difference=;
if(firstL>secondL) {
difference=firstL-secondL;
firstT=first;
secondT=second;
for(int i=;i<difference;i++)
firstT=firstT->next;
} else if(firstL<secondL) {
difference=secondL-firstL;
firstT=first;
secondT=second;
for(int i=;i<difference;i++)
secondT=secondT->next;
}
printf("%d, %d",firstT->v,secondT->v);
/**
* 同步指针之后,同时向前移动,firstT==secondT的时候
* 表示第一个公共节点,由于已经现在已经确定肯定有公共
* 节点,所以可以这样判断
* */
while(firstT!=NULL && secondT!=NULL) {
if(firstT==secondT)
break;
firstT=firstT->next;
secondT=secondT->next;
}
return firstT;
}

出题:实现字符集删除函数,要求在第一个字符串first中删除所有在第二个字符串second中出现的字符;

分析:

  • 解法1:可以使用暴力解法,遍历first中每一个字符,并与second中的字符比较,如果有相同则删除,然后比较first中下一个字符,时间复杂度为O(mn);
  • 解法2:由于是判断某一个字符是否存在的情况,所以可以使用Hash Table,将first映射到256大小的HT中,然后将second也映射过去,一旦遇到已经存在的字符,则删除;最后遍历first并比较HT中的 结果然后输出。此方法时间复杂度为O(N),但是需要使用额外内存保存最终结果,并不会改变原有的字符串;
  • 解法3:由于解法2需要使用额外的空间保存字符串,当字符串较大的时候不可取;所以需要考虑如何快速的在原有字符串上进行字符删除,可以考虑两个指针 left和right,使用right对first字符串进行遍历并通过hashTable(仅仅映射second字符串)判断是否需要删除当前字符,如 果需要进行删除则right指针直接跳过,如果不需要删除则将*left=*right,然后处理下一个字符,所以left指针记录了最终保存的字符串, 这样可以在O(N)的时间内完成,并且不需要任何额外内存;

解题:

 void charSetDelete(char *first, char *second) {
/**
* 由于char集一般为256,所以初始化一个bool数组
* 并且缺省所有的值都为false
* */
bool hashTable[];
for(int i=;i<;i++)
hashTable[i]=false;
/**
* 首先映射first字符串,true表示需要输出
* 然后映射second字符串,false表示不需要输出
* */
char *firstT=first, *secondT=second;
while(*firstT != '\0') {
hashTable[*firstT]=true;
firstT++;
}
while(*secondT != '\0') {
hashTable[*secondT]=false;
secondT++;
}
/**
* 通过hashTable对应元素是true还是false进行
* 输出判断
* */
firstT=first;
printf("\n");
while(*firstT != '\0') {
if(hashTable[*firstT])
printf("%c",*firstT);
firstT++;
}
printf("\n");
} void betterVersion(char *first, char *second) {
bool hashTable[];
for(int i=;i<;i++)
hashTable[i]=false;
/**
* hashTable用于记录second中指定需要
* 删除的字符集合
* */
char *secondT=second;
while(*secondT!='\0') {
hashTable[*secondT]=true;
secondT++;
}
/**
* 使用两个指针完成字符串的删除移动操作
* 仅当right索引的字符不需要删除的时候
* 才将其复制给left指向的位置
* */
char *left=first, *right=first;
while(*right!='\0') {
if(!hashTable[*right]) {
*left=*right;
left++;
}
right++;
}
*left='\0';
char *temp=first;
while(*temp!='\0')
printf("%c",*temp++);
} int main() {
char first[]="abcdefgghjjjj";
char second[]="agj";
charSetDelete(first, second);
betterVersion(first, second);
return ;
}

笔试算法题(19):判断两条单向链表的公共节点 & 字符集删除函数的更多相关文章

  1. 笔试算法题(13):反转链表 & 左旋转字符串

    出题:反转链表(递归和非递归解法): 分析:有递归跟非递归实现,注意对原始链表头节点的处理,因为其他节点都指向下一个节点,其需要指向NULL: 解题: struct Node { int v; Nod ...

  2. 算法题 19 二叉平衡树检查 牛客网 CC150

    算法题 19 二叉平衡树检查 牛客网 CC150 实现一个函数,检查二叉树是否平衡,平衡的定义如下,对于树中的任意一个结点,其两颗子树的高度差不超过1. 给定指向树根结点的指针TreeNode* ro ...

  3. Newtonsoft.Json C# Json序列化和反序列化工具的使用、类型方法大全 C# 算法题系列(二) 各位相加、整数反转、回文数、罗马数字转整数 C# 算法题系列(一) 两数之和、无重复字符的最长子串 DateTime Tips c#发送邮件,可发送多个附件 MVC图片上传详解

    Newtonsoft.Json C# Json序列化和反序列化工具的使用.类型方法大全   Newtonsoft.Json Newtonsoft.Json 是.Net平台操作Json的工具,他的介绍就 ...

  4. POJ1269:Intersecting Lines(判断两条直线的关系)

    题目:POJ1269 题意:给你两条直线的坐标,判断两条直线是否共线.平行.相交,若相交,求出交点. 思路:直线相交判断.如果相交求交点. 首先先判断是否共线,之后判断是否平行,如果都不是就直接求交点 ...

  5. C# 判断两条直线距离

    本文告诉大家获得两条一般式直线距离 一般式的意思就是 Ax+By+C=0" role="presentation">Ax+By+C=0Ax+By+C=0 如果有两个 ...

  6. 2018-7-31-C#-判断两条直线距离

    title author date CreateTime categories C# 判断两条直线距离 lindexi 2018-07-31 14:38:13 +0800 2018-05-08 10: ...

  7. 笔试算法题(27):判断单向链表是否有环并找出环入口节点 & 判断两棵二元树是否相等

    出题:判断一个单向链表是否有环,如果有环则找到环入口节点: 分析: 第一个问题:使用快慢指针(fast指针一次走两步,slow指针一次走一步,并判断是否到达NULL,如果fast==slow成立,则说 ...

  8. 笔试算法题(54):快速排序实现之单向扫描、双向扫描(single-direction scanning, bidirectional scanning of Quick Sort)

    议题:快速排序实现之一(单向遍历) 分析: 算法原理:主要由两部分组成,一部分是递归部分QuickSort,它将调用partition进行划分,并取得划分元素P,然后分别对P之前的部分和P 之后的部分 ...

  9. 笔试算法题(25):复制拥有多个指针的链表 & 判断二元树B是否为A的子树

    出题:定义一个复杂链表:在单向链表的基础上,每个节点附加一个指向链表中其他任意节点的指针sibling,实现CNode* Clone(Cnode *head)函数复制这个复杂链表: 分析: 解法1:将 ...

随机推荐

  1. Linux 常用命令七 grep

    一.grep命令 grep(global search regular expression(RE) and print out the line,全面搜索正则表达式并把行打印出来)是一种强大的文本搜 ...

  2. bzoj 2743: [HEOI2012]采花【树状数组】

    离线,按照l排序 注意到在区间里出现两次的颜色才有贡献,所以记录一个ne[i]表示i后第一个和i同色的花,维护一个l,每次处理询问的时候l单调右移,树状数组维护,在ne[ne[i]]位置++,在ne[ ...

  3. VMware 12安装Mac OS X 10.11&解决上网的问题

    近日想在Win10上安装Mac OS 玩玩,于是上网搜了相关资源,查看了相关经验分享,开始着手安装.系统很快成功安装,但最大问题是虚拟机中的Mac OS无法上网.费了很长时间,最终看到Ping通结果, ...

  4. jira以及jira API简单介绍

    最近需要预言:是否可以通过jira API实现用例管理,对jira的应用.API.扩展等进行了一定的了解. Jira介绍: jira是目前比较流行的基于Java架构的管理系统(Atlassian公司支 ...

  5. 《windows核心编程系列》十五谈谈windows线程栈

    谈谈windows线程栈. 当系统创建线程时会为线程预订一块地址空间区域,注意仅仅是预订.默认情况下预定的这块区域的大小是1MB,虽然预订这么多,但是系统并不会给全部区域调拨物理存储器.默认情况下,仅 ...

  6. [ZPG TEST 115] 种树【差分约束】

    4. 种树 (trees.pas/c/cpp) [问题描述] 一条街的一边有几座房子.因为环保原因居民想要在路边种些树.路边的地区被分割成块,并被编号为1..n.每个块的大小为一个单位尺寸并最多可种一 ...

  7. 题解报告:poj 3320 Jessica's Reading Problem(尺取法)

    Description Jessica's a very lovely girl wooed by lots of boys. Recently she has a problem. The fina ...

  8. ORA-01144_表空间数据文件超出最大限制

    Oracle11gR2扩展表空间报ORA-01144错误. 数据块大小为8K的数据库,单个数据文件大小限制在32GB内. 解决办法: 1.增加表空间数据文件的方式: 2.创建BIGFILE表空间:

  9. Java中的流(3)字符流-Reader和Writer

    java中提供了处理以16位的Unicode码表示的字符流的类,即以Reader和Writer 为基类派生出的一系列类.  1.Reader和Writer  这两个类是抽象类,只是提供了一系列用于字符 ...

  10. Xcode配置SVN详细步骤

    转载:http://blog.csdn.net/weiqubo/article/details/8288635   Xcode 默认自带Git 与 SVN,我们本篇介绍SVN的详细配置步骤如下: 1. ...