Функция W3CSchool неверна. Сбой, если существует несколько файлов cookie с одинаковым суффиксом, например:
ffoo=bar; foo=baz
При поиске foo
возвращается значение ffoo вместо foo .
Теперь вот что я хотел бы сделать: во-первых, вам нужно узнать синтаксис, как транспортируются куки. Исходная спецификация Netscape (есть только копии, доступные как , эта на haxx.se ) использует точки с запятой для разделения нескольких файлов cookie, в то время как каждая пара имя / значение имеет следующий синтаксис:
ИМЯ =
VALUE
Эта строка представляет собой последовательность символов, исключая точку с запятой, запятую и пробел. Если необходимо поместить такие данные в имя или значение, рекомендуется использовать какой-либо метод кодирования, например кодирование в стиле URL %XX
, хотя кодировка не определяется или не требуется.
Таким образом, разделение строки document.cookie
на точки с запятой или запятую является приемлемым вариантом.
Кроме того, RFC 2109 также указывает, что файлы cookie разделяются либо точками с запятой, либо запятыми:
cookie = "Cookie:" cookie-version
1*((";" | ",") cookie-value)
cookie-value = NAME "=" VALUE [";" path] [";" domain]
cookie-version = "$Version" "=" value
NAME = attr
VALUE = value
path = "$Path" "=" value
domain = "$Domain" "=" value
Хотя оба допускаются, предпочтительны запятые, поскольку они являются разделителем по умолчанию для элементов списка в HTTP.
Примечание: для обратной совместимости, разделитель в заголовке Cookie
это точка с запятой (;
) везде. Сервер также должен принимать запятую (,
)
в качестве разделителя между значениями cookie для будущей совместимости.
Более того, пара имя / значение имеет некоторые дополнительные ограничения, поскольку VALUE также может быть строкой в кавычках, как указано в RFC 2616 :
attr = token
value = token | quoted-string
Таким образом, эти две версии файлов cookie должны рассматриваться отдельно:
if (typeof String.prototype.trimLeft !== "function") {
String.prototype.trimLeft = function() {
return this.replace(/^\s+/, "");
};
}
if (typeof String.prototype.trimRight !== "function") {
String.prototype.trimRight = function() {
return this.replace(/\s+$/, "");
};
}
if (typeof Array.prototype.map !== "function") {
Array.prototype.map = function(callback, thisArg) {
for (var i=0, n=this.length, a=[]; i<n; i++) {
if (i in this) a[i] = callback.call(thisArg, this[i]);
}
return a;
};
}
function getCookies() {
var c = document.cookie, v = 0, cookies = {};
if (document.cookie.match(/^\s*\$Version=(?:"1"|1);\s*(.*)/)) {
c = RegExp.$1;
v = 1;
}
if (v === 0) {
c.split(/[,;]/).map(function(cookie) {
var parts = cookie.split(/=/, 2),
name = decodeURIComponent(parts[0].trimLeft()),
value = parts.length > 1 ? decodeURIComponent(parts[1].trimRight()) : null;
cookies[name] = value;
});
} else {
c.match(/(?:^|\s+)([!#$%&'*+\-.0-9A-Z^`a-z|~]+)=([!#$%&'*+\-.0-9A-Z^`a-z|~]*|"(?:[\x20-\x7E\x80\xFF]|\\[\x00-\x7F])*")(?=\s*[,;]|$)/g).map(function($0, $1) {
var name = $0,
value = $1.charAt(0) === '"'
? $1.substr(1, -1).replace(/\\(.)/g, "$1")
: $1;
cookies[name] = value;
});
}
return cookies;
}
function getCookie(name) {
return getCookies()[name];
}