随便敲敲算法(一)

                      这些是自己慢慢的日积月累的,总之要相信好的算法的基础也是靠慢慢积累和思考而来的,这里面的东西有些是别人博客上的,有些是面试题,自己百度的

1.五只猴子分桃。半夜,第一只猴子先起来,它把桃分成了相等的五堆,多出一只。于是,它吃掉了一个,拿走了一堆; 第二只猴子起来一看,只有四堆桃。于是把四堆合在一起,分成相等的五堆,又多出一个。于是,它也吃掉了一个,拿走了一堆;.....其他几只猴子也都是 这样分的。问:这堆桃至少有多少个?(朋友说,这是小学奥数题)。
  参考答案:先给这堆桃子加上4个,设此时共有X个桃子,最后剩下a个桃子.这样:
  第一只猴子分完后还剩:(1-1/5)X=(4/5)X;
  第二只猴子分完后还剩:(1-1/5)2X;
  第三只猴子分完后还剩:(1-1/5)3X;
  第四只猴子分完后还剩:(1-1/5)4X;
  第五只猴子分完后还剩:(1-1/5)5X=(1024/3125)X;
  得:a=(1024/3125)X;
  要使a为整数,X最小取3125.
  减去加上的4个,所以,这堆桃子最少有3121个。

 

 

2.有A到J十个人手拉手构成一个环,其中A和B不相邻,问一共有多少种排列方式?
  解:基本的环排列算法。公式:N!/N(N 即为参与排列的人)

  就本题而言,首先十个人环排列10!/10 ,然后将AB当成一个人进行环排列9!*2!/9

  排除AB相邻的即为答案:10!/10 -9!*2!/9

 

 

3.

给定字符串,可以通过插入字符,使其变成回文,求最少插入字符的数量。例如:ab最少插入1个字符,变为bab;aa最少插入0个字符;abcd最少插入3个字符,dcbabcd。

分析:首尾指针法——设两个指针pBegin和pEnd分别指向字符串的首尾,比较首尾指针指向的值,这时会有种情况:


   1)*pBegin==*pEnd 则二者均向中间移动,即pBegin++;pEnd--;


   2)*pBegin!=*pEnd,则有两种处理方法:一、在pEnd后插入和*pBegin相等的字符,然后pBegin++继续比较,统计剩余字符插入数量;二、在pBegin钱插入和*pEnd相等的字符,然后pEnd--继续比较,统计剩余字符插入数量。最少插入字符的数量即为一和二两种情况的最小者。

重复上面的过程,直到pBegin和pEnd相遇。

 

#include<stdio.h>
#include<string.h>

int minChange(char *str,char *pBegin,char *pEnd)
{
     if (str==NULL||pBegin==NULL||pEnd==NULL)
     {
     	return -1;
     }
     else if (*str=='\0'||pBegin>=pEnd)
     {
     	return 0;
     }
     else{
     	//*pBegin==*pEnd则二者均向中间移动,即pBegin++;pEnd--;
     	while(pBegin<pEnd&&*pBegin==pEnd)
     	{
     		pBegin++;
     		pEnd--;
     	}
     	//*pBegin!=*pEnd,判断是在pBegin前插入还是在pEND后插入
     	if(pBegin<pEnd)
     	{
     		int min1 = minChange(str,pBegin,pEnd-1);
     		int min2 = minChange(str,pBegin+1,pEnd);
     		return min1<min2?min1+1:min2+1;

     	}
     	else{
     		return 0;
     	}
     }

}

int main(int argc, char const *argv[])
{
	int min,n;
	char *pBegin=NULL,*pEnd=NULL;
	char str[100];
    gets(str);
    n=strlen(str);
    pBegin=str;
    pEnd=&str[n-1];
    min = minChange(str,pBegin,pEnd);
    printf("最少添加%d个字符\n",min);
	return 0;
}


 

 

4.题目:输入一个正数n,输出所有和为n连续正数序列。

例如输入15,由于1+2+3+4+5=4+5+6=7+8=15,所以输出3个连续序列1-5、4-6和7-8。

对于这个题目可以采用两个变量表示下标然后移动的方法,一个表示序列的开头,一个表示序列的结尾。当开头大于n的一半时终止移动

 

#include <iostream>
using namespace std;

void print_sq(int start,int end){//打印序列
	for(int i=start;i<=end;i++)
		cout<<i<<" ";
	cout<<endl;
}
void find_the_sq(int num){ //检索。若是sum大了,则start右移,若是小了end右移
	int start=1,end=2,mid=num/2;
	int sum=start+end;
	while(start<=mid){
		if (sum<num)
		{
		    sum += ++end;
		}
		else if (sum>num)
		{
			sum -= start++;
		}
		else{
			print_sq(start,end);
			sum -= start++;
		}
	}
}
int main(){
	find_the_sq(25);
	return 0;
}


5.题目:输入两个字符串,从第一字符串中删除第二个字符串中所有的字符。例如,输入”They are students.”和”aeiou”,则删除之后的第一个字符串变成”Thy r stdnts.”。

#include <stdio.h>
#include <string.h>
#include <memory.h>

void delete_ch(char *src,char *del_chs){
	int hash[256];
	int i;
	char *pstrat,*pend;
	int del_len=strlen(del_chs);
	pstrat=src;
	pend=src;
	memset(hash,0,sizeof(int)*256);//初始化
	for (i = 0; i < del_len; i++)
	{
		hash[del_chs[i]]=1;
	}
	while(*pend){
		while(hash[*pend]==1) //找到一个不需要删除字符
			*pend++;
		*pstrat++ =*pend++; //pstart和pend同时移动
	}
	*pstrat='\0';

}
int main(int argc, char const *argv[])
{
	char src[] = "Theeeey are students";
	char del_chs[] = "Taeiou";
	delete_ch(src, del_chs);
	printf("%s\n", src); 
	return 0;
}


 

 6 插入排序一段代码

 

#include<stdio.h>
#include<time.h>
#include<stdlib.h>
void insertsort(int array[],int n)   // 实现排序的函数
{
    int i,j,min=0;
    for(i=0;i<n;i++)
	{
		min=array[i];
		for(j=i-1;j>=0&&array[j]>min;j--)
			array[j+1]=array[j];
		array[j+1]=min;
	}

}
int main()          // 主函数
{
	int array[100];
	int i=0,n;
	srand ( (unsigned)time (NULL) );
    printf("请你输入要排序的个数: ");
	scanf("%d",&n);
	printf("随机产生%d个数: ",n);
	for(i=0;i<n;i++)
	{
		array[i]=rand()%10+10; //产生10~20之间的数
		printf("%d ",array[i]);
	}
	insertsort(array,n);   // 实现排序的函数
	printf("\n排序之后为:",n);
	for(i=0;i<n;i++)
	{
      printf("%d ",array[i]);
	}
    printf("\n");
	return 0;
}

 

 

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值