位运算

^ 异或运算符

异0 同1 (又称无进位相加)

0 ^ 0 -> 0
1 ^ 1 -> 0
0 ^ 1 -> 1
1 ^ 0 -> 1
10110 ^ 00111 -> 10001

在这里插入图片描述
解释:
2) a ^ b = b ^ a ; (a ^ b) ^ c = a ^( b ^ c)
交换a、b的值 (只要内存位置不一样都可以 否则会0)

//用^交换变量值
int a=17;
int b=56;
a=a ^ b;  //a=a.^b.;               b=b.;
b=a ^ b;  //a=a.^b.;               b=a.^b.^b.=a.^(b.^b.)=a.^0=a.;
a=a ^ b;  //a=a.^b.^a.=b.;         b=a.;

4)、5)时间复杂度O(n),空间复杂度O(1):

for{}
int eor = 0;
eor = eor ^ arr[i]

eor即是所求数

public class data16
{
public static void main(String[] args)
{
int a=15;
int b=2;
System.out.println("a 与 b 异或的结果是:"+(a^b));
}
}

运行结果
a 与 b 异或的结果是:13
分析上面的程序段:a 的值是15,转换成二进制为1111,而b 的值是2,转换成二进制为0010,根据异或的运算规律,可以得出其结果为1101 即13。


~ : 取反 运算符

如果位为0,结果是1,如果位为1,结果是0
~0 = 1;
~1 = 0;
~1010=0101;
~1010+1=0110;

public class data15
{
public static void main(String[] args)
{
int a=2;
System.out.println("a 非的结果是:"+(~a));
}
}

& 与 运算符

两个操作数中 位 都为1,结果才为1,否则结果为0

~1010+1 & 1010= 0010 取一个数右侧第一个1 ~a+1&a

public class data13
{
public static void main(String[] args)
{
int a=129;
int b=128;
System.out.println("a 和b 与的结果是:"+(a&b));
}
}

运行结果
a 和b 与的结果是:128
下面分析这个程序:
“a”的值是129,转换成二进制就是10000001,而“b”的值是128,转换成二进制就是10000000。根据与运算符的运算规律,只有两个位都是1,结果才是1,可以知道结果就是10000000,即128。


| 或 运算符

或运算符用符号“|”表示,其运算规律如下:

两个位只要有一个为1,那么结果就是1,否则就为0,下面看一个简单的例子。


public class data14
{
public static void main(String[] args)
{
int a=129;
int b=128;
System.out.println("a 和b 或的结果是:"+(a|b));
}
}

运行结果
a 和b 或的结果是:129
下面分析这个程序段:
a 的值是129,转换成二进制就是10000001,而b 的值是128,转换成二进制就是10000000,根据或运算符的运算规律,只有两个位有一个是1,结果才是1,可以知道结果就是10000001,即129。


综合案例:

一个数组中有两种数出现了奇数次,其他数都出现了偶数次,怎么找到这两个数

public class Code07_EvenTimesOddTimes {

	public static void printOddTimesNum1(int[] arr) {
		int eO = 0;
		for (int cur : arr) {
			eO ^= cur;
		}
		System.out.println(eO);
	}
	public static void printOddTimesNum2(int[] arr) {
		int eO = 0, eOhasOne = 0;
		for (int curNum : arr) {
			eO ^= curNum;
		}
		//eO=a^b
		//eO!=0
		//eO 必然有一个位置上是1
		int rightOne = eO & (~eO + 1);//提取出最右的1
		for (int cur : arr) {
			if ((cur & rightOne) != 0) {
				eOhasOne ^= cur;
			}
		}
		System.out.println(eOhasOne + " " + (eO ^ eOhasOne));
	}

	public static void main(String[] args) {
		int a = 5;
		int b = 7;
		a = a ^ b;
		b = a ^ b;
		a = a ^ b;
		System.out.println(a);
		System.out.println(b);
		int[] arr1 = { 3, 3, 2, 3, 1, 1, 1, 3, 1, 1, 1 };
		printOddTimesNum1(arr1);

		int[] arr2 = { 4, 3, 4, 2, 2, 2, 4, 1, 1, 1, 3, 3, 1, 1, 1, 4, 2, 2 };
		printOddTimesNum2(arr2);
	}
  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 2
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值