Menú

Mostrar Mensajes

Esta sección te permite ver todos los mensajes escritos por este usuario. Ten en cuenta que sólo puedes ver los mensajes escritos en zonas a las que tienes acceso en este momento.

Mostrar Mensajes Menú

Mensajes - Eleкtro

#1741
Cita de: **Aincrad** en 21 Septiembre 2017, 20:39 PMUmm. no había pensado hacer eso. en verdad podría ser mas optimo , ya que con la solución que encontré tarda mucho.

La metodología propuesta por el compañero NEBIRE de obtener una colección con todos los tipos de archivos e ir filtrando según la extensión, tiene sus ventajas pero también sus desventajas, resultando así en una solución óptima en muchos casos, y en otros no. Todo depende del escenario / circunstancias donde vayas a aplicar esa (o cualquier otra) solución.




Cita de: **Aincrad** en 21 Septiembre 2017, 20:39 PMla solución que encontré tarda mucho.

Si el rendimiento (en general, como la velocidad de ejecución) es un factor de importancia para ti, entonces deberías utilizar un tipo de evaluación vaga (o Lazy Evaluation por su nombre en Inglés) como sería la interfáz IEnumerable, y hacer uso de la programación asincrónica o del paralelismo.

Voy a compartir contigo un pequeño extracto del código fuente de mi framework comercial ElektroKit (el cual se puede encontrar en mi firma de usuario del foro). Este código es una implementación personal (bueno, es un wrapper mejorado de la función Directory.GetFiles() ) de un buscador de archivos y directorios, aunque solo he dejado visible dos métodos en el código fuente para que puedas buscar archivos.

Aparte de los grandes beneficios en velocidad que podrás comprobar por ti miso en los tests aquí abajo del todo de este comentario, este buscador también soporta la característica de buscar por múltiples patrones de nombre y por múltiples patrones de extensiones al mismo tiempo, así como la capacidad de poder excluir sin miedo algunos cualquier directorio en el que el usuario no disponga de permisos de lectura (con la posibilidad adicional de lanzar una excepción de acceso no autorizado si así se desea hacer), cosa que con las funciones built-in de .NET Framework no se puede controlar de forma natural.

Simplemente copiar y pegar. Aquí tienes:

Código (vbnet) [Seleccionar]
' ***********************************************************************
' Author   : Elektro
' Modified : 14-November-2015
' ***********************************************************************

#Region " Option Statements "

Option Strict On
Option Explicit On
Option Infer Off

#End Region

#Region " Imports "

Imports System.Collections.Concurrent
Imports System.IO
Imports System.Linq
Imports System.Security
Imports System.Threading.Tasks

#End Region

#Region " ElektroKit.Core.IO.Tools.Files "

Namespace ElektroKit.Core.IO.Tools

   ''' ----------------------------------------------------------------------------------------------------
   ''' <summary>
   ''' Contains file related utilities.
   ''' </summary>
   ''' ----------------------------------------------------------------------------------------------------
   Public NotInheritable Class Files

#Region " Constructors "

       ''' ----------------------------------------------------------------------------------------------------
       ''' <summary>
       ''' Prevents a default instance of the <see cref="Files"/> class from being created.
       ''' </summary>
       ''' ----------------------------------------------------------------------------------------------------
       <DebuggerNonUserCode>
       Private Sub New()
       End Sub

#End Region

#Region " Public Methods "

       ''' ----------------------------------------------------------------------------------------------------
       ''' <summary>
       ''' Gets the files those matches the criteria inside the specified directory and/or sub-directories.
       ''' </summary>
       ''' ----------------------------------------------------------------------------------------------------
       ''' <param name="dirPath">
       ''' The root directory path to search for files.
       ''' </param>
       '''
       ''' <param name="searchOption">
       ''' The searching mode.
       ''' </param>
       '''
       ''' <param name="fileNamePatterns">
       ''' The file name pattern(s) to match.
       ''' </param>
       '''
       ''' <param name="fileExtPatterns">
       ''' The file extension pattern(s) to match.
       ''' </param>
       '''
       ''' <param name="ignoreCase">
       ''' If <see langword="True"/>, ignores the comparing case of <paramref name="fileNamePatterns"/> and <paramref name="fileExtPatterns"/> patterns.
       ''' </param>
       '''
       ''' <param name="throwOnError">
       ''' If set to <see langword="True"/>, exceptions will be thrown, like access denied to file or directory.
       ''' </param>
       ''' ----------------------------------------------------------------------------------------------------
       ''' <returns>
       ''' An <see cref="IEnumerable(Of FileInfo)"/> instance containing the files information.
       ''' </returns>
       ''' ----------------------------------------------------------------------------------------------------
       ''' <exception cref="ArgumentException">dirPath or searchOption
       ''' </exception>
       ''' ----------------------------------------------------------------------------------------------------
       <DebuggerStepThrough>
       Friend Shared Function GetFiles(ByVal dirPath As String,
                                       ByVal searchOption As SearchOption,
                                       Optional ByVal fileNamePatterns As String() = Nothing,
                                       Optional ByVal fileExtPatterns As String() = Nothing,
                                       Optional ByVal ignoreCase As Boolean = True,
                                       Optional ByVal throwOnError As Boolean = False) As IEnumerable(Of FileInfo)

           ' Analyze and resolve path problems. (eg. 'C:' -> 'C:\')
           Files.AnalyzePath(dirPath)

           ' Analyze the passed arguments.
           Files.AnalyzeArgs(dirPath, searchOption)

           ' Get and return the files.
           Dim queue As New ConcurrentQueue(Of FileInfo)
           Files.CollectFiles(queue, dirPath, searchOption, fileNamePatterns, fileExtPatterns, ignoreCase, throwOnError)
           Return queue.AsEnumerable

       End Function

       ''' ----------------------------------------------------------------------------------------------------
       ''' <summary>
       ''' Gets the filepaths those matches the criteria inside the specified directory and/or sub-directories.
       ''' </summary>
       ''' ----------------------------------------------------------------------------------------------------
       ''' <param name="dirPath">
       ''' The root directory path to search for files.
       ''' </param>
       '''
       ''' <param name="searchOption">
       ''' The searching mode.
       ''' </param>
       '''
       ''' <param name="fileNamePatterns">
       ''' The file name pattern(s) to match.
       ''' </param>
       '''
       ''' <param name="fileExtPatterns">
       ''' The file extension pattern(s) to match.
       ''' </param>
       '''
       ''' <param name="ignoreCase">
       ''' If <see langword="True"/>, ignores the comparing case of <paramref name="fileNamePatterns"/> and <paramref name="fileExtPatterns"/> patterns.
       ''' </param>
       '''
       ''' <param name="throwOnError">
       ''' If set to <see langword="True"/>, exceptions will be thrown, like access denied to file or directory.
       ''' </param>
       ''' ----------------------------------------------------------------------------------------------------
       ''' <returns>
       ''' An <see cref="IEnumerable(Of String)"/> instance containing the filepaths.
       ''' </returns>
       ''' ----------------------------------------------------------------------------------------------------
       ''' <exception cref="ArgumentException">dirPath or searchOption
       ''' </exception>
       ''' ----------------------------------------------------------------------------------------------------
       <DebuggerStepThrough>
       Friend Shared Function GetFilePaths(ByVal dirPath As String,
                                           ByVal searchOption As SearchOption,
                                           Optional ByVal fileNamePatterns As String() = Nothing,
                                           Optional ByVal fileExtPatterns As String() = Nothing,
                                           Optional ByVal ignoreCase As Boolean = True,
                                           Optional ByVal throwOnError As Boolean = False) As IEnumerable(Of String)

           ' Analyze and resolve path problems. (eg. 'C:' -> 'C:\')
           Files.AnalyzePath(dirPath)

           ' Analyze the passed arguments.
           Files.AnalyzeArgs(dirPath, searchOption)

           ' Get and return the filepaths.
           Dim queue As New ConcurrentQueue(Of String)
           Files.CollectFilePaths(queue, dirPath, searchOption, fileNamePatterns, fileExtPatterns, ignoreCase, throwOnError)
           Return queue.AsEnumerable

       End Function

#End Region

#Region " Private Methods "

       ''' ----------------------------------------------------------------------------------------------------
       ''' <summary>
       ''' Analyzes a directory path and perform specific changes on it.
       ''' </summary>
       ''' ----------------------------------------------------------------------------------------------------
       ''' <param name="refDirPath">
       ''' The directory path.
       ''' </param>
       ''' ----------------------------------------------------------------------------------------------------
       ''' <exception cref="ArgumentNullException">dirPath;Value is null, empty, or white-spaced.
       ''' </exception>
       ''' ----------------------------------------------------------------------------------------------------
       <DebuggerStepThrough>
       Private Shared Sub AnalyzePath(ByRef refDirPath As String)

           If String.IsNullOrEmpty(refDirPath) OrElse String.IsNullOrWhiteSpace(refDirPath) Then
               Throw New ArgumentNullException("dirPath", "Value is null, empty, or white-spaced.")

           Else
               ' Trim unwanted characters.
               refDirPath = refDirPath.TrimStart({" "c}).TrimEnd({" "c})

               If Path.IsPathRooted(refDirPath) Then
                   ' The root paths contained on the returned FileInfo objects will start with the same string-case as this root path.
                   ' So just for a little visual improvement, I'll treat this root path as a Drive-Letter and I convert it to UpperCase.
                   refDirPath = Char.ToUpper(refDirPath.First()) & refDirPath.Substring(1)
               End If

               If Not refDirPath.EndsWith("\"c) Then
                   ' Possibly its a drive letter without backslash ('C:') or else just a normal path without backslash ('C\Dir').
                   ' In any case, fix the ending backslash.
                   refDirPath = refDirPath.Insert(refDirPath.Length, "\"c)
               End If

           End If

       End Sub

       ''' ----------------------------------------------------------------------------------------------------
       ''' <summary>
       ''' Analyzes the specified directory values.
       ''' </summary>
       ''' ----------------------------------------------------------------------------------------------------
       ''' <param name="dirPath">
       ''' The root directory path to search for files.
       ''' </param>
       '''
       ''' <param name="searchOption">
       ''' The searching mode.
       ''' </param>
       ''' ----------------------------------------------------------------------------------------------------
       ''' <exception cref="ArgumentException">dirPath or searchOption
       ''' </exception>
       ''' ----------------------------------------------------------------------------------------------------
       <DebuggerStepThrough>
       Private Shared Sub AnalyzeArgs(ByVal dirPath As String, ByVal searchOption As SearchOption)

           If Not Directory.Exists(dirPath) Then
               Throw New ArgumentException(String.Format("Directory doesn't exists: '{0}'", dirPath), "dirPath")

           ElseIf (searchOption <> SearchOption.TopDirectoryOnly) AndAlso (searchOption <> SearchOption.AllDirectories) Then
               Throw New ArgumentException(String.Format("Value of '{0}' is not valid enumeration value.", CStr(searchOption)), "searchOption")

           End If

       End Sub

       ''' ----------------------------------------------------------------------------------------------------
       ''' <summary>
       ''' Tries to instance the by-reference <see cref="DirectoryInfo"/> object using the given directory path.
       ''' </summary>
       ''' ----------------------------------------------------------------------------------------------------
       ''' <param name="dirPath">
       ''' The directory path used to instance the by-reference <see cref="DirectoryInfo"/> object.
       ''' </param>
       '''
       ''' <param name="refDirInfo">
       ''' The by-reference <see cref="DirectoryInfo"/> object to instance it using the given directory path.
       ''' </param>
       '''
       ''' <param name="throwOnError">
       ''' If set to <see langword="True"/>, exceptions will be thrown, like access denied to directory.
       ''' </param>
       ''' ----------------------------------------------------------------------------------------------------
       <DebuggerStepThrough>
       Private Shared Sub SetupDirInfoObject(ByVal dirPath As String,
                                             ByRef refDirInfo As DirectoryInfo,
                                             ByVal throwOnError As Boolean)

           Try
               refDirInfo = New DirectoryInfo(dirPath)

           Catch ex As Exception

               Select Case ex.GetType ' Handle or suppress exceptions by its type,

                   ' I've wrote different types just to feel free to extend this feature in the future.
                   Case GetType(ArgumentNullException),
                        GetType(ArgumentException),
                        GetType(SecurityException),
                        GetType(PathTooLongException),
                        ex.GetType

                       If throwOnError Then
                           Throw
                       End If

               End Select

           End Try

       End Sub

       ''' ----------------------------------------------------------------------------------------------------
       ''' <summary>
       ''' Tries to instance the by-reference <paramref name="refCol"/> object using the given directory path.
       ''' </summary>
       ''' ----------------------------------------------------------------------------------------------------
       ''' <typeparam name="A">
       ''' The type of the <paramref name="refCol"/> object used to cast and fill the by-reference collection.
       ''' </typeparam>
       '''
       ''' <param name="objectAction">
       ''' The method to invoke, only for <see cref="FileInfo"/> or <see cref="DirectoryInfo"/> objects, this parameter can be <see langword="Nothing"/>.
       ''' </param>
       '''
       ''' <param name="sharedAction">
       ''' The method to invoke, only for filepaths or directorypaths, this parameter can be <see langword="Nothing"/>.
       ''' </param>
       '''
       ''' <param name="dirPath">
       ''' The directory path used to instance the by-reference <paramref name="refCol"/> object.
       ''' </param>
       '''
       ''' <param name="searchPattern">
       ''' The search pattern to list files or directories.
       ''' </param>
       '''
       ''' <param name="refCol">
       ''' The by-reference <see cref="IEnumerable(Of A)"/> object to instance it using the given directory path.
       ''' </param>
       '''
       ''' <param name="throwOnError">
       ''' If set to <see langword="True"/>, exceptions will be thrown, like access denied to file or directory.
       ''' </param>
       ''' ----------------------------------------------------------------------------------------------------
       <DebuggerStepThrough>
       Private Shared Sub SetupFileDirCollection(Of A)(ByVal objectAction As Func(Of String, SearchOption,
                                                                                             IEnumerable(Of A)),
                                                   ByVal sharedAction As Func(Of String, String,
                                                                                         SearchOption,
                                                                                         IEnumerable(Of A)),
                                                   ByVal dirPath As String,
                                                   ByVal searchPattern As String,
                                                   ByRef refCol As IEnumerable(Of A),
                                                   ByVal throwOnError As Boolean)

           Try
               If objectAction IsNot Nothing Then
                   refCol = objectAction.Invoke(searchPattern, SearchOption.TopDirectoryOnly)

               ElseIf sharedAction IsNot Nothing Then
                   refCol = sharedAction.Invoke(dirPath, searchPattern, SearchOption.TopDirectoryOnly)

               Else
                   Throw New ArgumentException("Any Action has been defined.")

               End If

           Catch ex As Exception

               Select Case ex.GetType ' Handle or suppress exceptions by its type,

                   ' I've wrote different types just to feel free to extend this feature in the future.
                   Case GetType(UnauthorizedAccessException),
                        GetType(DirectoryNotFoundException),
                        ex.GetType

                       If throwOnError Then
                           Throw
                       End If

               End Select

           End Try

       End Sub

       ''' ----------------------------------------------------------------------------------------------------
       ''' <summary>
       ''' Determines whether at least one of the specified patterns matches the given value.
       ''' </summary>
       ''' ----------------------------------------------------------------------------------------------------
       ''' <param name="value">
       ''' The value, which can be a filename, file extension, direcrory path, or directory name.
       ''' </param>
       '''
       ''' <param name="patterns">
       ''' The patterns to match the given value.
       ''' </param>
       '''
       ''' <param name="ignoreCase">
       ''' If set to <see langword="True"/>, compares ignoring string-case rules.
       ''' </param>
       ''' ----------------------------------------------------------------------------------------------------
       ''' <returns>
       ''' <see langword="True"/> at least one of the specified patterns matches the given value; <see langword="False"/> otherwise.
       ''' </returns>
       ''' ----------------------------------------------------------------------------------------------------
       <DebuggerStepThrough>
       Private Shared Function IsMatchPattern(ByVal value As String,
                                              ByVal patterns As IEnumerable(Of String),
                                              ByVal ignoreCase As Boolean) As Boolean

           ' Iterate the filename pattern(s) to match each name pattern on the current name.
           For Each pattern As String In patterns

               ' Supress consecuent conditionals if pattern its an asterisk.
               If pattern.Equals("*", StringComparison.OrdinalIgnoreCase) Then
                   Return True

               ElseIf ignoreCase Then ' Compare name ignoring string-case rules.
                   If value.ToLower Like pattern.ToLower Then
                       Return True
                   End If

               Else ' Compare filename unignoring string-case rules.
                   If value Like pattern Then
                       Return True
                   End If

               End If ' ignoreCase

           Next pattern

           Return False

       End Function

       ''' <summary>
       ''' Runs the next collector tasks synchronouslly.
       ''' </summary>
       <DebuggerStepThrough>
       Private Shared Sub RunNextTasks(Of T)(ByVal action As Action(Of ConcurrentQueue(Of T),
                                                                    String,
                                                                    SearchOption,
                                                                    IEnumerable(Of String),
                                                                    IEnumerable(Of String),
                                                                    Boolean,
                                                                    Boolean),
                                             ByVal queue As ConcurrentQueue(Of T),
                                             ByVal dirPath As String,
                                             ByVal firstPatterns As IEnumerable(Of String),
                                             ByVal secondPatterns As IEnumerable(Of String),
                                             ByVal ignoreCase As Boolean,
                                             ByVal throwOnError As Boolean)

           Try
               Task.WaitAll(New DirectoryInfo(dirPath).
                                GetDirectories.
                                Select(Function(dir As DirectoryInfo)
                                           Return Task.Factory.StartNew(Sub()
                                                                            action.Invoke(queue,
                                                                                          dir.FullName, SearchOption.AllDirectories,
                                                                                          firstPatterns, secondPatterns,
                                                                                          ignoreCase, throwOnError)
                                                                        End Sub)
                                       End Function).ToArray)

           Catch ex As Exception

               Select Case ex.GetType ' Handle or suppress exceptions by its type,

                   ' I've wrote different types just to feel free to extend this feature in the future.
                   Case GetType(UnauthorizedAccessException),
                        GetType(DirectoryNotFoundException),
                        ex.GetType

                       If throwOnError Then
                           Throw
                       End If

               End Select

           End Try

       End Sub

       ''' <summary>
       ''' Collects the files those matches the criteria inside the specified directory and/or sub-directories.
       ''' </summary>
       <DebuggerStepThrough>
       Private Shared Sub CollectFiles(ByVal queue As ConcurrentQueue(Of FileInfo),
                                       ByVal dirPath As String,
                                       ByVal searchOption As SearchOption,
                                       ByVal fileNamePatterns As IEnumerable(Of String),
                                       ByVal fileExtPatterns As IEnumerable(Of String),
                                       ByVal ignoreCase As Boolean,
                                       ByVal throwOnError As Boolean)

           ' Initialize a FileInfo collection.
           Dim fileInfoCol As IEnumerable(Of FileInfo) = Nothing

           ' Initialize a DirectoryInfo.
           Dim dirInfo As DirectoryInfo = Nothing
           SetupDirInfoObject(dirPath, dirInfo, throwOnError)

           If fileExtPatterns IsNot Nothing Then
               ' Decrease time execution by searching for files that has extension.
               SetupFileDirCollection(Of FileInfo)(AddressOf dirInfo.GetFiles, Nothing,
                                                   dirInfo.FullName, "*.*", fileInfoCol, throwOnError)
           Else
               ' Search for all files.
               SetupFileDirCollection(Of FileInfo)(AddressOf dirInfo.GetFiles, Nothing,
                                                   dirInfo.FullName, "*", fileInfoCol, throwOnError)
           End If

           ' If the fileInfoCol collection is not empty then...
           If fileInfoCol IsNot Nothing Then

               ' Iterate the files.
               For Each fInfo As FileInfo In fileInfoCol

                   ' Flag to determine whether a filename pattern is matched. Activated by default.
                   Dim flagNamePattern As Boolean = True

                   ' Flag to determine whether a file extension pattern is matched. Activated by default.
                   Dim flagExtPattern As Boolean = True

                   ' If filename patterns collection is not empty then...
                   If fileNamePatterns IsNot Nothing Then
                       flagNamePattern = IsMatchPattern(fInfo.Name, fileNamePatterns, ignoreCase)
                   End If

                   ' If file extension patterns collection is not empty then...
                   If fileExtPatterns IsNot Nothing Then
                       flagExtPattern = IsMatchPattern(fInfo.Extension, fileExtPatterns, ignoreCase)
                   End If

                   ' If fileName and also fileExtension patterns are matched then...
                   If flagNamePattern AndAlso flagExtPattern Then
                       queue.Enqueue(fInfo) ' Enqueue this FileInfo object.
                   End If

               Next fInfo

           End If ' fileInfoCol IsNot Nothing

           ' If searchOption is recursive then...
           If searchOption = SearchOption.AllDirectories Then
               RunNextTasks(Of FileInfo)(AddressOf CollectFiles,
                                         queue, dirInfo.FullName, fileNamePatterns, fileExtPatterns, ignoreCase, throwOnError)
           End If

       End Sub

       ''' <summary>
       ''' Collects the filepaths those matches the criteria inside the specified directory and/or sub-directories.
       ''' </summary>
       <DebuggerStepThrough>
       Private Shared Sub CollectFilePaths(ByVal queue As ConcurrentQueue(Of String),
                                           ByVal dirPath As String,
                                           ByVal searchOption As SearchOption,
                                           ByVal fileNamePatterns As IEnumerable(Of String),
                                           ByVal fileExtPatterns As IEnumerable(Of String),
                                           ByVal ignoreCase As Boolean,
                                           ByVal throwOnError As Boolean)

           ' Initialize a filepath collection.
           Dim filePathCol As IEnumerable(Of String) = Nothing

           If fileExtPatterns IsNot Nothing Then
               ' Decrease time execution by searching for files that has extension.
               SetupFileDirCollection(Of String)(Nothing, AddressOf Directory.GetFiles,
                                                 dirPath, "*.*", filePathCol, throwOnError)
           Else
               ' Search for all files.
               SetupFileDirCollection(Of String)(Nothing, AddressOf Directory.GetFiles,
                                                 dirPath, "*", filePathCol, throwOnError)
           End If

           ' If the filepath collection is not empty then...
           If filePathCol IsNot Nothing Then

               ' Iterate the filepaths.
               For Each filePath As String In filePathCol

                   ' Flag to determine whether a filename pattern is matched. Activated by default.
                   Dim flagNamePattern As Boolean = True

                   ' Flag to determine whether a file extension pattern is matched. Activated by default.
                   Dim flagExtPattern As Boolean = True

                   ' If filename patterns collection is not empty then...
                   If fileNamePatterns IsNot Nothing Then
                       flagNamePattern = IsMatchPattern(Path.GetFileNameWithoutExtension(filePath), fileNamePatterns, ignoreCase)
                   End If

                   ' If file extension patterns collection is not empty then...
                   If fileExtPatterns IsNot Nothing Then
                       flagExtPattern = IsMatchPattern(Path.GetExtension(filePath), fileExtPatterns, ignoreCase)
                   End If

                   ' If fileName and also fileExtension patterns are matched then...
                   If flagNamePattern AndAlso flagExtPattern Then
                       queue.Enqueue(filePath) ' Enqueue this filepath.
                   End If

               Next filePath

           End If ' filePathCol IsNot Nothing

           ' If searchOption is recursive then...
           If searchOption = SearchOption.AllDirectories Then
               RunNextTasks(Of String)(AddressOf CollectFilePaths,
                                       queue, dirPath, fileNamePatterns, fileExtPatterns, ignoreCase, throwOnError)
           End If

       End Sub

#End Region

   End Class

End Namespace

#End Region





Puedes testear la velocidad de mi función y compararlo con las funciones Directory.GetFiles() y My.Computer.FileSystem.GetFiles() mediante el siguiente código:

Código (vbnet) [Seleccionar]
Imports System.Collections.ObjectModel
Imports System.IO

Module Module1

   Sub Main()

       Dim dirPath As String = "C:\"

       Dim methodDirectoryGetFiles As String()
       Dim methodMyComputerGetFiles As ReadOnlyCollection(Of String)
       Dim methodElektroGetFiles As IEnumerable(Of String)
       Dim sw As New Stopwatch()

       sw.Start()
       methodDirectoryGetFiles = Directory.GetFiles(dirPath, "*", IO.SearchOption.AllDirectories)
       sw.Stop()
       Debug.WriteLine(String.Format("method DirectoryGetFiles  - Elapsed Time: {0}", sw.Elapsed.ToString("mm\:ss\:ffff")))

       sw.Reset()
       sw.Start()
       methodMyComputerGetFiles = My.Computer.FileSystem.GetFiles(dirPath, FileIO.SearchOption.SearchAllSubDirectories, "*")
       sw.Stop()
       Debug.WriteLine(String.Format("method MyComputerGetFiles - Elapsed Time: {0}", sw.Elapsed.ToString("mm\:ss\:ffff")))

       sw.Reset()
       sw.Start()
       methodElektroGetFiles = ElektroKit.Core.IO.Tools.Files.GetFilePaths(dirPath, IO.SearchOption.AllDirectories, Nothing, {"*"}, True, True)
       sw.Stop()
       Debug.WriteLine(String.Format("method ElektroGetFiles    - Elapsed Time: {0}", sw.Elapsed.ToString("mm\:ss\:ffff")))

   End Sub

End Module


Aquí tienes unos tests que he realizado:

Directorio: P:\
Patrón de búsqueda: "*"
Búsqueda recursiva: Sí
Cantidad de archivos a analizar: 142.880 y 7.374 carpetas
Tamaño total de los archivos: 238 Gigabytes
Tamaño por Colección/Array: 142.880 elementos
------------------------------------------------------------
method DirectoryGetFiles  - Elapsed Time: 00:01:4564 ++
method MyComputerGetFiles - Elapsed Time: 02:59:5497 +++
method ElektroGetFiles    - Elapsed Time: 00:00:5751 +


Directorio: G:\
Patrón de búsqueda: "*"
Búsqueda recursiva: Sí
Cantidad de archivos a analizar: 373.384 y 28.721 carpetas
Tamaño total de los archivos: 1.58 Terabytes
Tamaño por Colección/Array: 373.384 elementos
------------------------------------------------------------
method DirectoryGetFiles  - Elapsed Time: 00:04:5291 ++
method MyComputerGetFiles - Elapsed Time: 24:00:5846 +++ (si, eso han sido 24 ***** minutos los que ha tardado)
method ElektroGetFiles    - Elapsed Time: 00:01:7399 +


Directorio: C:\Windows\System32
Patrón de búsqueda: "*.dll"
Búsqueda recursiva: No
Cantidad de archivos a analizar: 3.465
Tamaño de archivos: 1.60 Gigabytes
Tamaño por Colección/Array: 2.620 elementos
------------------------------------------------------------
method DirectoryGetFiles  - Elapsed Time: 00:00:0138 +
method MyComputerGetFiles - Elapsed Time: 00:00:0608 +++
method ElektroGetFiles    - Elapsed Time: 00:00:0151 ++


Como puedes comprobar, mi implementación es altamente velóz cuanto mayor es la cantidad de archivos a analizar, y es casi igual de eficiente a la función Directory.GetFiles() cuando se trata de una cantidad pequeña de archivos a analizar, con una diferencia ínfima de milisegundos... lo que hace que merezca mucho la pena usarlo si necesitamos buscar varias extensiones de archivo al mismo tiempo.

Bueno, y de la función My.Computer.FileSystem.GetFiles mejor ni hablemos, los resultados hablan por si mismos. Los miembros expuestos en  "My.Computer" al igual que en Microsoft.VisualBasic están muy poco optimizados y ningún programador de VB.NET al que le preocupe el rendimiento de su app debería usarlos. De hecho en el foro he repetido mil veces que los miembros del namespace Microsoft.VisualBasic solo existen por temas de compatibilidad y conformidad por gente que migre de VB6 a VB.NET y lo tengan más facil para orientarse, que ningún programador de VB.NET debería utilizar el namespace Microsoft.VisualBasic más que para mostrar un MsgBox y listo xD, se creerán que hablo por hablar o algo... pues bueno, pueden ver por ustedes mismos el nefasto código fuente en la referencia online del source de .NET Framework.

PD: Estos tests se han realizado bajo circunstancias óptimas, realizando dos veces consecutivas cada llamada a cada uno de los tres métodos para que la caché de archivos enumerados del disco se actualizase y evitar así diferencias dispares en el tiempo de ejecución entre cada método.

Un saludo.
#1742
Hola.

Cita de: Suncar007 en 22 Septiembre 2017, 04:19 AM¿Que es mejor? Un IDE o Compilador

La pregunta carece de sentido puesto que un IDE o entorno de programación y un compilador son cosas diferentes, lo que ocurre es que la mayoría de IDEs hoy en día (sino todos) integran funcionalidades de compilación, pero que vayan en conjunto y cojidos de la mano no significa que sean lo mismo, un compilador es algo aparte, individual.

La definición de IDE (o en su defecto un editor de texto) se podría reducir a ser una herramienta para la escritura del código, y el compilador para compilar el código... como su propio nombre indica por si solo.

Cita de: Suncar007 en 22 Septiembre 2017, 04:19 AM¿es mejor un IDE o usar un editor de texto y un compilador?.

Un editor de texto no es un IDE. Un editor de texto está enfocado a la edición de texto (o código) en general, mientras que un IDE es una herramienta especializada en el desarrollo de algoritmos programáticos mediante "X" lenguaje de programación. Creo que esto es suficiente para poder autoresponderse a la pregunta por uno mismo: ¿qué es mejor para programar, un IDE o un editor de texto?.

Hoy en día existen bastantes editores de texto "cómodos" que cumplen de forma superficial y parcialmente con algunas funcionalidades básicas de un IDE (ej. Sublime Text, Notepad++), entre ellas el resaltado de sintaxis, el autocompletado de miembros, el análisis sintáctico del código, y la compilación de código, y algunos otros también soportan la depuración de código (ej. Visual Studio Code, o los editores anteriormente mencionados... con ciertos plugins de terceros), pero no se les puede llamar IDE, sencillamente por que no lo son de forma completa, de lo contrario se llamarían IDE.

Algunos programadores de lenguajes de markups como HTML o XML y/o de algunos lenguajes de scripting como Python o Ruby los hay quienes prefieren usar un editor de texto (ej. Sublime Text) antes que un IDE especializado en dicho lenguaje de programación (ej. PyCharm, RubyMine), pero aquí no estamos cuestionando los gustos o el sentido de conformidad personal de cada individuo, sino qué tipo de herramienta es más óptima para llevar a cabo la tarea de programación. y para muchos lenguajes de programación el hecho de preferir usar un editor de texto antes que un IDE es sinónimo de haber adoptado malos hábitos de programación y ser altamente conformistas con una tipo de herramienta inferior al tipo de herramienta que realmente está destinada a beneficiarnos en tal propósito: un IDE. De hecho, para algunos lenguajes de programación de alto nivel resultaría absurdo intentar desarrollar una aplicación mediante un simple editor de texto... sencillamente sería de locos, por que se necesita al IDE para la automatización de ciertos procedimientos y la autogeneración de código en segundo plano que nos ahorraría el equivalente a estar varias horas (o dias, o semanas, según la embergadura del proyecto) escribiendo de forma manual el código en un editor de texto, y encima sin la posibilidad de poder depurarlo eficientemente en éste.

Por otro lado también hay que mencionar que existen muchas IDEs que son malas o al menos yo las considero malas (y esto por lo general también va acompañado a que sean IDEs gratuitas) tanto en la estética como en su intento por optimizar el rendimiento del usuario final. Las hay que son de todo excepto cómodas... y es que un IDE promete ser más óptimo, pero no más cómodo que un editor de texto ligero.

Un IDE profesional proporciona toda una amplia gama de características que no se verán en editor de texto (o un intento de IDE como Sublime Text o Notepad++), sin ir más lejos: algo tan necesario como es un diseñador de tipo WYSIWYG (what you see is what you get) de interfaces gráficas (o GUI builder por su nombre en Inglés), o un sistema sofisticado de depuración (mucho más que lo que tenga cualquier editor de texto), o algo de lo más básico como es el procedimiento inicial de creación de proyectos personalizados para un lenguaje específico (ej. una aplicación de consola, o una librería), o algo más avanzado como las herramientas de diagnóstico para la evaluación del consumo de memoria y rendimiento del CPU (conocido como Profilling por su nombre en Inglés), o funcionalidades de evaluación de unidades de código (o Unit Testing por su nombre en Inglés), o emuladores de dispositivos Android, o herramientas de automatización gráfica de la aplicación (conocido como Coded UI por su nombre en Inglés, aunque por el momento solo lo he visto disponible exclusivamente en el IDE Visual Studio), o herramientas de evaluación de la complejidad ciclomática de tu código, y así podriamos seguir hasta llenar la hoja entera de un cuaderno por delante y por detrás.

Un editor de texto no es un IDE, que eso nos quede claro. Si vas a programar de forma seria, entonces usa un IDE, pero si no necesitas las funcionalidades adicionales de un IDE y prefieres usar un editor de texto por que te sientes más...cómodo o simplemente por que te es suficiente para escribir pequeños códigos que no necesitan de tratamiento profesional en un IDE, pues hazlo a tu manera como tú prefieras, ya que al final en mi opinión lo único que importa es utilizar la herramienta que mejor nos consiga hacer optimizar nuestro tiempo y rendimiento.

Saludos.
#1743
.NET (C#, VB.NET, ASP) / Re: Aprender VB.NET O VB.6?
22 Septiembre 2017, 08:25 AM
Hola.

Cita de: n1sen en 22 Septiembre 2017, 00:49 AMhoy vengo a molestar preguntando cual de los 2 lenguajes es mejor, cual es mas optimo, etc.

Si esta pregunta hubiese sido formulada en el año 2002 todavía sería comprensible, pero en pleno año 2017 tras 19 años desde la muerte o descontinuación de VB6 y +15 años de desarrollo y evolución de la plataforma .NET Framework la verdad cuesta creer que siga existiendo dudas y desinformación entre las nuevas generaciones de programadores.

Puedes tomarte lo que acabo de decir como una respuesta a tus dudas. Visual Basic.NET es el sucesor y evolución directa de Visual Basic, no hay mucho más que añadir a este argumento de peso. Se podría explicar en profundidad todos los detalles y diferencias al respecto entre ambos lenguajes, pero toda la información necesaria la tienes al alcance de tu mano en la World Wide Web, la puedes buscar por ti mismo en Google...

Cita de: n1sen en 22 Septiembre 2017, 02:37 AMMe he enterado que existen muchas versiones de VB.NET, pero cual elijo?

A la hora de elegir más bien no elijes una versión del lenguaje de forma individual, sino una versión de la IDE, y la respuesta a eso es muy simple: la versión más actual, es decir: Visual Studio 2017 / VB15.

Al momento de desarrollar código y de compilar una aplicación deberás tomar otro tipo de decisión: ¿qué versión de .NET Framework debo soportar?. Al igual que antes, cuanto más actual sea dicha versión más beneficios obtendrás en tu entorno de programación. La versión más actual es .NET Framework 4.7, pero para poder utilizarla necesitas estar bajo una release actual de Windows 10 (ya que la instalación de los targeting packs de .NET Framework 4.7 no son compatible con otros sistemas operativos ni tampoco con las primeras releases de Windows 10). Con que soportes la versión 4.5 o 4.6 de .NET Framework será más que suficiente para poder disfrutar de todas las mejoras actuales, sobre todo las mejoras de programación asincrónica con los keywords Async / Await.

Lo que en realidad debería preocuparte un poco es qué tecnología elegir para compilar tus aplicaciones y desempeñar los progresos de tu aprendizaje: Windows Forms, Windows Presentation Foundation, ASP.NET, etc. Mi recomendación para el desarrollo de aplicaciones de escritorio: WPF. Y si quieres desarrollar aplicaciones multiplataforma entonces deberás utilizar una versión reducida y portable de .NET Framework, conocida como: .NET Core.

Saludos.
#1744
Cita de: warcry. en 21 Septiembre 2017, 19:37 PM
como se nota que el señor Gates no limpia el teclado con el ordenador encendido  :silbar:

...y por lo que parece con el teclado activado también xD. ¿Va siendo hora de cambiarse a un teclado con botón de ON/OFF?, yo solo lo comento. :P

Saludos!
#1745
Hola.

Esta pregunta trata sobre herramientas de e-mail marketing. Me gustaría saber si alguien conoce alguna aplicación para Windows que sea GRATUITA y certificada (es decir, con validaciones de confianza, sin falsos positivos) que permita verificar la existencia de direcciones de correo de grandes listas de cientos de millones de e-mails en formato .txt o .csv. Y a ser posible opcionálmente que sea capaz de identificar direcciones de trampas de Spam (spam traps por su nombre en Inglés).

¿Alguien sabe algo?.

Gracias por leer.
#1746
.NET (C#, VB.NET, ASP) / Re: actualizar access
21 Septiembre 2017, 04:08 AM
Cita de: PETTETE en 20 Septiembre 2017, 17:00 PMcomo puedo hacer para que al leer un valor alfanumerico de un acces este se incremente en uno y se ponga en el excel?

Teniendo un patrón tal que: "Mi_Prefijo_0000"

1. Separar el String en dos partes o tokens, la alfabética (el prefijo), y la numérica (el índice).   "Mi_Prefijo_"  +  "0000"

2. Incrementar el valor de la parte numérica preservando la longitud del caracter de relleno, en este caso el Cero. ( ej. 0000 -> 0001 -> 0002 -> etc... )

3. Concatenar de nuevo la parte alfabética con la numérica, dando como resultado: "Mi_Prefijo_0001"

Fin.

No es algo complicado. Aquí tienes una función de uso genérico que he desarrollado la cual puedes adaptar a tus necesidades:

Código (vbnet) [Seleccionar]
Public Shared Function IncrementIndexPattern(ByVal pattern As String, ByVal position As Integer, ByVal fillChar As Char) As String

   Dim curSuffix As String = pattern.Substring(position)
   Dim newSuffix As String
   Dim suffixLen As Integer = curSuffix.Length
   Dim index As Integer

   If Not Integer.TryParse(If(fillChar = "0"c, curSuffix, curSuffix.TrimStart(fillChar)), index) Then
       Throw New FormatException("The value does not have a valid numeric format.")
   End If

   newSuffix = Convert.ToString(index + 1).PadLeft(suffixLen, fillChar)
   If (newSuffix.Length > suffixLen) Then
       ' -= Deposite su control de errores aquí =-
       ' La longitud del nuevo índice es mayor que la capacidad de ceros.
       ' Ej. "pattern_12345" es más largo que "pattern_000".
       Throw New IndexOutOfRangeException()

   Else
       Return pattern.Substring(0, position) & newSuffix

   End If

End Function


Ejemplo de uso:
Código (vbnet) [Seleccionar]
' Programatically building a pattern.
Dim prefix As String = "My_Pattern_"
Dim fillChar As Char = "#"c
Dim fillCount As Integer = 4
Dim firstIdx As String = New String(fillChar, (fillCount - 1)) & "0" ' ###0
Dim pattern As String = (prefix & firstIdx) ' My_Pattern_###0
Debug.WriteLine(pattern)

' Setting the loop values.
Dim idxStartPos As Integer = prefix.Length ' Or also: (pattern.LastIndexOf("_"c) + 1)
Dim maxIndexCount As Integer = Convert.ToInt32(New String("9"c, fillCount)) ' Max possible index to fill.

For i As Integer = 0 To (maxIndexCount - 1)
   pattern = IncrementIndexPattern(pattern, idxStartPos, fillChar)
   Debug.WriteLine(pattern)
Next i


Resultado de ejecución:
My_Pattern_###0
My_Pattern_###1
My_Pattern_###2
My_Pattern_###3
My_Pattern_###4
My_Pattern_###5
My_Pattern_###6
My_Pattern_###7
My_Pattern_###8
My_Pattern_###9
My_Pattern_##10
...
My_Pattern_9990
My_Pattern_9991
My_Pattern_9992
My_Pattern_9993
My_Pattern_9994
My_Pattern_9995
My_Pattern_9996
My_Pattern_9997
My_Pattern_9998
My_Pattern_9999


Saludos.
#1747
Hola.

Yo el primer problema no lo entiendo, basicamente por que aquí ya estás utilizando una iteración recursiva en busca de archivos con distintas extensiones:


Así que en teoría eso ya sabes como resolverlo, pero bueno, te hago saber que la función System.IO.Directory.GetFiles() tiene una sobrecarga para que le indiques si la búsqueda debe realizarse de forma recursiva o no:


Dicho método no provee ninguna sobrecarga para especificar más de una extensión o patrón de búsqueda, para eso ya estabas utilizando la otra función, My.Computer.FileSystem.GetFiles().

Si quieres sustituir la función My.Computer.FileSystem.GetFiles() por System.IO.Directory.GetFiles() y conservando la funcionalidad de buscar varias extensiones, pues sí o sí tienes que llamar a dicha función más de una vez para concatenar los resultados en una colección. Ejemplo:

Código (vbnet) [Seleccionar]
Dim dirPath As String = ".\"
Dim fileExts As String() = {"*.vbs", "*.cmd", "*.js", "*.wsf", "*.ink", "*.bat"}

Dim filePaths As New List(Of String)
For Each fileExt As String In fileExts
   filePaths.AddRange(Directory.GetFiles(dirPath, String.Format("*.{0}", fileExt), SearchOption.AllDirectories))
Next

For Each filePath As String In filePaths
   Debug.WriteLine(filePath)
Next





Cita de: **Aincrad** en 20 Septiembre 2017, 22:05 PM
Código (vbnet) [Seleccionar]
File.SetAttributes(dirPath, FileAttributes.Normal)

Eso es incorrecto, ya que así no solo estás eliminando el atributo Hidden, sino también todos los demás atributos que hayan sido asignados al archivo.

Tienes que conservar esos otros atributos. Ejemplo:

Código (vbnet) [Seleccionar]
Dim filePath As String = "C:\file.txt"
Dim attribs As FileAttributes = File.GetAttributes(filePath)

File.SetAttributes(filePath, attribs And Not FileAttributes.Hidden)





Cita de: **Aincrad** en 20 Septiembre 2017, 20:11 PM

el code tiene que implementar la funcion de :
Código (vbnet) [Seleccionar]
Dim dirPath As String = Form2.FolderBrowserDialog1.SelectedPath

para el directorio. no se si me entienden.  :silbar:

No, no lo entiendo bien. Supongo que te refieres a que quieres visualizar las carpetas ocultas en el árbol de carpetas de la ventana de diálogo FolderBrowserDialog, pues bien, eso no puedes hacerlo. El componente FolderBrowserDialog respeta la configuración del usuario, así que si el usuario no tiene activada la opción para ver archivos y carpetas ocultas, el componente no las mostrará.

De todas formas, una de las peores decisiones que un programador de .NET bajo tecnología WinForms puede tomar es utilizar el componente FolderBrowserDialog, ya que es límitadísimo y obsoleto, pues no le han modificado el diseño desde los 90. ¡Ni siquiera puedes introducir una ruta de directorio para navegar directamente!, no, tienes que clickar en los nodos hasta aburrirte. Es de lo peor, todo lo opuesto a productivo.



Por ese motivo te recomiendo encarecidamente que utilices los componentes gratuitos de Ooki, que son un wrapper de los diálogos modernos del "buscador" de carpetas:




PD: Insisto, ningún diálogo de carpetas te va a mostrar carpetas ocultas a menos que la opción de mostrar archivos y carpetas ocultas esté activada en el sistema. Puedes activarla tu mismo, aquí abajo te muestro como hacerlo, pero ten en cuenta que llevar a cabo este tipo de actos se considera intrusismo (un programa intrusivo):

Código (vbnet) [Seleccionar]
Imports Microsoft.Win32

Código (vbnet) [Seleccionar]
Using reg As RegistryKey = RegistryKey.OpenBaseKey(RegistryHive.CurrentUser, RegistryView.Default)
   reg.OpenSubKey("Software\Microsoft\Windows\CurrentVersion\Explorer\Advanced", writable:=True).
       SetValue("Hidden", 1, RegistryValueKind.DWord)
End Using


Saludos.
#1748
Cita de: crazykenny en 20 Septiembre 2017, 19:19 PM
Bueno, Elektro, dejando de lado el debate, queria responder para agradecerte por el aporte, ya que siempre ayuda, y mas viendo el motivo por el cual has abierto este tema y por como van los temas relacionados con politica (entre otras cosas).  :D :) ;)

Saludos.

Gracias CrazyKenny. Tú siempre tan correcto, jeje :). El motivo de comprartir esos videos era que la gente se mentalizase del timo que existe con respecto al Yoga, pero que detrás del timo existe un arte ancestral cuya práctica es...honesta y sus fines éticos. A mi el señor del video me parece muy honrado.




Cita de: Orubatosu en 20 Septiembre 2017, 22:26 PMPor eso venden las medicinas milagrosas, las píldoras que adelgazan, las que agradan el pene y mil timos mas. La gente quiere cosas sencillas, rápidas y sin esfuerzo

Y luego está esto otro...

http://www.elrellano.com/videos_online/13708/como-perder-peso-sin-dieta-ni-ejercicios.html

( no tiene desperdicio )

Saludos
#1749
Cita de: #!drvy en 20 Septiembre 2017, 13:04 PMPues si mi pareja me hace eso, seguramente la denuncie. Que sea mi pareja y que yo sienta algo por ella no quiere decir que puede espiarme y meterse de pleno en mi vida privada. No la da derecho a estar las 24 horas encima mio ni mucho menos "sobreprotegerme".

Si no te gusta lo que hace tu pareja pues se lo dices y punto, para eso se supone que dos personas se juntan y forman una relación sentimental, para solucionar los conflictos mediante el díalogo, el Amor, respeto y comprensión, ¿no?. Si el celoso/desconfiado/espía no respeta a su pareja y no cesa en espiar pues cortas con él/ella y lo mandas a...tomar por saco, pero no vas y le denuncias sin pensar precisamente en las consecuencias de esa denuncia que acabará por arruinarle la vida y su futuro de negocio convirtiéndolo en un supuesto "criminal" ante cualquier futura entrevista de trabajo, por que entonces nunca has querido a esa persona.

Cita de: #!drvy en 20 Septiembre 2017, 13:04 PMNo es un error cualquiera, es un abuso, y como tal, tiene sus consecuencias.

Es un error cualquiera, todo el mundo comete todo tipo de errores (cosas de las que luego se arrepiente) por Amor, desde la simple e inocente mentira piadosa para no herir los sentimientos de tu pareja (ej. "cariño, no se me ha olvidado nuestro aniversario, lo que ocurre es que... { deposite su excusa favorita aquí }"), hasta este tipo de errores que muchas veces no se llegan a cometer con maldad sino simplemente por miedo a perder a tu pareja, por sentirte inferior o que no te la mereces y se podría enamorar de otro, o cosas parecidas. La condena que ese tio se merece no es ir a la cárcel, sino ponerle a cargo de un profesional (psicólogo) para que reciba un poco de apoyo moral y recapacité sobre los actos que ha cometido ya está.

Esto no se puede llamar justicia lo mires por donde lo mires, me da igual si le instaló uno o cinco troyanos como si le pinchó el telefono a ella y a su abuela, esto es una injusticia, qué digo, ¡UNA VERGUENZA!, y la jueza esa es una hija de la grandísima PU... por que los problemas de este tipo son psicológicos, no vandálicos, y requieren que sean analizados como tal, como un problema mental de una persona que requiere AYUDA, no ser enjaulado en una celda para privarle de su libertad y "trastornarle" más de lo que ya lo está.
En mi opinión, aunque desde la ignorancia, la jueza al ser una mujer probablemente le faltó sentido de imparcialidad en sus observaciones, ya que la primera impresión que da todo este asunto es que ella no se puso en la piel del acusado, simplemente defendió los derechos de la mujer cual Feminazi y ale, a la cárcel, por que todos los hombres somos unos maltratadores en potencia y nos lo merecemos.  Pf. . .

Un error así no se merece unas consecuencias tan INMERECIDAS, para NADIE, ni para el hombre ni para la mujer que espíe, y mira que de ese tipo de mujeres celosas hay muchas más que hombres, todo el día cotilleando las conversaciones en el móvil y retorciéndose el cerebro pensando en cuantas veces les ponemos los cuernos con las amigas y hechándotelo en cara una y otra vez... ¿pero acaso algún hombre denuncia esas cosas?, ¿no verdad?, pues ya está.

En fin...
#1750
Foro Libre / ¿Donde comprar Ethereum?
20 Septiembre 2017, 08:07 AM
Buenas.

¿ Alguien sabe en qué página web se puede comprar la criptomoneda Ethereum a cambio de €uros ?. Ojo, no busco vendedores individuales de Ethereum, sino un servicio web ( tipo bit2me.com ). No importa si el pago se debe realizar con tarjeta de crédito, Paypal, o Bitcoin.

Gracias por leer. Saludos.