Pregunta

En los marcos de aplicaciones sigo viendo marcos que le permiten pasar múltiples valores Int (generalmente usados ??en lugar de una enumeración) a una función.

Por ejemplo:

public class Example
{ 
    public class Values
    {
        public static final int ONE = 0x7f020000;
        public static final int TWO = 0x7f020001;
        public static final int THREE = 0x7f020002;
        public static final int FOUR = 0x7f020003;
        public static final int FIVE = 0x7f020004;
    }

    public static void main(String [] args)
    {
        // should evaluate just Values.ONE
        Example.multiValueExample(Values.ONE);

        // should evalueate just values Values.ONE,  Values.THREE, Values.FIVE
        Example.multiValueExample(Values.ONE | Values.THREE | Values.FIVE);

        // should evalueate just values Values.TWO , Values.FIVE
        Example.multiValueExample(Values.TWO | Values.FIVE);
    }

    public static void multiValueExample(int values){
        // Logic that properly evaluates bitwise values
        ...
    }
}

Entonces, ¿qué lógica debería existir en multiValueExample para que yo pueda evaluar adecuadamente los valores int múltiples que se pasan utilizando el operador bit a bit?

¿Fue útil?

Solución

Sus valores deben ser potencias de 2.

De esa manera, no pierdes ninguna información cuando las haces bit a bit.

public static final int ONE   = 0x01;
public static final int TWO   = 0x02;
public static final int THREE = 0x04;
public static final int FOUR  = 0x08;
public static final int FIVE  = 0x10;

etc.

Entonces puedes hacer esto:

public static void main(String [] args) {
    Example.multiValueExample(Values.ONE | Values.THREE | Values.FIVE);
}

public static void multiValueExample(int values){
    if ((values & Values.ONE) == Values.ONE) {
    }

    if ((values & Values.TWO) == Values.TWO) {
    }

    // etc.
}

Otros consejos

Como ya se mencionó, considere el uso de enumeraciones en lugar de valores de bits.

Según Java 2 efectivo : " Artículo 32: Utilice EnumSet en lugar de los campos de bits "

El uso de EnumSet es bastante eficaz para el uso de memoria y muy conveniente.

Aquí hay un ejemplo:

package enums;

import java.util.EnumSet;
import java.util.Set;

public class Example {
  public enum Values {
    ONE, TWO, THREE, FOUR, FIVE
  }

  public static void main(String[] args) {
    // should evaluate just Values.ONE
    Example.multiValueExample(EnumSet.of(Values.ONE));

    // should evalueate just values Values.ONE, Values.THREE, Values.FIVE
    Example.multiValueExample(EnumSet.of(Values.ONE, Values.THREE, Values.FIVE));

    // should evalueate just values Values.TWO , Values.FIVE
    Example.multiValueExample(EnumSet.of(Values.TWO, Values.FIVE));
  }

  public static void multiValueExample(Set<Values> values) {
    if (values.contains(Values.ONE)) {
      System.out.println("One");
    }

    // Other checks here...

    if (values.contains(Values.FIVE)) {
      System.out.println("Five");
    }
  }
}

Configura los valores enteros para que sean potencias de dos para que cada valor enumerado sea un solo bit en la representación binaria.

int ONE = 0x1;    //0001
int TWO = 0x2;    //0010
int THREE = 0x4;  //0100
int FOUR = 0x8;   //1000

Luego, usa OR a nivel de bits para combinar valores y AND a nivel de bits para probar valores establecidos.

int test_value = (ONE | FOUR);   //-> 1001
bool has_one = (test_value & ONE) != 0;  //-> 1001 & 0001 -> 0001 -> true

Los valores que combina con | (OR binario, OR no lógico [que es ||]) no debe tener "1" superpuestos en su representación de bits. Por ejemplo,

ONE = 0x1 =   0000 0001
TWO = 0x2 =   0000 0010
THREE = 0x3 = 0000 0011
FOUR = 0x4 =  0000 0100

Entonces puede combinar UNO y DOS, por ejemplo:

ONE | TWO = 0000 0011

Pero no puedes distinguir ONE | DOS de TRES, porque hay bits superpuestos. Por lo tanto, los números que combine deben ser potencias de dos, de modo que no se superpongan cuando OR se combinen. Para probar si se pasó un número en " valores " ;, haga:

if (values & ONE) {
    // ... then ONE was set
}

Para comprender mejor por qué y cómo funciona esto, le recomiendo que lea un poco sobre representación binaria y lógica. Un buen lugar es Capítulo 3 del Arte de la Asamblea .

Bueno, si son potencias de 2, harías algo como la " pantalla " método en el siguiente código.

Aquí también hay un enlace en wikipedia sobre el tema que debería explica por qué quieres poderes de 2.

public class Main
{
    private static final int A = 0x01;
    private static final int B = 0x02;
    private static final int C = 0x04;

    public static void main(final String[] argv)
    {
        display(A);
        display(B);
        display(C);
        display(A | A);
        display(A | B);
        display(A | C);
        display(B | A);
        display(B | B);
        display(B | C);
        display(C | A);
        display(C | B);
        display(C | C);
        display(A | A | A);
        display(A | A | B);
        display(A | A | C);
        display(A | B | A);
        display(A | B | B);
        display(A | B | C);
        display(A | C | A);
        display(A | C | B);
        display(A | C | C);
        display(B | A | A);
        display(B | A | B);
        display(B | A | C);
        display(B | B | A);
        display(B | B | B);
        display(B | B | C);
        display(B | C | A);
        display(B | C | B);
        display(B | C | C);
        display(C | A | A);
        display(C | A | B);
        display(C | A | C);
        display(C | B | A);
        display(C | B | B);
        display(C | B | C);
        display(C | C | A);
        display(C | C | B);
        display(C | C | C);
    }

    private static void display(final int val)
    {
        if((val & A) != 0)
        {
            System.out.print("A");
        }

        if((val & B) != 0)
        {
            System.out.print("B");
        }

        if((val & C) != 0)
        {
            System.out.print("C");
        }

        System.out.println();
    }
}

El uso de máscaras de bits era popular cuando cada bit contaba. Otra forma de hacer esto hoy es usar enumeraciones que son más fáciles de manipular y extender.

import static Example.Values.*;
import java.util.Arrays;

public class Example {
    public enum Values { ONE, TWO, THREE, FOUR, FIVE }

    public static void main(String [] args) {
        // should evaluate just Values.ONE
        multiValueExample(ONE);

        // should evaluate just values Values.ONE,  Values.THREE, Values.FIVE
        multiValueExample(ONE, THREE, FIVE);

        // should evaluate just values Values.TWO , Values.FIVE
        multiValueExample(TWO, FIVE);
    }

    public static void multiValueExample(Values... values){
        // Logic that properly evaluates
        System.out.println(Arrays.asList(values));
        for (Values value : values) {
            // do something.
        }
    }
}

Primero, no puede definir los valores de esa manera para hacer comparaciones bit a bit. En su lugar, establezca diferentes bits:

public static final int ONE   = 0x1;  // First bit is set
public static final int TWO   = 0x2;  // Second bit is set
public static final int THREE = 0x4;  // Third bit is set
public static final int FOUR  = 0x8;  // Fourth bit is set
public static final int FIVE  = 0x10; // Fifth bit is set

En segundo lugar, probablemente debería estar usando java.util.BitSet para este tipo de operaciones:

BitSet bits = new BitSet(5);
bits.set(2);
bits.set(4);

System.out.println("these bits are set: " + bits);
// Prints "these bits are set: {2, 4}"

BitSet otherBits = new BitSet(5);
otherBits.set(3);
otherBits.set(4);

System.out.println("these bits are set: " + bits.or(otherBits));
// Prints "these bits are set: {2, 3, 4}"

El capítulo del Tutorial de Java sobre operaciones bit a bit está en

http://java.sun.com/docs /books/tutorial/java/nutsandbolts/op3.html

Es muy conciso pero bueno como referencia.

Licenciado bajo: CC-BY-SA con atribución
No afiliado a StackOverflow
scroll top