HTML можно преобразовать в word_html для отображения текстовых документов.
HTML можно преобразовать в word_html для отображения текстовых документов.

Всем привет, мы снова встретились, я ваш друг Цюаньчжаньцзюнь.

Бэкэнд проекта использует Springboot и Maven, а внешний интерфейс использует редактор форматированного текста ckeditor. В настоящее время слово, преобразованное из HTML, имеет формат doc, а обработка изображений поддерживает формат docx, поэтому вам необходимо вручную сохранить документ как docx, прежде чем вы сможете заменить изображение.

1. Добавьте зависимости maven

В основном используются следующие зависимости, относящиеся к poi. Для облегчения получения элементов изображения html также используется jsoup:

Язык кода:javascript
копировать
<dependency>    <groupId>org.apache.poi</groupId>    <artifactId>poi</artifactId>    <version>3.14</version></dependency><dependency>    <groupId>org.apache.poi</groupId>    <artifactId>poi-scratchpad</artifactId>    <version>3.14</version></dependency><dependency>    <groupId>org.apache.poi</groupId>    <artifactId>poi-ooxml</artifactId>    <version>3.14</version></dependency><dependency>    <groupId>fr.opensagres.xdocreport</groupId>    <artifactId>xdocreport</artifactId>    <version>1.0.6</version></dependency><dependency>    <groupId>org.apache.poi</groupId>    <artifactId>poi-ooxml-schemas</artifactId>    <version>3.14</version></dependency><dependency>    <groupId>org.apache.poi</groupId>    <artifactId>ooxml-schemas</artifactId>    <version>1.3</version></dependency><dependency>    <groupId>org.jsoup</groupId>    <artifactId>jsoup</artifactId>    <version>1.11.3</version></dependency>

2. Преобразование слова в HTML

Создайте новую статическую папку в каталоге ресурсов проекта Springboot и вставьте в нее текстовый файл temp.docx, который необходимо преобразовать. Поскольку static является файлом ресурсов Springboot по умолчанию, нет необходимости настраивать его отдельно в файле. Если имя изменено на другое, соответствующую настройку необходимо выполнить в файле application.yml.

Преобразование формата документа в html:

Язык кода:javascript
копировать
public static String docToHtml() throws Exception {
    File path = new File(ResourceUtils.getURL("classpath:").getPath());
    String imagePathStr = path.getAbsolutePath() + "\\static\\image\\";
    String sourceFileName = path.getAbsolutePath() + "\\static\\test.doc";
    String targetFileName = path.getAbsolutePath() + "\\static\\test2.html";
    File file = new File(imagePathStr);    if(!file.exists()) {
        file.mkdirs();
    }
    HWPFDocument wordDocument = new HWPFDocument(new FileInputStream(sourceFileName));
    org.w3c.dom.Document document = DocumentBuilderFactory.newInstance().newDocumentBuilder().newDocument();
    WordToHtmlConverter wordToHtmlConverter = new WordToHtmlConverter(document);    //Сохраняем изображение и возвращаем относительный путь к изображению
    wordToHtmlConverter.setPicturesManager((content, pictureType, name, width, height) -> {        try (FileOutputStream out = new FileOutputStream(imagePathStr + name)) {
            out.write(content);
        } catch (Exception e) {
            e.printStackTrace();
        }        return "image/" + name;
    });    wordToHtmlConverter.processDocument(wordDocument);    org.w3c.dom.Document htmlDocument = wordToHtmlConverter.getDocument();    DOMSource domSource = new DOMSource(htmlDocument);    StreamResult streamResult = new StreamResult(new File(targetFileName));    TransformerFactory tf = TransformerFactory.newInstance();    Transformer serializer = tf.newTransformer();    serializer.setOutputProperty(OutputKeys.ENCODING, "utf-8");    serializer.setOutputProperty(OutputKeys.INDENT, "yes");    serializer.setOutputProperty(OutputKeys.METHOD, "html");    serializer.transform(domSource, streamResult);    return targetFileName;
}123456789101112131415161718192021222324252627282930313233

Конвертировать формат docx в html

Язык кода:javascript
копировать
public static String docxToHtml() throws Exception {
    File path = new File(ResourceUtils.getURL("classpath:").getPath());
    String imagePath = path.getAbsolutePath() + "\\static\\image";
    String sourceFileName = path.getAbsolutePath() + "\\static\\test.docx";
    String targetFileName = path.getAbsolutePath() + "\\static\\test.html";

    OutputStreamWriter outputStreamWriter = null;    try {
        XWPFDocument document = new XWPFDocument(new FileInputStream(sourceFileName));
        XHTMLOptions options = XHTMLOptions.create();        // Папка для хранения фотографий
        options.setExtractor(new FileImageExtractor(new File(imagePath)));        // Путь к изображению в html
        options.URIResolver(new BasicURIResolver("image"));
        outputStreamWriter = new OutputStreamWriter(new FileOutputStream(targetFileName), "utf-8");
        XHTMLConverter xhtmlConverter = (XHTMLConverter) XHTMLConverter.getInstance();
        xhtmlConverter.convert(document, outputStreamWriter, options);
    } finally {        if (outputStreamWriter != null) {
            outputStreamWriter.close();
        }
    }    return targetFileName;
}

После успешного преобразования будет создан соответствующий html-файл. Если вы хотите отобразить его во внешнем интерфейсе, просто прочитайте файл напрямую, преобразуйте его в строку и верните во внешний интерфейс.

Язык кода:javascript
копировать
public static String readfile(String filePath) {
    File file = new File(filePath);
    InputStream input = null;    try {
        input = new FileInputStream(file);
    } catch (FileNotFoundException e) {
        e.printStackTrace();
    }
    StringBuffer buffer = new StringBuffer();    byte[] bytes = new byte[1024];    try {        for (int n; (n = input.read(bytes)) != -1;) {
            buffer.append(new String(bytes, 0, n, "utf8"));
        }
    } catch (IOException e) {
        e.printStackTrace();
    }    return buffer.toString();
}

Эффект отображения в редакторе форматированного текста ckeditor:

3. Преобразование HTML в Word

Идея реализации состоит в том, чтобы сначала извлечь все элементы изображения в HTML и равномерно заменить их переменным символом «${imgReplace}». Если изображений несколько, их можно расположить по порядку, а затем создать соответствующий файл документа. (Я пробовал сгенерировать docx непосредственно перед тем, как файл не открывается, и хорошего решения этой проблемы пока не найдено), сохраняем его как docx-файл, а затем заменяем переменные изображениями:

Язык кода:javascript
копировать
public static String writeWordFile(String content) {
       String path = "D:/wordFile";
       Map<String, Object> param = new HashMap<String, Object>();
       if (!"".equals(path)) {
           File fileDir = new File(path);
           if (!fileDir.exists()) {
               fileDir.mkdirs();
           }
           content = HtmlUtils.htmlUnescape(content);
           List<HashMap<String, String>> imgs = getImgStr(content);
           int count = 0;
           for (HashMap<String, String> img : imgs) {
               count++;
               //Замена ручки с“/>”финальныйimgЭтикетка
               content = content.replace(img.get("img"), "${imgReplace" + count + "}");
               //Замена ручки с“>”финальныйimgЭтикетка
               content = content.replace(img.get("img1"), "${imgReplace" + count + "}");
               Map<String, Object> header = new HashMap<String, Object>();
               try {
                   File filePath = new File(ResourceUtils.getURL("classpath:").getPath());
                   String imagePath = filePath.getAbsolutePath() + "\\static\\";
                   imagePath += img.get("src").replaceAll("/", "\\\\");
                   //Если атрибуты ширины и высоты отсутствуют, значение по умолчанию — 400*300
                   if(img.get("width") == null || img.get("height") == null) {
                       header.put("width", 400);
                       header.put("height", 300);
                   }else {
                       header.put("width", (int) (Double.parseDouble(img.get("width"))));
                       header.put("height", (int) (Double.parseDouble(img.get("height"))));
                   }
                   header.put("type", "jpg");
                   header.put("content", OfficeUtil.inputStream2ByteArray(new FileInputStream(imagePath), true));
               } catch (FileNotFoundException e) {
                   e.printStackTrace();
               }
               param.put("${imgReplace" + count + "}", header);
           }
           try {
               // Создайте документ Word в формате doc, который необходимо вручную изменить на docx.
               byte by[] = content.getBytes("UTF-8");
               ByteArrayInputStream bais = new ByteArrayInputStream(by);
               POIFSFileSystem poifs = new POIFSFileSystem();
               DirectoryEntry directory = poifs.getRoot();
               DocumentEntry documentEntry = directory.createDocument("WordDocument", bais);
               FileOutputStream ostream = new FileOutputStream("D:\\wordFile\\temp.doc");
               poifs.writeFilesystem(ostream);
               bais.close();
               ostream.close();
               // Временный файл (файл docx изменен вручную)
               CustomXWPFDocument doc = OfficeUtil.generateWord(param, "D:\\wordFile\\temp.docx");
               //Окончательно сгенерированный текстовый файл с картинками
               FileOutputStream fopts = new FileOutputStream("D:\\wordFile\\final.docx");
               doc.write(fopts);
               fopts.close();
           } catch (Exception e) {
               e.printStackTrace();
           }
       }
       return "D:/wordFile/final.docx";
   }
   //Получаем информацию об элементе изображения в html
   public static List<HashMap<String, String>> getImgStr(String htmlStr) {
       List<HashMap<String, String>> pics = new ArrayList<HashMap<String, String>>();
       Document doc = Jsoup.parse(htmlStr);
       Elements imgs = doc.select("img");
       for (Element img : imgs) {
           HashMap<String, String> map = new HashMap<String, String>();
           if(!"".equals(img.attr("width"))) {
               map.put("width", img.attr("width").substring(0, img.attr("width").length() - 2));
           }
           if(!"".equals(img.attr("height"))) {
               map.put("height", img.attr("height").substring(0, img.attr("height").length() - 2));
           }
           map.put("img", img.toString().substring(0, img.toString().length() - 1) + "/>");
           map.put("img1", img.toString());
           map.put("src", img.attr("src"));
           pics.add(map);
       }
       return pics;
   }

В классе инструмента OfficeUtil я обнаружил, что метод записи в Интернете поддерживает изменение только одного изображения, и при добавлении нескольких изображений будет сообщено об ошибке. Причина в том, что изменился размер прогонов в методеprocessParagraphs. будет сообщено об исключении ArrayList, которое будет повторяться вместе с нами. Удаление элементов в списке вызовет исключение. Решение состоит в том, чтобы скопировать новый Arraylist и выполнить цикл:

Язык кода:javascript
копировать
package com.example.demo.util;  
import java.io.ByteArrayInputStream;import java.io.FileInputStream;import java.io.IOException;import java.io.InputStream;import java.util.ArrayList;import java.util.Iterator;import java.util.List;import java.util.Map;import java.util.Map.Entry;import org.apache.poi.POIXMLDocument;import org.apache.poi.hwpf.extractor.WordExtractor;import org.apache.poi.openxml4j.opc.OPCPackage;import org.apache.poi.xwpf.usermodel.XWPFParagraph;import org.apache.poi.xwpf.usermodel.XWPFRun;import org.apache.poi.xwpf.usermodel.XWPFTable;import org.apache.poi.xwpf.usermodel.XWPFTableCell;import org.apache.poi.xwpf.usermodel.XWPFTableRow;  
/** 
* Применимо к слову 2007
*/  public class OfficeUtil {  
   /** 
    * Генерировать на основе заданных значений параметров и шаблонов word документ 
    * @param param Переменные, которые необходимо заменить 
    * @param template шаблон 
    */  
   public static CustomXWPFDocument generateWord(Map<String, Object> param, String template) {  
       CustomXWPFDocument doc = null;        try {  
           OPCPackage pack = POIXMLDocument.openPackage(template);  
           doc = new CustomXWPFDocument(pack);  
           if (param != null && param.size() > 0) {  
               //обработка параграфа  
               List<XWPFParagraph> paragraphList = doc.getParagraphs();  
               processParagraphs(paragraphList, param, doc);  
               //обработка формы  
               Iterator<XWPFTable> it = doc.getTablesIterator();  
               while (it.hasNext()) {
                   XWPFTable table = it.next();  
                   List<XWPFTableRow> rows = table.getRows();  
                   for (XWPFTableRow row : rows) {  
                       List<XWPFTableCell> cells = row.getTableCells();  
                       for (XWPFTableCell cell : cells) {  
                           List<XWPFParagraph> paragraphListTable =  cell.getParagraphs();  
                           processParagraphs(paragraphListTable, param, doc);  
                       }  
                   }  
               }  
           }  
       } catch (Exception e) {  
           e.printStackTrace();  
       }  
       return doc;  
   }  
   /** 
    * Параграфы процесса 
    * @param paragraphList 
    */  
   public static void processParagraphs(List<XWPFParagraph> paragraphList,Map<String, Object> param,CustomXWPFDocument doc){  
       if(paragraphList != null && paragraphList.size() > 0){  
           for(XWPFParagraph paragraph:paragraphList){                //Интервал, преобразованный из poi, слишком велик, и его необходимо отрегулировать вручную.                if(paragraph.getSpacingBefore() >= 1000 || paragraph.getSpacingAfter() > 1000) {
                   paragraph.setSpacingBefore(0);
                   paragraph.setSpacingAfter(0);
               }                //Устанавливаем интервал слева и справа в слове
               paragraph.setIndentationLeft(0);
               paragraph.setIndentationRight(0);
               List<XWPFRun> runs = paragraph.getRuns();                //Добавлены картинки и изменен размер фрагментов абзаца, чтобы цикл не мог использовать пробеги.
               List<XWPFRun> allRuns = new ArrayList<XWPFRun>(runs);                for (XWPFRun run : allRuns) {
                   String text = run.getText(0);  
                   if(text != null){                        boolean isSetText = false;  
                       for (Entry<String, Object> entry : param.entrySet()) {  
                           String key = entry.getKey();  
                           if(text.indexOf(key) != -1){  
                               isSetText = true;  
                               Object value = entry.getValue();  
                               if (value instanceof String) {//Замена текста  
                                   text = text.replace(key, value.toString());  
                               } else if (value instanceof Map) {//Замена изображения  
                                   text = text.replace(key, "");  
                                   Map pic = (Map)value;  
                                   int width = Integer.parseInt(pic.get("width").toString());  
                                   int height = Integer.parseInt(pic.get("height").toString());  
                                   int picType = getPictureType(pic.get("type").toString());  
                                   byte[] byteArray = (byte[]) pic.get("content");  
                                   ByteArrayInputStream byteInputStream = new ByteArrayInputStream(byteArray);  
                                   try {  
                                       String blipId = doc.addPictureData(byteInputStream,picType);  
                                       doc.createPicture(blipId,doc.getNextPicNameNumber(picType), width, height,paragraph);
                                   } catch (Exception e) {  
                                       e.printStackTrace();  
                                   }  
                               }  
                           }  
                       }  
                       if(isSetText){  
                           run.setText(text,0);  
                       }  
                   }  
               }  
           }  
       }  
   }  
   /** 
    * В зависимости от типа изображения получите соответствующий код типа изображения. 
    * @param picType 
    * @return int 
    */  
   private static int getPictureType(String picType){  
       int res = CustomXWPFDocument.PICTURE_TYPE_PICT;  
       if(picType != null){  
           if(picType.equalsIgnoreCase("png")){  
               res = CustomXWPFDocument.PICTURE_TYPE_PNG;  
           }else if(picType.equalsIgnoreCase("dib")){  
               res = CustomXWPFDocument.PICTURE_TYPE_DIB;  
           }else if(picType.equalsIgnoreCase("emf")){  
               res = CustomXWPFDocument.PICTURE_TYPE_EMF;  
           }else if(picType.equalsIgnoreCase("jpg") || picType.equalsIgnoreCase("jpeg")){  
               res = CustomXWPFDocument.PICTURE_TYPE_JPEG;  
           }else if(picType.equalsIgnoreCase("wmf")){  
               res = CustomXWPFDocument.PICTURE_TYPE_WMF;  
           }  
       }  
       return res;  
   }  
   /** 
    * Запись данных из входного потока в массив байтов 
    * @param in 
    * @return 
    */  
   public static byte[] inputStream2ByteArray(InputStream in,boolean isClose){  
       byte[] byteArray = null;  
       try {  
           int total = in.available();  
           byteArray = new byte[total];  
           in.read(byteArray);  
       } catch (IOException e) {  
           e.printStackTrace();  
       }finally{  
           if(isClose){  
               try {  
                   in.close();  
               } catch (Exception e2) {  
                   System.out.println("Не удалось закрыть поток");  
               }  
           }  
       }  
       return byteArray;  
   }  
}

Я думаю, что причина, по которой word2003 не поддерживает замену изображений, заключается главным образом в том, что объект HWPFDocument версии 2003 объявлен как окончательный, и мы не можем переопределить его метод. Класс, обрабатывающий версию 2007, — это XWPFDocument, который можно унаследовать. Наследуя XWPFDocument и переопределив метод createPicture, можно добиться замены изображения. Ниже приведен соответствующий класс CustomXWPFDocument.

Язык кода:javascript
копировать
package com.example.demo.util;    
import java.io.IOException;  
import java.io.InputStream;  
import org.apache.poi.openxml4j.opc.OPCPackage;  
import org.apache.poi.xwpf.usermodel.XWPFDocument;  
import org.apache.poi.xwpf.usermodel.XWPFParagraph;  
import org.apache.xmlbeans.XmlException;  
import org.apache.xmlbeans.XmlToken;  
import org.openxmlformats.schemas.drawingml.x2006.main.CTNonVisualDrawingProps;  
import org.openxmlformats.schemas.drawingml.x2006.main.CTPositiveSize2D;  
import org.openxmlformats.schemas.drawingml.x2006.wordprocessingDrawing.CTInline;  
/** 
* Настроить XWPFDocument и переопределить метод createPicture() 
*/  public class CustomXWPFDocument extends XWPFDocument {    
   public CustomXWPFDocument(InputStream in) throws IOException {    
       super(in);    
   }    
   public CustomXWPFDocument() {    
       super();    
   }    
   public CustomXWPFDocument(OPCPackage pkg) throws IOException {    
       super(pkg);    
   }    
   /** 
    * @param ind 
    * @param width Ширина 
    * @param height высокий 
    * @param paragraph  абзац 
    */  
   public void createPicture(String blipId, int ind, int width, int height,XWPFParagraph paragraph) {    
       final int EMU = 9525;    
       width *= EMU;    
       height *= EMU;    
       CTInline inline = paragraph.createRun().getCTR().addNewDrawing().addNewInline();    
       String picXml = ""    
               + "<a:graphic xmlns:a=\"http://schemas.openxmlformats.org/drawingml/2006/main\">"    
               + "   <a:graphicData uri=\"http://schemas.openxmlformats.org/drawingml/2006/picture\">"    
               + "      <pic:pic xmlns:pic=\"http://schemas.openxmlformats.org/drawingml/2006/picture\">"    
               + "         <pic:nvPicPr>" + "            <pic:cNvPr id=\""    
               + ind    
               + "\" name=\"Generated\"/>"    
               + "            <pic:cNvPicPr/>"    
               + "         </pic:nvPicPr>"    
               + "         <pic:blipFill>"    
               + "            <a:blip r:embed=\""    
               + blipId    
               + "\" xmlns:r=\"http://schemas.openxmlformats.org/officeDocument/2006/relationships\"/>"    
               + "            <a:stretch>"    
               + "               <a:fillRect/>"    
               + "            </a:stretch>"    
               + "         </pic:blipFill>"    
               + "         <pic:spPr>"    
               + "            <a:xfrm>"    
               + "               <a:off x=\"0\" y=\"0\"/>"    
               + "               <a:ext cx=\""    
               + width    
               + "\" cy=\""    
               + height    
               + "\"/>"    
               + "            </a:xfrm>"    
               + "            <a:prstGeom prst=\"rect\">"    
               + "               <a:avLst/>"    
               + "            </a:prstGeom>"    
               + "         </pic:spPr>"    
               + "      </pic:pic>"    
               + "   </a:graphicData>" + "</a:graphic>";    
       inline.addNewGraphic().addNewGraphicData();    
       XmlToken xmlToken = null;    
       try {    
           xmlToken = XmlToken.Factory.parse(picXml);    
       } catch (XmlException xe) {    
           xe.printStackTrace();    
       }    
       inline.set(xmlToken);   
       inline.setDistT(0);      
       inline.setDistB(0);      
       inline.setDistL(0);      
       inline.setDistR(0);      
       CTPositiveSize2D extent = inline.addNewExtent();    
       extent.setCx(width);    
       extent.setCy(height);    
       CTNonVisualDrawingProps docPr = inline.addNewDocPr();      
       docPr.setId(ind);      
       docPr.setName("Изображение" + ind);      
       docPr.setDescr("Тест");   
   }    
}

Вышеуказанное представляет собой взаимное преобразование HTML и Word через POI. Проблема, связанная с невозможностью преобразования HTML в читаемый документ, еще не решена. Если у вас есть хорошее решение, вы можете поделиться им ~~~.

Заявление об авторских правах: Содержание этой статьи добровольно предоставлено пользователями Интернета, а мнения, выраженные в этой статье, представляют собой только точку зрения автора. Этот сайт предоставляет только услуги по хранению информации, не имеет никаких прав собственности и не принимает на себя соответствующие юридические обязательства. Если вы обнаружите на этом сайте какое-либо подозрительное нарушение авторских прав/незаконный контент, отправьте электронное письмо, чтобы сообщить. После проверки этот сайт будет немедленно удален.

Издатель: Full stack программист и руководитель стека, укажите источник для перепечатки: https://javaforall.cn/182887.html Исходная ссылка: https://javaforall.cn

boy illustration
Неразрушающее увеличение изображений одним щелчком мыши, чтобы сделать их более четкими артефактами искусственного интеллекта, включая руководства по установке и использованию.
boy illustration
Копикодер: этот инструмент отлично работает с Cursor, Bolt и V0! Предоставьте более качественные подсказки для разработки интерфейса (создание навигационного веб-сайта с использованием искусственного интеллекта).
boy illustration
Новый бесплатный RooCline превосходит Cline v3.1? ! Быстрее, умнее и лучше вилка Cline! (Независимое программирование AI, порог 0)
boy illustration
Разработав более 10 проектов с помощью Cursor, я собрал 10 примеров и 60 подсказок.
boy illustration
Я потратил 72 часа на изучение курсорных агентов, и вот неоспоримые факты, которыми я должен поделиться!
boy illustration
Идеальная интеграция Cursor и DeepSeek API
boy illustration
DeepSeek V3 снижает затраты на обучение больших моделей
boy illustration
Артефакт, увеличивающий количество очков: на основе улучшения характеристик препятствия малым целям Yolov8 (SEAM, MultiSEAM).
boy illustration
DeepSeek V3 раскручивался уже три дня. Сегодня я попробовал самопровозглашенную модель «ChatGPT».
boy illustration
Open Devin — инженер-программист искусственного интеллекта с открытым исходным кодом, который меньше программирует и больше создает.
boy illustration
Эксклюзивное оригинальное улучшение YOLOv8: собственная разработка SPPF | SPPF сочетается с воспринимаемой большой сверткой ядра UniRepLK, а свертка с большим ядром + без расширения улучшает восприимчивое поле
boy illustration
Популярное и подробное объяснение DeepSeek-V3: от его появления до преимуществ и сравнения с GPT-4o.
boy illustration
9 основных словесных инструкций по доработке академических работ с помощью ChatGPT, эффективных и практичных, которые стоит собрать
boy illustration
Вызовите deepseek в vscode для реализации программирования с помощью искусственного интеллекта.
boy illustration
Познакомьтесь с принципами сверточных нейронных сетей (CNN) в одной статье (суперподробно)
boy illustration
50,3 тыс. звезд! Immich: автономное решение для резервного копирования фотографий и видео, которое экономит деньги и избавляет от беспокойства.
boy illustration
Cloud Native|Практика: установка Dashbaord для K8s, графика неплохая
boy illustration
Краткий обзор статьи — использование синтетических данных при обучении больших моделей и оптимизации производительности
boy illustration
MiniPerplx: новая поисковая система искусственного интеллекта с открытым исходным кодом, спонсируемая xAI и Vercel.
boy illustration
Конструкция сервиса Synology Drive сочетает проникновение в интрасеть и синхронизацию папок заметок Obsidian в облаке.
boy illustration
Центр конфигурации————Накос
boy illustration
Начинаем с нуля при разработке в облаке Copilot: начать разработку с минимальным использованием кода стало проще
boy illustration
[Серия Docker] Docker создает мультиплатформенные образы: практика архитектуры Arm64
boy illustration
Обновление новых возможностей coze | Я использовал coze для создания апплета помощника по исправлению домашних заданий по математике
boy illustration
Советы по развертыванию Nginx: практическое создание статических веб-сайтов на облачных серверах
boy illustration
Feiniu fnos использует Docker для развертывания личного блокнота Notepad
boy illustration
Сверточная нейронная сеть VGG реализует классификацию изображений Cifar10 — практический опыт Pytorch
boy illustration
Начало работы с EdgeonePages — новым недорогим решением для хостинга веб-сайтов
boy illustration
[Зона легкого облачного игрового сервера] Управление игровыми архивами
boy illustration
Развертывание SpringCloud-проекта на базе Docker и Docker-Compose