Obtén contenido entre dos cadenas PHP
-
22-07-2019 - |
Pregunta
¿Cuál es la mejor manera de obtener el contenido entre dos cadenas, por ejemplo,
ob_start();
include('externalfile.html'); ## see below
$out = ob_get_contents();
ob_end_clean();
preg_match('/{FINDME}(.|\n*)+{\/FINDME}/',$out,$matches);
$match = $matches[0];
echo $match;
## I have used .|\n* as it needs to check for new lines. Is this correct?
## externalfile.html
{FINDME}
Text Here
{/FINDME}
Por alguna razón, esto parece funcionar en un lugar de mi código y no en otro. ¿Voy sobre esto de la manera correcta? ¿O hay una mejor manera?
¿También el buffer de salida es la forma de hacer esto o file_get_contents?
¡Gracias de antemano!
Solución
- Use
#
en lugar de/
para no tener que escapar de ellos. - El modificador
s
hace que.
y\ s
también incluyan nuevas líneas. -
{
y}
tiene varias funciones, como de n a m veces en{n, m}
. -
Lo básico
preg_match('#\\{FINDME\\}(.+)\\{/FINDME\\}#s',$out,$matches);
-
El avanzado para varias etiquetas, etc. (el estilo no es tan agradable con JavaScript).
$delimiter = '#'; $startTag = '{FINDME}'; $endTag = '{/FINDME}'; $regex = $delimiter . preg_quote($startTag, $delimiter) . '(.*?)' . preg_quote($endTag, $delimiter) . $delimiter . 's'; preg_match($regex,$out,$matches);
Pon este código en una función
- Para cualquier archivo que no desee ejecutar ningún código php perdido , debe usar file_get_contents. incluir / requerir ni siquiera debería ser una opción allí.
Otros consejos
También puede usar substr y strpos para esto.
$startsAt = strpos($out, "{FINDME}") + strlen("{FINDME}");
$endsAt = strpos($out, "{/FINDME}", $startsAt);
$result = substr($out, $startsAt, $endsAt - $startsAt);
Deberá agregar una comprobación de errores para manejar el caso en el que no ENCUENTRA.
Me encantan estas dos soluciones
function GetBetween($content,$start,$end)
{
$r = explode($start, $content);
if (isset($r[1])){
$r = explode($end, $r[1]);
return $r[0];
}
return '';
}
function get_string_between($string, $start, $end){
$string = " ".$string;
$ini = strpos($string,$start);
if ($ini == 0) return "";
$ini += strlen($start);
$len = strpos($string,$end,$ini) - $ini;
return substr($string,$ini,$len);
}
También hice algunos puntos de referencia con ambas soluciones anteriores y ambas están dando casi el mismo tiempo. Puedes probarlo también. Le di a ambas funciones un archivo para leer que tenía aproximadamente 60000 caracteres (revisado con el recuento de palabras de la Sra. Word) y ambas funciones resultaron en aproximadamente 0.000999 segundos para encontrar.
$startTime = microtime(true);
GetBetween($str, '<start>', '<end>');
echo "Explodin Function took: ".(microtime(true) - $startTime) . " to finish<br />";
$startTime = microtime(true);
get_string_between($str, '<start>', '<end>');
echo "Subsring Function took: ".(microtime(true) - $startTime) . " to finish<br />";
Los saltos de línea pueden causar problemas en RegEx, intente eliminarlos o reemplazarlos con \ n antes de procesarlos.
Me gusta evitar el uso de expresiones regulares si es posible, aquí hay una solución alternativa para buscar todas las cadenas entre dos cadenas y devuelve una matriz.
function getBetween($content, $start, $end) {
$n = explode($start, $content);
$result = Array();
foreach ($n as $val) {
$pos = strpos($val, $end);
if ($pos !== false) {
$result[] = substr($val, 0, $pos);
}
}
return $result;
}
print_r(getBetween("The quick brown {{fox}} jumps over the lazy {{dog}}", "{{", "}}"));
Resultados:
Array
(
[0] => fox
[1] => dog
)
function getInbetweenStrings($start, $end, $str){
$matches = array();
$regex = "/$start([a-zA-Z0-9_]*)$end/";
preg_match_all($regex, $str, $matches);
return $matches[1];
}
$str = "C://@@ad_custom_attr1@@/@@upn@@/@@samaccountname@@";
$str_arr = getInbetweenStrings('@@', '@@', $str);
print_r($str_arr);
Esta es una solución PHP que devuelve las cadenas encontradas entre etiquetas en un pajar. Funciona, pero no he probado la eficiencia. Necesitaba esto y me inspiró la respuesta de Adam Wright en esta página.
Devuelve una matriz () que contiene todas las cadenas encontradas entre $ tag y $ end_symbold. $ tag en $ haystack, o FALSE si no se encontró $ end_symbol. $ tag, por lo tanto, no existe ningún par de etiquetas en el $ haystack.
function str_between_tags($haystack, $tag, $end_symbol){
$c_end_tags = substr_count($haystack, $end_symbol.$tag);
if(!$c_end_tags) return FALSE;
for($i=0; $i<$c_end_tags; $i++){
$p_s = strpos($haystack, $tag, (($p_e)?$p_e+strlen($end_symbol.$tag):NULL) ) + strlen($tag );
$p_e = strpos($haystack, $end_symbol.$tag, $p_s);
$result[] = substr($haystack, $p_s, $p_e - $p_s);
}
return $result;
}
Forma rápida de poner todo en una cadena.
$newlines = array("\t","\n","\r","\x20\x20","\0","\x0B");
$one_string = str_replace($newlines, "", html_entity_decode($content));