Как видно из заголовка, у меня есть строка, и я хочу разбить ее на сегменты по n символов.
Например:
var str = 'abcdefghijkl';
после некоторой магии с n=3
, он станет
var arr = ['abc','def','ghi','jkl'];
Есть ли способ сделать это?
Как видно из заголовка, у меня есть строка, и я хочу разбить ее на сегменты по n символов.
Например:
var str = 'abcdefghijkl';
после некоторой магии с n=3
, он станет
var arr = ['abc','def','ghi','jkl'];
Есть ли способ сделать это?
var str = 'abcdefghijkl';
console.log(str.match(/.{1,3}/g));
Если вы не хотите использовать регулярное выражение...
var chunks = [];
for (var i = 0, charsLength = str.length; i < charsLength; i += 3) {
chunks.push(str.substring(i, i + 3));
}
... в противном случае решение regex довольно хорошо:)
str.match(/.{3}/g); // => ['abc', 'def', 'ghi', 'jkl']
Основываясь на предыдущих ответах на этот вопрос; следующая функция будет разделять строку (str
) n-number (size
) символов.
function chunk(str, size) {
return str.match(new RegExp('.{1,' + size + '}', 'g'));
}
(function() {
function chunk(str, size) {
return str.match(new RegExp('.{1,' + size + '}', 'g'));
}
var str = 'HELLO WORLD';
println('Simple binary representation:');
println(chunk(textToBin(str), 8).join('\n'));
println('\nNow for something crazy:');
println(chunk(textToHex(str, 4), 8).map(function(h) { return '0x' + h }).join(' '));
// Utiliy functions, you can ignore these.
function textToBin(text) { return textToBase(text, 2, 8); }
function textToHex(t, w) { return pad(textToBase(t,16,2), roundUp(t.length, w)*2, '00'); }
function pad(val, len, chr) { return (repeat(chr, len) + val).slice(-len); }
function print(text) { document.getElementById('out').innerHTML += (text || ''); }
function println(text) { print((text || '') + '\n'); }
function repeat(chr, n) { return new Array(n + 1).join(chr); }
function textToBase(text, radix, n) {
return text.split('').reduce(function(result, chr) {
return result + pad(chr.charCodeAt(0).toString(radix), n, '0');
}, '');
}
function roundUp(numToRound, multiple) {
if (multiple === 0) return numToRound;
var remainder = numToRound % multiple;
return remainder === 0 ? numToRound : numToRound + multiple - remainder;
}
}());
#out {
white-space: pre;
font-size: 0.8em;
}
<div id="out"></div>
function chunk(er){
return er.match(/.{1,75}/g).join('\n');
}
Выше функция - это то, что я использую для Chunking Base64. Он создаст прерывание строки, состоящее из 75 символов.
Мое решение (синтаксис ES6):
const source = "8d7f66a9273fc766cd66d1d";
const target = [];
for (
const array = Array.from(source);
array.length;
target.push(array.splice(0,2).join(''), 2));
Мы могли бы даже создать функцию с этим:
function splitStringBySegmentLength(source, segmentLength) {
if (!segmentLength || segmentLength < 1) throw Error('Segment length must be defined and greater than/equal to 1');
const target = [];
for (
const array = Array.from(source);
array.length;
target.push(array.splice(0,segmentLength).join('')));
return target;
}
Затем вы можете вызвать функцию легко в многоразовом режиме:
const source = "8d7f66a9273fc766cd66d1d";
const target = splitStringBySegmentLength(source, 2);
Приветствия
Здесь мы вставляем строку с другой строкой каждые n символов:
export const intersperseString = (n: number, intersperseWith: string, str: string): string => {
let ret = str.slice(0,n), remaining = str;
while (remaining) {
let v = remaining.slice(0, n);
remaining = remaining.slice(v.length);
ret += intersperseWith + v;
}
return ret;
};
если мы будем использовать выше, как так:
console.log(splitString(3,'|', 'aagaegeage'));
мы получаем:
AAG | AAG | AEG | ЕАГ | е
и здесь мы делаем то же самое, но нажимаем на массив:
export const sperseString = (n: number, str: string): Array<string> => {
let ret = [], remaining = str;
while (remaining) {
let v = remaining.slice(0, n);
remaining = remaining.slice(v.length);
ret.push(v);
}
return ret;
};
и затем запустите его:
console.log(sperseString(5, 'foobarbaztruck'));
мы получаем:
['fooba', 'rbazt', 'ruck']
если кто-то знает способ упростить приведенный выше код, lmk, но он должен хорошо работать со строками.
const chunkStr = (str, n, acc) => {
if (str.length === 0) {
return acc
} else {
acc.push(str.substring(0, n));
return chunkStr(str.substring(n), n, acc);
}
}
const str = 'abcdefghijkl';
const splittedString = chunkStr(str, 3, []);
Чистый раствор без REGEX
Некоторое чистое решение без использования регулярных выражений:
/**
* Create array with maximum chunk length = maxPartSize
* It work safe also for shorter strings than part size
**/
function convertStringToArray(str, maxPartSize){
const chunkArr = [];
let leftStr = str;
do {
chunkArr.push(leftStr.substring(0, maxPartSize));
leftStr = leftStr.substring(maxPartSize, leftStr.length);
} while (leftStr.length > 0);
return chunkArr;
};
Пример использования - https://jsfiddle.net/maciejsikora/b6xppj4q/.
Я также попытался сравнить свое решение с регулярным выражением, которое было выбрано в качестве правильного ответа. Некоторое испытание можно найти на jsfiddle - https://jsfiddle.net/maciejsikora/2envahrk/. Тесты показывают, что оба метода имеют схожую производительность, возможно, при первом образе regexp решение немного быстрее, но судите сами.
Здесь быстрая двухстрочная линия, которая запятнает числа:
function commafy(inVal){
var ary = String(inVal).match(/(\d{0,2})((?:\d{3})*)([^\d].*$)/i);
return (ary[1] == "" ? [] : [ary[1]]).splice(1, 0, ary[2].match(/[0-9]{3}/g)).join(",") + ary[3];
}
если он был распространен, он может выглядеть так:
function commafy(inVal){
var aryChunks = [];
var inVal = String(inVal);
var aryPart1 = inVal.match(/(\d{0,2})((?:\d{3})*)([^\d].*$)/i);
if(aryPart1[1] != ""){
aryChunks.push(aryPart1[1]);
}
var aryPart2 = aryPart1[2].match(/[0-9]{3}/g);
aryChunks.splice(1, 0, aryPart2);
var outVal = aryChunks.join(",");
outVal += aryPart1[3];
return outVal;
}
Во-первых, aryPart1 присваивается совпадением регулярных выражений, состоящим из [от 0 до 2 чисел], за которым следует [строка символов, длина которых кратная 3], а затем [ноль или один незнатный и все остальное].
Затем, если это не "", мы добавляем [0 to 2 numbers] к aryChunks.
После этого мы берем [строку символов, длина которых кратно 3], а регулярное выражение сопоставляет их в массив из [3-символьных фрагментов], называемый aryPart2, который мы затем соединяем с aryChunks.
Теперь aryChunks содержит куски, которые мы присоединяем к запятой и присваиваем outVal.
Все, что осталось сделать, это добавить [zero или one non-digit и все остальное] в outVal и вернуть outVal вызывающему.
В OP, {0,2} и {3} в регулярном выражении 1 и {3} в регулярном выражении 2 могут быть переменными, чтобы сделать куски любой длины, а \d в регулярных выражениях можно изменить на "". если вы хотите, чтобы он работал больше, чем просто цифры.
Я слишком долго писал это, так что может быть несколько сбоев. Направьте их, и я их подстрою.
С .split
:
var arr = str.split( /(?<=^(?:.{3})+)(?!$)/ ) // [ 'abc', 'def', 'ghi', 'jkl' ]
и .replace
будет:
var replaced = str.replace( /(?<=^(.{3})+)(?!$)/g, ' || ' ) // 'abc || def || ghi || jkl'
/(?!$)/
- остановить до конца /$/
без:
var arr = str.split( /(?<=^(?:.{3})+)/ ) // [ 'abc', 'def', 'ghi', 'jkl' ] // I don't know why is not [ 'abc', 'def', 'ghi', 'jkl' , '' ], comment?
var replaced = str.replace( /(?<=^(.{3})+)/g, ' || ') // 'abc || def || ghi || jkl || '
игнорирование группы /(?:
... )/
не нужно в .replace
но в .split
добавляет группы в arr:
var arr = str.split( /(?<=^(.{3})+)(?!$)/ ) // [ 'abc', 'abc', 'def', 'abc', 'ghi', 'abc', 'jkl' ]