“<<”,“|” ,"&" 运算 简单应用场景

本文介绍了一种使用位运算来标记对象属性状态的方法,适用于Android应用管理和客户端与服务器交互场景,通过位运算实现高效的数据传输。


简介:

近日,偶然学习到一个知识点,就是用“与”运算 算出来的结果来传值,Android framework中 通过包管理器来获得 应用程序信息,如下:

PackageManager pm = this.getPackageManager();
List<PackageInfo> appinfos = pm.getInstalledPackages(0);
for(PackageInfo p:appinfos){
    String packageName  = p.packageName ;
    Drawable icon = p.applicationInfo.loadIcon(pm);
    String lableName = p.applicationInfo.loadLabel(pm).toString();
    int flags = p.applicationInfo.flags; //应用程序的 属性(比如是系统应用还是用户的应用,是不是游戏,安装在哪里了等等)标记
    if((flags & ApplicationInfo.FLAG_SYSTEM )==0){
        // & 运算只要有一个为0,就是0,只有都是1才为1. 因此如果这个&运算 为0 ,则说明 flags的最后一位
        //等于0  表示false ,就是不是system 应用
    } else{
        //不是系统应用当然是 用户安装的应用
    }

    if((flags & ApplicationInfo.FLAG_EXTERNAL_STORAGE )==0){
        //表示没有装在 外存储设备上,就是 安装在了内存中
    }else{
        //表示装在 外存储设备上
    }
}

由此上面代码可以看到 ,一个对象的属性如果是多种状态的时候 ,(比如flags 来标记 应用程序状态属性等),且这个状态是可以用bool类型描述,(比如 不是系统应用就是自己安装应用,是不是游戏 ,是不是安装在外存储设备 等等), 就可以用定义 二进制类型的整数来表示。再通过& 运算 计算可以得出 一个对象属性的状态。 我自己总结为:

 一、一个对象的某个属性有多种状态

二、这个对象的属性的状态 可以用bool类型描述

三、这个对象的属性的状态 是不能认为改变的,比如 不能将一个系统应用改变一个用户普通应 用 (这样显然是不合理的)。


那么就可以用这种 像ApplicationInfo.FLAG_XXX的形式来设计 ,尤其是当这个对象的属性的状态多的时候,这种设计显得更方便,简洁。


利用这种形式设计 代替枚举类型的传值:(例子形式描述)

如何来实现这种设计呢,我总结为下面几个点:

一、用位移形式  定义对象属性状态的值。感觉像定义协议差不多。

一个对象属性选项如下:

public static final  int A = 1<<0;  

public static final  int B = 1<<1;  

public static final  int C = 1<<2;  

................. 都和 1 来 做“与” 运算。如果想给这个对象传递A和C ,那么传递( 1<<0) + (1<<2) 就可以了,拿到值后 经过计算 可以得到 是(1<<0) 和(1<<2)两个值

,并且只能是(1<<0) 和(1<<2)两个值,其他的都是非法的,这是由这种算法方式决定的。


 我们来假设一种场景,客户端 和服务器端交互时,客户端有个对象的属性有多种状态,例如一个视频的某个属性 是:是否免费,是否适合18+看 等等。这时如何设计呢,可以设计为用json 字段来描述 每个状态,

例如:{isFree:true,isAllowedAdult:true},。

如果用上述 方式来传递的话,需要事先和客户端定义好协议,

比如:

public static final int  isFree = 1<<0 ;                        //是否免费

public static final int  isAllowedAdult  = 1<<1 ;        //是否适合18岁以上人看

只需要传递 (1<<0)一个整数就 可以了,这样 节省了很多传输量。效率不言而喻。


二、对象属性的状态 在程序间传递(一个数字)

三、对象属性的状态 的获得 

其实 二、三 步骤都是 已经固定了。下面是例子:


package com.example.administrator.applicationname;

import android.util.ArrayMap;

import java.util.ArrayList;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Map;
import java.util.Random;
import java.util.Set;

/**
 * Created by Administrator on 2015/8/26.
 */
public class Test {

    //都和1来做 “与”运算
    public static final int A = 1<<0;
    public static final int B = 1<<1;
    public static final int C = 1<<2;
    public static final int D = 1<<3;
    public static final int E = 1<<4;
    public static final int F = 1<<5;

    /**
     *用来当做协议表,产生选项的时候可以用
     */
    static HashMap<String,Integer> map = null;
    /**
     * 用来当做协议表,拿到传递的值后解析用。
     */
    static HashMap<Integer,String> map2 = null;
    static Random r = new Random();
    public static void main(String[] a){

        //初始化协议 表
        map = new HashMap<String,Integer>();
        map2 = new HashMap<Integer,String>();
        map.put("A",A);
        map.put("B",B);
        map.put("C",C);
        map.put("D",D);
        map.put("E",E);
        map.put("F",F);
        map2.put(A,"A");
        map2.put(B,"B");
        map2.put(C,"C");
        map2.put(D,"D");
        map2.put(E, "E");
        map2.put(F, "F");

        //测试 :
        HashSet<String> set = new HashSet<String>();
        set.add("A");
        set.add("B");
        set.add("B");
        set.add("C");

        System.out.println("测试:选择的选项是" + set.toString());
        decode(encode(set));


    }

    /**
     * 测试 随机选择的项
     * @param whichToChoose
     * @return
     */
    private static int encode(HashSet<String> whichToChoose){
        if(whichToChoose!=null && whichToChoose.size()>6){
            throw  new IllegalArgumentException("最多只能限制6个选项或者参数不合法");
        }
        int sum = 0;
        for(String s : whichToChoose){
            Integer i = map.get(s);
            sum += i;
        }
        System.out.println("计算后产生的数为:" + sum);
        return sum;
    }

    /**
     *
     * @param param
     */
    private static void decode(int param){

        ArrayList<Integer> list = new ArrayList<Integer>();
        int tmp = 1;
        for(int i = 0 ; i < param; i++){ //32

            if((tmp&param) != 0){
                System.out.println("选项为:" + i);
                list.add((1<<i));
            }
            tmp = tmp << 1;
        }

        if(list.size()>0){
            for (Integer  integer : list) {
                String selected =map2.get(integer);
                System.out.println("选择的:"+selected);
            }
        }
    }

}

运行结果:




再来看下& 运算:

                int c = 2 << 3;
		int d = 2 << 4;
		int e = 2 << 5;
		int f = 2 << 6;
		
		int temp = 2 << 5;
		System.out.println((temp & c)!=0?"选中了":"没选中");
		System.out.println((temp & d)!=0?"选中了":"没选中");
		System.out.println((temp & e)!=0?"选中了":"没选中");
		System.out.println((temp & f)!=0?"选中了":"没选中");

执行结果:

没选中
没选中
选中了
没选中

由此 ,我们可以做这么一个应用场景,假如传递过来一个值 param,  需要判断是否 是 c ,d e, f 中的 一个。简单的是 方法是

一个一个的判断:

if(c==param){

}else if(d==param){

}else if(e==param){

}else if(f==param){}else{....}      


    或者   switch(param){

  case c:

break;

...................

               }


如果是用& 这个运算符来 做,那么在特定的业务场景下,比如单选,只需要考虑正确的判断逻辑即可, 于是就可以写成:(加入判断是否选中某个选项--单选c)

if(param &  c != 0){

  // 表示 选中了c

}


而不必再一个一个判断。





&lt;think&gt;好的,用户的问题是关于如何在C语言中正确表示条件&ldquo;10 &lt; x &lt; 100 或者 x &lt; 0&rdquo;。首先,我需要回忆一下C语言的语法规则,特别是关于逻辑表达式的部分。 用户提到的&ldquo;10 &lt; x &lt; 100&rdquo;在数学上是常见的写法,但在编程中,尤其是C语言里,这样的写法并不正确。因为C语言中的关系运算符是从左到右结合的,所以&ldquo;10 &lt; x &lt; 100&rdquo;会被解释为&ldquo;(10 &lt; x) &lt; 100&rdquo;。这里,10 &lt; x的结果是一个布尔值,即0或1,然后再100比较。这显然不符合用户的意图,因为无论x的值如何,这个表达式的结果都是1 &lt; 100,也就是总是为真。所以必须纠正这个写法。 正确的做法应该是将两个条件分开,并用逻辑运算符(&amp;&amp;)连接,即&ldquo;x &gt; 10 &amp;&amp; x &lt; 100&rdquo;。然后再用逻辑或运算符(||)x &lt; 0连接起来。因此,整个条件表达式应该是&ldquo;(x &gt; 10 &amp;&amp; x &lt; 100) || x &lt; 0&rdquo;。需要注意的是,虽然括号在某些情况下可能不是必须的,但为了确保运算顺序的正确性和代码的可读性,加上括号是一个好的做法。 接下来,我需要考虑用户可能的误区。例如,用户可能误以为数学中的连续不等式可以直接用在C语言中,或者可能忽略运算符的优先级问题。因此,在回答时需要明确说明这一点,并强调使用逻辑运算符的重要性。 另外,用户可能需要一个具体的代码示例来验证他们的理解。因此,提供一个简单的if语句示例会很有帮助。例如: if ((x &gt; 10 &amp;&amp; x &lt; 100) || x &lt; 0) { // 条件满足时的操作 } 这样可以让用户更直观地看到正确的写法。同时,需要提醒用户注意运算符的优先级,虽然逻辑&amp;&amp;)的优先级高于逻辑或(||),但使用括号可以避免混淆,尤其是在复杂的表达式中。 总结来说,回答的重点在于指出数学表达式C语言表达式的区别,正确使用逻辑运算符,并提供示例代码。同时,可能需要预见到用户后续的问题,比如运算符优先级或其他逻辑条件的组合,因此提前在相关问题上进行简要说明也是有必要的。&lt;/think&gt;在C语言中,数学表达式$10&lt;x&lt;100$需要拆解为逻辑表达式。正确的表达方式为: ```c (x &gt; 10 &amp;&amp; x &lt; 100) || x &lt; 0 ``` ### 分步说明: 1. **数学表达式拆解**: $10&lt;x&lt;100$在数学中是连续不等式,但在C语言中需拆解为两个关系表达式并用逻辑运算符连接,即`x &gt; 10 &amp;&amp; x &lt; 100` 2. **逻辑运算符选择**: - `&amp;&amp;`表示逻辑&quot;&quot;(两个条件同时满足) - `||`表示逻辑&quot;或&quot;(任意一个条件满足) 3. **运算符优先级**: 关系运算符(`&gt;`、`&lt;`)优先级高于逻辑运算符(`&amp;&amp;`、`||`),但建议用括号明确优先级,例如: `(x &gt; 10 &amp;&amp; x &lt; 100) || x &lt; 0` ### 错误写法示例: - `10 &lt; x &lt; 100`:会被解析为`(10 &lt; x) &lt; 100`,由于`10 &lt; x`返回布尔值(0或1),最终等价于`1 &lt; 100`,永远为真 - `x &gt; 10 || x &lt; 100`:逻辑错误,因为$x&lt;100$包含所有小于100的数,包括负数 ### 扩展应用场景: 该表达式常用于参数范围校验,例如输入过滤、物理量阈值判断等[^1]。
评论 1
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值