侯体宗的博客
  • 首页
  • Hyperf版
  • beego仿版
  • 人生(杂谈)
  • 技术
  • 关于我
  • 更多分类
    • 文件下载
    • 文字修仙
    • 中国象棋ai
    • 群聊
    • 九宫格抽奖
    • 拼图
    • 消消乐
    • 相册

用注解编写创建表的SQL语句

数据库  /  管理员 发布于 5年前   343

今晚读了think in java 的章节,感觉很不错,我就敲了下来,贴上代码给以后一个回顾: 

建议提前读一下think in java 注解 。 

说明创建注解我在第一个注解说明下,以后的注解不在说明。‘ 

DBTable 注解: 

/*** Project Name:myannotation* File Name:DBTable.java* Package Name:com.iflytek.db* Date:2016-8-28下午08:20:54* Copyright (c) 2016, [email protected] All Rights Reserved.**/package com.iflytek.db;import java.lang.annotation.ElementType;import java.lang.annotation.Retention;import java.lang.annotation.RetentionPolicy;import java.lang.annotation.Target;/**@Target:   @Target说明了Annotation所修饰的对象范围:Annotation可被用于 packages、types(类、接口、枚举、Annotation类型)、类型成员(方法、构造方法、成员变量、枚举值)、方法参数和本地变量(如循环变量、catch参数)。在Annotation类型的声明中使用了target可更加明晰其修饰的目标。  作用:用于描述注解的使用范围(即:被描述的注解可以用在什么地方)  取值(ElementType)有:    1.CONSTRUCTOR:用于描述构造器    2.FIELD:用于描述域    3.LOCAL_VARIABLE:用于描述局部变量    4.METHOD:用于描述方法    5.PACKAGE:用于描述包    6.PARAMETER:用于描述参数    7.TYPE:用于描述类、接口(包括注解类型) 或enum声明 @Retention:  @Retention定义了该Annotation被保留的时间长短:某些Annotation仅出现在源代码中,而被编译器丢弃;而另一些却被编译在class文件中;编译在class文件中的Annotation可能会被虚拟机忽略,而另一些在class被装载时将被读取(请注意并不影响class的执行,因为Annotation与class在使用上是被分离的)。使用这个meta-Annotation可以对 Annotation的“生命周期”限制。  作用:表示需要在什么级别保存该注释信息,用于描述注解的生命周期(即:被描述的注解在什么范围内有效)  取值(RetentionPoicy)有:    1.SOURCE:在源文件中有效(即源文件保留)    2.CLASS:在class文件中有效(即class保留)    3.RUNTIME:在运行时有效(即运行时保留)  Retention meta-annotation类型有唯一的value作为成员,它的取值来自java.lang.annotation.RetentionPolicy的枚举类型值 */@Target(ElementType.TYPE)@Retention(RetentionPolicy.RUNTIME)public @interface DBTable{  public String name() default "";}

Constraints 约束注解: 

/*** Project Name:myannotation* File Name:Constraints.java* Package Name:com.iflytek.db* Date:2016-8-28下午08:27:08* Copyright (c) 2016, [email protected] All Rights Reserved.**/package com.iflytek.db;import java.lang.annotation.ElementType;import java.lang.annotation.Retention;import java.lang.annotation.RetentionPolicy;import java.lang.annotation.Target;@Target(ElementType.FIELD)@Retention(RetentionPolicy.RUNTIME)public @interface Constraints{  boolean primaryKey() default false;    boolean allowNull() default true;    boolean unique() default false;}

 SQLInteger int注解: 

/*** Project Name:myannotation* File Name:SQLInteger.java* Package Name:com.iflytek.db* Date:2016-8-29下午10:24:11* Copyright (c) 2016, [email protected] All Rights Reserved.**/package com.iflytek.db;import java.lang.annotation.ElementType;import java.lang.annotation.Retention;import java.lang.annotation.RetentionPolicy;import java.lang.annotation.Target;@Target(ElementType.FIELD)@Retention(RetentionPolicy.RUNTIME)public @interface SQLInteger{  String name() default "";    Constraints constraints() default @Constraints;}

 SQLString 字符注解: 

/*** Project Name:myannotation* File Name:SQLString.java* Package Name:com.iflytek.db* Date:2016-8-29下午10:28:04* Copyright (c) 2016, [email protected] All Rights Reserved.**/package com.iflytek.db;import java.lang.annotation.ElementType;import java.lang.annotation.Retention;import java.lang.annotation.RetentionPolicy;import java.lang.annotation.Target;@Target(ElementType.FIELD)@Retention(RetentionPolicy.RUNTIME)public @interface SQLString{  int value() default 0;    String name() default "";    Constraints constraints() default @Constraints;}

 创建表的处理器:

/*** Project Name:myannotation* File Name:TableCreator.java* Package Name:com.iflytek.table* Date:2016-8-29下午10:57:52* Copyright (c) 2016, [email protected] All Rights Reserved.**/package com.iflytek.table;import java.lang.annotation.Annotation;import java.lang.reflect.Field;import java.util.ArrayList;import java.util.List;import com.iflytek.db.Constraints;import com.iflytek.db.DBTable;import com.iflytek.db.SQLInteger;import com.iflytek.db.SQLString;public class TableCreator{  public static void main(String[] args)  {    createTable(Member.class);  }    //创建表SQL语句  private static void createTable(Class<?> cl)  {    //获取DBTable注解    DBTable dbTable = cl.getAnnotation(DBTable.class);    //判断DBTable注解是否存在    if (dbTable == null)    {      System.out.println("没有找到关于DBTable");      return;    }        //如果@DBTable注解存在获取表明     String tableName = dbTable.name();    //判断表名是否存在    if (tableName.length() < 1)    {      //不存在,说明默认就是类名,通过 cl.getSimpleName()获取类名并且大写      tableName = cl.getSimpleName().toUpperCase();    }        //定义获取column的容器    List<String> columnDefs = new ArrayList<String>();    //循环属性字段    //说明:getDeclaredFields()获得某个类的所有申明的字段,即包括public、private和proteced,但是不包括父类的申明字段。     //getFields()获得某个类的所有的公共(public)的字段,包括父类。     for (Field field : cl.getDeclaredFields())    {      //定义表字段名称变量      String columnName = null;      //获取字段上的注解(现在字段允许多个注解,因此返回的是数组)      Annotation[] anns = field.getDeclaredAnnotations();      //判断属性是否存在注解      if (anns.length < 1)        continue;//判断是否是我们定义的数据类型      if (anns[0] instanceof SQLInteger)      {        //获取SQLInteger 注解        SQLInteger sInt = (SQLInteger)anns[0];        //判断是否注解的name是否有值        if (sInt.name().length() < 1)        {          //如果没有值,说明是类的属性字段,获取属性并转换大写          columnName = field.getName().toUpperCase();        }        else        { //如果有值,获取设置的name值          columnName = sInt.name();        }        //放到属性的容器内        columnDefs.add(columnName + " INT " + getConstraints(sInt.constraints()));      }//同上SQLInteger,这里不写注释了      if (anns[0] instanceof SQLString)      {        SQLString sString = (SQLString)anns[0];        if (sString.name().length() < 1)        {          columnName = field.getName().toUpperCase();        }        else        {          columnName = sString.name();        }        columnDefs.add(columnName + " VARCHAR(" + sString.value() + ")" + getConstraints(sString.constraints()));      }//定义生成创建表的SQL语句      StringBuilder createCommand = new StringBuilder("CREATE TABLE " + tableName + "(");      //循环上面属性容器,      for (String columnDef : columnDefs)      {        //把属性添加到sql语句中        createCommand.append("\n  " + columnDef + ",");        //去掉最后一个逗号        String tableCreate = createCommand.substring(0, createCommand.length() - 1) + ");";        //打印        System.out.println("Table creation SQL for " + cl.getName() + " is :\n" + tableCreate);      }    }  }    private static String getConstraints(Constraints con)  {    String constraints = "";    //判断是否为null    if (!con.allowNull())    {      constraints += " NOT NULL ";    }    //判断是否是主键    if (con.primaryKey())    {      constraints += " PRIMARY KEY ";    }    //是否唯一    if (con.unique())    {      constraints += " UNIQUE ";    }        return constraints;  }}

以上代码拷贝出来,就可以运行了! 

上面虽然是简单的创建表语句,但我们可以蔓延到hibernate的domain类里的注解,各种CURD ,何尝不是这样处理的呢,只是hibernate有很多东西,但是万变不离其宗,以后有机会研究一下hibernate 。 

收获: 

读了以后,对于注解知道为什么要这么用了,其实顾名思义就是一个注解,只是有一个处理器来处理这个注解,这对我以后用到注解方面应该有帮助的, 

时间不早了,就写到这里!

结果:

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。


  • 上一条:
    SQL Server成功与服务器建立连接但是在登录过程中发生错误的快速解决方案
    下一条:
    SQLServer2005创建定时作业任务
  • 昵称:

    邮箱:

    0条评论 (评论内容有缓存机制,请悉知!)
    最新最热
    • 分类目录
    • 人生(杂谈)
    • 技术
    • linux
    • Java
    • php
    • 框架(架构)
    • 前端
    • ThinkPHP
    • 数据库
    • 微信(小程序)
    • Laravel
    • Redis
    • Docker
    • Go
    • swoole
    • Windows
    • Python
    • 苹果(mac/ios)
    • 相关文章
    • 分库分表的目的、优缺点及具体实现方式介绍(0个评论)
    • DevDB - 在 VS 代码中直接访问数据库(0个评论)
    • 在ubuntu系统中实现mysql数据存储目录迁移流程步骤(0个评论)
    • 在mysql中使用存储过程批量新增测试数据流程步骤(0个评论)
    • php+mysql数据库批量根据条件快速更新、连表更新sql实现(0个评论)
    • 近期文章
    • 在go语言中使用api.geonames.org接口实现根据国际邮政编码获取地址信息功能(1个评论)
    • 在go语言中使用github.com/signintech/gopdf实现生成pdf分页文件功能(0个评论)
    • gmail发邮件报错:534 5.7.9 Application-specific password required...解决方案(0个评论)
    • 欧盟关于强迫劳动的规定的官方举报渠道及官方举报网站(0个评论)
    • 在go语言中使用github.com/signintech/gopdf实现生成pdf文件功能(0个评论)
    • Laravel从Accel获得5700万美元A轮融资(0个评论)
    • 在go + gin中gorm实现指定搜索/区间搜索分页列表功能接口实例(0个评论)
    • 在go语言中实现IP/CIDR的ip和netmask互转及IP段形式互转及ip是否存在IP/CIDR(0个评论)
    • PHP 8.4 Alpha 1现已发布!(0个评论)
    • Laravel 11.15版本发布 - Eloquent Builder中添加的泛型(0个评论)
    • 近期评论
    • 122 在

      学历:一种延缓就业设计,生活需求下的权衡之选中评论 工作几年后,报名考研了,到现在还没认真学习备考,迷茫中。作为一名北漂互联网打工人..
    • 123 在

      Clash for Windows作者删库跑路了,github已404中评论 按理说只要你在国内,所有的流量进出都在监控范围内,不管你怎么隐藏也没用,想搞你分..
    • 原梓番博客 在

      在Laravel框架中使用模型Model分表最简单的方法中评论 好久好久都没看友情链接申请了,今天刚看,已经添加。..
    • 博主 在

      佛跳墙vpn软件不会用?上不了网?佛跳墙vpn常见问题以及解决办法中评论 @1111老铁这个不行了,可以看看近期评论的其他文章..
    • 1111 在

      佛跳墙vpn软件不会用?上不了网?佛跳墙vpn常见问题以及解决办法中评论 网站不能打开,博主百忙中能否发个APP下载链接,佛跳墙或极光..
    • 2017-06
    • 2017-08
    • 2017-09
    • 2017-10
    • 2017-11
    • 2018-01
    • 2018-05
    • 2018-10
    • 2018-11
    • 2020-02
    • 2020-03
    • 2020-04
    • 2020-05
    • 2020-06
    • 2020-07
    • 2020-08
    • 2020-09
    • 2021-02
    • 2021-04
    • 2021-07
    • 2021-08
    • 2021-11
    • 2021-12
    • 2022-02
    • 2022-03
    • 2022-05
    • 2022-06
    • 2022-07
    • 2022-08
    • 2022-09
    • 2022-10
    • 2022-11
    • 2022-12
    • 2023-01
    • 2023-03
    • 2023-04
    • 2023-05
    • 2023-07
    • 2023-08
    • 2023-10
    • 2023-11
    • 2023-12
    • 2024-01
    • 2024-03
    Top

    Copyright·© 2019 侯体宗版权所有· 粤ICP备20027696号 PHP交流群

    侯体宗的博客