JavaScript ( (__ = !$ + $)[+$] + ({} + $)[_/_] +({} + $)[_/_] )
($=[$=[]][(__=!$+$)[_=-~-~-~$]+({}+$)[_/_]+
($$=($_=!''+$)[_/_]+$_[+$])])()[__[_/_]+__
[_+~$]+$_[_]+$$](_/_)
注意,上面這段看起來(lái)很混亂的代碼并不是自動(dòng)換行,而是三行(當(dāng)然,你寫在同一行也沒(méi)有錯(cuò))。編寫一個(gè)頁(yè)面運(yùn)行一下(據(jù)說(shuō) IE 下不行),你就會(huì)發(fā)現(xiàn)這段代碼的功能等同于
alert(1)
為什么會(huì)這樣呢?我們來(lái)把這段代碼拆開(kāi)來(lái)分析。
$=[] // $ 被賦值為一個(gè)空數(shù)組,所以 !$ 的值為 false.
__ = !$ + $ // 加號(hào)會(huì)把 !$ 和 $ 都轉(zhuǎn)換成字符串,所以 __ 的值變成了字符串 “false”
_ = -~-~-~$ // 這里有一個(gè) ~ 操作符,它表示 -($+1),所以 -~$ 的值為 1. _ 的值為 3.
由此可以推導(dǎo):
(__ = !$ + $ )[ _ = -~-~-~$] => (”false”)[_] => (”false”)[3] => “false”[3] = “s”
({} + $)[_/_] => (”[object Object]“)[_/_] => (”[object Object]“)[1] => “[object Object]“[1] = “o”
接下來(lái)再拆 $$=($_=!”+$)[_/_]+$_[+$] :
$_=!”+$ // 注意,!” 中是兩個(gè)單引號(hào),也就是對(duì)一個(gè)空字符串做非運(yùn)算。所以變量 $_ 被賦值為字符串 “true”。
由此可推:
$$=($_=!”+$)[_/_]+$_[+$] => $$ = ( “true”)[1] + “true”[0] => “r” + “t” = “rt”
所以 (__=!$+$)[_=-~-~-~$]+({}+$)[_/_]+($$=($_=!”+$)[_/_]+$_[+$]) 就是 “s” + “o” + “rt” ,也就是 “sort”.
所以原來(lái)的表達(dá)式
($=[$=[]][(__=!$+$)[_=-~-~-~$]+({}+$)[_/_]+
($$=($_=!''+$)[_/_]+$_[+$])])()[__[_/_]+__
[_+~$]+$_[_]+$$](_/_)
可以被替換成:
($=[[]]["sort"])()[__[_/_]+__
[_+~$]+$_[_]+$$](_/_)
接下來(lái)我們看 [__[_/_]+__[_+~$]+$_[_]+$$](_/_) 是什么東西。
前面我們已經(jīng)得知:
__ = “false”
_ = 3
~$ = -1
$_ = “true”
$$ = “rt”
所以 [__[_/_]+__[_+~$]+$_[_]+$$](_/_) => ["false"[1] + “false”[3-1] + “true”[3] + “rt”](3/3) => ["a" + "l" + "e" + "rt"](1) => ["alert"](1)
所以原來(lái)的表達(dá)式最終可以被替換成:
($=[[]]["sort"])()["alert"](1)
這段代碼是如何執(zhí)行的呢?我們來(lái)逐步分析:
a = [[]] // 創(chuàng)建一個(gè)數(shù)組
b = a["sort"] // 獲取數(shù)組的 sort 方法
c = b() // 調(diào)用數(shù)組的 sort 方法,這里 b() 返回的是 window 對(duì)象
d = c["alert"] // 獲取 window.alert 方法
d(1) // 調(diào)用 window.alert 方法。
所以這堆亂七八遭的表達(dá)式最終的執(zhí)行結(jié)果就是 window.alert(1).
更多 請(qǐng)看原文 和 Reddit上的討論。
原文評(píng)論里也有人貼出了一個(gè)日本開(kāi)發(fā)者寫的小工具,可以把一段 JavaScript 代碼編碼成各種表情符號(hào),而且可以執(zhí)行,enjoy it.
BTW, 上面這段代碼除了做 XSS 攻擊之外作用不大,但是可以從分析這段代碼學(xué)習(xí)一點(diǎn)兒數(shù)據(jù)類型轉(zhuǎn)換相關(guān)的東西,也可以領(lǐng)略到 JavaScript 的靈活。
詳細(xì)代碼解析:
(
$=[$=[]] // $ = []
[
(__= !$ + $) // __ = "false"
[_=-~-~-~$] // _ = 3
// (__)[3] = "s"
+
({} + $) // ({} + $) = [object Object]
[_/_] // _/_ = 1
+ // ([object Object])[1] = "o"
($$ = //
($_=!'' + $) // !''+ $ = "true" ; $_ = "true"
[_/_] // _/_ = 1;
// $_[1] = "r"
+
$_[+$] // +$ = 0; $_[0] = "t"
) // $$= "rt"
] // ["sort"]
// []["sort"] = [].sort = function sort() { [native code] }
// $ = []["sort"]
)() // ($)() = [object Window]
[
__[_/_] // __ = "false";
// __[1] = "a"
+
__ [_+~$] //_ = 3; ~$ = -1; _ + ~$ = 2
// __[2] = "l"
+
$_[_] // $_ = "true" ; _ = 3;
// $_[3] = "e"
+
$$ // $$ = "rt"
](_/_); // _ / _ = 1
// window["alert"](1)
這里需要注意的幾點(diǎn):
1. javascript 里面 $, _ 均可以作為變量名
2. 函數(shù)還可以這樣調(diào)用: 比如 [1,2,4,1,9,1].sort() 可以寫成 [1,2,4,1,9,1]["sort"]();
3. ~ 對(duì)數(shù)字按位取反
4. javascript 中在對(duì)不同類型的變量 使用 + 時(shí)候的規(guī)則
5. 最后一個(gè)針對(duì)某些native code寫的方法,執(zhí)行會(huì)返回 window對(duì)象, 比如
var s = [].sort ; t=s(); 則 t 是 window
相關(guān)文章
基于BootStrap Metronic開(kāi)發(fā)框架經(jīng)驗(yàn)小結(jié)【七】數(shù)據(jù)的導(dǎo)入、導(dǎo)出及附件的查看處理
在很多系統(tǒng)模塊里面,我們可能都需要進(jìn)行一定的數(shù)據(jù)交換處理,這樣可以很好的達(dá)到用戶操作體驗(yàn)感,接下來(lái)通過(guò)本文給大家介紹基于BootStrap Metronic開(kāi)發(fā)框架經(jīng)驗(yàn)小結(jié)【七】數(shù)據(jù)的導(dǎo)入、導(dǎo)出及附件的查看處理相關(guān)知識(shí),非常具有參考價(jià)值,感興趣的朋友一起學(xué)習(xí)吧2016-05-05漂亮的widgets,支持換膚和后期開(kāi)發(fā)新皮膚
漂亮的widgets,支持換膚和后期開(kāi)發(fā)新皮膚...2007-04-04使用 bootstrap modal遇到的問(wèn)題小結(jié)
bootstrap提供了一個(gè)寫好的css文件和js文件然而在使用時(shí)遇到了一下并不是很好的問(wèn)題,今天在使用彈出對(duì)話框時(shí)遇到了一個(gè)這樣的問(wèn)題,今天小編給大家分享下使用 bootstrap modal遇到的問(wèn)題小結(jié),一起看看吧2016-11-11JavaScript 小型打飛機(jī)游戲?qū)崿F(xiàn)原理說(shuō)明
這次為大家?guī)?lái)的小游戲是:打飛機(jī)。呃。。。我本人就寫不出什么驚天大作的游戲的了,只能寫寫小游戲,代碼量小,又可以學(xué)習(xí),主要是想法思路,代碼量大,估計(jì)也沒(méi)啥人會(huì)去研究學(xué)習(xí)。。。2010-10-10微信小程序動(dòng)畫(Animation)的實(shí)現(xiàn)及執(zhí)行步驟
這篇文章主要介紹了微信小程序動(dòng)畫(Animation) 的實(shí)現(xiàn)及執(zhí)行步驟,需要的朋友可以參考下2018-10-10JavaScript?中的?parseInt()?函數(shù)詳解
parseInt函數(shù)將其第一個(gè)參數(shù)轉(zhuǎn)換為一個(gè)字符串,對(duì)該字符串進(jìn)行解析,然后返回一個(gè)整數(shù)或?NaN,這篇文章主要介紹了JavaScript?的?parseInt()?函數(shù),需要的朋友可以參考下2023-05-05JavaScript 監(jiān)聽(tīng)組合按鍵思路及代碼實(shí)現(xiàn)
這篇文章主要介紹了JavaScript 監(jiān)聽(tīng)組合按鍵思路及代碼實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-07-07