1、设计一个算法,删除单链表L(有头结点)中的一个最小值结点。P48
LinkList Delete_Min(LinkList &L){
//L带头结点的单链表,本算法删除其最小值结点
LNode *pre=L,*p=pre->next; //p为工作指针,pre指向其前驱
LNode *minpre=pre,*minp=p; //保存最小值结点及其前驱
while(p!=NULL){
if(p->data<minp->data){
minp=p; //找到比之前找到的最小值结点更小的结点
minpre=pre;
}
pre=p;
p=p->next; //继续扫描下一个结点
}
minpre->next=minp->next; //删除最小值结点
free(minp);
return L;
}
2、如果在循环队列的两端都可以进行插入和删除操作,写出从队尾出队元素的算法。
Status DeQueue(SqQueue &Q){
//若队列不空,则删除Q的队尾元素,并返回ok
//否则返回ERROR
if(Q.front == Q.rear)
return ERROR;
Q.rear = (Q.rear - 1 + MAXQSIZE) % MAXQSIZE;
return OK;
}
3、假设二叉树采用二叉链表存储结构,设计一个算法,计算一棵给定二叉树的所有叶子结点数。
Status POLeafNodeNum(int &i,BiTree &T){
if(T){
if(!T->lchild && !T->rchild)
i++;
POLeafNodeNum(i,T->lchild);
POLeafNodeNum(i,T->rchild);
}
return OK;
}
4、奇偶交换排序是一种交换排序。它的第一趟对序列中的所有奇数项i扫描,第二趟对序列中的所有偶数项i扫描,若A[i]>A[i+1],则交换它们。第三趟对序列中的所有奇数项扫描,第四题对序列中的所有偶数项i扫描,…,如此反复,直至整个序列全部排好序为止。写出奇偶交换排序的算法。
此算法是用Java所写,尚未找到C语言版的
private void core(int[] array) {
int arrayLength = array.length;
boolean oddSorted = false;
boolean evenSorted = false;
while(!oddSorted || !evenSorted) {
int base = 0;
oddSorted = true;
evenSorted = true;
for (int i = base; i < arrayLength - 1; i += 2) {
if (array[i] > array[i + 1]) {
ArrayUtils.swap(array, i, i + 1);
oddSorted = false;
}
}
base = 1;
for (int i = base; i < arrayLength - 1; i += 2) {
if (array[i] > array[i + 1]) {
ArrayUtils.swap(array, i, i + 1);
evenSorted = false;
}
}
}
}
#include
using namespace std;
int n,flag=0;
int A[100]={
0};
int shulu(){
cin>>n;
cout<<"n的值是"<<n<<end1;
for(int i=1;i<=n;i++){
cout<<"第"<<i<<end1;
cin>>A[i];
}
return 0;
}
int paixu()[
flag=0;
for(int i=1;i+1<=n;i+=2){
if(A[i]>A[i+1]){
int p;
p=A[i];
A[i]=A[i+1];
A[i+1]=p;
flag=1;
}
else
continue;
}
for(i=2;i+1<=n;i+=2){
if(A[i]>a[i+1]){
int p;
p=A[i];
A[i]=A[i+1];
A[i+1]=p;
flag=1;
}
else
continue;
}
return 0;
}
int main(){
shulu();
flag=1;
cout<<"开始排序"<<end1;
while(flag!=0){
paixu();
}
cout<<end1;
for(int i=1;i<=n;i++)
cout<<A[i]<<" ";
return 0;
}
1、有一个递增非空链表,设计一个算法删除值域重复的结点。例如,{1,1,2,3,3,3,4,4,7,7,7,9,9}。
void Del_Same(LinkList &L){
//L是递增有序的单链表,本算法删除表中数值相同的元素
LNode *p=L->nest,*q;
if(p==NULL)
return;
while(p->next!=NULL){
q=p->next; //q指向*p的后继结点
if(p->data==q->data){
//找到重复值的结点
p->next=q->next; //释放*q结点
free(q); //释放相同元素值的结点
}
else
p=p->next;
}
}
2、写一个函数,逆序打印单链表中的数据,假设指针L指向单链表的开始结点(用递归算法)。
void reprintList(LNode *L){
if(L!=NULL){
reprintList(L->next); //递归逆序打印开始结点之后的结点数据
print("%d\t",L->data); //打印结点数据
}
}
3、在二叉树中查找值为x的结点。假设值为x的结点不多于一个,打印出值为x的结点所在的层次。
算法思路 :结点查找–>>遍历二叉树;要判断当前所在层次,利用函数参数和返回值传递信息的作用实现。
int h=0;
int find_node_level(BTNode *bt, ElemType x,int h){
if(bt == NULL)
return 0;
else if(bt->data == x)
return h;
else{
int l = find_node_level(bt->lchild, x, h+1);
if(l != 0)
return l;
else
return find_node_level(bt->rchild, x, h+1);
}
}
4、写出直接插入排序的算法。
不带“哨兵”, 此方法优于带“哨兵”
void InsertSort(ElemType A[],int n){
int i,j,temp;
for(i=1;i<n;i++) //将各元素插入已排好序的的序列中
if(A[i]<A[i-1]){
//若A[i]关键字小于前驱
temp=A[i]; //用temp暂存A[i]
for(j=i-1;j>=0 && A[j]>temp;--j) //检查所有前面已排好序的元素
A[j+1]=A[j]; //所有大于temp的元素都向后挪位
A[j+1]=temp; //复制到插入位置
}
}
带“哨兵”
void InsertSort(ElemType A[],int n){
int i,j;
for(i=2;i<=n;i++) //依次将A[2]~A[n]插入到前面已排好序序列
if(A[i]<A[i-1]){
//若A[i]关键码小于其前驱,将A[i]插入有序表
A[0]=A[i]; //复制为哨兵,A[0]不存放元素
for(j=i-1;A[0]<A[j];--j) //从后往前查找待插入位置
A[j+1]=A[j]; //向后挪位
A[j+1]=A[0]; //复制到插入位置
}
}
1、设计一个算法,从一给定的顺序表L中删除下标i到j(i<=j,包括i,j)之间的所有元素,假定i,j都是合法的。
使用单链表存储顺序表L中的所有值,要删除i到j之间的所有元素,因为下标i到j是连续的,故可以将第i个结点的下一个指针next指向下标为j+1的结点。
bool Del_List(SqList &l,ElemType i,ElemType j){
//删除顺序表L中下标在i到j之间(要去i<=j)的所有元素
int m,k=0;
if(i>j || L.length==0)
return false; //顺序表为空或当下标i>j时,返回
for(m=i; m<=j; m++)
k++;
L.data[i]=L.data[j+1]; //当前元素移动到下标为i的结点位置
L.length-=k; //长度减小
return true;
}
2、设有一元素为整数的线性表L,存放在一维数组A[0…n-1]中,设计一个算法,以A[n-1]为参考值,将该数组分为左右两部分,其中左半部分的元素均小于等于A[n-1],右半部分的元素均大于等于A[n-1],A[n-1]位于这两部分之间,要求结果扔存放在A数组中。
算法思路:类似于“快速排序”的一趟排序,经过一趟排序后A[n-1]确定了其最终位置。
void Partition(ElemType A[]. int low, int high){
ElemType pivot = A[high]; //将当前表中的第n个(即下标为n-1)的元素设为枢轴,对表进行划分
while(low<high){
//循环跳出的条件
while(low<high && A[low]<=pivot)
++low;
A[high] = A[low]; //将比枢轴大的元素移动到右端
while(low<high && A[high]>=pivot)
--high;
A[low] = A[high]; //将比枢轴小的元素移动到左端
}
A[high] = pivot; //枢轴元素(即下标为n-1的元素)存放的最终位置
}
3、设计一个算法,对给定的一个整型m*n矩阵A,统计这个矩阵中具有下列特征的元素个数并输出它们的坐标及数值:它们既是所有行中的最小值,又是所在列中的最小值。
算法思路:只需要找出某行中的最小值,然后再判断其是否为该列的最小值即可。
void Find_Min(int A[][MAXSIZE], int m, int n){
int rowMin;
int count = 0; //计数器,记录符合条件的元素数目
for(int i=0; i<m; i++){
//遍历行
rowMin = A[i][0]; //将每行的第一个元素赋予为其所在行的最小值
int x = i, y = 0; //记录行最小值的行下标和列下标
for(int j=0; j<n; j++){
//遍历列
if(A[i][j] < rowMin){
rowMin = A[i][j];
x = i;
y = j;
}
}
int temp = rowMin;
for(int q=0; q<m; q++){
if(A[q][y] < rowMin)
rowMin = A[q][y];
}
if(temp == rowMin){
count++;
printf("%d,[%d][%d]\n",temp,x,y);
}
printf("既是所在行又是所在列最小值的元素的个数为:%d\n",count);
}
4、设一棵二叉树以二叉链表为存储结构,结点结构为lchild|data|rchild。设计一个算法,求在前根序列中处于第k个位置的结点。
设置一个全局变量i记录已访问过的结点的序号,其初值是根结点在先序序列中的序号,即1。
int i=1;
ElemType PreNode(BiTree b, int k){
if(b==NULL)
return '#'; //空结点,返回特殊字符
if(i==k)
return b->data; //相等,则当前结点即为第k个结点
i++; //下一个结点
ch = PreNode(b->lchild, k); //左子数中递归寻找
if(ch!=‘#’)
return ch; //在左子树中,则返回该值
ch = PreNode(b->rchild, k);
return ch; //在右子树中递归寻找
}
1、设计一个算法,将一个头结点为A的单链表(其数据域为整数)分解为两个单链表A和B,使得A链表只含有原来链表中的data域为奇数的结点,而B链表只含有原来data域为偶数的结点,且保持原来相对顺序。
算法思路:根据结点的数据域奇偶性来判断插入哪个链表中,使用尾插法来保持顺序不变。
LinkList DisCreat_l(LinkList &A){
B=(LinkList)malloc(sizeof(LNode)); //创建B表表头
B->next=NULL; //B表初始化
LNode *ra=A, *rb=B; //ra和rb将分别指向将创建的A表和B表的尾结点
p=A->next;
while(p!=NULL){
if(p->data%2==0){
rb->next=p; //在B的表尾插入新结点
rb=p; //rb指向新的尾结点
}
else{
ra->next=p;
ra=p; //在A表尾插入新结点
}
p=p->next; //将p恢复为指向新的待处理结点
}
ra->next=NULL;
rb->next=NULL;
return A;
return B;
}
2、设有浮点型数组A[0···n],设计求数组A中最大值的递归算法。
int i=0, j=n;
float FindMax(flaot A[], int i, int j){
float max;
if(i==j)
return A[i];
else{
max=FloatMax(A, i+1, j);
if(A[i]>max)
return A[i];
else
return max;
}
}
3、在二叉树中查找值为x的结点。假设值为x的结点不多于一个,打印出值为x的结点的所有祖先。
算法思路:采用非递归后序遍历,最后访问根结点,访问到值为x的结点时,栈中所有元素均为该结点的祖先,依次出栈打印即可。 王道书P163
typedef struct{
BiTree t;
int tag;
}stack; //tag=0表示左子女被访问,tag=1表示右子女被访问
void Search(BiTree bt,ElemType x){
//在二叉树bt中,查找值为x的结点,并打印其左右祖先
stack s[];
top=0;
while(bt!=NULL||top>0){
while(bt!=NULL&&bt->data!=x){
s[++top].t=bt;
s[top].tag=0;
bt=bt->lchild; //沿左分支向下
}
if(bt->data==x){
printt("所查结点的所有祖先结点的值为:\n");
for(i=1;i<=top;i++)
printf("%d",s[i].t->data);
exit(1);
}
while(top!=0 && s[top].tag==1)
top--; //退栈(空遍历)
if(top!=0){
s[top].tag=1;
bt=s[top].t->rchild; //沿右分支向下遍历
}
}
}
4、设有一组初始记录关键字序列(K1,K2,…,Kn),要求设计一个算法能够在O(n)的时间复杂度内将线性表划分成两部分,其中左半部分的每个关键字均小于Ki,右半部分的关键字均大于等于Ki。
void QuickPass(int K[], int s, int t){
int i=s;j=t,x=K[s];
while(i<j){
while(i<j && K[i]>x){
j=j-1;
if(i<j){
K[i]=K[j];
i=i+1;
}
}
while(i<j && K[i]<x){
i=i+1;
if(i<j){
K[j]=K[i];
j=j-1;
}
}
}
K[i]=x;
}
1、设计在单链表中删除值相同的多余结点的算法。
void DeleteSame(LinkList *L){
LNode *p,*q,*s;
p=(*L)->next;
for(p;p!=NULL;p=p->next){
s=p; //s指向要删除结点的前驱
for(q=p->next;q!=NULL;){
if(q->data==p->data){
s->next=q->next;
free(q);
q=s->next;
}
else{
s=q;
q=q->next;
}
}
}
}
2、设计一个求结点x在二叉树中的双亲结点算法。
typedef struct BiTNode{
ElemType data;
struct BiTNode *lchild,*rchild;
}BiTNode *BiTree;
BiTNode *parent(BiTree T,ElemType x){
BiTNode *ans;
if(T==NULL)
return NULL;
if(T->lchild==NULL&&T->rchild==NULL)
return NULL;
else{
if(T->lchid->data==x||T->rchild->data==x)
return T;
else{
ans=parent(T->lchid,x);
if(ans)
return ans;
ans=parent(T->rchild,x);
if(ans)
return ans;
return NULL;
}
}
}
3、写一个算法利用折半查找方法在一个有序表中插入元素x,且保持表的有序性。
void InsertSort(int num[],int n,int x){
int mid;
int low=0,high=n-1;
int flag=0; //标记是否发生插入
int pos;
while(low<=high&&flag==0){
mid=(low+high)/2;
if(num[mid]==x)
flag=1; //若关键字等于x,则将flag置为1
else if(num[mid]>x)
high=mid-1;
else
low=mid+1;
}
if(flag==1)
pos=mid;
else
pos=low; //若x最大,且不在原有序表中,则low指向最后一个元素下标的下一个位置
for(int i=n-1;i>pos;i--)
num[i+1]=num[i];
num[pos]=x;
}
4、写出冒泡排序算法。
void swap(int &a,int &b){
int temp=a;
a=b;
b=temp;
}
void BubbleSort(ElemType A[],int n){
for(i=0;i<n-1;i++){
bool flag=false;
for(int j=n-1;j>i;j--)
if(A[j-1]>A[j]){
swap(A[j-1],A[j]);
flag=true;
}
if(flag==false)
return;
}
}