【C语言刷题】找单身狗、模拟实现atoi

news2025/1/11 19:47:52

目录

一、找单身狗

1.暴力循环法

2.分组异或法

二、模拟实现atoi

1.atoi函数的功能

2.模拟实现atoi


一、找单身狗

题目描述:给定一个数组中只有两个数字是出现一次,其他所有数字都出现了两次。

编写一个函数找出这两个只出现一次的数字。

比如:[1,1,2,2,3,3,4,4,5,6]

单身狗是:5和6

1.暴力循环法

对于这道题,我们最容易想到的方法就是暴力循环遍历了。

我们设置两个下标或者两个指针,两层for循环来进行遍历求解。

但是缺陷是时间复杂度过大。当数据量很大的时候,效率很低

这种方法的代码实现也比较简单,这里也就不在赘述了

2.分组异或法

我们其实在前面的文章种已经讲解过一个找单身狗的问题了,只不过那个问题是一个数组中只有一个单身狗。我们可以根据异或操作符的特性,直接将整个数组进行异或就可以求解问题。

但是这道题目,我们发现有两个单身狗。因此我们不能单纯的将整体数组进行异或从而进行求解。

但是我们可以采用类似的思路。巧妙的使用异或操作符

我们是这样思考的,我们有没有什么办法能将两个单身狗给分离出去呢?也就是说将一个数组给分割成两个数组

每一个数组都需要满足这两个条件,两个单身狗需要在两个数组中,不能再、在一个数组中,其次我们还需要满足相同的两个数在同一组中

分成了两组以后,那么问题就简单了,两组分别异或,就得到了两个单身狗

那么如何将一个数组分割成满足这些条件的两个数组呢?我们要根据什么进行分组呢?

其实我们仍然可以根据异或操作符进行分组

我们对整个数组进行使用异或操作符,那么我们就可以得到两个单身狗异或的结果。这个结果的二进制序列中有1的位就是这两个单身狗的区别。根据这个区别,我们就可以将整个数组中,对应的这个位为1的分为一组,对应的这个位为0的分为一组。这样我们就完成了分组

比如说:[1,1,2,2,3,3,4,4,5,6]这个数组中

第一步:将整体异或得到101^110=011

第二步:我们发现011中有两个位不一样,我们随便选定一个即可,我们选定第一位

第三步分组:将上面的数组全部写成二进制的形式

        [001,001,010,010,011,011,100,100,101,110]

        根据第一位的不同分组结果如下:

        第一组:[001,001,011,011,101],即[1,1,3,3,5]

        第二组:[010,010,100,100,110],即[2,2,4,4,6]

第四步:分别异或

第一组异或为5,第二组异或为6

最终代码如下:

#include<stdio.h>
void find_signal_dog(int* arr, int sz, int* p)
{
	int i = 0;
	int ret = 0;

	for (i = 0; i < sz; i++)
	{
		ret ^= arr[i];
	}
	int pos = 0;
	for (i = 0; i < 32; i++)
	{
		if (((ret >> i) & 1 )== 1)
		{
			pos = i;
			break;
		}
	}
	for (i = 0; i < sz; i++)
	{
		if (((arr[i] >> pos) & 1 )== 1)
		{
			p[0] ^= arr[i];
		}
		else
		{
			p[1] ^= arr[i];
		}
	}
}
int main()
{
	int arr[] = { 1,1,2,2,3,3,4,4,5,6 };
	int sz = sizeof(arr) / sizeof(arr[0]);
	int signal_dog[2] = { 0 };
	find_signal_dog(arr, sz, signal_dog);
	printf("%d %d", signal_dog[0], signal_dog[1]);
	return 0;
}

 

二、模拟实现atoi

1.atoi函数的功能

如下图所示,atoi的功能是将一个字符串转化为整型数字

2.模拟实现atoi

如何模拟实现atoi是一个难点

因为我们会遇到很多种情况

1.如果遇到空指针

2.如果遇到空字符串

3.如果遇到空白字符

4.如果遇到+-

5.如果遇到数据溢出

为了解决上面的问题,我们可以一点一点来思考,首先是空指针,我们直接使用断言即可

然后当我们遇到一个空字符串也就是''\0''的时候,我们可以直接返回0,但是这时候,我们会发现一个问题,这个0也有可能是正儿八经的'0'字符转化得到的,所以我们要进行区分

我们定义一个状态值。用来加以区分。我们默认是非法的数据

当我们遇到空白字符的时候,我们需要跳过这些空白字符,当我们遇到+- 的时候,我们就需要判断正负号了

处理完这些情况以后,就是正常的转化了,在转化的时候,我们先要保证str还有值,然后我们使用isdigit函数判断是否为数字字符,如果是,则直接转化,在转化的时候要注意越界,为了判断是否越界,我们将ret设置为longlong类型,这样就可以根据int的最大最小值进行判断了。处理好了以后,如果str所指向的为非数字字符,那么就是说,转化完成了,我们直接就可以返回这个ret了,这里的ret我们认为是非法的字符串转化得到的。最后当str遇到\0的时候,那么也是转化完成了,这样的转化我们认为是合法的。所以设置状态值为VALID,返回ret

#include<stdio.h>
#include<stdlib.h>
#include<assert.h>
#include<ctype.h>
enum State
{
	VALID,
	INVALID
}state=INVALID;//默认是非法的
int my_atoi(const char* str)
{
	assert(str);
	if (str == '\0')
	{
		return 0;
	}
	//跳过空白字符
	while (isspace(*str))
	{
		str++;
	}
	//确认正负号
	int flag = 1;
	if (*str == '+')
	{
		flag = 1;
		str++;
	}
	if (*str == '-')
	{
		flag = -1;
		str++;
	}
	long long ret = 0;
	while (*str)
	{
		if (isdigit(*str))
		{
			ret = ret*10 + flag * (*str - '0');
			if (ret > INT_MAX)
			{
				return INT_MAX;
			}
			else if (ret < INT_MIN)
			{
				return INT_MIN;
			}
		}
		else
		{
			return ret;
		}
		str++;
	}
	state = VALID;
	return (int)ret;
}
int main()
{
	char* p = "1234";
	int a = my_atoi(p);
	if (state == VALID)
	{
		printf("%d\n", a);
	}
	else
	{
		printf("不合法的字符串:%d\n", a);
	}
	return 0;
}

 

 


本节内容就到这里

如果对你有帮助的话,不要忘记点赞加收藏哦!!!

想获得更多优质内容,一定要关注我哦!!!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/373773.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【Maven】(三)Maven仓库概念及私服安装与使用 附:Nexus安装包下载地址

文章目录1.前言2.Maven的仓库2.1.仓库类型3.私服Nexus3.1.Nexus的安装与配置3.1.1.使用安装包安装3.1.2.使用Docker安装3.2.Nexus配置3.2.1.仓库配置在这里插入图片描述4.私服的使用4.1.修改Maven配置4.2.从私服中下载构件4.3.推送构件到私服5.小结1.前言 本系列文章记录了 Ma…

超级困惑:单品牌好还是多品牌好?

超级困惑&#xff1a;单品牌好还是多品牌好&#xff1f; 相当于&#xff1a;买一套房好还是多套房好&#xff1f; 品牌是增加被消费者选择的优势 同一公司多品牌名之间&#xff0c;要区分明显 趣讲大白话&#xff1a;品牌要花大笔银子滴 【安志强趣讲信息科技87期】 **********…

?? JavaScript 双问号(空值合并运算符)

?? JavaScript 双问号&#xff08;空值合并运算符) 一、简述 在网上浏览 JavaScript 代码时或者学习其他代码时&#xff0c;可能会发现有的表达式用了两个问号&#xff08;??&#xff09;如下所示&#xff1a; let username; console.log(username ?? "Guest"…

kafka-console-ui v1.0.6发布

前言kafka-console-ui 是一款web版的kafka管理平台&#xff0c;从第一次发布到现在已经两年了&#xff0c;断断续续也更新了7个版本了&#xff08;v1.0.0~v1.0.6&#xff09;。一些常用的功能也陆续完善了不少&#xff0c;相对最新的kafka版本&#xff0c;某些功能上还是有所欠…

认识vue

认识vue.js框架 它是目前非常流行的一个框架。主要用于前端的MVVM的开发&#xff0c;也是前端的SPA开发框架&#xff0c;它是一个专 门用于服务前端的一个框架 能够实现SPA的框架目前有3个 1. vue.js 2. react.js 3. angular M:model V:view VM:viewmodel 网页代码相当于v…

你好 ETHDenver的2023年BUIDLathon已经准备就位开始建造

这是一站式 BUILDer 指南&#xff0c;可以帮助你做好迎接 Cartesi 在 ETHDenver 的挑战并且还可以使你充分的体验它你好&#xff0c;全球的以太坊社区。我们很高兴的将Cartesi 技术带到了丹佛市ETHDenver 2023的BUIDLathon中。如果你是一名喜欢测试新技术并且喜欢钻研创新的开发…

《Spring源码深度分析》第2章 容器的基本实现

目录标题前言一、容器的基本用法1、创建一个简单的Spring项目2、功能分析3、spring-beans模块1.核心类&#xff1a;DefaultListableBeanFactorya.容器加载相关类图b.XmlBeanFactory类2.核心类&#xff1a;XmlBeanDefinitionReader4、容器的基础 XmlBeanFactorya.配置文件封装b.…

1029 旧键盘 C++中find函数的使用

题目链接&#xff1a; 一、自己的想法&#xff1a;&#xff08;弱化版双指针&#xff09; 思路为用两个“指针”i, j分别指向原来字符串和实际输入字符串的第一个字符&#xff0c;然后判断i&#xff0c;j所指字符是否一致&#xff0c;若是则i, j同时&#xff0c;若否则将i所指…

浏览器多线程到事件循环机制

浏览器与js运行机制 进程与线程 进程 进程是CPU分配资源的最小单位&#xff0c;它是一个可以自己独立运行且拥有自己资源空间的任务程序&#xff1b;包括程序以及程序所使用的内存及系统资源 线程 线程是CPU调度的最小单位&#xff0c;它就是程序中的一个执行流&#xff1…

2023年三月份图形化二级打卡试题

活动时间 从2023年3月1日至3月21日&#xff0c;每天一道编程题。 本次打卡的规则如下&#xff1a; 小朋友每天利用10~15分钟做一道编程题&#xff0c;遇到问题就来群内讨论&#xff0c;我来给大家答疑。 小朋友做完题目后&#xff0c;截图到朋友圈打卡并把打卡的截图发到活动群…

基础数据结构--线段树(Python版本)

文章目录前言特点操作数据存储updateLazy下移查询实现前言 月末了&#xff0c;划个水&#xff0c;赶一下指标&#xff08;更新一些活跃值&#xff0c;狗头&#xff09; 本文主要是关于线段树的内容。这个线段树的话&#xff0c;主要是适合求解我们一个数组的一些区间的问题&am…

java中的类型转换

java的基本数据类型 1.数值型&#xff1a;byte&#xff0c;short&#xff0c;int&#xff0c;long&#xff0c;float&#xff0c;double 2.字符型&#xff1a;char 3.布尔型&#xff1a;boolean 数据类型占据字节数byte1个字节short2个字节int4个字节long8个字节float4个字节…

开发一个问答式的node脚本

前言 我们公司一般有早上知识分享的规定&#xff0c;那天有个同事分享了如何通过Node脚本实现国际化替换 。 起因是这样的&#xff0c;有一个已经成熟的项目了&#xff0c;突然被要求实现中英文切换。前端中英文切换基本上就是通过 vue-i18n 来实现&#xff08;不熟悉的可以看…

安装配置DHCP

本次实验采用CentOS71.检查在安装DHCP之前先使用rpm命令查看系统中已有的DHCP软件包rpm -qa | grep dhcp由此可知&#xff0c;系统中尚未安装DHCP软件包2.安装我们可以使用yum命令为系统安装DHCP软件包yum -y install dhcp安装完成后再次检查可以看到DHCP软件包3.配置dhcp配置文…

20230225在WIN10下安装PR2023失败的解决

20230225在WIN10下安装PR2023失败的解决 2023/2/25 23:42 对于Adobe Premiere Pro 2023&#xff0c;就算你安装在早起的Windows 10上&#xff0c;也会安装失败的&#xff01; 对于WIN7&#xff0c;就不要再想安装PR2023了&#xff0c;根本不支持呀&#xff01; Adobe Installer…

php 基于ICMP协议实现一个ping命令

php 基于ICMP协议实现一个ping命令 网络协议是什么ICMP 协议什么是ICMP?ICMP 的主要功能ICMP 在 IPv4 和 IPv6 的封装Wireshark抓包ICMP 请求包分析PHP构建 ICMP 数据包php中的 pack & unpack 函数字节和字符packunpackICMP计算校验和步骤总结网络协议是什么 网络协议&…

_hand-1

实现防抖函数&#xff08;debounce&#xff09; 防抖函数原理&#xff1a;把触发非常频繁的事件合并成一次去执行 在指定时间内只执行一次回调函数&#xff0c;如果在指定的时间内又触发了该事件&#xff0c;则回调函数的执行时间会基于此刻重新开始计算 防抖动和节流本质是不一…

Socket通信详解

Socket通信详解 文章目录Socket通信详解Socket流程介绍函数介绍编程实例Socket流程介绍 socket通信类似于电话通信&#xff0c;其服务器基本流程就是 Created with Raphal 2.3.0安装电话socket()分配电话号码bind()连接电话线listen()拿起话筒accept()函数介绍 socket() 其中…

行测-判断推理-图形推理-样式规律-加减异同

图1图2图3选D图1图2都有的线&#xff0c;则消除图1图2只有一幅图里有的线&#xff0c;则保留选C第一列和第二列都有的线&#xff0c;则消除第一列和第二列只有一幅图里有的线&#xff0c;则保留选A第一列顺时针旋转90&#xff0c;再与第二列去同存异选D第一列和第二列去同存异&…

二叉树、队列、栈、广义表(二)数据结构与算法(十八)

数据结构与算法&#xff08;一&#xff09;-软件设计&#xff08;十七&#xff09;https://blog.csdn.net/ke1ying/article/details/129220378 线性表-队列与栈 队列&#xff1a;先进先出。 栈&#xff1a;先进后出。 循环队列&#xff1a;队投和队尾连接起来。 队空的条件&…