Задать вопрос
  • Чем заменить функция file_put_contents?

    Попробуйте глянуть в эту сторону Работаем с файлами INI в PHP , возможно это то что Вам нужно
    Ответ написан
    1 комментарий
  • Чем заменить функция file_put_contents?

    KorsaR-ZN
    @KorsaR-ZN
    Я уже втрой раз вижу этот код тут, и просто не могу задать вопроса, вы когда-нибудь слышали про отрицание в логических условиях?

    Т.е ваш код можно сократить и улучшить читаемость...
    $tank = "bd/".$text.".txt";
    if (!file_exists($tank)) {
        file_put_contents($tank, $text = "<span class='$random'>$text</span><br>");
    }
    Ответ написан
    5 комментариев
  • Почему не срабатывает флаг [L] в htaccess?

    madmages
    @madmages Автор вопроса
    Человек прямоходящий
    в следующий раз буду сначала гуглить;)
    stackoverflow.com/questions/6797998/rewriterule-la...
    Ответ написан
    Комментировать
  • Как указать максимальный размер одного слова из текста в textarea?

    @PaulJE
    Дело в том, что нативно HTML5 не потдерживает проверку на длину каждого слова, только если всего текста внутри текстареа, следовательно есть 2 пути решения: проверка на клиенте и на бакэнде.
    1й способ прост в исполнении, на сабмит формы вешать обработчик на javascript. Проблема только 1: пользователь может легко обойти такую проверку
    2й Способ более эффективен но более сложен в плане проработки нюансов, т.к. нужно вернуть ту же страницу с указанием ошибки, при этом не терять текста внутри текстареа.
    Лучше всего комбинировать такие веши: простая проверка на JS + проверка на бакенде
    Ответ написан
    5 комментариев
  • Как разобрать текст на php без повторений?

    akubintsev
    @akubintsev
    Опытный backend разработчик
    $string = 'кораблик плавал плавал и утонул';
    $wordArray = explode(' ', $string);
    $wordsCount = count($wordArray);
    $result = $dupArray = [];
    
    
    for ($i = 0; $i <= $wordsCount; $i++) {
        for ($j = 0; $j < $i; $j++) {
            $pos = $wordsCount - $i + 1; // length для slice в php относительно offset, а не начала массива
            $stroke = implode(' ', array_slice($wordArray, $j, $pos));
    
            if (!isset($dupArray[$stroke])) {
                $dupArray[$stroke] = array_push($result, $stroke);
            }
        }
    }
    
    echo implode("\n", $result);
    Ответ написан
    4 комментария
  • Какие есть сайты наподобие toster?

    butteff
    @butteff
    Раз в тысячу лет заправляю свитер в носки
    hashcode.ru askdev.ru
    Ответ написан
    Комментировать
  • Как разделить текст без повторений?

    Petroveg
    @Petroveg
    Миром правят маленькие с#@&ки
    var start = ('кораблик плавал плавал и утонул').split(' ');
    
    for (var i = 0, result = [], _result = {}, stroke; i <= start.length; i++) {
    	for (var j = 0; j < i; j++) {
    		stroke = start.slice(j, j + start.length - i + 1).join(' ');
    
    		if (!_result[stroke]) {
    			_result[stroke] = result.push(stroke);
    		}
    	}
    }
    
    console.log(result);


    jsfiddle.net/petroveg/agwtd8vz
    Ответ написан
  • Как убрать в url .php через .htaccess?

    htaccess не убирает .php, а помогает найти нужный url, из которого .php убран средствами вашего движка.
    Ответ написан
  • (Unicode) Как сжать код?

    Scorpi
    @Scorpi
    На PHP не пишу, сделал всё что мог :)
    function mb_str_split($string) {
    	$l = mb_strlen($string, "utf-8");
    	$arr = array();
    	for ($i = 0; $i < $l; $i++)
    		array_splice($arr, $i, 1, mb_substr($string,$i,1,"utf-8"));
    	return $arr;
    }
    
    function incode($string) {
    	$string = array_map(function($e) {
    		$result = unpack("N", mb_convert_encoding($e, "UCS-4BE", "UTF-8"));
    		if (is_array($result) === true) return $result[1];
    	}, mb_str_split($string));
    	return implode("", $string);  // $string функция
    }
    Ответ написан
    8 комментариев
  • Как собрать многоуровневый массив?

    @eandr_67
    web-программист (*AMP, Go, JavaScript, вёрстка).
    Зачем такие сложности? Всё делается намного проще - без рекурсий и проверок множества дополнительных условий:

    $input = 'Шла Саша по шоссе и сосала сушку';
    
    $inp_arr = preg_split('#[[:space:]]+#s', trim($input)); // Список обрабатываемых слов
    $current = array(); // Список уже обработанных слов
    $result  = array(implode(' ', $inp_arr)); // Добавляем в результат исходную фразу
    while (count($inp_arr) > 1) { // Пока в списке обрабатываемых не менее 2 слов
      $head = array(); // Голова списка обрабатываемых слов
      $tail = $inp_arr; // Хвост списка обрабатываемых слов
      while (count($tail) > 1) { // Пока в хвосте не менее 2 слов
        $head[]   = array_shift($tail); // Перенос первого слова хвоста в конец головы
        // Добавляем комбинацию в результат
        $result[] = array_merge($current, array(implode(' ', $head), implode(' ', $tail)));
      }
      $current[] = array_shift($inp_arr); // Перенос первого слова обрабатываемых в уже обработанные
    }
    
    var_export($result);


    Вариант, который перебирает все комбинации, всё же удобнее сделать рекурсивным, но алгоритмически он проще, т.к. не надо заботиться о том, чтобы в $current было по одному слову:

    $input="Шла Саша по шоссе и сосала сушку";
    $result=generate(preg_split('#[[:space:]]+#s', trim($input)));
    var_export($result);
     
    function generate($inp){
      $out=array(implode(' ', $inp));
      $head=array();
      while(count($inp)>1){
        $head[]=array_shift($inp);
        foreach(generate($inp) as $val){
          $out[]=array_merge(array(implode(' ', $head)), (array)$val);
        }
      }
      return $out;
    }

    Если нужно сортировать по длинам строк, то достаточно изменить порядок обработки: предыдущий вариант начинает перебор с пустой головы и длинного хвоста, а надо с длиной головы и пустого хвоста:

    function generate($inp){
      $out=array(implode(' ', $inp));
      $tail=array();
      while(count($inp)>1){
        array_unshift($tail, array_pop($inp));
        foreach(generate($tail) as $val){
          $out[]=array_merge(array(implode(' ', $inp)), (array)$val);
        }
      }
      return $out;
    }

    Если же нужно отсортировать результат по длинам строк независимо от того, в какой позиции эта строка находится, то проще всего - как и сказал @Fesor - использовать отдельную сортировку результата. Например, вот так:

    usort($result, function($val_1, $val_2){
      // Получение кол-ва слов в строках сравниваемых вариантов
      $len_1=array_map(function($data){ return count(preg_split('#[[:space:]]+#s', $data)); }, (array)$val_1);
      $len_2=array_map(function($data){ return count(preg_split('#[[:space:]]+#s', $data)); }, (array)$val_2);
    
      // Массивы кол-ва слов, отсортированные по убыванию
      $sort_len_1=$len_1;
      $sort_len_2=$len_2;
      rsort($sort_len_1);
      rsort($sort_len_2);
    
      // Проверяем, что два варианта имеют разное кол-во слов в строках
      for($i=0; $i<count($sort_len_1); $i++){
        if($sort_len_1[$i]!=$sort_len_2[$i]){
          return $sort_len_2[$i]-$sort_len_1[$i];
        }
      }
    
      // К данному моменту варианты по длинам строк уже отсортированы
      // Этот цикл сортирует варианты с одинаковыми длинами по местоположению длинных строк
      // Если такая сортировка не требуется - удалить цикл
      for($i=0; $i<count($len_1); $i++){
        if($len_1[$i]!=$len_2[$i]){
          return $len_2[$i]-$len_1[$i];
        }
      }
    
      return 0;
    });
    Ответ написан
  • Как собрать многоуровневый массив?

    @HaveFun
    $text = 'Шла саша по шоссе и сосала сушку';
    
    $arWords = explode(' ',$text);
    function implodeAndNest($arWords,$firstElement = null){
      $arResult = array();
      foreach($arWords as $wordNum => $word){
        if($wordNum == 0){
          $result = array(implode(' ',array_slice($arWords,$wordNum)));
        }else{
          $result = array(implode(' ',array_slice($arWords,0,$wordNum)),implode(' ',array_slice($arWords,$wordNum)));
        }
        if(isset($firstElement))
          $result = array_merge(array($firstElement),$result);
        $arResult[] = $result;
      }
      return $arResult;
    }
    
    $arResult = array();
    foreach($arWords as $wordNum => $word){
      if($wordNum == 0){
        $arResult = implodeAndNest(array_slice($arWords,$wordNum));
      }else{
        $arResult = array_merge($arResult,array_values(implodeAndNest(array_slice($arWords,$wordNum),implode(' ',array_slice($arWords,0,$wordNum)))));
      }
    }
    
    // удаляем лишние результаты
    $arResult = array_values(array_map("unserialize", array_unique(array_map("serialize", $arResult))));
    
    echo '<pre>';
    print_r($arResult);
    echo '</pre>';
    Ответ написан
    5 комментариев
  • Как собрать многоуровневый массив?

    Fesor
    @Fesor
    Full-stack developer (Symfony, Angular)
    Я слегка к вечеру начал туго соображать, но давайте попробуем. Возможно я допущу какое грубое невежество или моя версия алгоритма не слишком интересная... но мне откровенно лень много думать. Так что задачу я решу почти в лоб. Для начала определимся что мы должны сделать...

    $words = explode(' ', 'a b c d'); // сразу представим строку как массив для более удобной работы с оным
    $expectedResult = [
        // 1st level
        [
            'a b c d e',
        ],
        // 2-ой уровень
        [
            'a',
            'b c d e'
        ],
        [
            'a b',
            'c d',
        ],
        [
            'a b c',
            'd'
        ],
        // 3-ий уровень
        [
            'a',
            'b',
            'c d',
        ],
        [
            'a',
            'b c',
            'd'
        ],
        [
            'a b',
            'c',
            'd'
        ],
        // 4-ый уровень
        [
            'a',
            'b',
            'c',
            'd'
        ]
    ];


    каждый уровень представляет из себя комбинации предыдущего:
    то есть для строки "a b c d" первый элемент третьего уровня:
    [
        'a',
        'b',
        'c d'
    ]

    можно составить как конкатенацию первого элементов второго уровня строки 'b c d' с добавлением оторванной части
    ['a'] + ['b', 'c d']
    Следовательно можно упростить алгоритм введя рекурсию постоянно уменьшая сложность. Разделяй и влавствуй как говориться. Поскольку нас теперь заботят только элементы первого и второго уровня алгорим существенно упрощается (можно хоть 666 уровней делать, реализация от этого сложнее не становится. Единственный минус - на большой вложенности можно упереться в лимит по стэку - решается заменой стэка на очередь вызовов но это не здесь и не сейчас).

    Единственное о чем нам стоит подумать - о логике склеивания кусков и условия выхода из рекурсии.

    Допустим у нас 4 слова в строке. Нам нужно собрать все варианты элементов третьего уровня. Для этого нам надо:
    1) ['a'] + ['b', 'c d'] - есть наборы для строк "a" и "b c d"
    2) ['a'] + ['b с', 'd'] - есть наборы для строк "a" и "b c d"
    3) ['a b'] + ['c', 'd'] - есть наборы для строк "a b" и "c d"
    На третьем стэпе мы достигли ситуации, при которой у нас вторая часть строки содержит количество слов равной требуемому уровню (два слова для второго уровня). Собирать комбинации больше неизчего. Сделаем это нашим уловием выхода из цикла.

    Давайте теперь попробуем соорудить функцию, которая будет корректно отрабатывать первый и второй уровни

    function buildLevel(array $words, $level) {
        // с первым уровнем все просто
        if ($level === 1) return [implode(' ', $words)];
        if ($level === 2) {
            // со вторым чуть по сложнее...
            $result = [];
            $chunk = [];
            while(count($words) >= $level) {
                // отделяем первое слово из "строки" и зановим его к первому "слагаемому"
                array_push($chunk, array_shift($words));
                $result[] = array_merge([implode(' ', $chunk)], [implode(' ', $words)]);
            }
    
            return $result;
        }
    
        throw new \Exception(sprintf('Not implemented for level %d for now', $level));
    }


    Проверям работу: ideone.com/ggFzJd
    array(1) {
      [0]=>
      string(7) "a b c d"
    }
    array(3) {
      [0]=>
      array(2) {
        [0]=>
        string(1) "a"
        [1]=>
        string(5) "b c d"
      }
      [1]=>
      array(2) {
        [0]=>
        string(3) "a b"
        [1]=>
        string(3) "c d"
      }
      [2]=>
      array(2) {
        [0]=>
        string(5) "a b c"
        [1]=>
        string(1) "d"
      }
    }

    все ок, только результаты надо склеить.... Но давайте вспомним одно из основных правил программирования! DRY - do not repeat your self. Что мы видим? Одинаковый код! В обработке второго уровня у нас используется код, который мы уже использовали в обработке первого уровня! Причем дважды! Кошмар! Рефакторим...

    $result[] = array_merge(buildLevel($chunk, 1), buildLevel($words, 1));


    Теперь вспомним наш замысел... Мы хотели огранизовать рекурсию, и она родимая у нас и выходит. Функция вызывает самою себя понижая уровень. Сложность решения уменьшается, все хорошо. По такому же принципу добавляем обработку третьего уровня. Попробуйте сначала сами и самостоятельно попробуйте побороть проблему "склейки" результатов третьего и второго уровня. Зачем просто унифицируйте решения. Для того что бы поставить логическую точку, я все же предоставлю готовое решение, но постарайтесь все же сами сделать сначала:

    ideone.com/fwS4u6 - к сожалению времени не так много... может позже доделаю склейку. Пока так. Суть надеюсь будет ясна. Если кто допилит - буду рад.
    Ответ написан