- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我想写这样的东西:
if [[ ( releases["token"] & $MASK ) -eq 1 ]]; then
但我得到的错误是:
unexpected token `&', conditional binary operator expected
如何在 if 语句中使用按位运算符?
最佳答案
乍一看,正如@chepner 的 comment 中指出的那样,问题可能只是语法之一,这会导致编译错误(unexpected token '&', conditional binary operator expected
)。事实上,@kev's answer通过使用 arithmetic expansion 来解决这个问题,应用于 @Gustavo's answer 中的 If
语句.
但是,问题“如何在 if 语句中使用位运算符” 是以更一般的方式表述的,并请求关于如何< em>使用位运算符来检查$MASK
(这不是“使用二进制比较时如何避免语法错误”的问题)。
其实可以假设示例代码
if [[ ( releases["token"] & $MASK ) -eq 1 ]]; then
是一项正在寻找解决方案的工作,并被明确标记为“类似...”。现在@kev 的任意假设
releases["token"]=3
MASK=5
可能掩盖了一个事实,即首先使用 -eq 1
也存在逻辑上的误解。因此,虽然@kev 的答案适用于所选值,但输入的结果例如4
行在
(((5&4)==1)) && echo YES || echo NO
将打印 NO,即使在这种情况下也期望 YES,因为 4
和 5
都设置了第 3 位.
因此,这个答案解决了问题示例中的这个逻辑错误,并尝试对问题的标题进行一般性回答。
to understand bitwise comparison it is helpful to visualise numbers in their bit representation (listed top-down in the following example):
|-----------------------------------------------------------|
| | hexadecimal representation of the value |
| bit val | 0 1 2 3 4 5 6 7 8 9 A B C D E F |
|---------|---------------------- bit ----------------------|
| | shows if the given value has the given bit set |
| 0 | - x - x - x - x - x - x - x - x |
| 1 | - - x x - - x x - - x x - - x x |
| 2 | - - - - x x x x - - - - x x x x |
| 3 | - - - - - - - - x x x x x x x x |
|---------|---------------------- val ----------------------|
| shows the value that the given bit adds to the number |
| 0 1 | 0 1 0 1 0 1 0 1 0 1 0 1 0 1 0 1 |
| 1 2 | 0 0 2 2 0 0 2 2 0 0 2 2 0 0 2 2 |
| 2 4 | 0 0 0 0 4 4 4 4 0 0 0 0 4 4 4 4 |
| 3 8 | 0 0 0 0 0 0 0 0 8 8 8 8 8 8 8 8 |
|---------|-------------------------------------------------|
|sum: 15 | 0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 |
| | decimal representation of the value |
|===========================================================|
由以下代码片段创建:
( # start a subshell to encapsulate vars for easy copy-paste into the terminal
# do not use this in a script!
echo ;
echo "|-----------------------------------------------------------|";
echo "| | hexadecimal representation of the value |";
echo "| bit val | 0 1 2 3 4 5 6 7 8 9 A B C D E F |";
echo "|---------|---------------------- bit ----------------------|";
echo "| | shows if the given value has the given bit set |";
mask=0;
for (( bit=0; bit<4; bit++)); do
mask=$((1<<bit));
echo -n "| $bit |"
for ((x=0;x<16;x++)); do ((((x&mask)>0))&&echo -n ' x'||echo -n ' -'); done
echo " |";
done ;
echo "|---------|---------------------- val ----------------------|";
echo "| shows the value that the given bit adds to the number |";
mask=0;
for (( bit=0; bit<4; bit++)); do
mask=$((1<<bit));
echo -n "| $bit $mask |"
for ((x=0;x<16;x++)); do echo -n " $((x&mask))"; done
echo " |";
done ;
echo "|---------|-------------------------------------------------|";
echo "|sum: 15 | 0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 |";
echo "| | decimal representation of the value |";
echo "|===========================================================|";
echo ;
)
3 & 5
与 4 & 5
的奇怪案例==1
vs >0
和&
, |
的区别, 和 ^
"if we now check the special case of the examples mentioned, we'll see the shortcoming:
|------------------ value: 3 / mask: 5 -------------------|
| | value | mask | and | or | xor | xnor |
| | 3 | 5 | 3 & 5 | 3 | 5 | 3 ^ 5 | |
| bit val |bit:val|bit:val|bit:val|bit:val|bit:val|bit:val|
|---------|-------|-------|-------|-------|-------|-------|
| 0 1 | 1 1 | 1 1 | 1 1 | 1 1 | 0 0 | 0 0 |
| 1 2 | 2 1 | 0 0 | 0 0 | 1 2 | 1 2 | 1 2 |
| 2 4 | 0 0 | 4 1 | 0 0 | 1 4 | 1 4 | 1 4 |
| 3 8 | 0 0 | 0 0 | 0 0 | 0 0 | 0 0 | 0 0 |
|---------|-------|-------|-------|-------|-------|-------|
|sum: | 3 | 5 | ==> 1 | 7 | 6 | 6 |
|---------|-----------------------------------------------|
|check: | 3 & 5 == 1 ? YES | |
| | 3 & 5 >= 1 ? YES | ==> 3 & 5 > 0 ? YES |
|=========================================================|
|------------------ value: 4 / mask: 5 -------------------|
| | value | mask | and | or | xor | xnor |
| | 4 | 5 | 4 & 5 | 4 | 5 | 4 ^ 5 | |
| bit val |bit:val|bit:val|bit:val|bit:val|bit:val|bit:val|
|---------|-------|-------|-------|-------|-------|-------|
| 0 1 | 0 0 | 1 1 | 0 0 | 1 1 | 1 1 | 1 1 |
| 1 2 | 0 0 | 0 0 | 0 0 | 0 0 | 0 0 | 0 0 |
| 2 4 | 4 1 | 4 1 | 1 4 | 1 4 | 0 0 | 0 0 |
| 3 8 | 0 0 | 0 0 | 0 0 | 0 0 | 0 0 | 0 0 |
|---------|-------|-------|-------|-------|-------|-------|
|sum: | 4 | 5 | ==> 4 | 5 | 1 | 1 |
|---------|-----------------------------------------------|
|check: | 4 & 5 == 1 ? NO | |
| | 4 & 5 >= 1 ? YES | ==> 4 & 5 > 0 ? YES |
|=========================================================|
请特别注意以下检查:
|---------|-----------------------------------------------|
|check: | 3 & 5 == 1 ? YES | |
| | 3 & 5 >= 1 ? YES | ==> 3 & 5 > 0 ? YES |
---------|------------------------------------------------|
|check: | 4 & 5 == 1 ? NO | |
| | 4 & 5 >= 1 ? YES | ==> 4 & 5 > 0 ? YES |
|---------|-----------------------------------------------|
上面的输出是由以下代码片段创建的:
( # start a sub-shell to encapsulate vars for easy copy-paste into the terminal
# do not use this in a script!
echo ;
for o in 3 4; do
echo "|------------------ value: $o / mask: 5 -------------------|";
echo "| | value | mask | and | or | xor | xnor |";
echo "| | $o | 5 | $o & 5 | $o | 5 | $o ^ 5 | |";
echo "| bit val |bit:val|bit:val|bit:val|bit:val|bit:val|bit:val|";
echo "|---------|-------|-------|-------|-------|-------|-------|";
mask=0;
for (( bit=0; bit<4; bit++)); do
mask=$((1<<bit));
echo -n "| $bit $mask "
echo -n " | $(($o&mask)) $((($o&mask)>0))"
echo -n " | $((5&mask)) $(((5&mask)>0))"
echo -n " | $(((($o&mask)&(5&mask))>0)) $((($o&mask)&(5&mask)))"
echo -n " | $(((($o&mask)|(5&mask))>0)) $((($o&mask)|(5&mask)))"
echo -n " | $(((($o&mask)^(5&mask))>0)) $((($o&mask)^(5&mask)))"
echo -n " | $(((($o&mask)^(5&mask))>0)) $((($o&mask)^(5&mask)))"
echo " |";
done ;
echo "|---------|-------|-------|-------|-------|-------|-------|";
echo -n "|sum: | $o | 5 |";
echo " ==> $(($o&5)) | $(($o|5)) | $(($o^5)) | $(($o^5)) |";
echo "|---------|-----------------------------------------------|";
echo -n "|check: | $o & 5 == 1 ? $(((($o&5)==1))&&echo YES||echo "NO ") ";
echo "| |";
echo -n "| | $o & 5 >= 1 ? $(((($o&5)>=1))&&echo YES||echo "NO ") ";
echo "| ==> $o & 5 > 0 ? $(((($o&5)>0))&&echo YES||echo "NO ") |";
echo "|=========================================================|";
echo ;
done
echo ;
)
So how do we do it now?! Let's imagine, we have the following option set:
# option set:
option_1=1
option_2=2
option_3=4
option_4=8
option_5=16
option_6=32
option_7=64
option_8=128
option_9=256
We could for example set a selection of those options in a function and return the combined code as a return value. Or the other way round: pass a selection of options as one numeric parameter. Whatever your use case is, the options would be summed together:# set options:
option = option_1
+ option_4
+ option_5
+ option_9
How do we best check, which options are set (1
,4
,5
, &9
)? It depends of course again on your use case, but i kinda like thecase
construct! Something like ...case $option in
0) echo "no option set!";;
1) echo "option 1";;
2) echo "option 2";;
3) echo "option 1 and 2";;
*) echo "...";;
esaccould work, but is not very nice, as we would have to construct every combination!
案例
”We can use
case
in the following way instead ...echo "check for options using >0:"
case 1 in
$(( (option & option_1) >0 )) ) echo "- option_1 is set";;&
$(( (option & option_2) >0 )) ) echo "- option_2 is set";;&
$(( (option & option_3) >0 )) ) echo "- option_3 is set";;&
$(( (option & option_4) >0 )) ) echo "- option_4 is set";;&
$(( (option & option_5) >0 )) ) echo "- option_5 is set";;&
$(( (option & option_6) >0 )) ) echo "- option_6 is set";;&
$(( (option & option_7) >0 )) ) echo "- option_7 is set";;&
$(( (option & option_8) >0 )) ) echo "- option_8 is set";;&
$(( (option & option_9) >0 )) ) echo "- option_9 is set";;&
esacPlease note that
- the spaces within
$(( (option & option_1) >0 )) )
are completely optional and are added for readability. The last closing bracket)
is for thecase
construct.- the command-lists are terminated with
;;&
, in order to continue evaluation with the next option test. In case you want to abord further processing of the list, setoption=0
or to your currentoption=option_X
.... we get the following result:
check for options using >0:
- option_1 is set
- option_4 is set
- option_5 is set
- option_9 is set欢呼! :-)
如果
我是个有钱人!echo "# to use it in an if statement:";
if (((option&option_5)>0)); then
echo "- option_5 is set"
else
echo "- option_5 is NOT set"
fi
# to use it in an if statement:
- option_5 is set
echo "# or to use it just for conditional execution:";
(((option&option_6)>0)) \
&& echo "- option_6 is set" \
|| echo "- option_6 is NOT set"
# or to use it just for conditional execution:
- option_6 is NOT set
(卡夫卡,1975:181)
So it would be perfectly possible to use the solution as posted in the question, by the slight change as follows:
(
declare -A releases=([token]=4)
declare -i MASK=5
if [[ $(( releases["token"] & $MASK )) -gt 0 ]]; then
echo YES
else
echo NO
fi
)The changes are the following: - use
$(())
instead of()
to do the test, which will return the value of the bitwise comparison - use-gt 0
instead of-eq 1
完全在行动:
( # start a sub-shell to encapsulate vars for easy copy-paste into the terminal
# do not use this in a script!
echo ;
for ((i=0; i<9;i++)); do
o="option_$((i+1))"
declare -i $o=$((1<<$i))
echo "$o = ${!o}"
done
echo ;
echo ;
echo "# set options:"
echo "option = option_1"
echo " + option_4"
echo " + option_5"
echo " + option_9"
option=option_1+option_4+option_5+option_9
echo ;
echo ;
echo "check for options using >0:"
case 1 in
$(( (option & option_1) >0 )) ) echo "- option_1 is set";;&
$(( (option & option_2) >0 )) ) echo "- option_2 is set";;&
$(( (option & option_3) >0 )) ) echo "- option_3 is set";;&
$(( (option & option_4) >0 )) ) echo "- option_4 is set";;&
$(( (option & option_5) >0 )) ) echo "- option_5 is set";;&
$(( (option & option_6) >0 )) ) echo "- option_6 is set";;&
$(( (option & option_7) >0 )) ) echo "- option_7 is set";;&
$(( (option & option_8) >0 )) ) echo "- option_8 is set";;&
$(( (option & option_9) >0 )) ) echo "- option_9 is set";;&
esac
echo ;
echo ;
echo "# to use it in an if statement:";
echo " => if (((option&option_5)>0));";
if (((option&option_5)>0)); then
echo "- option_5 is set"
else
echo "- option_5 is NOT set"
fi
echo ;
echo ;
declare -A releases=([token]=4)
declare -i MASK=5
echo "# Does 4 pass the mask 5 in the test construct?";
echo " => if [[ $(( releases["token"] & $MASK )) -gt 0 ]];";
if [[ $(( releases["token"] & $MASK )) -gt 0 ]]; then
echo YES
else
echo NO
fi
echo ;
echo ;
echo "# or to use it just for conditional execution:";
echo " => (((option&option_6)>0)) && do || dont";
(((option&option_6)>0)) \
&& echo "- option_6 is set" \
|| echo "- option_6 is NOT set"
)
Exit 0
关于bash - 如何在 if 语句中使用位运算符?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/14318451/
创建一个“海盗对话”,可以选择左手或右手。我希望它对“左”和“右”的不同拼写做出积极的回答(正如您将在代码中看到的那样),但是,当我为所有非“右”或“左”的输入添加最终的“else”代码时,它给了我一
With 语句 对一个对象执行一系列的语句。 With object statements End With 参数 object 必需的部分
While...Wend 语句 当指定的条件为 True 时,执行一系列的语句。 While condition  ; Version [stat
所以我正在处理的代码有一个小问题。 while True: r = input("Line: ") n = r.split() if r == " ":
我有一个对象数组: var contacts = [ { "firstName": "Akira", "lastName": "Laine", "number"
int main() { int f=fun(); ... } int fun() { return 1; return 2; } 在上面的程序中,当从main函数中调用一个
我的项目中有很多 if 语句、嵌套 if 语句和 if-else 语句,我正在考虑将它们更改为 switch 语句。其中一些将具有嵌套的 switch 语句。我知道就编译而言,switch 语句通常更
Rem 语句 包含程序中的解释性注释。 Rem comment 或 ' comment comment 参数是需要包含的注释文本。在 Rem 关键字和 comment 之间应有一个空格。
ReDim 语句 在过程级中声明动态数组变量并分配或重新分配存储空间。 ReDim [Preserve] varname(subscripts) [, varname(subscripts)]
Randomize 语句 初始化随机数生成器。 Randomize [number] number 参数可以是任何有效的数值表达式。 说明 Randomize 使用 number 参数初始
Public 语句 定义公有变量并分配存储空间。在 Class 块中定义私有变量。 Public varname[([subscripts])][, varname[([subscripts])
Sub 语句 声明 Sub 过程的名称、参数以及构成其主体的代码。 [Public [Default]| Private] Sub name [( arglist )]
Set 语句 将对象引用赋给一个variable或property,或者将对象引用与事件关联。 Set objectvar = {objectexpression | New classname
我有这个代码块,有时第一个 if 语句先运行,有时第二个 if 语句先运行。我不确定为什么会这样,因为我认为 javascript 是同步的。 for (let i = 0; i < dataObje
这是一个 javascript 代码,我想把它写成这样:如果此人回答是,则回复“那很酷”,如果此人回答否,则回复“我会让你开心”,如果此人回答的问题包含"is"或“否”,请说“仅键入”是或否,没有任何
这是我的任务,我尝试仅使用简短的 if 语句来完成此任务,我得到的唯一错误是使用“(0.5<=ratio<2 )”,除此之外,构造正确吗? Scanner scn = new Scanner(
有没有办法在 select 语句中使用 if 语句? 我不能在这个中使用 Case 语句。实际上我正在使用 iReport 并且我有一个参数。我想要做的是,如果用户没有输入某个参数,它将选择所有实例。
这个问题在这里已经有了答案: 关闭 11 年前。 Possible Duplicate: If vs. Switch Speed 我将以 C++ 为例,但我要问的问题不是针对特定语言的。我的意思是一
Property Set 语句 在 Class 块中,声明名称、参数和代码,这些构成了将引用设置到对象的 Property 过程的主体。 [Public | Private] Pro
Property Let 语句 在 Class 块中,声明名称、参数和代码等,它们构成了赋值(设置)的 Property 过程的主体。 [Public | Private] Prop
我是一名优秀的程序员,十分优秀!