Estoy escribiendo una aplicación C ++. La mayoría de las aplicaciones leen y escriben datos citas de necesarias y esta no es una excepción. Creé un diseño de alto nivel para el modelo de datos y la lógica de serialización. Esta pregunta solicita una revisión de mi diseño con estos objetivos específicos en mente:
Tener una manera fácil y flexible de leer y escribir modelos de datos en formatos arbitrarios: binario sin formato, XML, JSON, et. Alabama. El formato de los datos se debe desacoplar de los datos en sí, así como del código que solicita la serialización.
Para garantizar que la serialización esté tan libre de errores como sea razonablemente posible. La E / S es intrínsecamente riesgosa por una variedad de razones: ¿mi diseño introduce más formas de que falle? Si es así, ¿cómo podría refactorizar el diseño para mitigar esos riesgos?
Este proyecto usa C ++. Ya sea que lo ames o lo odies, el lenguaje tiene su propia forma de hacer las cosas y el diseño tiene como objetivo trabajar con el lenguaje, no en contra de él .
Finalmente, el proyecto está construido sobre wxWidgets . Si bien estoy buscando una solución aplicable a un caso más general, esta implementación específica debería funcionar bien con ese kit de herramientas.
Lo que sigue es un conjunto muy simple de clases escritas en C ++ que ilustran el diseño. Estas no son las clases reales que he escrito parcialmente hasta ahora, este código simplemente ilustra el diseño que estoy usando.
Primero, algunos DAO de muestra:
#include <iostream>
#include <map>
#include <memory>
#include <string>
#include <vector>
// One widget represents one record in the application.
class Widget {
public:
using id_type = int;
private:
id_type id;
};
// Container for widgets. Much more than a dumb container,
// it will also have indexes and other metadata. This represents
// one data file the user may open in the application.
class WidgetDatabase {
::std::map<Widget::id_type, ::std::shared_ptr<Widget>> widgets;
};
A continuación, defino clases virtuales puras (interfaces) para leer y escribir DAO. La idea es abstraer la serialización de datos de los datos en sí ( SRP ).
class WidgetReader {
public:
virtual Widget read(::std::istream &in) const abstract;
};
class WidgetWriter {
public:
virtual void write(::std::ostream &out, const Widget &widget) const abstract;
};
class WidgetDatabaseReader {
public:
virtual WidgetDatabase read(::std::istream &in) const abstract;
};
class WidgetDatabaseWriter {
public:
virtual void write(::std::ostream &out, const WidgetDatabase &widgetDb) const abstract;
};
Finalmente, aquí está el código que obtiene el lector / escritor adecuado para el tipo de E / S deseado. Habría subclases de los lectores / escritores también definidos, pero estos no agregan nada a la revisión del diseño:
enum class WidgetIoType {
BINARY,
JSON,
XML
// Other types TBD.
};
WidgetIoType forFilename(::std::string &name) { return ...; }
class WidgetIoFactory {
public:
static ::std::unique_ptr<WidgetReader> getWidgetReader(const WidgetIoType &type) {
return ::std::unique_ptr<WidgetReader>(/* TODO */);
}
static ::std::unique_ptr<WidgetWriter> getWidgetWriter(const WidgetIoType &type) {
return ::std::unique_ptr<WidgetWriter>(/* TODO */);
}
static ::std::unique_ptr<WidgetDatabaseReader> getWidgetDatabaseReader(const WidgetIoType &type) {
return ::std::unique_ptr<WidgetDatabaseReader>(/* TODO */);
}
static ::std::unique_ptr<WidgetDatabaseWriter> getWidgetDatabaseWriter(const WidgetIoType &type) {
return ::std::unique_ptr<WidgetDatabaseWriter>(/* TODO */);
}
};
Según los objetivos establecidos de mi diseño, tengo una preocupación específica. Las secuencias de C ++ se pueden abrir en modo de texto o binario, pero no hay forma de verificar una secuencia ya abierta. A través del error del programador, podría ser posible proporcionar, por ejemplo, una secuencia binaria a un lector / escritor XML o JSON. Esto podría causar errores sutiles (o no tan sutiles). Preferiría que el código fallara rápidamente, pero no estoy seguro de que este diseño lo haga.
Una forma de evitar esto podría ser descargar la responsabilidad de abrir la transmisión al lector o escritor, pero creo que eso viola SRP y haría que el código sea más complejo. Al escribir un DAO, al escritor no le debe importar a dónde va la transmisión: podría ser un archivo, salida estándar, una respuesta HTTP, un socket, cualquier cosa. Una vez que esa preocupación se encapsula en la lógica de serialización, se vuelve mucho más compleja: debe conocer el tipo específico de flujo y a qué constructor llamar.
Aparte de esa opción, no estoy seguro de cuál sería una mejor manera de modelar estos objetos que sea simple, flexible y que ayude a evitar errores lógicos en el código que lo usa.
El caso de uso con el que debe integrarse la solución es un cuadro de diálogo de selección de archivo simple . El usuario selecciona "Abrir ..." o "Guardar como ..." en el menú Archivo, y el programa abre o guarda la WidgetDatabase. También habrá opciones "Importar ..." y "Exportar ..." para widgets individuales.
Cuando el usuario selecciona un archivo para abrir o guardar, wxWidgets devolverá un nombre de archivo. El controlador que responde a ese evento debe ser un código de propósito general que tome el nombre del archivo, adquiera un serializador y llame a una función para realizar el trabajo pesado. Idealmente, este diseño también funcionaría si otra pieza de código realiza E / S sin archivos, como enviar una WidgetDatabase a un dispositivo móvil a través de un zócalo.
¿Se guarda un widget en su propio formato? ¿Interopera con los formatos existentes? ¡Si! Todas las anteriores. Volviendo al diálogo de archivo, piense en Microsoft Word. Microsoft era libre de desarrollar el formato DOCX como quisiera dentro de ciertas limitaciones. Al mismo tiempo, Word también lee o escribe formatos heredados y de terceros (por ejemplo, PDF). Este programa no es diferente: el formato "binario" del que hablo es un formato interno aún por definir, diseñado para la velocidad. Al mismo tiempo, debe poder leer y escribir formatos estándar abiertos en su dominio (irrelevante para la pregunta) para poder trabajar con otro software.
Finalmente, solo hay un tipo de Widget. Tendrá objetos secundarios, pero estos serán manejados por esta lógica de serialización. El programa nunca cargará Widgets y Sprockets. Este diseño solo tiene que ver con Widgets y WidgetDatabases.