现代化历险记:策略+将COBOL转换为Java的示例

在Keyhole Software,我们在很大程度上是一家现代化公司。 我们拥有一些顾问,他们专门研究将旧的代码迁移到新的,翻新的残旧代码库,并为大多数已经被供应商锁定的企业设计更光明的未来。

作为这些经验的有趣的副作用,我们遇到了一些重复的模式和策略,以了解如何实现遗留系统的现代化。

在此博客中,我们将介绍一种看起来非常流行的策略Re-Platforming ,并将用我们开发的Keyhole Labs产品进行演示。 这篇文章的基本流程是:

  • 现代化概论
  • 重新平台战略的高层定义
  • 使用Keyhole语法树变压器进行重新平台化的示例
  • 结束语
  • 摘要

“我要进行一次现代化……不等,也许要两个……”

当我们第一次围绕现代化主题吸引客户时,我们会看到他们在此过程中实际想要完成的工作的定义各不相同。 这些范围包括从大型机应用程序退出,从ESB /经典SOA架构过渡到基于云的PaaS实施,再到从供应商锁定/分层架构迁移到DevOps /微服务架构。

随着十年前最近更新其技术堆栈的公司遇到成功运营或成长的一些关键问题,所有这些情况的发生频率越来越高:

  • 部署问题:所有内容都必须作为一个单元进行部署,这是一个痛苦的过程,并且/或者与其所有基础架构紧密耦合
  • 可伸缩性问题:可伸缩性的垂直极限受到了打击–这意味着机器无法足够快地变大以应对容量的增加
  • 性能问题:系统中消息/事务的数量正在增加延迟,在某些情况下会导致级联故障
  • 资源问题:使用此系统的工程师和计算机科学家本来就不在或即将退休,并且不再在学校教授编程语言


因此,请进入现代化计划。 让我们首先回顾一下Re-Platforming策略及其优缺点。

这就像修理我的靴子吗?”

重新平台有时也称为升降机。 从根本上讲,重新平台是将一种代码语言转换为另一种代码语言,即进行翻译。 作为现代化策略,这意味着将较旧的代码语言转换为较新的代码语言。

由于各种原因,大型机在一些大型企业中仍然很普遍,因此,像COBOL这样的旧代码库也仍然存在。 摆脱这些较旧的代码库和大型机的原因通常是以下之一:

  • 资源问题(如上所述):大型机程序员正变得稀缺,并且这些语言集在任何现代课程中都没有涉及。 招聘新开发人员比较困难,尤其是在快速变化和技术选择日趋广泛的情况下。 更少的员工愿意研究某些被淘汰的技术。
  • 对于任何规模的企业而言,大型机都是一笔巨额费用,而垂直增长则是唯一的增长选择,有时这是昂贵的。
    大多数现代体系结构中常见的灾难恢复和高可用性策略可能会对大型机造成成本上的限制。
  • 程序语言构造(OOP,函数式编程,反应式编程等)中不能轻易利用更新的编程模式-因此限制了选择的范围。
  • SDLC的变化–即从瀑布过渡到敏捷过程以保持竞争力。

因此,长话短说-说“重新平台化”实际上是什么意思?

在此过程中,将分析较旧的代码库以确定代码库中的语法或模式。

一旦定义了语法树或一组代码模式,便会通过某些单步或多步编译器软件运行原始代码库(即COBOL),以将旧代码转换为所需的最终状态-通常是Java, C#或更高版本的等效语言。

从业务角度来看,这可能非常有吸引力。 无需让产品所有者和开发人员组成团队来逐步用新语言重新编写每个旧代码位,此方法有望通过几次按钮操作来完成所有繁重的工作。 听起来不错!

好吧,教授,请稍等片刻–在继续之前,这种方法存在一些固有的问题,需要提到。 最难的事情是:

代码翻译不一定能解决技术问题!

在某些情况下,这些旧代码库可能已经存在了20多年。 这可能是20多年的错误决定,或者是特定于大型机的决策被纳入您的代码中。

所有的翻译过程都将为您提供那些潜在的代码地雷,这些地雷现在正在使用新的语言,它们可能无法从大型机的慷慨和能力中受益。

代码看起来可能比大型机差!

在此过程中运行代码有时可能最终看起来像是被削木机扔掉了。 一些大型机和遗留代码构造/行为无法很好地转换或根本无法转换成较新的代码库。 (例如:在一个最近的客户处,我们找到了一个示例,其中在一个代码库中x / 0的数学运算返回0!)

即使代码可以转换并看起来不错,这也不意味着它将始终运行!

仅仅翻译成另一种语言并不能保证执行成功–一次成功的翻译通常并不意味着语法错误。

可能需要一些调整,其他基础结构来帮助代码工作和构建。

运行中!=执行中

同样,如果我们让它运行并构建起来,那么在我们的试点转换中,一切似乎都很棒。 一旦我们处理了数百万笔交易并进行了记录,您就会发现存储桶中的所有漏洞。

这个过程很可能不会降低复杂性!

在此过程中,您很可能会从处理其所有复杂性的过程(在某些情况下几乎没有或没有I / O损失)过渡到对资源的​​慷慨解囊。

将这些代码库转移到较新的语言中,通常涉及一些关注点分离:

  • 数据访问层与嵌入式SQL语句相对
  • 与基于文件的数据存储相反的潜在新关系数据存储
  • 与UI代码相反的表示层
  • 服务/业务逻辑层作为其自己的层

可能需要一些其他基础架构来处理大型机免费提供的功能

像消息传递,容器或虚拟机编排,队列和AD / LDAP / OAuth集成等。

因此,现在您可能会感觉就像您刚走进一家医药商业广告,我在那儿说道:

“这种小药丸可以解决您所有的背痛和黄色的趾甲问题。 潜在的副作用可能包括呕吐,眼睛和/或耳朵出血,暂时性视力丧失,自发性脱发以及对字母“ A”的痛苦敏感性。”

但是,如果您专注于以下方面,这可能是一个成功的旅程:

  1. 如果您有使用旧版/大型机语言编写的大型代码库,则此过程可以将您的代码库很快地转变为更现代的代码库。
  2. 从这时起,您的开发团队就可以通过自己现在可以读取代码的简单事实,来以所需的最终状态更新应用程序。

如果您选择一个可以将语法树用于初始转换的过程,则……

您只需调整语法并重新运行即可快速调整和调整更新后的输出。

有时,基于模式的转换是唯一的选择。 但是,在许多情况下,可以生成语法树–然后您只需调整语法即可,而不是一次性调整输出或单个模式。

Keyhole的Syntax Tree Transformer及其专有的COBOL语法分析器基于语法,并且可以做到这一点!

这是使您分阶段实施的可行选择。

特别是如果您的组织没有人员来处理可能将数千个程序转换为新堆栈的工作。

通过在短时间内转换所有旧代码,您可以更快地淘汰旧技术。 然后,您可以重新分配这些资源,以分析和重新编写或清理具有最大业务价值和ROI的代码部分。

这使组织可以对业务真正重要的方面做出更有目的的决策。

提供对您的代码库中应用的业务逻辑的宝贵见解和分析。

在某些情况下,业务逻辑可能与代码库一样古老,并且不再适用。 大多数客户从中发现了很多价值,最终仅通过分析活动就将他们的代码库减少了10-25%。

有机会引入DevOps作为转换的一部分。

根据所需的代码最终状态,在转换过程之外,有机会引入DevOps作为转换的一部分可能会有所帮助。 有时,“有能力”站起来使用一些工具或实施新流程最终会成为注入最佳实践的机会,而无需经历过多繁琐的工作或走闸。

这些新的流程和工具可以被业务的其他领域所利用,并通过提高敏捷性和引起某些文化转变来增加价值。

这个过程可以是短期预算双赢。

由于可以快速转换和淘汰大型机和较旧的技术,因此可以收回资本支出和维护成本。

使代码进入转换后状态的总开发成本通常比手动团队重写要小。

需要注意的是,从长远来看,这可能是一项更昂贵的工作,因为现在新语言和基础架构中的代码量很大–可能需要新的/额外的资源来维护和扩展代码库。 –但是至少您应该能够找到它们!

该策略的要旨是:

如果您确定自己意识到该流程实际上可以做什么,并且选择了一个强大的基于语法的工具(例如Keyhole语法树变形器和我们的Parser –只是说出来),那么您将获得非常可预测的结果,从而可以节省预算和时间胜。

既然我们已经了解了实施此策略的定义和利弊,那么实际上让我们轻描淡写吧。 本文的用例将使用Keyhole语法树转换器从COBOL转换为JAVA。

“让我们已经重新平台!”

为了开始这个示例,我们将从COBOL的样本位开始,该样本已被我们的专有语法解析器转换为JSON语法树。 COBOL程序仅读取DB2数据存储并返回员工列表。 我们将不会显示COBOL到JSON的实际转换-而是从已经转换的COBOL程序开始。

(对不起,这是博客文章的秘诀–因此,我们将以这种烹饪表演的方式,从昨晚已经准备好的火鸡开始!如果您对组织的流程感兴趣或想要演示–请与我们联系 )。

首先,我们需要介绍几个设置项目:

  • 对于此示例,您将需要克隆此存储库: https : //github.com/in-the-keyhole/khs-syntax-tree-transformer
  • 您将需要使用支持Docker的机器(Windows 10,各种版本的Linux,Mac)。 这是针对DB2示例的,如果您不想弄混Docker,则在回购中有一个简单的COBOL示例。
  • 这是一个人为的例子! 它并不意味着可以治愈任何疾病或在任何生产环境中使用! 它旨在演示该机制,并说明如何从语法树到Java应用程序。

好的,让我们开始吧!

步骤1:
克隆存储库后,将其作为Maven项目导入Eclipse,STS或Intellij。

第二步:
使用JSON输入文件的命令行参数和发出的Java包名称执行main方法。 像这样:

这将在项目目录中生成一个发出的Program.java program

package khs.res.example.Programpublic class Program   {private Double CONST-PI = null;  private Double WORK-1 = 0;  private Double WORK-2 = 0;  private Double PRINT-LINE = null;  public void static main(String[] args) {Program job = new Program ();job.A-PARA ();}   public void A-PARA () {WORK-1 = 123.46WORK-2 = WORK-2+2WORK-2 = WORK-3*3C-PARA()}public void B-PARA () {CONST-PI = Math.PI;EDT-ID = ZERO}public void C-PARA () {B-PARA()}}

以下是程序将使用的秘密酱汁分析器创建的输入demo.json

{"name" : "Program","typeName" : "CLASS","variables" : [ {"name" : "CONST-PI","typeName" : "VARIABLE","value" : null,"isLocal" : false,"isWorking" : true,"isArray" : false,"fileLevel" : null,"variables" : [ ]}, {"name" : "WORK-1","typeName" : "VARIABLE","value" : "ZERO","isLocal" : false,"isWorking" : true,"isArray" : false,"fileLevel" : null,"variables" : [ ]}, {"name" : "WORK-2","typeName" : "VARIABLE","value" : "ZERO","isLocal" : false,"isWorking" : true,"isArray" : false,"fileLevel" : null,"variables" : [ ]}, {"name" : "PRINT-LINE","typeName" : "VARIABLE","value" : null,"isLocal" : false,"isWorking" : true,"isArray" : true,"fileLevel" : null,"variables" : [ {"name" : "EDT-ID","typeName" : "VARIABLE","value" : "SPACES","isLocal" : false,"isWorking" : true,"isArray" : false,"fileLevel" : null,"variables" : [ ]}, {"name" : "FILLER","typeName" : "VARIABLE","value" : "' Perimeter '","isLocal" : false,"isWorking" : true,"isArray" : false,"fileLevel" : null,"variables" : [ ]}, {"name" : "EDT-3-15-CIR","typeName" : "VARIABLE","value" : null,"isLocal" : false,"isWorking" : true,"isArray" : false,"fileLevel" : null,"variables" : [ ]}, {"name" : "FILLER","typeName" : "VARIABLE","value" : "' Radius '","isLocal" : false,"isWorking" : true,"isArray" : false,"fileLevel" : null,"variables" : [ ]}, {"name" : "EDT-3-15-RAD","typeName" : "VARIABLE","value" : null,"isLocal" : false,"isWorking" : true,"isArray" : false,"fileLevel" : null,"variables" : [ ]}, {"name" : "FILLER","typeName" : "VARIABLE","value" : "' Pi '","isLocal" : false,"isWorking" : true,"isArray" : false,"fileLevel" : null,"variables" : [ ]}, {"name" : "EDT-1-15-PI","typeName" : "VARIABLE","value" : null,"isLocal" : false,"isWorking" : true,"isArray" : false,"fileLevel" : null,"variables" : [ ]} ]} ],"functions" : [ {"name" : "A-PARA","typeName" : "FUNCTION","methods" : [ {"name" : "123.46TOWORK-1","typeName" : "METHOD","type" : {"name" : null,"typeName" : "MOVE","varName" : "WORK-1","value" : "123.46"}}, {"name" : "2TOWORK-2","typeName" : "METHOD","type" : {"typeName" : "ADD","value" : "2","var1" : "WORK-2","var2" : null}}, {"name" : "3GIVINGWORK-3","typeName" : "METHOD","type" : {"typeName" : "MULTI","value" : "3","var1" : "WORK-2","var2" : "WORK-3"}}, {"name" : "C-PARA","typeName" : "METHOD","type" : {"name" : "C-PARA","typeName" : "CALL"}} ]}, {"name" : "B-PARA","typeName" : "FUNCTION","methods" : [ {"name" : "PITOCONST-PI","typeName" : "METHOD","type" : {"name" : null,"typeName" : "MOVE","varName" : "CONST-PI","value" : "PI"}}, {"name" : "ZEROTOEDT-ID","typeName" : "METHOD","type" : {"name" : null,"typeName" : "MOVE","varName" : "EDT-ID","value" : "ZERO"}} ]}, {"name" : "C-PARA","typeName" : "FUNCTION","methods" : [ {"name" : "B-PARA","typeName" : "METHOD","type" : {"name" : "B-PARA","typeName" : "CALL"}} ]} ]
}

DB2示例

现在,在持久性方面迈出了一步,我们将转换简单的DB2程序以演示使用DB2 Express的Java代码。

这是示例DB2 Cobol应用程序:

* --------------------------------------------------------------
* Selects a single employee into a record's detail fields, and
* then displays them by displaying the record.
*
* Demonstrates Cobol-to-Java translation of a DB2 SELECT INTO
* the detail fields of a parent record.
*
* Java has no native notion of a record aggregate. A SQL
* SELECT INTO similarly lacks a record construct.
*
* Lou Mauget, January 31, 2017
* --------------------------------------------------------------IDENTIFICATION DIVISION.PROGRAM-ID. COBOLDB2.DATA DIVISION.WORKING-STORAGE SECTION.EXEC SQLINCLUDE SQLCAEND-EXEC.EXEC SQLINCLUDE EMPLOYEEEND-EXEC.EXEC SQL BEGIN DECLARE SECTIONEND-EXEC.01 WS-EMPLOYEE-RECORD.05 WS-EMPNO PIC XXXXXX.05 WS-LAST-NAME PIC XXXXXXXXXXXXXXX.05 WS-FIRST-NAME PIC XXXXXXXXXXXX.EXEC SQL END DECLARE SECTIONEND-EXEC.PROCEDURE DIVISION.EXEC SQLSELECT EMPNO, LASTNAME, FIRSTNMEINTO :WS-EMPNO, :WS-LAST-NAME, :WS-FIRST-NAME FROM EMPLOYEEWHERE EMPNO=200310END-EXEC.IF SQLCODE = 0DISPLAY WS-EMPLOYEE-RECORDELSEDISPLAY 'Error'END-IF.STOP RUN.

使用我们的Antlr解析器已将其转换为JSON语法树。 使用khs.transformer.CommandLine.java对象将语法树JSON转换为以下Java应用程序。

/*** Java source, file COBOLDB2.java generated from Cobol source, COBOLDB2.cbl** @version 0.0.3* @author Keyhole Software LLC*/
public class COBOLDB2   {private static Logger Log = LoggerFactory.getLogger("COBOLDB2");// SQLCAprivate int sqlcode;// Level 05private String v_ws_empno;// Level 05private String v_ws_last_name;// Level 05private String v_ws_first_name;// Level 01private InItem[] v_ws_employee_record = new InItem[]{ () -> v_ws_empno, () -> v_ws_last_name, () -> v_ws_first_name };// Procedure division entry:public static void main(String[] args) {try {COBOLDB2 instance = new COBOLDB2();instance.m_procdiv();} catch (Exception e) {e.printStackTrace();}}private void m_procdiv () throws Exception {final String sql = "SELECT EMPNO, LASTNAME, FIRSTNME FROM EMPLOYEE WHERE EMPNO=200310";final OutItem[] into = new OutItem[]{s -> v_ws_empno = (String)s,s -> v_ws_last_name = (String)s,s -> v_ws_first_name = (String)s};sqlcode = Database.getInstance().selectInto( sql, into );if ( sqlcode == 0 ) {Display.display( v_ws_employee_record );} else {Display.display( "Error" );}// EXIT ...System.exit(0);}}

以下步骤描述了如何设置DB2以执行该应用程序。 DB2 Express在Docker容器中运行。 没有池连接。 这只是一个演示。 ☺

Docker DB2 Express容器

确保您有权访问Docker 。

使用此Docker映像进行初始DB2绑定: https : //hub.docker.com/r/ibmcom/db2express-c/

docker run --name db2 -d -it -p 50000:50000 -e DB2INST1_PASSWORD=db2inst1-pwd -e LICENSE=accept -v  $(pwd)/dbstore:/dbstore ibmcom/db2express-c:latest db2start
docker exec -it db2 bash

创建运行中的Docker DB2 Express容器守护程序,并登录到bash会话,如上所示。

发行su db2inst1
发出db2sampl (花一些时间来创建数据库“ SAMPLE”)。

[db2inst1@6f44040637fc /]$ db2samplCreating database "SAMPLE"...Connecting to database "SAMPLE"...Creating tables and data in schema "DB2INST1"...Creating tables with XML columns and XML data in schema "DB2INST1"...'db2sampl' processing complete.

完成烟雾测试安装后:

以Java运行: khs.transformer.CheckDb2Connection

控制台上显示以下内容:

一旦在Docker容器上安装并验证了数据库,就可以执行转换后的Cobol / DB2到Java程序khs.res.db2demo.COBOLDB2.java 。 一旦执行该程序,我们将得到以下输出:


基本上是魔术!
同样,这是人为的,但是我们采用了一个COBOL程序,该程序被转换为JSON语法树,然后最终得到了一个Java应用程序,该程序从DB2数据库返回了我们的数据–正是COBOL程序所做的!

结论

希望在本文和上面的示例之后,我们都对重新平台战略有了更好的了解。 此策略是否适合您的组织是另一个话题(顺便提一下,我们很乐意与您联系)。

我要打动的重点是,即使听起来很酷,代码转译也不是遗留代码狼的灵丹妙药! 我还想告诉您,虽然充满了危险,但是如果正确使用正确的方法并使用可靠的工具(ahem – 锁Kong语法树变压器和解析),这可能是一个非常可行的策略。

“那么,我们在这里完成了什么?”

总而言之,我们涵盖了以下内容:

  • 现代化简介
  • 回顾现代化的重平台战略
  • 使用Keyhole语法树变压器的重新平台示例
  • 关于此策略的价值/风险的其他总结想法

我们肯定希望您能像我们一样喜欢它。 请,如果您有任何问题或反馈,请在下面发布或直接与我们联系。

谢谢,记得负责任地现代化!

资源/参考:该演示也可以在这里找到: https : //github.com/in-the-keyhole/khs-syntax-tree-transformer

翻译自: https://www.javacodegeeks.com/2017/02/adventures-modernization-strategy-example-converting-cobol-java.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/350352.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

String类的特点和使用步骤

概述 java.lang.String 类代表字符串。Java程序中所有的字符串文字(例如 "abc" )都可以被看作是实现此类的实例 类 String 中包括用于检查各个字符串的方法,比如用于比较字符串,搜索字符串,提取子字符串以及…

Ostu最大类间差方法

Ostu方法又名最大类间差方法,通过统计整个图像的直方图特性来实现全局阈值T的自动选取,其算法步骤为: 1) 先计算图像的直方图,即将图像所有的像素点按照0~255共256个bin,统计落在每个bin的像素点数量 2) 归一化直方图…

pads 文本不能修改_斜框检测经典网络(文本)- R2CNN

论文题目:R2CNN:Rotational Region CNN for Orientation Robust Scene Text Detection论文来源:2017CVPR论文地址:https://arxiv.org/abs/1706.09579论文代码:https://github.com/DetectionTeamUCAS/R2CNN_Faster-RCNN…

【题解】炮兵阵地

【题解】炮兵阵地 luogu 题解 暴力状压记录上一行和上上行信息,暴力判断是否存在不合法的阵地,暴力滚动数组,暴力统计,暴力转移即可。 看了yyb代码,我感觉我不曾写过代码... 加入说不要那么暴力也可以,可以…

matlab reshape矩阵维度变换

B reshape(A,m,n) 将矩阵A的元素返回到一个mn的矩阵B。如果A中没有mn个元素则返回一个错误。 B reshape(A,m,n,p,...) or B reshape(A,[m n p ...]) 把A中元素进行重塑成mnp…的矩阵,特别地,指定的维数mnp…的积必须与prod(size(A))相同。 …

使用Arquillian测试Spring Data + Spring Boot应用程序(第2部分)

在上一篇文章中 ,我写了关于如何使用Arquillian Cube和 Docker一起测试Spring Data应用程序的信息。 测试看起来像: RunWith(SpringRunner.class) SpringBootTest(classes PingPongController.class, webEnvironment RANDOM_PORT) ContextConfigurati…

OpenCV circle图像上画圆

OpenCV中circle与rectangle函数显示,只不过rectangle在图像中画矩形,circle在图像中画圆。 void circle(Mat img, Point center, int radius, Scalar color, int thickness1, int lineType8, int shift0) img为源图像 center为画圆的圆心坐标 radius…

如何得到长整数逆序后的整数

目录 一 如何得到长整数逆序后的整数注:原创不易,转载请务必注明原作者和出处,感谢支持! 一 如何得到长整数逆序后的整数 最近在写代码遇到这样一个问题:如何得到得到一个长整型数逆序后的长整型数?比如输入…

python网页表格读取_是否可以读取网页html表格数据?

我目前正在考虑一些自动读取网页数据。因此,有没有可能从网页中读取以下类型的表格:excel应该有一个值name of condion,Operator and Expressions。在编辑>>> from urllib import urlopen>>> from bs4 import BeautifulSoup>>&g…

matlab exist()判断目录文件是否存在

if exist(abc.doc,file)0error(display(文件不存在));end判断当前目录中是否存在x文件夹,若不存在则创建if exist(x,dir)0mkdir(x);end

20190608笔试题のCSS-属性继承

以下的CSS属性哪些可以继承?(单选) A. font-sizeB. marginC. widthD. padding emmm,这题答案是A,看到这题我是能选对的,但又不由让我想到一件事情,在全部的CSS属性里有哪些是不可被继承…

群晖套件来源_群晖安装IPKG包管理器及第三方社区安装包步骤

运行环境为群晖DS1517,系统版本为DSM6.2.3。群晖虽然是基于Linux开发的,但与Red Hat或者Ubuntu这样的公开发行版不同,群晖除了官方的图形化包管理器外,并没有提供类似yum或者apt-get等类似包管理器。为了通过命令行安装诸如nano 等…

tf.app.flags和tf.app.run的使用

tf.app.flags和tf.app.run的使用 tf.app.flags主要用于处理命令行参数的解析工作,其实可以理解为一个封装好了的argparse包(argparse是一种结构化的数据存储格式,类似于Json、XML)。 我们通过tf.app.flags来调用这个flags.py文件…

redis 持久化性能_高性能持久消息

redis 持久化性能总览 尽管有许多可用于Java的高性能消息传递系统,但大多数都避免引用基准,包括持久消息传递和消息的序列化/反序列化。 这样做有多种原因。 1)您并不总是需要或想要持久消息2)您希望使用自己的序列化选项。 避免使…

python标签使用教程_怎样用Python做标签云

怎样用Python做标签云标签云是比较直观的频率分布表现方式,很多网站和APP在年度盘点和总结时会使用。Python生成标签云有一个比较易用的库 pytagcloud。Python做标签云的具体方法如下:1、导入头文件from pytagcloud import create_tag_image, make_tagsf…

057-while循环

<?php$x1; //初始化变量while($x<5){ //执行while循环echo "$x<br />";$x;} ?> 转载于:https://www.cnblogs.com/tianpan2019/p/10995180.html

OpenCV显示中文汉字,未使用CvxText和FreeType库

OpenCV显示中文汉字&#xff0c;未使用CvxText和FreeType库 采用windows的GDI显示系统的TrueType字体&#xff0c;没有封装&#xff0c;就两个函数&#xff0c;分成了h和cpp文件&#xff0c;可以自己编辑文件名和函数名&#xff0c;亦可以直接将cpp的代码复制到你需要的程序中…

使用Arquillian测试Spring Data + Spring Boot应用程序(第1部分)

Spring Data的使命是为数据访问提供一个熟悉且一致的&#xff0c;基于Spring的编程模型&#xff0c;同时仍保留基础数据存储的特​​殊特征。 它提供了与一些后端技术的集成&#xff0c;例如JPA&#xff0c;Rest&#xff0c;MongoDB&#xff0c;Neo4J或Redis。 因此&#xff0…

生日祝福小程序_广告配音剪映零基础入门教程第二十六篇:如何给朋友制作生日祝福视频...

经常听到小伙伴问到生日祝福视频怎么做&#xff0c;当然我想既然要为他人做生日祝福视频&#xff0c;那么这个人必定是自己身边比较重要的人&#xff0c;而生日又是每个人都是非常重要的&#xff0c;在这种充满意义的时刻&#xff0c;我们想给自己极其重要的人送上一份祝福&…

putty WinScp 免密登录远程 Linux

该方法的原理是预先生成一对公钥和私钥&#xff0c;私钥以文件的形式保存在本地&#xff0c;公钥保存在远程机器上。这样每次登录只需指定私钥文件&#xff0c;远程机器通过比对公钥和私钥来验证登录的合法性。 Putty 免密登录 第一步 生成公钥/私钥对 使用 putty 安装目录中的…