Флаги перечисления в JavaScript

Мне нужно имитировать тип перечисления в Javascript, и подход кажется довольно простым:

var MyEnum = {Left = 1; Right = 2; Top = 4; Bottom = 8}

Теперь в C # я мог бы комбинировать эти значения следующим образом:

MyEnum left_right = MyEnum.Left | MyEnum.Right

а затем я могу проверить, имеет ли перечисление определенное значение:

if (left_right & MyEnum.Left == MyEnum.Left) {...}

Могу ли я сделать что-то подобное в Javascript?


person Andrey    schedule 26.10.2009    source источник
comment
Обратите внимание, что синтаксис объекта для MyEnum неправильный. CMS уже предоставила исправленный пример в его ответе.   -  person Justin Love    schedule 26.10.2009


Ответы (6)


В javascript вы сможете объединить их как:

var left_right = MyEnum.Left | MyEnum.Right;

Тогда тестирование будет таким же, как в вашем примере

if ( (left_right & MyEnum.Left) == MyEnum.Left) {...}
person Mike Clark    schedule 26.10.2009

Вам просто нужно использовать побитовые операторы:

var myEnum = {
  left: 1,
  right: 2,
  top: 4,
  bottom: 8
}

var myConfig = myEnum.left | myEnum.right;

if (myConfig & myEnum.right) {
  // right flag is set
}

Больше информации:

person Christian C. Salvadó    schedule 26.10.2009

Да, в Javascript работает побитовая арифметика. Вы должны быть осторожны с этим, потому что Javascript имеет только тип данных Number, который реализован как тип с плавающей запятой. Но значения преобразуются в 32-битные подписанные значения для побитовых операций. Так что до тех пор, пока вы не пытаетесь использовать более 31 бита, все будет в порядке.

person Warren Young    schedule 26.10.2009

Я попытался создать пример, демонстрирующий распространенный вариант использования, когда можно использовать перечисления битовых масок для управления подробностью ведения журнала. Он демонстрирует использование поразрядных операций JavaScript: См. На JSFiddle

/*
 * Demonstration of how a Flags enum can be simulated in JavaScript and 
 * Used to control what gets logged based on user passed value
 */

// A Flags Enum (sort-of)
var LogLevels = {
    NONE: 0,
    INFO: 1,
    TRACE: 2,
    DEBUG: 4,
    WARN: 8,
    ERROR: 16,
    FATAL: 32
};

// Initialize
var currLogLevel = LogLevels.NONE;

// User Sets a log level
var logLevel = LogLevels.WARN;

// Convert the configured logLvel to a bit-masked enum value
switch (logLevel) {
    case LogLevels.INFO:
        currLogLevel = LogLevels.INFO | LogLevels.TRACE | LogLevels.DEBUG | LogLevels.WARN | LogLevels.ERROR | LogLevels.FATAL;
        break;
    case LogLevels.TRACE:
        currLogLevel = LogLevels.TRACE | LogLevels.DEBUG | LogLevels.WARN | LogLevels.ERROR | LogLevels.FATAL;
        break;
    case LogLevels.DEBUG:
        currLogLevel = LogLevels.DEBUG | LogLevels.WARN | LogLevels.ERROR | LogLevels.FATAL;
        break;
    case LogLevels.WARN:
        currLogLevel = LogLevels.WARN | LogLevels.ERROR | LogLevels.FATAL;
        break;
    case LogLevels.ERROR:
    case LogLevels.FATAL:
    default:
        currLogLevel = LogLevels.ERROR | LogLevels.FATAL;
}

// Example: log verbosity set to WARN, so this would NOT be logged
if ((currLogLevel & LogLevels.DEBUG) == LogLevels.DEBUG) {
    console.log("log DEBUG");
}
person Sudhanshu Mishra    schedule 18.07.2014

Вот моя реализация в машинописном тексте:

export class FlagEnumService {

    constructor(private value: number = 0) {
    }

    public get() {
        return this.value;
    }

    public set(value: number): this {
        this.value = value;
        return this;
    }

    public has(key: number): boolean {
        return !!(this.value & key);
    }

    public add(key: number): this {
        this.value |= key;
        return this;
    }

    public delete(key: number): this {
        this.value &= ~key;
        return this;
    }

    public toggle(key: number): this {
        this.has(key) ? this.delete(key) : this.add(key);
        return this;
    }
}

И тесты на выяснение

import { FlagEnumService } from './flag-enum.service';

enum Test {
    None = 0,
    First = 1,
    Second = 2,
    Third = 4,
    Four = 8
}

describe('FlagEnumService', () => {
    let service: FlagEnumService;
    beforeEach(() => service = new FlagEnumService());

    it('should create with initial value', () => {
        service = new FlagEnumService(Test.First);
        expect(service.get()).toBe(Test.First);
    });

    it('should return true if has flag', () => {
        service = new FlagEnumService(Test.First);
        expect(service.has(Test.First)).toBe(true);
    });

    it('should return false if doesn\'t have flag', () => {
        service = new FlagEnumService(Test.First);
        expect(service.has(Test.Second)).toBe(false);
    });

    it('should add', () => {
        expect(service.add(Test.First).add(Test.Second).get()).toBe(Test.First + Test.Second);
    });

    it('should not add the same value twice', () => {
        expect(service.add(Test.First).add(Test.First).get()).toBe(Test.First);
    });

    it('should remove', () => {
        expect(
            service
                .add(Test.First)
                .add(Test.Second)
                .delete(Test.Second)
                .get()
        )
            .toBe(Test.First);
    });

    it('should return 0 when add then remove the same value', () => {
        expect(service.add(Test.First).delete(Test.First).get()).toBe(0);
    });

    it('should not remove not added values', () => {
        expect(service.add(Test.First).delete(Test.Second).get()).toBe(Test.First);
    });
});
person MarkosyanArtur    schedule 10.02.2020

Пометить перечисления в JavaScript

Перечисления флагов: значения должны увеличиваться в степени 2

var SEASONS = {
  Spring : 1,
  Summer : 2,
  Fall   : 4,
  Winter : 8
};

Нельзя использовать 0 в побитовой & операции для проверки флага b / c, он всегда будет приводить к нулю. Однако его можно использовать для логических сравнений.

Примеры использования (надуманные)

var getSeasonsSelected = function( seasons ) {
  var selected = [];

  // The perens are needed around the bitwise operation due to the
  // greater operator precedence of `===`
  if ( (seasons & SEASONS.Spring) === SEASONS.Spring ) selected.push('Spring');
  if ( (seasons & SEASONS.Summer) === SEASONS.Summer ) selected.push('Summer');
  if ( (seasons & SEASONS.Fall)   === SEASONS.Fall )   selected.push('Fall');
  if ( (seasons & SEASONS.Winter) === SEASONS.Winter ) selected.push('Winter');

  return selected;
};

var s1 = getSeasonsSelected( SEASONS.Spring | SEASONS.Fall );
console.log(s1);
//=> ["Spring", "Fall"]
person Farrukh Malik    schedule 24.09.2020