Es posible que desee ver esta pregunta en la que he presentado una muestra de código que utiliza la recursividad para representar una estructura de directorio en un TreeView. La lógica debería ser la misma en la mayoría de los casos.
Cerebrus
55
El problema con esto es que se rompe muy fácilmente si no tiene acceso a un solo directorio: sin resultados ...
Este artículo cubre todo lo que necesitas. Excepto en lugar de buscar los archivos y comparar nombres, simplemente imprima los nombres.
Se puede modificar así:
staticvoidDirSearch(string sDir){try{foreach(string d inDirectory.GetDirectories(sDir)){foreach(string f inDirectory.GetFiles(d)){Console.WriteLine(f);}DirSearch(d);}}catch(System.Exception excpt){Console.WriteLine(excpt.Message);}}
Añadido por barlop
GONeale menciona que lo anterior no enumera los archivos en el directorio actual y sugiere colocar la parte de la lista de archivos fuera de la parte que obtiene los directorios. Lo siguiente haría eso. También incluye una línea de Writeline que puede descomentar, que ayuda a rastrear dónde se encuentra en la recursión que puede ayudar a mostrar las llamadas para ayudar a mostrar cómo funciona la recursividad.
DirSearch_ex3("c:\\aaa");staticvoidDirSearch_ex3(string sDir){//Console.WriteLine("DirSearch..(" + sDir + ")");try{Console.WriteLine(sDir);foreach(string f inDirectory.GetFiles(sDir)){Console.WriteLine(f);}foreach(string d inDirectory.GetDirectories(sDir)){DirSearch_ex3(d);}}catch(System.Exception excpt){Console.WriteLine(excpt.Message);}}
Este método no enumera los archivos para el directorio inicial, solo son subdirectorios y versiones inferiores. Me gustaría mover GetFiles fuera de GetDirectories
GONeale
1
A veces uno no quiere los archivos para el directorio inicial, en cuyo caso esto es perfecto para estructuras razonablemente pequeñas. Para listas muy grandes, use algo como la solución de Marc Gravell: stackoverflow.com/a/929418/91189
Joseph Gabriel el
2
@GONeale es correcto. Es mucho menos plausible que un usuario no espere la lista de archivos del directorio raíz de entrada. La palabra de entrada es clave aquí. Ha sido ingresado por una razón.
Florin Mircea
2
Tuve que agregar un intento de captura alrededor del bucle foreach interno, de lo contrario no continuará con los errores de acceso denegado
Shaun Vermaak
3
Debería evitar capturar una excepción: ¿realmente le gustaría atrapar una excepción OutOfMemoryException, por ejemplo? Solo atrapa lo que puedas manejar.
alastairtree
435
Tenga en cuenta que en .NET 4.0 hay (supuestamente) funciones de archivo basadas en iterador (en lugar de matriz) integradas:
Por el momento usaría algo como a continuación; el método recursivo incorporado se rompe demasiado fácilmente si no tiene acceso a un solo subdirectorio ...; el Queue<string>uso evita demasiada recursividad de pila de llamadas, y el bloque iterador evita que tengamos una gran matriz.
@soandos En el punto de reparación recursiva EnumerateFiles arroja IOException "El sistema no puede resolver el nombre del archivo"
SerG
55
Para todos los que quieran saber si *.*también incluye archivos sin extensión de archivo: Sí, sí, probado hace un minuto.
Tobias Knauss
1
Para usar esto, necesitará agregarusing System.IO;
Restablecer Monica - Goodbye SE
77
@Wikis y para usar Consolenecesitará agregar using System;, pero dado que el IDE puede agregar todas las usingdirectivas necesarias para usted (ctrl +.), Y dado que no estamos usando nada exótico aquí, es común no incluirlas. Diablos, también necesitarás una classdefinición, etc. Solo digo
Marc Gravell
1
@MarcGravell Estamos en el núcleo .NET y Visual Studio Código mundo ahora, para incluyendo el uso de declaraciones, siempre es bienvenido en cualquier ejemplo de código .NET para guardar una serie de búsquedas y sin sentido "de afeitar de yak"
Cómo evitar el error si el usuario de inicio de sesión no tiene acceso en algunas de las carpetas.
Romil Kumar Jain
55
@Romil No creo que este fragmento de código intente indicar la funcionalidad completa, solo la funcionalidad sin procesar que buscaba el OP. Gracias por compartir, Pescuma!
kayleeFrye_onDeck
@kayleeFrye_onDeck, pongo solo una inquietud en caso de que haya una elevación para alguna de las carpetas mientras obtengo archivos. Debido a esta preocupación, implementamos nuestra función recursiva personalizada.
Romil Kumar Jain
3
Recibirá "UnauthorizedAccessException" con esta solución. Debería tener una solución que pueda manejar errores como este.
Kairan
13
En .NET 4.5, al menos, existe esta versión que es mucho más corta y tiene la ventaja adicional de evaluar cualquier criterio de archivo para su inclusión en la lista:
En Framework 2.0 puede usar (enumera los archivos de la carpeta raíz, es mejor la respuesta más popular):
staticvoidDirSearch(string dir){try{foreach(string f inDirectory.GetFiles(dir))Console.WriteLine(f);foreach(string d inDirectory.GetDirectories(dir)){Console.WriteLine(d);DirSearch(d);}}catch(System.Exception ex){Console.WriteLine(ex.Message);}}
Algunas respuestas excelentes, pero estas respuestas no resolvieron mi problema.
Tan pronto como surge un problema de permiso de carpeta: "Permiso denegado" el código falla. Esto es lo que solía solucionar el problema "Permiso denegado":
/// <summary>/// Scans a folder and all of its subfolders recursively, and updates the List of files/// </summary>/// <param name="sFullPath">Full path of the folder</param>/// <param name="files">The list, where the output is expected</param>internalstaticvoidEnumerateFiles(string sFullPath,List<FileInfo> fileInfoList){try{DirectoryInfo di =newDirectoryInfo(sFullPath);FileInfo[] files = di.GetFiles();foreach(FileInfo file in files)
fileInfoList.Add(file);//Scan recursivelyDirectoryInfo[] dirs = di.GetDirectories();if(dirs ==null|| dirs.Length<1)return;foreach(DirectoryInfo dir in dirs)EnumerateFiles(dir.FullName, fileInfoList);}catch(Exception ex){Logger.Write("Exception in Helper.EnumerateFiles", ex);}}
Está haciendo manualmente lo que DirectoryInfo.GetFiles () hará por usted de forma inmediata: simplemente use la sobrecarga con SearchOption.AllDirectories y se repetirá por sí solo. Entonces esa es una solución complicada .
philw
2
Prefiero usar DirectoryInfo porque puedo obtener FileInfo's, no solo cadenas.
¿Por qué el parámetro fileses ref? No hay necesidad.
Massimiliano Kraus
@MassimilianoKraus Yo diría que, aunque no es obligatorio, deja en claro que su método se alterará filesy que ya no se puede dar new List<FileInfo>()como parámetro, lo que sería inútil. Podría permitir alguna suboptimización y evitar crear un nuevo objeto a menos que sea necesario.
jeromej
@JeromeJ si sabe qué es OOP, sabe que cada vez que pasa un objeto a un método, ese método puede cambiar las propiedades / campos del objeto. Entonces refno aclara nada. El refpropósito es cambiar todo el filespuntero incluso para la persona que llama al método: es una operación peligrosa y aquí no hay necesidad de eso: simplemente puede llenar la Lista, no necesita volver a apuntarla a otra Lista en el montón refdebe usarse solo en casos muy particulares; la mayoría de las veces solo necesitas implementar las cosas de una manera más paradigmática funcional.
Massimiliano Kraus
1
Para evitar UnauthorizedAccessException, yo uso:
var files =GetFiles(@"C:\","*.*",SearchOption.AllDirectories);foreach(var file in files){Console.WriteLine($"{file}");}publicstaticIEnumerable<string>GetFiles(string path,string searchPattern,SearchOption searchOption){var foldersToProcess =newList<string>(){
path
};while(foldersToProcess.Count>0){string folder = foldersToProcess[0];
foldersToProcess.RemoveAt(0);if(searchOption.HasFlag(SearchOption.AllDirectories)){//get subfolderstry{var subfolders =Directory.GetDirectories(folder);
foldersToProcess.AddRange(subfolders);}catch(Exception ex){//log if you're interested}}//get filesvar files =newList<string>();try{
files =Directory.GetFiles(folder, searchPattern,SearchOption.TopDirectoryOnly).ToList();}catch(Exception ex){//log if you're interested}foreach(var file in files){yieldreturn file;}}}
Si solo necesita nombres de archivo y dado que realmente no me gustaron la mayoría de las soluciones aquí (características o legibilidad), ¿qué tal esta perezosa?
privatevoidFoo(){var files =GetAllFiles("pathToADirectory");foreach(string file in files){// Use can use Path.GetFileName() or similar to extract just the filename if needed// You can break early and it won't still browse your whole disk since it's a lazy one}}/// <exception cref="T:System.IO.DirectoryNotFoundException">The specified path is invalid (for example, it is on an unmapped drive).</exception>/// <exception cref="T:System.UnauthorizedAccessException">The caller does not have the required permission.</exception>/// <exception cref="T:System.IO.IOException"><paramref name="path" /> is a file name.-or-A network error has occurred.</exception>/// <exception cref="T:System.IO.PathTooLongException">The specified path, file name, or both exceed the system-defined maximum length. For example, on Windows-based platforms, paths must be less than 248 characters and file names must be less than 260 characters.</exception>/// <exception cref="T:System.ArgumentNullException"><paramref name="path" /> is null.</exception>/// <exception cref="T:System.ArgumentException"><paramref name="path" /> is a zero-length string, contains only white space, or contains one or more invalid characters as defined by <see cref="F:System.IO.Path.InvalidPathChars" />.</exception>[NotNull]publicstaticIEnumerable<string>GetAllFiles([NotNull]string directory){foreach(string file inDirectory.GetFiles(directory)){yieldreturn file;// includes the path}foreach(string subDir inDirectory.GetDirectories(directory)){foreach(string subFile inGetAllFiles(subDir)){yieldreturn subFile;}}}
Aquí está mi ángulo, basado en Hernaldo, si necesita encontrar archivos con nombres de cierto patrón, como archivos XML que en algún lugar de su nombre contienen una cadena particular:
// call this like so: GetXMLFiles("Platypus", "C:\\");publicstaticList<string>GetXMLFiles(string fileType,string dir){string dirName = dir;var fileNames =newList<String>();try{foreach(string f inDirectory.GetFiles(dirName)){if((f.Contains(fileType))&&(f.Contains(".XML"))){
fileNames.Add(f);}}foreach(string d inDirectory.GetDirectories(dirName)){GetXMLFiles(fileType, d);}}catch(Exception ex){MessageBox.Show(ex.Message);}return fileNames;}
Listado de archivos y carpetas para modelar, implementación personalizada.
Esto crea una lista completa de todos los archivos y carpetas a partir de su directorio de inicio.
salida de la función (el contenido de la carpeta5 se excluye debido al límite de lvl y el contenido de la carpeta3 se excluye porque está en el conjunto de carpetas excluidas):
Una solución muy simple, devuelve una lista de archivos.
publicstaticList<string>AllFilesInFolder(string folder){var result =newList<string>();foreach(string f inDirectory.GetFiles(folder)){
result.Add(f);}foreach(string d inDirectory.GetDirectories(folder)){
result.AddRange(AllFilesInFolder(d));}return result;}
staticvoidMain(string[] args){string[] array1 =Directory.GetFiles(@"D:\");string[] array2 =System.IO.Directory.GetDirectories(@"D:\");Console.WriteLine("--- Files: ---");foreach(string name in array1){Console.WriteLine(name);}foreach(string name in array2){Console.WriteLine(name);}Console.ReadLine();}
Respuestas:
Este artículo cubre todo lo que necesitas. Excepto en lugar de buscar los archivos y comparar nombres, simplemente imprima los nombres.
Se puede modificar así:
Añadido por barlop
GONeale menciona que lo anterior no enumera los archivos en el directorio actual y sugiere colocar la parte de la lista de archivos fuera de la parte que obtiene los directorios. Lo siguiente haría eso. También incluye una línea de Writeline que puede descomentar, que ayuda a rastrear dónde se encuentra en la recursión que puede ayudar a mostrar las llamadas para ayudar a mostrar cómo funciona la recursividad.
fuente
Tenga en cuenta que en .NET 4.0 hay (supuestamente) funciones de archivo basadas en iterador (en lugar de matriz) integradas:
Por el momento usaría algo como a continuación; el método recursivo incorporado se rompe demasiado fácilmente si no tiene acceso a un solo subdirectorio ...; el
Queue<string>
uso evita demasiada recursividad de pila de llamadas, y el bloque iterador evita que tengamos una gran matriz.fuente
*.*
también incluye archivos sin extensión de archivo: Sí, sí, probado hace un minuto.using System.IO;
Console
necesitará agregarusing System;
, pero dado que el IDE puede agregar todas lasusing
directivas necesarias para usted (ctrl +.), Y dado que no estamos usando nada exótico aquí, es común no incluirlas. Diablos, también necesitarás unaclass
definición, etc. Solo digofuente
En .NET 4.5, al menos, existe esta versión que es mucho más corta y tiene la ventaja adicional de evaluar cualquier criterio de archivo para su inclusión en la lista:
Usar así:
fuente
fuente
En Framework 2.0 puede usar (enumera los archivos de la carpeta raíz, es mejor la respuesta más popular):
fuente
Algunas respuestas excelentes, pero estas respuestas no resolvieron mi problema.
Tan pronto como surge un problema de permiso de carpeta: "Permiso denegado" el código falla. Esto es lo que solía solucionar el problema "Permiso denegado":
Espero que esto ayude a otros.
fuente
Una solución simple y limpia.
fuente
Prefiero usar DirectoryInfo porque puedo obtener FileInfo's, no solo cadenas.
Hago esto en caso de que en el futuro necesite un filtrado futuro ... basado en las propiedades de FileInfo.
También puedo recurrir a cadenas si es necesario. (y todavía estoy preparado para el futuro para filtros / cosas de cláusula where.
Tenga en cuenta que " . " Es un patrón de búsqueda válido si desea archivar por extensión.
fuente
fuente
files
esref
? No hay necesidad.files
y que ya no se puede darnew List<FileInfo>()
como parámetro, lo que sería inútil. Podría permitir alguna suboptimización y evitar crear un nuevo objeto a menos que sea necesario.ref
no aclara nada. Elref
propósito es cambiar todo elfiles
puntero incluso para la persona que llama al método: es una operación peligrosa y aquí no hay necesidad de eso: simplemente puede llenar la Lista, no necesita volver a apuntarla a otra Lista en el montónref
debe usarse solo en casos muy particulares; la mayoría de las veces solo necesitas implementar las cosas de una manera más paradigmática funcional.Para evitar
UnauthorizedAccessException
, yo uso:fuente
Si solo necesita nombres de archivo y dado que realmente no me gustaron la mayoría de las soluciones aquí (características o legibilidad), ¿qué tal esta perezosa?
fuente
Registro más corto
fuente
Aquí está mi ángulo, basado en Hernaldo, si necesita encontrar archivos con nombres de cierto patrón, como archivos XML que en algún lugar de su nombre contienen una cadena particular:
fuente
Listado de archivos y carpetas para modelar, implementación personalizada.
Esto crea una lista completa de todos los archivos y carpetas a partir de su directorio de inicio.
Método:
Uso:
fuente
Solución corta y simple
fuente
Este me ayudó a obtener todos los archivos en un directorio y subdirectorios. Puede ser útil para alguien. [Inspirado en las respuestas anteriores]
fuente
fuente
Algunas versiones mejoradas con max lvl para bajar en el directorio y la opción de excluir carpetas:
directorio de entrada:
salida de la función (el contenido de la carpeta5 se excluye debido al límite de lvl y el contenido de la carpeta3 se excluye porque está en el conjunto de carpetas excluidas):
fuente
Aquí hay una versión del código de B. Clay Shannon no estático para archivos de Excel:
fuente
Una solución muy simple, devuelve una lista de archivos.
fuente
fuente