在学C语言的时候,学过两数交换:《小朋友学C语言(25):两数交换》 https://www.jianshu.com/p/64bc70f0abfe 在学C++的时候,也学了两数交换:《小朋友学C++(14 ):两数交换》 https://www.jianshu.com/p/9a071870e0a0 这里咱们要考虑的是,两个数相等的交换情况。 ********** 2 1 4 3 **********交换相同的值********** 5 5 6 6 0 0 0 0 9 9 结果显示,前两组数值不同的两个数,能交换成功。 后五组相同的值,交换成功的有3组。不成功的有两组,并且值都变成了0。 这是什么原因呢? 观察最后三对的数,都是数组里的同一元素交换。 可见自己写的两个函数,一定存在bug。 仔细观察,发现a3 = 5, b3 = 5,这是两个不同的变量(在内存里的地址不同),而myswap2(d[0], d[0]),这是对同一个地址里的数交换。
如果是-0的话,因为0无所谓正负,-0就是0,这会导致计算机里面有两种方式表示0。这是不允许的。那么这个数到底表示多少呢? 事实上,这个数很特殊。 接下来因为减掉之后,数变小了,又变回正常的正数了。 从这里可以看出,int类型的取值范围为[-2147483648, 2147483647], 即[-2^31, 2^31-1] (五)结论 (1)使用(三)中的方法交换两数后,如果数很大的话 ,中间a = (2)如果数很大的话,比如a = 2147483647; b = 1;a = a + b的期望结果是214748364,但是因为这个数不在int的取值范围内,所以得到了一个负数。 所以,建议不要使用(三)中的方法来交换两个数。
之前学C语言的时候,咱们直接在main函数中使用“异或”位运算符,很容易实现了两数交换。 本节课将在此基础上,把交换两个数的算法,封装到swap函数中。 这样不管是哪个地方想要交换两个数,调用swap函数就可以了。 在swap内,m = a = 1, n = b = 2,交换后m = 2, n = 1。但是,m和n的值并不会传回给a和b,导致a和b的值没有被对换。 解决方案,使用上节课讲过的引用。 address of n: 0x7fffd009de9c Before Swap: m = 1 n = 2 After Swap: m = 2 n = 1 a = 2 b = 1 分析: 使用引用后,达到了交换 交换m和n的值,就是交换a和b的值。
如果不引入第三个变量,你还有几种方法交换ab两个数? int a=10; int b=20; 怎么交换a,b?
异或运算可以达到交换两数的目的,代码如下: ? 但不推荐使用这种方式,附上常用的临时变量方法对比说明。 临时变量方法: ? 对于临时变量法,每次赋值只要读取一个变量的值到寄存器,然后再从寄存器写回到另一个变量中即可,前后涉及两次内存写入操作;但是对于异或运算操作,每次都需要读取两个数据到寄存器中,再进行运算操作,之后把结果写回到变量中 如果使用C语言实现上述两种方法,并用gcc编译器编译,可以使用命令 gcc -S swap.c 查看相应的汇编代码,临时变量法代码行数更少,另外使用 gcc 编译器时,用异或运算交换数组会出错,参见链接 在不引入临时变量的基础上,交换两数的值还可以使用三次加减法,代码如下: ? 这种方式同样需要三次内存写入操作,同时代码可读性也较差。 最后附上两张三种方法编译后对应汇编代码对比图(平台:Ubuntu14.04,gcc 4.8.4),图中 swap1.c 文件对应临时变量法,swap2.c 文件对应加减方法,swap3.c 文件对应异或方法
https://blog.csdn.net/FE_dev/article/details/71419588 说明 这里说的几种两数交换的方法,并不仅仅适用于JavaScript中,在其他语言中同样是适用的 ,两两交换,便达到了我们的目的,而a=[b,b=a][0];只是他的一种简写罢了。 总结 方法2 (求和交换),方法3 (按位异或),这两个方法比较局限,只能换数字,如果变量不是数字,就不能用了。 总结一下上面所有方法的思路,交换两数,不是凭空就能换的,终究还是要依靠一下别的数,只是不用把这个数表现出来,不用再声明一个变量而已。 交换两数,如果大家还有什么有意思的方法,欢迎在评论区,分享出来。
已知两个1~30之间的数字,甲知道两数之和,乙知道两数之积。 甲问乙:”你知道是哪两个数吗?”乙说:”不知道”; 乙问甲:”你知道是哪两个数吗?” 甲说:”也不知道”; 于是,乙说:”那我知道了”; 随后甲也说:”那我也知道了”; 这两个数是什么? 答案: 答案1:为x=1,y=6;甲知道和A=x+y=7,乙知道积B=x*y=6 答案2:为x=1,y=8;甲知道和A=x+y=9,乙知道积B=x*y=8 解: 设这两个数为x,y 甲知道两数之和 A=x+y; 乙知道两数之积 B=x*y; 该题分两种情况 : 允许重复, 有(1 <= x <= y <= 30); 不允许重复,有(1 <= x < y <= 2)由题设条件:甲不知道答案 <=> A=x+y 解不唯一 => A >= 5; 分两种情况: A=5,A=6时x,y有双解 A>=7 时x,y有三重及三重以上解 假设
用临时变量交换2个数 示例代码: int a = 20; int b = 15; System.out.println("a="+a+",b="+b); //交换前2个数的值 int temp = a; a = b; b = temp; System.out.println("a="+a+",b="+b); //交换后2个数的值 这种方式也是我们经常用到的,比较容易理解的一种方法。 不用临时变量交换2个数 方式一 示例代码: int a = 20; int b = 15; System.out.println("a="+a+",b="+b); //交换前2个数的值 a = 方式二 示例代码: int a = 20; int b = 15; System.out.println("a="+a+",b="+b); //交换前2个数的值 a = a ^ b; b = a ^ b; a = a ^ b; System.out.println("a="+a+",b="+b); //交换后2个数的值 本方法使用了位运算的相关知识。
看到了一个这样的提问:一个表中有id和name两个字段,id是连续非空不重复的,想要交换相邻表记录的name值(单双两两交换)。 -- 查询数据 select * from diosos; 实现相邻表记录交换(单双两两交换),原理就是根据对id取余判断奇偶和相邻id加减做连接关联,更新行记录的name。 -- 更新,两两交换,不成对不交换 update diosos d1 set d1.name = ( select d2.name from diosos d2 where (mod
LeetCode01两数之和 题目描述: 给定一个整数数组 nums 和一个目标值 target,请你在该数组中找出和为目标值的那 两个 整数,并返回他们的数组下标。 本题主要有暴力和哈希两种方法: 法一:暴力法 把所有两两配对的问题全部遍历出来,知道找到满足题意得结果为止,时间复杂度O(n2) ? 因为如果从数的特性来看: 数是一对形式出现的 一对有前后位置之分,在遍历到前的时候不一定会找到后面的元素,但是遍历到后面的元素前面一定被我们存储了。 ? } else { map.put(nums[i], i); } } return a; } LeetCode02两数之加 题目描述: 给出两个 非空 的链表用来表示两个非负的整数。
问:有变量 a 和 b,将 a 和 b 的值调换 # 解法1:使用其他变量 c c = a a = b b = c # 解法2:不使用其他变量 a = a + b b = a - b a = a - b # 解法3:拆包(python特有的方法) a, b = (b, a) a, b = b, a # =右边实质是元组
给你两个 非空 的链表,表示两个非负的整数。它们每位数字都是按照 逆序 的方式存储的,并且每个节点只能存储 一位 数字。 请你将两个数相加,并以相同形式返回一个表示和的链表。 你可以假设除了数字 0 之外,这两个数都不会以 0 开头。 示例1: 输入:l1 = [2,4,3], l2 = [5,6,4] 输出:[7,0,8] 解释:342 + 465 = 807. 先将l1和l2头节点的值加起来赋值给新链表的头节点 遍历两个链表,只要有一个链表还没有遍历到末尾,就继续遍历 2.每次遍历生成一个当前节点cur的下一个节点,其值为两链表对应节点的和再加上当前节点cur =NULL,那么p1=l1->next,否则就说明l1位数少,需要补上0 //p1=l1->next是对l1链表的每个节点进行遍历,获得当前节点上的数字 p1 = p1->next ! output(head->next); cout << head->val ; } 加法模板 <公式> 当前位 = (A 的当前位 + B 的当前位 + 进位carry) % 10 注意,AB两数都加完后
两数相加 题目描述 给出两个 非空 的链表用来表示两个非负的整数。其中,它们各自的位数是按照 逆序 的方式存储的,并且它们的每个节点只能存储 一位 数字。 如果,我们将这两个数相加起来,则会返回一个新的链表来表示它们的和。 您可以假设除了数字 0 之外,这两个数都不会以 0 开头。 示意图 对两数相加的可视化:342 + 465 = 807,每个结点都包含一个数字,并且数字按位逆序存储。 就像在纸上计算两个数字一样,首先从最低有效位也就是 链表 l1 和 l2 的头部开始相加。 由于每位数字都应当处于 0~9 的范围内,所以在计算两个数字的和时可能会出现”溢出“。 进位 carry 必定是0或1,这是因为两个数字相加(考虑到进位)可能出现的最大和为 9 + 9 + 1 = 19。
题意 给一个整数数组,找到两个数使得他们的和等于一个给定的数 target。 你需要实现的函数 twoSum 需要返回这两个数的下标, 并且第一个下标小于第二个下标。 思路 可以用一个 Map 集合,遍历数组,先记录下当前数与目标数的差值与角标,然后寻找与这个差值相同的数,找到后,将这两个数的角标加 1 后返回即可。 numbers[i], i); } int[] nums = {}; return nums; } } 原题地址 LintCode:两数之和
01 题目描述 给定一个整数数组 nums 和一个目标值 target,请你在该数组中找出和为目标值的那 两个整数,并返回他们的数组下标。 你可以假设每种输入只会对应一个答案。 但是,数组中同一个元素不能使用两遍。 7, 11, 15], target = 9 输出:[0, 1] 因为 nums[0] + nums[1] = 2 + 7 = 9 02 暴力枚举 这里想必大家很快就能得到思路也就是双指针遍历所有两两相加判断是否与目标值相等 return new int[]{i, j}; } } } return new int[0]; } 03 Hash表 但实际上按照上面我们去到数组当中找两个数相加为目标值的方式也就是在确定
题目 难度级别:简单 给定一个整数数组 nums 和一个目标值 target,请你在该数组中找出和为目标值的那 两个 整数,并返回他们的数组下标。 你可以假设每种输入只会对应一个答案。 但是,数组中同一个元素不能使用两遍。
两两交换链表中的节点 给定一个链表,两两交换其中相邻的节点,并返回交换后的链表。 你不能只是单纯的改变节点内部的值,而是需要实际的进行节点交换。 curTmp; pre = curTmp; cur = curTmp.next; } return headNode.next; }; 思路 通过迭代的方式实现两两交换链表中的节点 next赋值为nextNode,将nextNode的next赋值为curNode,最后将preNode赋值为curNode,curNode赋值为curNode的next,注意此时的curNode其实已经被交换换成了 ,是两个节点中的后一个节点,最后等待循环完成后返回头结点的next即可。
给定一个整数数组 nums 和一个整数目标值 target,请你在该数组中找出 和为目标值 target 的那 两个 整数,并返回它们的数组下标。 你可以假设每种输入只会对应一个答案。
给定一个整数数组 nums 和一个整数目标值 target,请你在该数组中找出 和为目标值 target 的那 两个 整数,并返回它们的数组下标。 你可以假设每种输入只会对应一个答案。
题目 给出两个 非空 的链表用来表示两个非负的整数。其中,它们各自的位数是按照 逆序 的方式存储的,并且它们的每个节点只能存储 一位 数字。 如果,我们将这两个数相加起来,则会返回一个新的链表来表示它们的和。 您可以假设除了数字 0 之外,这两个数都不会以 0 开头。