¿Cómo aplicar el principio de segregación de interfaz en C?


15

Tengo un módulo, digamos 'M', que tiene algunos clientes, digamos 'C1', 'C2', 'C3'. Quiero distribuir el espacio de nombres del módulo M, es decir, las declaraciones de las API y los datos que expone, en archivos de encabezado de tal manera que:

  1. para cualquier cliente, solo los datos y las API que requiere están visibles; el resto del espacio de nombres del módulo está oculto para el cliente, es decir, se adhieren al principio de segregación de interfaz .
  2. una declaración no se repite en varios archivos de encabezado, es decir, no viola DRY .
  3. El módulo M no tiene dependencias de sus clientes.
  4. un cliente no se ve afectado por los cambios realizados en partes del módulo M que no utiliza.
  5. Los clientes existentes no se ven afectados por la adición (o eliminación) de más clientes.

Actualmente trato con esto dividiendo el espacio de nombres del módulo según los requisitos de sus clientes. Por ejemplo, en la imagen a continuación se muestran las diferentes partes del espacio de nombres del módulo requerido por sus 3 clientes. Los requisitos del cliente se superponen. El espacio de nombres del módulo se divide en 4 archivos de encabezado separados: 'a', '1', '2' y '3' .

Particionamiento del espacio de nombres del módulo

Sin embargo, esto viola algunos de los requisitos antes mencionados, es decir, R3 y R5. Se infringe el requisito 3 porque esta partición depende de la naturaleza de los clientes; también al agregar un nuevo cliente, esta partición cambia y viola el requisito 5. Como se puede ver en el lado derecho de la imagen de arriba, con la adición de un nuevo cliente, el espacio de nombres del módulo ahora se divide en 7 archivos de encabezado: 'a ',' b ',' c ',' 1 ',' 2 * ',' 3 * 'y' 4 ' . Los archivos de encabezado significaban para 2 de los cambios de clientes anteriores, lo que desencadena su reconstrucción.

¿Hay alguna manera de lograr la segregación de interfaz en C de manera no artificial?
En caso afirmativo, ¿cómo abordaría el ejemplo anterior?

Una solución hipotética irreal que imagino sería:
el módulo tiene 1 archivo de encabezado grueso que cubre todo su espacio de nombres. Este archivo de encabezado se divide en secciones y subsecciones direccionables, como una página de Wikipedia. Luego, cada cliente tiene un archivo de encabezado específico diseñado para ello. Los archivos de encabezado específicos del cliente son solo una lista de hipervínculos a las secciones / subsecciones del archivo de encabezado grueso. Y el sistema de compilación debe reconocer un archivo de encabezado específico del cliente como 'modificado' si se modifica alguna de las secciones a las que apunta en el encabezado del Módulo.


1
¿Por qué este problema es específico de C? ¿Es porque C no tiene herencia?
Robert Harvey

Además, ¿violar el ISP hace que su diseño funcione mejor?
Robert Harvey

2
C realmente no admite intrínsecamente los conceptos de OOP (como las interfaces o la herencia). Nos conformamos con hacks crudos (pero creativos). Buscando un truco para simular interfaces. Por lo general, todo el archivo de encabezado es la interfaz de un módulo.
work.bin

1
structes lo que usas en C cuando quieres una interfaz. Por supuesto, los métodos son un poco difíciles. Puede encontrar esto interesante: cs.rit.edu/~ats/books/ooc.pdf
Robert Harvey

No pude encontrar una interfaz equivalente usando structy function pointers.
work.bin

Respuestas:


5

La segregación de interfaz, en general, no debe basarse en los requisitos del cliente. Debe cambiar todo el enfoque para lograrlo. Yo diría que modularice la interfaz agrupando las características en grupos coherentes . Es decir, la agrupación se basa en la coherencia de las características en sí, no en los requisitos del cliente. En ese caso, tendrá un conjunto de interfaces, I1, I2, ... etc. El Cliente C1 puede usar I2 solo. El cliente C2 puede usar I1 e I5, etc. Tenga en cuenta que, si un cliente usa más de un Ii, no es un problema. Si ha descompuesto la interfaz en módulos coherentes, ahí es donde está el meollo del asunto.

Nuevamente, el ISP no está basado en el cliente. Se trata de descomponer la interfaz en módulos más pequeños. Si esto se hace correctamente, también garantizará que los clientes estén expuestos a la menor cantidad de funciones que necesiten.

Con este enfoque, sus clientes pueden aumentar a cualquier número, pero usted no se ve afectado. Cada cliente utilizará una o alguna combinación de las interfaces en función de sus necesidades. ¿Habrá casos en que un cliente, C, deba incluir decir I1 e I3, pero no usar todas las características de estas interfaces? Sí, eso no es un problema. Solo usa la menor cantidad de interfaces.


Seguramente quisiste decir grupos disjuntos o no superpuestos , supongo.
Doc Brown

Sí, disjunto y no superpuesto.
Nazar Merza

3

El principio de segregación de interfaz dice:

Ningún cliente debería verse obligado a depender de métodos que no utiliza. El ISP divide las interfaces que son muy grandes en otras más pequeñas y más específicas para que los clientes solo tengan que conocer los métodos que les interesan.

Hay algunas preguntas sin respuesta aquí. Uno es:

¿Cuán pequeño?

Tu dices:

Actualmente trato con esto dividiendo el espacio de nombres del módulo según los requisitos de sus clientes.

A este manual lo llamo escribir pato . Construye interfaces que exponen solo lo que un cliente necesita. El principio de segregación de la interfaz no es simplemente escribir manualmente.

Pero el ISP tampoco es simplemente un llamado a interfaces de roles "coherentes" que puedan reutilizarse. Ningún diseño de interfaz de roles "coherente" puede proteger perfectamente contra la adición de un nuevo cliente con sus propias necesidades de roles.

ISP es una forma de aislar a los clientes del impacto de los cambios en el servicio. Su objetivo era hacer que la compilación fuera más rápida a medida que realiza cambios. Claro que tiene otros beneficios, como no romper clientes, pero ese fue el punto principal. Si estoy cambiando la count()firma de la función de servicios , es bueno si los clientes que no usan count()no necesitan ser editados y recompilados.

Esto es POR QUÉ me importa el Principio de segregación de interfaz. No es algo que considero importante como la fe. Resuelve un problema real.

Entonces, la forma en que debe aplicarse debería resolver un problema para usted. No hay una forma de muerte cerebral de aplicar ISP que no pueda ser derrotada con el ejemplo correcto de un cambio necesario. Se supone que debe observar cómo está cambiando el sistema y tomar decisiones que permitan que las cosas se calmen. Exploremos las opciones.

Primero pregúntese: ¿es difícil hacer cambios en la interfaz de servicio en este momento? Si no, sal y juega hasta que te calmes. Este no es un ejercicio intelectual. Asegúrese de que la cura no sea peor que la enfermedad.

  1. Si muchos clientes usan el mismo subconjunto de funciones, eso argumenta a favor de interfaces reutilizables "coherentes". El subconjunto probablemente se centra en una idea que podemos considerar como el rol que el servicio está proporcionando al cliente. Es bueno cuando esto funciona. Esto no siempre funciona.

  2.  

    1. Si muchos clientes usan diferentes subconjuntos de funciones, es posible que el cliente realmente esté usando el servicio a través de múltiples roles. Eso está bien, pero hace que los roles sean difíciles de ver. Encuéntralos e intenta separarlos. Eso puede volver a ponernos en el caso 1. El cliente simplemente usa el servicio a través de más de una interfaz. Por favor, no comience a enviar el servicio. En todo caso, eso significaría pasar el servicio al cliente más de una vez. Eso funciona, pero me hace preguntar si el servicio no es una gran bola de lodo que necesita ser dividida.

    2. Si muchos clientes usan diferentes subconjuntos, pero no ve roles que permiten que los clientes usen más de uno, entonces no tiene nada mejor que escribir en pato para diseñar sus interfaces. Esta forma de diseñar las interfaces garantiza que el cliente no esté expuesto ni siquiera a una función que no esté utilizando, pero casi garantiza que agregar un nuevo cliente siempre implicará agregar una nueva interfaz que, aunque la implementación del servicio no necesita saber al respecto, la interfaz que agrega las interfaces de rol lo hará. Simplemente hemos cambiado un dolor por otro.

  3. Si muchos clientes usan diferentes subconjuntos, se superponen, se espera que se agreguen nuevos clientes que necesitarán subconjuntos impredecibles, y usted no está dispuesto a dividir el servicio y luego considerar una solución más funcional. Dado que las dos primeras opciones no funcionaron y realmente estás en un mal lugar donde nada sigue un patrón y se están produciendo más cambios, entonces considera proporcionar a cada función su propia interfaz. Terminar aquí no significa que el ISP haya fallado. Si algo falló fue el paradigma orientado a objetos. Las interfaces de método único siguen al ISP en extremo. Es un poco digno de teclado, pero es posible que de repente esto haga que las interfaces sean reutilizables. Nuevamente, asegúrese de que no haya

Entonces resulta que pueden volverse muy pequeños.

He tomado esta pregunta como un desafío para aplicar ISP en los casos más extremos. Pero tenga en cuenta que es mejor evitar los extremos. En un diseño bien pensado que aplica otros principios SÓLIDOS, estos problemas generalmente no ocurren ni importan, casi tanto.


Otra pregunta sin respuesta es:

¿Quién posee estas interfaces?

Una y otra vez veo interfaces diseñadas con lo que yo llamo una mentalidad de "biblioteca". Todos hemos sido culpables de la codificación mono-ver-mono-hacer donde solo estás haciendo algo porque así es como lo viste hacer. Somos culpables de lo mismo con las interfaces.

Cuando miro una interfaz diseñada para una clase en una biblioteca, solía pensar: oh, estos tipos son profesionales. Esta debe ser la forma correcta de hacer una interfaz. Lo que no entendía es que el límite de una biblioteca tiene sus propias necesidades y problemas. Por un lado, una biblioteca ignora por completo el diseño de sus clientes. No todos los límites son iguales. Y a veces, incluso el mismo límite tiene diferentes formas de cruzarlo.

Aquí hay dos formas simples de ver el diseño de la interfaz:

  • Interfaz de propiedad del servicio. Algunas personas diseñan cada interfaz para exponer todo lo que un servicio puede hacer. Incluso puede encontrar opciones de refactorización en IDE que escribirán una interfaz para usted usando cualquier clase que alimente.

  • Interfaz propiedad del cliente. El ISP parece argumentar que esto es correcto y que el servicio es incorrecto. Debe dividir cada interfaz con las necesidades de los clientes en mente. Como el cliente posee la interfaz, debe definirla.

Entonces, ¿quién tiene razón?

Considere los complementos:

ingrese la descripción de la imagen aquí

¿Quién posee las interfaces aquí? ¿Los clientes? ¿Los servicios?

Resulta que ambos.

Los colores aquí son capas. Se supone que la capa roja (derecha) no sabe nada sobre la capa verde (izquierda). La capa verde se puede cambiar o reemplazar sin tocar la capa roja. De esa manera, cualquier capa verde se puede conectar a la capa roja.

Me gusta saber qué se supone que debe saber sobre qué y qué se supone que no debe saber. Para mí, "¿qué sabe sobre qué?", ​​Es la pregunta arquitectónica más importante.

Dejemos claro un poco de vocabulario:

[Client] --> [Interface] <|-- [Service]

----- Flow ----- of ----- control ---->

Un cliente es algo que usa.

Un servicio es algo que se usa.

Interactor Resulta ser ambos.

ISP dice romper las interfaces para los clientes. Bien, apliquemos eso aquí:

  • Presenter(un servicio) no debe dictar a la Output Port <I>interfaz. La interfaz debe reducirse a lo que Interactornecesita (aquí actuando como cliente). Eso significa que la interfaz SABE sobre el Interactory, para seguir al ISP, debe cambiar con él. Y esto está bien.

  • Interactor(aquí actuando como un servicio) no debería dictar a la Input Port <I>interfaz. La interfaz debe reducirse a lo que Controller(un cliente) necesita. Eso significa que la interfaz SABE sobre el Controllery, para seguir al ISP, debe cambiar con él. Y esto no está bien.

El segundo no está bien porque no se supone que la capa roja sepa sobre la capa verde. Entonces, ¿está mal el ISP? Así un poco. Ningún principio es absoluto. Este es un caso en el que los tontos a quienes les gusta la interfaz para mostrar todo lo que el servicio puede hacer resultan ser correctos.

Al menos, tienen razón si Interactorno hace nada más que este caso de uso. Si Interactorhace cosas para otros casos de uso, no hay razón Input Port <I>para saber esto. No estoy seguro de por qué Interactorno puede centrarse solo en un caso de uso, por lo que este no es un problema, sino que sucede algo.

Pero la input port <I>interfaz simplemente no puede esclavarse a sí mismaController cliente y hacer que este sea un verdadero complemento. Este es un límite de 'biblioteca'. Una tienda de programación completamente diferente podría estar escribiendo la capa verde años después de la publicación de la capa roja.

Si está cruzando un límite de 'biblioteca' y siente la necesidad de aplicar ISP a pesar de que no posee la interfaz en el otro lado, tendrá que encontrar una manera de estrechar la interfaz sin cambiarla.

Una forma de lograrlo es con un adaptador. Ponlo entre clientes como Controlery la Input Port <I>interfaz. El adaptador acepta Interactorcomo Input Port <I>y delega su trabajo. Sin embargo, expone solo lo que los clientes Controllernecesitan a través de una interfaz de rol o interfaces propiedad de la capa verde. El adaptador no sigue al ISP por sí mismo, pero permite clases más complejas como Controllerdisfrutar del ISP. Esto es útil si hay menos adaptadores que clientes Controllerque los usan y cuando se encuentra en una situación inusual en la que está cruzando el límite de una biblioteca y, a pesar de ser publicada, la biblioteca no dejará de cambiar. Mirándote Firefox. Ahora esos cambios solo rompen sus adaptadores.

Entonces, ¿qué significa esto? Significa honestamente que no me has proporcionado suficiente información para decirte lo que debes hacer. No sé si no seguir a ISP te está causando un problema. No sé si seguirlo no te causaría más problemas.

Sé que estás buscando un principio rector simple. ISP intenta ser eso. Pero deja mucho sin decir. Yo creo en eso. Sí, ¡no obligue a los clientes a depender de métodos que no utilizan, sin una buena razón!

Si tiene una buena razón, como el diseño de algo para aceptar complementos, tenga en cuenta los problemas que no siguen las causas del ISP (es difícil cambiar sin romper clientes) y las formas de mitigarlos (mantener Interactoro al menos Input Port <I>centrarse en uno estable caso de uso).


Gracias por el aporte. Tengo un módulo de prestación de servicios que tiene múltiples clientes. Su espacio de nombres tiene límites lógicamente coherentes, pero las necesidades del cliente atraviesan estos límites lógicos. Por lo tanto, dividir el espacio de nombres sobre la base de los límites lógicos no ayuda con el ISP. Por lo tanto, he dividido el espacio de nombres en función de las necesidades del cliente como se muestra en el diagrama de la pregunta. Pero esto lo hace dependiente de los clientes y una mala forma de acoplar clientes al servicio, ya que los clientes podrían agregarse / eliminarse con relativa frecuencia, pero los cambios en el servicio serán mínimos.
work.bin

Ahora me estoy inclinando hacia el servicio que proporciona una interfaz gruesa, como en su espacio de nombres completo y depende del cliente acceder a estos servicios a través de adaptadores específicos del cliente. En términos de C, sería un archivo de envoltorios de funciones propiedad del cliente. Los cambios en el servicio forzarían la recompilación del adaptador, pero no necesariamente el cliente. .. <
contd

<contd> .. Esto definitivamente mantendrá mínimos los tiempos de compilación y mantendrá el acoplamiento entre el cliente y el servicio 'suelto' a costa del tiempo de ejecución (llamando a una función de envoltura intermedia), aumenta el espacio de código, aumenta el uso de la pila y probablemente más espacio mental (programador) en el mantenimiento de los adaptadores.
work.bin

Mi solución actual satisface mis necesidades ahora, el nuevo enfoque requerirá más esfuerzo y puede estar violando YAGNI. Tendré que sopesar los pros y los contras de cada método y decidir qué camino tomar aquí.
work.bin

1

Entonces este punto:

existent clients are unaffected by the addition (or deletion) of more clients.

Renuncia a que estás violando otro principio importante que es YAGNI. Me importaría cuando tenga cientos de clientes. Pensar en algo por adelantado y luego resultará que no tienes clientes adicionales para este código supera el propósito.

Segundo

 partitioning depends on the nature of clients

¿Por qué su código no usa DI, inversión de dependencia, nada, nada en su biblioteca debería depender de la naturaleza de su cliente?

Eventualmente parece que necesita una capa adicional debajo de su código para satisfacer las necesidades de cosas superpuestas (DI, por lo que su código frontal depende solo de esta capa adicional, y sus clientes dependen solo de su interfaz frontal) de esta manera vence a DRY.
Esto lo odiarías de verdad. Así que haces lo mismo que usas en tu capa de módulo debajo de otro módulo. De esta manera, logrando una capa debajo de usted:

para cualquier cliente, solo los datos y las API que requiere están visibles; el resto del espacio de nombres del módulo está oculto para el cliente, es decir, se adhieren al principio de segregación de interfaz.

si

una declaración no se repite en varios archivos de encabezado, es decir, no viola DRY. El módulo M no tiene dependencias de sus clientes.

si

un cliente no se ve afectado por los cambios realizados en partes del módulo M que no utiliza.

si

Los clientes existentes no se ven afectados por la adición (o eliminación) de más clientes.

si


1

La misma información que se proporciona en la declaración siempre se repite en la definición. Es solo la forma en que funciona este lenguaje. Además, repetir una declaración en varios archivos de encabezado no viola DRY . Es una técnica bastante utilizada (al menos en la biblioteca estándar).

Repetir la documentación o la implementación violaría DRY .

No me molestaría con esto a menos que el código del cliente no esté escrito por mí.


0

Renuncio a mi confusión. Sin embargo, su ejemplo práctico dibuja una solución en mi cabeza. Si puedo expresarlo con mis propias palabras: todas las particiones en el módulo Mtienen una relación exclusiva de muchos a muchos con todos y cada uno de los clientes.

Estructura de la muestra

M.h      // fat header
 - P1    // Partition 1
 - P2    // ... 2
   - P21 // ... 2 section 1
 - P3    // ... 3
C1.c     // Client 1 (Needs to include P1, P3)
C2.c     // ... 2 (Needs to include P2)
C3.c     // ... 3 (Needs to include P1, P21, P3)

Mh

#ifdef P1
#define _PREF_ P1_             // Define Prefix ("PREF") = P1_
 void _PREF_init();            // Some partition specific function
#endif /* P1 */

#ifdef P2
#define _PREF_ P2_
 void _PREF_init();
#endif /* P2 */

#if defined(P21) || defined (P2) // Part 2.1
#define _PREF_ P2_1_
 void _PREF_oddone();
#endif /* P21 */

#ifdef P3
#define _PREF_ P3_
 void _PREF_init();
#endif /* P3 */

Mc

En el archivo Mc, en realidad no tendría que usar los #ifdefs porque lo que coloca en el archivo .c no afecta a los archivos del cliente siempre que las funciones que utilizan los archivos del cliente estén definidas.

#include "M.h"
#define _PREF_ P1_        
void _PREF_init() { ... };

#define _PREF_ P2_
void _PREF_init() { ... }

#define _PREF_ P2_1_
void _PREF_oddone() { ... }

#define _PREF_ P3_
void _PREF_init() { ... }

C1.c

#define P1     // "invite" P1
#define P3     // "invite" P3
#include "M.h" // Open the door, but only the invited come in.

void main()
{
    P1_init();
    //P2_init();
    //P2_1_oddone();
    P3_init();
}

C2.c

#define P2
#include "M.h

void main()
{
    //P1_init();
    P2_init();
    P2_1_oddone();
    //P3_init();
}

C3.c

#define P1
#define P21
#define P3  
#include "M.h" 

void main()
{
    P1_init();
    //P2_init();
    P2_1_oddone();
    P3_init();
}

Nuevamente, no estoy seguro si esto es lo que estás preguntando. Así que tómelo con cuidado.


¿Cómo se ve Mc? ¿Defines P1_init() y P2_init() ?
work.bin

@ work.bin Supongo que Mc se vería como un simple archivo .c con la excepción de definir el espacio de nombres entre funciones.
Sanchke Dellowar

Suponiendo que ambos existen C1 y C2 - lo que hace P1_init()y P2_init()enlace a?
work.bin

En el archivo Mh / Mc, el preprocesador se reemplazará _PREF_con lo que se definió por última vez. Entonces _PREF_init()será P1_init()debido a la última declaración #define. Luego, la siguiente instrucción define establecerá PREF igual a P2_, generando así P2_init().
Sanchke Dellowar
Al usar nuestro sitio, usted reconoce que ha leído y comprende nuestra Política de Cookies y Política de Privacidad.
Licensed under cc by-sa 3.0 with attribution required.