Antecedentes: soy partidario de la programación funcional que trabaja en una tienda VB.NET donde el modelo mental predominante es la programación imperativa. Siendo que la base de nuestro sistema es WinForms, puedo entender que no nos vamos a alejar por completo de la programación imperativa, pero aún así trato de usar FP (principalmente a través de Linq) siempre que sea posible porque creo en sus méritos.
Argumentos y contraargumentos contra FP
Uno podría notar que Linq fluido es menos eficiente que su contraparte imperativa en que este estilo procesa una secuencia hacia otra secuencia y la repite. En general, tomará algunas pasadas más que el enfoque imperativo, que puede optimizarse mejor para evitar repeticiones en una secuencia. Por esta razón, el líder no podía entender por qué elegiría un enfoque funcional que sea claramente "menos eficiente".
- Contraargumento : Argumenté que si bien a veces es menos eficiente en términos de ciclos de CPU, sentí que es más inteligible y fácil de seguir humanamente, ya que cada línea hace una sola cosa en su paso por la secuencia. Para mí esto se siente como tener una línea de montaje donde cada persona en su estación tiene un solo trabajo que hacer. Siento que el insignificante intercambio de eficiencia es recompensado por un código cuyas preocupaciones están claramente separadas.
El siguiente argumento contra FP que escucho en mi tienda es que es más difícil de depurar, lo cual es cierto. No es fácil pasar por alto el código Linq. Y a veces tengo que desentrañar una cadena de métodos para seguir y analizar mejor los problemas que no puedo detectar de inmediato.
- _Argumento del contador: en su mayor parte, aunque no tengo problemas con esto porque creo que el estilo funcional es más declarativo en cómo se lee y cuando se arroja un error dentro de una cadena funcional, generalmente puedo detectar el problema de inmediato.
Mi pregunta
He estado tratando de promover el estilo funcional en nuestra tienda y no siento que esté avanzando. He hecho ambos estilos de programación y solo recientemente incursioné en Haskell. A pesar de años de experiencia imperativa, ahora que estoy haciendo uso rutinario de FP en JavaScript, ha crecido en mí. Suena una nota de rectitud en mi núcleo cuando lo comparo con lo que podría haber hecho si me hubiera quedado con un estilo imperativo. He vuelto a entrenar mi cerebro hacia el pensamiento funcional, hacia la composición funcional.
Lo que no puedo entender es lo difícil que ha sido convencer a otros de los méritos de FP.
Por ejemplo, los desarrolladores en mi tienda usan Linq, pero creo que generalmente lo usan en el contexto de tratar con datos de dominio. Lo uso en un sentido más general y lo prefiero cada vez que estoy lidiando con secuencias / listas o estructuras de datos persistentes. No he podido convencer a mis compañeros de equipo para ampliar su uso de Linq.
Lo que estoy tratando de entender es lo que hace que a un desarrollador no le guste FP.
Me gustaría ver una respuesta de alguien que tenga mucha experiencia con FP pero que haya decidido a favor del estilo imperativo. ¿Qué impulsó la decisión de permanecer imperativo en lugar de usar funcional?
Aquí hay un ejemplo adicional que destaca las diferencias entre la programación imperativa y funcional.
Escribí el SelectedRows
método de nuestra cuadrícula en Linq como tal:
Public Property SelectedRows() As DataRow() Implements IDataSourceControl.SelectedRows
Get
Return Me.ugrBase.Selected.Rows.
OfType(Of Infragistics.Win.UltraWinGrid.UltraGridRow)().
Select(Function(ugr) ugr.ListObject).
OfType(Of DataRowView)().
Select(Function(drv) drv.Row).
ToArray
End Get
Sin embargo, este estilo de código hace que algunos de nuestros desarrolladores se sientan incómodos, por lo que nuestro líder lo reescribió a los más familiares:
Public Property SelectedRows() As DataRow() Implements IDataSourceControl.SelectedRows
Get
Dim plstRows As New List(Of DataRow)
For Each bugrLoop As Infragistics.Win.UltraWinGrid.UltraGridRow In Me.ugrBase.Selected.Rows
If bugrLoop.ListObject IsNot Nothing Then
plstRows.Add(CType(bugrLoop.ListObject, DataRowView).Row)
End If
Next
Return plstRows.ToArray()
End Get