【数组&&字符串&&宏练习题】

简介: 【数组&&字符串&&宏练习题】

1 模拟实现atoi

我们知道:atoi是将字符串中的数字转化成整形数字,但是有以下的情况需要特殊考虑:

1空指针

2空字符串

3遇到了非数字字符

4超出了范围

所以我们不妨用一种枚举类型state来判断当前的输入是否合法,符号的正负可以定义一个flag.

enum state
{
  VALID,
  UNVALID,
};
enum state s = VALID;
int MyAtoi(const char* ps)
{
  if (NULL == ps)
  {
    s = UNVALID;
      return 0;
    }
  int flag = 1;
    if (*ps == '+')
  {
    ps++;
  }
  else if (*ps == '-')
  {
       flag = -1;
         ps++;
    }
  while (isspace(*ps))
  {
    ps++;
  }
  long long n = 0;
  while (isdigit(*ps))
  {
    n = n * 10 + flag*(*ps - '0');
    if (n > INT_MAX || n < INT_MIN)
    {
      s = UNVALID;
        return 0;
    }
    ps++;
  }
  if (*ps == '\0')
    return (int)n;
  else
  {
    s = UNVALID;
    return 0;
    }
}
int main()
{
  const char* pc = "- a123456";
  int ret = MyAtoi(pc);
  if (s == VALID)
    {
    printf("%d\n", ret);
  }
  else
    printf("格式错误,无法打印\n");
  return 0;
}

2 寻找单身狗

题目要求:一个整形数组中只有两个元素是单身的(只出现了一次),其他元素均出现了两次,请找出这两个单身元素。

我们知道:如果只有一个单身元素的话,只需要将数组中每一个元素互相^就能解决问题。那么有两个单身元素又该怎么办呢?

我i们不妨假设一个整形数组{1,1,2,2,3,3,4,4,5,6},如果将数组中每个元素^就相当于5^6,得到的结果是3,解决该问题的关键在于如何将5和6分到不同的小组?

我们发现:将5^6的结果(3)二进制的每一位都&1(从右向左),如果&1的结果==1,就记住第一次==1的位置,假设该位置记做pos,然后让数组中每一个元素都向右移动pos位,得到的结果如果==1,就分到一个数组,不为1就分到另外一个数组,这样就将5和6分到了不同的小组了。

//求一个数组中哪两个是单身元素(其余重复元素只有两个)
void GudgSgle(int arr[], int len,int*p1,int*p2)
{
  int ret = 0;
  int i = 0;
  int pos = 0;
  for (i = 0; i < len; i++)
  {
    ret ^= arr[i];
  }
  for (i = 0; i < 32; i++)
  {
    if ((ret >> i & 1) == 1)
    {
      pos = i;
      break;
    }
  }
  for (i = 0; i <len; i++)
  {
    if ((arr[i] >> pos & 1) == 1)
      *p1 ^= arr[i];
    else
      *p2 ^= arr[i];
  }
}
int main()
{
  int arr[12] = { 1,2,3,4,5,6,1,2,3,4 };
  int len = sizeof(arr) / sizeof(int);
  int single1 = 0;
  int single2 = 0;
  GudgSgle(arr, len,&single1,&single2);
  printf("%d %d\n", single1, single2);
  return 0;
}

3 用宏将一个二进制位的奇数位与偶数位交换

首先来看:如何将一个二进制位奇数位与偶数位交换?如果能拿到所有奇数位以及所有偶数位,再将所有奇数位向左移动一位,所有偶数位向右移动一位,然后再相加就能解决问题。关键在于如何拿到所有的奇数位和所有的偶数位:给定一个二进制数1111 1111 1111 1111 1111 1111 1111 1111,如果要得到奇数位就将该二进制数&0101 0101 0101 0101 0101 0101 0101 0101(0x55555555)


如果要得到偶数位就将该二进制数&1010 1010 1010 1010 1010 1010 1010 1010 (0xaaaaaaaa)

//写一个宏:将一个整数的二进制位奇数与偶数交换
#define SWAP(num) (((num&0x55555555)<<1)+((num&0xaaaaaaaa)>>1))
int main()
{
  int num = 10;
  printf("%d\n", SWAP(num));
  return 0;
}


e810e690a7154105b35cb5b1a81f7ea2.png

得到的结果是5 符合要求。

4 用宏模拟实现offsetof

offsetof是计算结构体中某变量相对于首地址的偏移量。为了计算的简便,我们不妨将0强转成一种结构体指针,再通过该指针引用结构体中的变量,取出它的地址,该地址再数值上就等于所求结构体变量的偏移量。

//写一个宏,自己实现offsetof
struct state
{
  char a;
  char b;
  short c;
  int d;
};
#define OFFSETOF(stuname,memname) (int)&(((struct state*)0)->memname)
int main()
{
  printf("%d\n", OFFSETOF(struct state, a));
  printf("%d\n", OFFSETOF(struct state, b));
  printf("%d\n", OFFSETOF(struct state, c));
  printf("%d\n", OFFSETOF(struct state, d));
  return 0;
}

得到的结果:

476716694afb41c08b33f22cb1ab42d1.png

这里值得注意的是:将0强转成一种结构体指针并不是真正使用了0地址处的空间,只是模拟实现了计算偏移量而已,这里也可以将其他数字强转,只不过后面要再减去该数字。

下面这种写法也是正确的

54e06aa515d340d6944b47cab66e2c5a.png

4 序列中整数去重

给定一个整形数组,去除其中重复出现的数字:例如{1,2,2,2,5,5,3,9},去重后变为{1,2,5,3,9},

解题的思路是:遍历数组中每一个元素,将该元素与后面其他所有数字比较,如果相等,就将该元素被后面一个元素覆盖(该元素其后的元素均向前移动一位)

#include<stdio.h>
int main()
{
    int n=0;
    scanf("%d",&n);
    int i=0;
    int arr[1000]={0};
    for(i=0;i<n;i++)
    {
        scanf("%d",&arr[i]);
    }
    int j=0;
    int k=0;
    for(i=0;i<n;i++)
    {
        for(j=i+1;j<n;j++)
        {
            if(arr[j]==arr[i])
            {
                for(k=j+1;k<n;k++)
                    arr[k-1]=arr[k];
                n--;
                j--;
            }
        }
    }
    for(i=0;i<n;i++)
    {
        printf("%d ",arr[i]);
    }
    return 0;
}

其中容易出错的是当覆盖了一个元素后j--,就拿{1,2,2,2,5,5,3,9}来说,当拿第二个元素与第3个元素相比时,就会把第二个元素覆盖,此时数组变为{1,2,2,5,5,3,9},j如果不--的话会直接跳到现在的第3位元素,会让第3位元素与后面的进行比较,也就忽略了覆盖元素,就会出错。

5 有序列合并

题目要求:将两个按升序排列的整形数组合并成一个按升序排列的整形数组。

解题思路:可以创建一个整形数组,一个一个比较另外两个数组,将其中的较小值放在创建的数组中。

#include<stdio.h>
int main()
{
    int m=0;
    int n=0;
    int arr1[1000]={0};
    int arr2[1000]={0};
    int arr3[3000]={0};
    scanf("%d%d",&m,&n);
    int i=0;
    for(i=0;i<m;i++)
    {
        scanf("%d",&arr1[i]);
    }
    for(i=0;i<n;i++)
    {
        scanf("%d",&arr2[i]);
    }
    i=0;
    int j=0;
    int k=0;
    while(i<m && j<n)
    {
        if(arr1[i]>arr2[j])
        {
            arr3[k++]=arr2[j];
            j++;
        }
        else
        {
            arr3[k++]=arr1[i];
            i++;
        }
    }
    if(i==m)
        for(;j<n;j++)
            arr3[k++]=arr2[j];
    else
        for(;i<m;i++)
            arr3[k++]=arr1[i];
    for(i=0;i<k;i++)
        printf("%d ",arr3[i]);
    return 0;
}

如果其中一个数组遍历完了但是另一个数组还没有遍历完,就把没有遍历完的元素一 一赋给创建的数组。




01f7288885064b9fbf5dbab147c608e1.png

目录
相关文章
|
5天前
|
人工智能 定位技术 SEO
我学 GEO 第 15 天:终于知道AI GEO该如何做?
我是暴走的莉莉酱,边旅行边研究AI GEO的数字游民。专注普通人如何提升“AI可见度”——让AI在回答用户问题时准确识别、理解并推荐你。不讲玄学,只做可测、可调、可持续的GEO实践。
403 125
|
7天前
|
机器学习/深度学习 人工智能 调度
🐴 HappyHorse 1.1 现已上线阿里云百炼!快来查收模型使用指南,现在调用享 6 折~
HappyHorse 1.1 是新一代视频生成大模型,全面升级动态表现力、角色一致性、指令遵循、视觉质感与音画协同能力。支持I2V/T2V/R2V三类生成,适配短剧、电商广告、品牌营销等场景,提供高质、流畅、可控的AI视频生产力。
686 4
🐴 HappyHorse 1.1 现已上线阿里云百炼!快来查收模型使用指南,现在调用享 6 折~
|
5天前
|
缓存 人工智能 运维
阿里云618百炼大模型Qwen3.7-Max功能、免费试用、订阅计费、配置接入详解
Qwen3.7-MAX是阿里云百炼平台推出的通义千问3.7系列旗舰大语言模型,专为智能体时代复杂任务打造,依托阿里云全域算力与自研技术,在逻辑推理、长文本处理、代码工程、长周期自主执行等领域达到行业顶尖水平。2026年618期间,该模型推出多重免费试用权益、按量计费5折、订阅套餐优惠等专属福利,覆盖个人开发者、团队与企业全场景需求,以下从核心功能、免费试用、订阅计费、配置接入四方面展开详细解析。
398 123
|
3天前
|
人工智能 自然语言处理 API
阿里云Token Plan团队版解析:功能、三档套餐与省钱订阅指南
阿里云百炼平台推出的Token Plan团队版,是面向企业与团队的AI大模型订阅服务,以Credits为统一计量单位,整合文本与图像生成模型,提供团队管理、数据安全、多工具兼容等核心能力,解决团队零散订阅AI服务的管理混乱、成本失控、数据安全等痛点。本文将从核心定位、套餐详情、计费规则、团队管理、工具兼容、便宜订阅技巧等方面,全面解析Token Plan团队版,帮助企业与团队高效、低成本地使用AI服务。
298 108
|
18天前
|
缓存 测试技术 API
Qwen 3.7 Plus 与 Max 实测:性价比与多模态能力差异解析(2026)
2026 年 6 月 1 日,阿里悄无声息地发布了 Qwen 3.7 Plus,距 Qwen 3.7 Max 上线刚好 11 天。同样的 1M 上下文,同样的 35 小时自治上限。但价格才是头条:Plus 是 0.40/M输入,Max是 2.50/M——便宜约 6 倍——并且还能看图、看视频。Vision Arena 上 Plus 已经排到 #16。所以这周真正值得讨论的问题不是”要不要为视觉能力买单”,而是”Max 凭什么用 6 倍价格换来 2 个百分点的 benchmark 领先”。
|
4天前
|
存储 人工智能 数据可视化
别再手动复制 Skill 了:多 Agent 时代的 Skill 管理方案
多 Agent 场景下 Skill 的统一管理与同步。
236 124
|
11天前
|
缓存 人工智能 运维
GLM 5.2自托管全流程实战:硬件选型、vLLM/SGLang部署与成本盈亏测算
2026年智谱发布GLM 5.2超大混合专家模型,区别于以往仅开放API的闭源大模型,该模型权重以MIT开源协议对外发布,企业与开发者可完整下载、本地审计、私有化部署,实现数据不出环境、自定义微调、自主调度推理资源。GLM 5.2拥有753B总参数,原生支持百万级上下文窗口,在代码生成、长文档推理、数学逻辑等多项基准测试中对标国际顶尖商用模型,是首款可完整自托管的前沿代码向大模型。
887 0
|
4天前
|
SQL 存储 运维
日志能不能改?SLS LogStore 原生支持更新和删除了
随着日志承载的业务语义越来越多,数据订正、回填、清理等需求变得越来越常见。SLS 现已为 LogStore 提供原生 update/delete 能力——支持按 RowID 精确修改,按查询条件批量操作,类似计费调账、标签刷新、反馈回填等场景都可以直接在 LogStore 内完成闭环。
205 124