求单独出现的数,冒泡排序优化,strncpy的优化

求未配对的数(1):

在一组数据中有 只有一个数出现了一次,其余的数都是成对的出现,请找出这个数:这个题很容易解决,只要把这组数据全部异或(相同为零,不同为1),所以出现两次的数异或后就为零,最后剩下的就是出现一次的数。

#include<stdio.h>
#include<stdlib.h>

int find_once_num(int arr[], int len )
{
                 int num = 0;
                 for (int i = 0; i < len; i++)
                {
                                num ^= arr[i];
                }
                 return num;
}
int main()
{
                int arr[9] = { 8, 4, 2, 7, 9, 8, 4, 2, 7 };
                int ret=find_once_num(arr,9);
                printf( "%d\n",ret);
                system( "pause");
                return 0;
}

(2)

现在将题目变一下,一组数据中只有两个数出现了一次,其余数字都是成对出现的,请找出这两个数,这时就不能直接异或了,必须将这一组数据分成两块,每块只包含一个单独出现的数,这样剩下的问题就和第一题一样了,可关键是怎么分呢???

分析:我们可以先对这组数据进行排序,这样成对出现的数就相邻在一起了,我们再一对一对的比较,直到找到第一对不相等的数,然后从中分开,这样就分成两块了。

void bubble_sort(int arr[], int n )
{
   int flag = 0;
   int i = 0;
   int j = 0;
   for (i = 0; i < n - 1; i++)
   {
      flag = 1;
      for (j = 0; j < n - i - 1; j++)
      {
        if (arr [j]>arr[j + 1])
        {
          arr[j] = arr [j] + arr[j + 1];
          arr[j + 1] = arr [j] - arr[j + 1];
          arr[j] = arr [j] - arr[j + 1];
          flag = 0;
        }
       }
      if (flag == 1)
        break;
    }
}
int find_one_num(const int *p1,int len)
{
   assert(p1 );
   int num = 0;
    for (int i = 0; i <=len; i++)
   {
    num ^= p1[i];
   }
   return num;
}
int main()
{
  int arr[10] = {0 ,1, 13, 18, 18,13,1,46,0,3};
  int i = 0;
  int *p = arr;
  bubble_sort(arr, 10);
  for (i = 0; i < 10 - 1;)
  {
    if (p[i] == p[i + 1])
    {
      i += 2;
    }
    else
      break;
  }
  printf( "%d   %d \n", find(p, i),find((p+i+1),9-i-1));
  system( "pause");
  return 0;
}

冒泡排序的优化:

上面对数组进行排序时,用到了冒泡排序法,通常我们使用冒泡排序法时会将每个数都与后面的数进行比较,这样虽然可以实现冒泡排序,但是却不是最优的方法,因为冒泡排序还可以进行优化。

优化:以升序为例

void bubble_sort(int arr[], int n )
{
   int flag = 0;
   int i = 0;
   int j = 0;
   for (i = 0; i < n - 1; i++)
   {
     flag = 1;        //设置一个标记flag                
     for (j = 0; j < n - i - 1; j++)                  
     {
     //如果第1,2有序,第2,3也有序,后面的都有序,则不进入if,flag就不会变成0
              if (arr [j]>arr[j + 1])                              {                                               arr[j] = arr [j] + arr[j + 1];                          arr[j + 1] = arr [j] - arr[j + 1];                       arr[j] = arr [j] - arr[j + 1];
               flag = 0;                                                //只要前一个数大于后一个数,进入if,flag会变成1
        }
      }
     if (flag == 1)                                       break;        //如果flag为1,则就说明这组数据已经有序                
  }
}

strncpy的优化:

当n很大很大时,如果一个字节一个字节的copy效率会非常低下,所以我们要想一个办法,当n特别大时,使得一次性可以copy多个字节。其实这个办法就是强制类型转换,将char *转换为int *(在c中试了一下转换为double*,显示无法转换),这样一次就可以copy 4个字节,而当n小于4时再转换为char*进行copy.

#include<stdio.h>
#include<stdlib.h>
#include<assert.h>
char* my_strncpy(char *dst, const char *src, int len)
{
	assert(dst);
	assert(src);
	int* p1= (int *)dst;
	const int* p2 = (int *)src;
	while (len)
	{
		if (len < 4)
		{
			(char *)*p1=(char *)*p2;
			((char* )p1)++;
			((char *)p2)++;
			len--;
		}
		else
		{
			*p1++ = *p2++;
			len -= 4;
		}

	}
	(char *)*p1 = ‘\0‘;
	return  dst;
}
int main()
{
	char arr1[100];
	char arr2[50];
	int n = 0;
	scanf("%s%s%d", arr1, arr2,&n);
	char *ret = my_strncpy(arr1, arr2, n);
	printf("%s\n",ret);
	system("pause");
	return 0;
}
时间: 2024-10-14 21:03:53

求单独出现的数,冒泡排序优化,strncpy的优化的相关文章

求单独出现的数,strncpy的优化

求未配对的数(1): 在一组数据中有 只有一个数出现了一次,其余的数都是成对的出现,请找出这个数:这个题很容易解决,只要把这组数据全部异或(相同为零,不同为1),所以出现两次的数异或后就为零,最后剩下的就是出现一次的数. #include<stdio.h> #include<stdlib.h> int find_once_num(int arr[], int len ) {   int num = 0;   for (int i = 0; i < len; i++)   {

【总结】冒泡排序及冒泡排序的两种优化

------------------------------------------------------------------------------------------------------ 冒泡排序(bubble sort)算法的运作如下:从前往后一次比较相邻的两个元素,如果第二个比第一个元素小,则交换这两个元素,一直到与最后一个元素比较完成,这样最大的元素就放到了最后:这样重复比较(n-1)次即可完成排序. -----------------------------------

Java_冒泡排序_原理及优化

冒泡排序及其优化 一.原理及优化原理 1.原理讲解 冒泡排序即:第一个数与第二个数进行比较,如果满足条件位置不变,再把第二个数与第三个数进行比较.不满足条件则替换位置,再把第二个数与第三个数进行比较,以此类推,执行完为一个趟,趟数等于比较的个数减一. 2.冒泡排序原理图示:(以98765序列为例,排序结果从小到大) 3.冒泡排序优化 优化版:每一次减少一次循环(即红色部分不需要在进行比较) 4.冒泡排序最终版 最终版:每一趟减少一次循环(删除线不需要再执行) 二.实现代码 1.冒泡排序实现主要代

一个自然数在1700和1800之间,且被5除余3,被7除余4,被11除余6,求符合条件的数

昨天晚上看了一道逻辑题:一个自然数在1700和1800之间,且被5除余3,被7除余4,被11除余6,求符合条件的数.题目后面写着,有人看了几分钟便给出了答案.我很好奇,此人是如何解答的. 我自己先琢磨了下,拿笔算了半天,最后一个巧合的情况下,得到了答案.此题的一个关键且明显的推论是:能被5除余3的数,肯定最后一位是3或者8.那么接下来怎么推呢?我从网上搜集了答案. 方案1: 这个数被5除余3,则此数个位数为3或8, 设这个数十位为x,则此数可表达为1703+10x,或1708+10x当此数为17

hdu 3987 求最小割条数最小

题意:    一个人要从起点  0  到达 n-1   n个点  m条路  ,我们求最少破坏路的条数使无法 从起点到达终点.题意很明显  ,求最小割条数最少,由于最小割流量虽然固定,但是其条数却不固定,可以破坏3条路,也可以破坏4条路,他们总流量相同才会出现这种情况. 题解:由于上述的情况,他们总流量相同但是条数不同,现在我们需要改变边的容量使得条数少边才是最小割,条数多的将不会是最小割. 官方题解有两种 ,我选择的是在残余网络中进行扩充流的操作,使的两个最小割不同,残余网络中,我进行所有边流量

寻找单独出现的数——通用技巧

请使用最快的方法,寻找单独出现的数. 例1:在一组数据中,只有一个数出现一次,其余数都出现两次,请找出这个单独出现的数. 例2:在一组数据中,只有一个数出现一次,其余数都出现三次,请找出这个单独出现的数. 例3:在一组数据中,只有一个数出现一次,其余数都出现四次,请找出这个单独出现的数. 例4:在一组数据中,只有一个数出现一次,其余数都出现五次,请找出这个单独出现的数. ........... 分析: 像这种问题都可以通过排序来解决,但是毫无疑问,排序不是最快的方法.针对例1,大部分人都能想到,

hdu 4587 2013南京邀请赛B题/ / 求割点后连通分量数变形。

题意:求一个无向图的,去掉两个不同的点后最多有几个连通分量. 思路:枚举每个点,假设去掉该点,然后对图求割点后连通分量数,更新最大的即可.算法相对简单,但是注意几个细节: 1:原图可能不连通. 2:有的连通分量只有一个点,当舍去该点时候,连通分量-1: 复习求割点的好题! #include<iostream> #include<cstdio> #include<vector> using namespace std; int n,m; vector<vector&

hdu3709(求区间内平衡数的个数)数位dp

题意:题中平衡数的定义: 以一个位置作为平衡轴,然后左右其他数字本身大小作为重量,到平衡轴的距离作为全职,实现左右平衡(即杠杆原理平衡).然后为区间[x,y]内平衡数的个数. (0 ≤ x ≤ y ≤ 1018) 解法:数位dp.如果一个数的平衡数,那么它的平衡轴位置是确定的.原来一直尝试数位dp在dfs时候列举平衡轴的位置,后来才意识到可以提前枚举平衡轴位置,然后再dfs,这样比较好写.dp[mid][pre][wei];表示对称轴是mid,计算第pre个位置以后需要力矩大小wei的数的个数.

求100内的数和

//1+2+3+...+nstatic int add(int n) { if(n == 1) { return 1; } else { return n + add(n-1); } } 求100内的数和