精华内容
下载资源
问答
  • 经百度后发现了可行的方式.1: 需要借助于ganymed-ssh2-build210.jar jar2: 默认一次只能执行一个命令(不然会抛出A remote execution has already started.) 要想实现执行多个命令 需要执行多次 或者 写成批文件的...

    突发奇想,想用JAVA代码实现执行Linux脚本,实现 增量文件的上传然后重启Tomcat服务器. 经百度后发现了可行的方式.

    1: 需要借助于 ganymed-ssh2-build210.jar jar

    2: 默认一次只能执行一个命令(不然会抛出A remote execution has already started.) 要想实现执行多个命令 需要执行多次 或者 写成批文件的形式 再或者 几条命令合并成为一条语句执行package ssh;

    import java.io.BufferedReader;

    import java.io.BufferedWriter;

    import java.io.ByteArrayOutputStream;

    import java.io.File;

    import java.io.FileInputStream;

    import java.io.FileNotFoundException;

    import java.io.IOException;

    import java.io.InputStreamReader;

    import java.io.OutputStream;

    import java.io.OutputStreamWriter;

    import java.io.PrintWriter;

    import java.util.ArrayList;

    import java.util.Collection;

    import java.util.List;

    import java.util.Objects;

    import java.util.Scanner;

    import java.util.Set;

    import java.util.concurrent.TimeUnit;

    import java.util.logging.Level;

    import java.util.logging.Logger;

    import java.util.stream.Collectors;

    import ch.ethz.ssh2.Connection;

    import ch.ethz.ssh2.SCPClient;

    import ch.ethz.ssh2.Session;

    import ch.ethz.ssh2.StreamGobbler;

    public class Scpclient {

    // IP

    private String ip;

    // 端口

    private int port;

    // 用户名

    private String username;

    // 密码

    private String password;

    // 实例

    static private Scpclient instance;

    static synchronized public Scpclient getInstance(String IP, int port, String username, String passward) {

    if (instance == null) {

    instance = new Scpclient(IP, port, username, passward);

    }

    return instance;

    }

    public Scpclient(String IP, int port, String username, String passward) {

    this.ip = IP;

    this.port = port;

    this.username = username;

    this.password = passward;

    }

    public void getFile(String remoteFile, String localTargetDirectory) {

    Connection conn = null;

    try {

    conn = new Connection(ip, port);

    conn.connect();

    boolean isAuthenticated = conn.authenticateWithPassword(username, password);

    if (isAuthenticated == false) {

    System.err.println("authentication failed");

    }

    SCPClient client = new SCPClient(conn);

    client.get(remoteFile, localTargetDirectory);

    conn.close();

    } catch (IOException ex) {

    Logger.getLogger(SCPClient.class.getName()).log(Level.SEVERE, null, ex);

    }finally {

    if (conn != null) conn.close();

    }

    }

    public void putFile(String localFile, String remoteTargetDirectory) {

    Connection conn = null;

    try {

    conn = new Connection(ip, port);

    conn.connect();

    boolean isAuthenticated = conn.authenticateWithPassword(username, password);

    if (isAuthenticated == false) {

    System.err.println("authentication failed");

    }

    SCPClient client = new SCPClient(conn);

    client.put(localFile, remoteTargetDirectory);

    conn.close();

    } catch (IOException ex) {

    Logger.getLogger(SCPClient.class.getName()).log(Level.SEVERE, null, ex);

    }finally {

    if (conn != null) conn.close();

    }

    }

    /**

    * 上传文件

    * @param localFile

    * @param remoteFileName

    * @param remoteTargetDirectory

    * @param mode

    * Author:YCM

    * 2017年12月28日 上午11:12:08

    */

    public void putFile(String localFile, String remoteFileName, String remoteTargetDirectory, String mode) {

    Connection conn = null;

    Session sess = null;

    try {

    conn = new Connection(ip, port);

    conn.connect();

    // 进行认证

    boolean isAuthenticated = conn.authenticateWithPassword(username, password);

    if (isAuthenticated == false) {

    System.err.println("authentication failed");

    }

    SCPClient client = new SCPClient(conn);

    if ((mode == null) || (mode.length() == 0)) {

    // 默认执行模式

    mode = "0600";

    }

    client.put(localFile, remoteFileName, remoteTargetDirectory, mode);

    //重命名

    sess = conn.openSession();

    String tmpPathName = remoteTargetDirectory + File.separator + remoteFileName;

    String newPathName = tmpPathName.substring(0, tmpPathName.lastIndexOf("."));

    sess.execCommand("mv " + remoteFileName + " " + newPathName);//重命名回来

    sess.close();

    conn.close();

    } catch (IOException ex) {

    Logger.getLogger(SCPClient.class.getName()).log(Level.SEVERE, null, ex);

    }finally {

    if (sess != null) sess.close();

    if (conn != null) conn.close();

    }

    }

    /**

    * 上传文件,不支持重命名

    * @param collections

    * @param serverRootPath

    * @param zlRootPath

    * @param mode

    * Author:YCM

    * 2017年12月28日 上午11:10:49

    */

    public void putFile(Collection collections,String serverRootPath,String zlRootPath,String mode) {

    Connection conn = null;

    try {

    conn = new Connection(ip, port);

    conn.connect();

    boolean isAuthenticated = conn.authenticateWithPassword(username, password);

    if (isAuthenticated == false) {

    System.err.println("authentication failed");

    }

    SCPClient client = new SCPClient(conn);

    if ((mode == null) || (mode.length() == 0)) {

    mode = "0600";

    }

    int index = 1;

    for (File file : collections) {

    // 本地路径

    String diskPath = file.getPath();

    System.out.println("......上传增量文件[" + index + "][" + diskPath + "]");

    // 相对路径

    String relativePath = diskPath.replace(zlRootPath, "").replace("\\", "/");

    relativePath = relativePath.substring(0, relativePath.lastIndexOf("/"));

    // 文件名称

    String fileName = file.getName();

    // 上传文件

    client.put(diskPath, fileName, serverRootPath + relativePath, mode);

    index++;

    }

    conn.close();

    } catch (IOException ex) {

    Logger.getLogger(SCPClient.class.getName()).log(Level.SEVERE, null, ex);

    }finally {

    if (conn != null) conn.close();

    }

    }

    /**

    * 创建文件夹

    */

    public void createFolder(Collection collections) {

    Connection conn = null;

    Session sess = null;

    try {

    conn = new Connection(ip, port);

    conn.connect();

    boolean isAuthenticated = conn.authenticateWithPassword(username, password);

    if (isAuthenticated == false) {

    System.err.println("authentication failed");

    }

    sess = conn.openSession();

    // 不能执行多次命令,一次只能执行一条命令(不然会抛出A remote execution has already started.)

    String command = "mkdir -p ";

    for (String remoteTargetDirectory : collections) {

    if (remoteTargetDirectory.contains(".")) {

    String folderPath = remoteTargetDirectory.substring(0, remoteTargetDirectory.lastIndexOf("/"));

    command += folderPath + " ";

    }

    }

    sess.execCommand(command);

    sess.close();

    conn.close();

    } catch (IOException ex) {

    Logger.getLogger(SCPClient.class.getName()).log(Level.SEVERE, null, ex);

    }finally {

    if (sess != null) sess.close();

    if (conn != null) conn.close();

    }

    }

    public static byte[] getBytes(String filePath) {

    byte[] buffer = null;

    try {

    File file = new File(filePath);

    FileInputStream fis = new FileInputStream(file);

    ByteArrayOutputStream byteArray = new ByteArrayOutputStream(1024 * 1024);

    byte[] b = new byte[1024 * 1024];

    int i;

    while ((i = fis.read(b)) != -1) {

    byteArray.write(b, 0, i);

    }

    fis.close();

    byteArray.close();

    buffer = byteArray.toByteArray();

    } catch (FileNotFoundException e) {

    e.printStackTrace();

    } catch (IOException e) {

    e.printStackTrace();

    }

    return buffer;

    }

    public void readWell(File file, List filesList) {

    if (file.isDirectory()) {

    File[] listFiles = file.listFiles();

    for (File fle : listFiles) {

    if (fle.isDirectory()) {

    readWell(fle, filesList);

    } else {

    filesList.add(fle);

    }

    }

    } else {

    filesList.add(file);

    }

    }

    /**

    * 输出日志

    */

    public void tailViewTomcatLog(String shell) {

    Connection conn = null;

    Session sess = null;

    try {

    conn = new Connection(ip, port);

    conn.connect();

    boolean isAuthenticated = conn.authenticateWithPassword(username, password);

    if (isAuthenticated == false) {

    System.err.println("authentication failed");

    }

    sess = conn.openSession();

    if (shell.contains(".sh")) {

    sess.requestDumbPTY();

    sess.startShell();

    PrintWriter out = new PrintWriter(sess.getStdin());

    out.write(shell);

    }else

    sess.execCommand(shell);

    System.err.println("[EXEC RESULT:" + shell + "]");

    try (StreamGobbler stdout = new StreamGobbler(sess.getStdout());

    BufferedReader br = new BufferedReader(new InputStreamReader(stdout))) {

    while (true) {

    String line = br.readLine();

    if (line == null) break;

    //buffer.append(line);

    //buffer.append(System.getProperty("line.separator"));

    System.out.println(line);

    }

    }

    } catch (IOException ex) {

    Logger.getLogger(SCPClient.class.getName()).log(Level.SEVERE, null, ex);

    } finally {

    if (sess != null) sess.close();

    if (conn != null) conn.close();

    }

    }

    public void restartServer(String exeShell) {

    Connection conn = null;

    Session sess = null;

    try {

    Connection connection = new Connection("120.120.120.120");

    connection.connect();

    boolean authenticateWithPass = connection.authenticateWithPassword("root", "password");

    if (!authenticateWithPass) { throw new RuntimeException("Authentication failed. Please check hostName, userName and passwd"); }

    Session session = connection.openSession();

    session = connection.openSession();

    session.requestDumbPTY();

    session.startShell();

    OutputStream stdins = session.getStdin();

    BufferedWriter bw = new BufferedWriter(new OutputStreamWriter(stdins));

    bw.write(exeShell + " \n\r");

    bw.flush();

    } catch (IOException ex) {

    Logger.getLogger(SCPClient.class.getName()).log(Level.SEVERE, null, ex);

    } finally {

    if (sess != null) sess.close();

    if (conn != null) conn.close();

    }

    }

    public void exeHander() {

    }

    public String getUsername() {

    return username;

    }

    public void setUsername(String username) {

    this.username = username;

    }

    public String getPassword() {

    return password;

    }

    public void setPassword(String password) {

    this.password = password;

    }

    public int getPort() {

    return port;

    }

    public void setPort(int port) {

    this.port = port;

    }

    public static void main(String[] args) {

    // String zlRootPath = "C:\\Users\\Administrator\\Desktop\\CS_20171227\\wechat\\ROOT";

    // String wechatServerRootPath = "/mydata/tomcat/wechat/apache-tomcat-8.5.24/webapps/ROOT/";

    Scanner in = null;

    try {

    System.err.println("!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!");

    System.err.println("!!!!     注意:增量包需放置在桌面上,并输入增量文件夹名称(大小写忽略)      !!!!");

    System.err.println("!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!");

    in =  new Scanner(System.in);

    String zlFolderName = "";

    System.out.print("请输入增量文件夹名称:");

    zlFolderName = in.nextLine();

    if (zlFolderName == null || Objects.equals("", zlFolderName.trim())) {

    System.err.println("请输入增量文件夹名称");

    return;

    }

    // 转大写

    zlFolderName = zlFolderName.toUpperCase();

    System.out.print("请输入将要上传的项目名称:");

    String inProjectName = in.nextLine();

    if (inProjectName == null || Objects.equals("", inProjectName.trim())) {

    System.err.println("请输入有效的项目名称");

    return;

    }

    String zlRoot = "C:\\Users\\Administrator\\Desktop\\" + zlFolderName,

    serverRoot = "/mydata/tomcat",

    zlRootPath = "",

    serverRootPath = "",

    bashFilePath = "";

    switch (inProjectName.trim()) {

    case "admin":

    zlRootPath = zlRoot + "\\admin\\ROOT";

    serverRootPath = serverRoot + "/admin/apache-tomcat-8.0.42/webapps/ROOT";

    break;

    case "wechat":

    zlRootPath = zlRoot + "\\wechat\\ROOT";

    serverRootPath = serverRoot + "/wechat/apache-tomcat-8.5.24/webapps/ROOT";

    break;

    case "core-server":

    zlRootPath = zlRoot + "\\core-server\\ROOT";

    serverRootPath = serverRoot + "/core-server/apache-tomcat-8.0.42/webapps/ROOT";

    break;

    default:

    break;

    }

    if (zlRootPath == null || serverRootPath == null ||

    Objects.equals("", zlRootPath) || Objects.equals("", serverRootPath)) {

    System.err.println("请输入有效的项目名称");

    return;

    }

    // 可执行文件路径

    bashFilePath = serverRootPath.replace("/webapps/ROOT", "");

    // 判断增量文件夹是否存在

    File zlRootFile = new File(zlRootPath);

    if(!zlRootFile.exists()) {

    System.err.println("增量文件夹不存在");

    return;

    }

    System.out.println("执行上传Linux脚本[开始]");

    Scpclient scp = Scpclient.getInstance("120.120.120.120", 22, "root", "password");

    List filesList = new ArrayList();

    // 递归收集文件

    System.out.println("...收集增量文件[开始]");

    scp.readWell(zlRootFile, filesList);

    System.out.println("...收集增量文件[结束],文件数:" + filesList.size());

    System.out.println("...创建不存在文件夹[开始]");

    String finalServerRootPath = serverRootPath,finalZlRootPath = zlRootPath;

    Set folderNames = filesList.stream().map(file -> finalServerRootPath + file.getPath().replace(finalZlRootPath, "").replace("\\", "/"))

    .collect(Collectors.toSet());

    scp.createFolder(folderNames);

    System.out.println("...创建不存在文件夹[结束]");

    System.out.println("...上传增量文件[开始]");

    scp.putFile(filesList,serverRootPath,zlRootPath,null);

    System.out.println("...上传增量文件[结束]");

    System.out.println("执行上传Linux脚本[结束]");

    System.out.print("是否执行重启操作[y/n]:");

    String restart = in.nextLine();

    if (Objects.equals("y", restart) || Objects.equals("Y", restart)) {

    System.out.println("...执行命令重启服务器[开始]");

    System.out.println("...另起线程获取服务器实时日志到本地[开始]");

    SSHAgent sshAgent = new SSHAgent();

    sshAgent.setExeShell("tail -f " + bashFilePath + "/logs/catalina.out \n\r");

    new Thread(sshAgent).start();

    TimeUnit.SECONDS.sleep(5);

    System.out.println("......停止[" + inProjectName + "]服务器[开始]");

    //scp.tailViewTomcatLog("cd " + bashFilePath + " && bin/shutdown.sh");

    scp.restartServer(bashFilePath + "/bin/shutdown.sh");

    System.out.print("...是否执行启动命令[y/n]:");

    String incontinue = in.nextLine();

    if (Objects.equals("y", incontinue) || Objects.equals("Y", incontinue)) {

    System.out.println("......重启[" + inProjectName + "]服务器[开始]");

    scp.restartServer(bashFilePath + "/bin/startup.sh");

    }

    }

    System.out.print("按任意键退出:");

    in.nextLine();

    } catch (Exception e) {

    e.printStackTrace();

    } finally {

    if (in != null) in.close();

    }

    }

    }

    展开全文
  • 满意答案homewellhhx2014....建议如下:如果你这A()调用的接口一次只允许一访问,那么需要在A所在的类里设置一静态成员变量,如static String object = "some"。然后在A()方法一开始用synchronized (object) {...

    满意答案

    02ae427d08e371d7e90d5b995e828d6d.png

    homewellhhx

    2014.01.01

    02ae427d08e371d7e90d5b995e828d6d.png

    采纳率:58%    等级:13

    已帮助:8827人

    本来用DWR调用方法就是多线程的,线程总数与J2EE容器配置的有关。

    建议如下:

    如果你这个A()调用的接口一次只允许一个访问,那么需要在A所在的类里设置一个静态成员变量,如static String object = "some"。

    然后在A()方法一开始用synchronized (object) {.. }把代码都包含进来。

    这样可以确保一次只有一个访问。

    如果接口一次最多运行固定数目的访问,如10个。那么复杂一些,不过我估计你不是这个情况。一般建议这个固定数目与J2EE容器配置的线程数一致即可。

    上述方法都是为了防止接口被同时访问,但这样的后果就是前端用户会等待,甚至线程满。前端等待是没办法的,后面慢,前面只有排队了。线程满的话,最好的方式是采用非阻塞的IO(NIO),不过那个很难做到。你提出的10秒终止方法是一种方案,不过关键在于如何终止一个方法的运行。

    很遗憾,Java不能任意终止一个方法的运行,不过对于接口操作,可以如下处理:

    a) 如果接口是TCP/IP,那么可以通过强行关闭socket来终止。如:

    timer.schedule( new TimerTask() {

    public void run() {

    socket.close();

    }

    }, 10000);

    b) 如果接口是用类库的话,看看它有没有设置timeout的地方,如果有,那么设置一下,如果没有,那么没办法了。

    补充:

    你用了axis的setTimeout,理论上超过10秒后call.invoke会抛出异常的。

    假设后台很慢,每次都需要10秒,假设weblogic线程池大小为50,那么如果同时访问的人超过50个,或者每秒访问量超过5个,那么就会发生线程阻塞。

    这是系统性能问题,可以将weblogic的线程池最大数量设置高些来增加吞吐量。不过这种方式治标不治本,如果你的程序预计到并发访问量很大,那么后台响应超过1秒就不太合理了,这个改动起来就很麻烦了。

    00分享举报

    展开全文
  • 使用PreparedStatement实现CRUD操作 操作和访问数据库 数据库连接被用于向数据库服务器发送...PrepatedStatement:SQL 语句被预编译并存储在此对象中,可以使用此对象多次高效地执行该语句。 CallableStatement:用于

    使用PreparedStatement实现CRUD操作

    操作和访问数据库

    数据库连接被用于向数据库服务器发送命令和 SQL 语句,并接受数据库服务器返回的结果。其实一个数据库连
    接就是一个Socket连接。
    在 java.sql 包中有 3 个接口分别定义了对数据库的调用的不同方式:
    Statement:用于执行静态 SQL 语句并返回它所生成结果的对象。
    PrepatedStatement:SQL 语句被预编译并存储在此对象中,可以使用此对象多次高效地执行该语句。
    CallableStatement:用于执行 SQL 存储过程
    在这里插入图片描述

    使用Statement操作数据表的弊端

    通过调用 Connection 对象的 createStatement() 方法创建该对象。该对象用于执行静态的 SQL 语句,并且返
    回执行结果。
    Statement 接口中定义了下列方法用于执行 SQL 语句:
    但是使用Statement操作数据表存在弊端:
    问题一:存在拼串操作,繁琐
    问题二:存在SQL注入问题
    SQL 注入是利用某些系统没有对用户输入的数据进行充分的检查,而在用户输入数据中注入非法的 SQL 语句段
    或命令(如:SELECT user, password FROM user_table WHERE user=‘a’ OR 1 = ’ AND password = ’ OR ‘1’ =
    ‘1’) ,从而利用系统的 SQL 引擎完成恶意行为的做法。
    对于 Java 而言,要防范 SQL 注入,只要用 PreparedStatement(从Statement扩展而来) 取代 Statement 就可
    以了。

    PreparedStatement的使用

    PreparedStatement介绍
    可以通过调用 Connection 对象的 preparedStatement(String sql) 方法获取 PreparedStatement 对象
    PreparedStatement 接口是 Statement 的子接口,它表示一条预编译过的 SQL 语句
    PreparedStatement 对象所代表的 SQL 语句中的参数用问号(?)来表示,调用 PreparedStatement 对象的
    setXxx() 方法来设置这些参数. setXxx() 方法有两个参数,第一个参数是要设置的 SQL 语句中的参数的索引(从 1
    开始),第二个是设置的 SQL 语句中的参数的值
    ** PreparedStatement vs Statement**
    代码的可读性和可维护性。
    PreparedStatement 能最大可能提高性能:
    DBServer会对预编译语句提供性能优化。因为预编译语句有可能被重复调用,所以语句在被DBServer的
    编译器编译后的执行代码被缓存下来,那么下次调用时只要是相同的预编译语句就不需要编译,只要将参
    数直接传入编译过的语句执行代码中就会得到执行。
    在statement语句中,即使是相同操作但因为数据内容不一样,所以整个语句本身不能匹配,没有缓存语句的意
    义.事实是没有数据库会对普通语句编译后的执行代码缓存。这样每执行一次都要对传入的语句编译一次。
    (语法检查,语义检查,翻译成二进制命令,缓存)
    PreparedStatement 可以防止 SQL 注入
    Java与SQL对应数据类型转换表
    在这里插入图片描述

    使用PreparedStatement实现增、删、改操作
     // 使用PreparedStatement来替换Statement,实现对数据表的增删改 操作
    public class PreparedStatementTest {
        public static void main(String[] args) throws SQLException, IOException, ClassNotFoundException, ParseException {
    String sql="delete from customers where id=?";
    change(sql,19);
    
        }
    
        //获得数据库连接
        public static Connection getConnection() throws IOException, ClassNotFoundException, SQLException {
            //1.加载配置文件
            InputStream is =
                    ClassLoader.getSystemClassLoader().getResourceAsStream("jdbc.properties");
            Properties pros = new Properties();
            pros.load(is);
            //2.读取配置信息
            String user = pros.getProperty("user");
            String password = pros.getProperty("password");
            String url = pros.getProperty("url");
            String driverClass = pros.getProperty("driverClass");
            //3.加载驱动
            Class.forName(driverClass);
            //4.获取连接
            Connection conn = DriverManager.getConnection(url, user, password);
            return conn;
        }
        //向customers表中添加一条数据
        public static void testInsert() throws SQLException, IOException, ClassNotFoundException, ParseException {
            //1.获得数据库连接
            Connection connection = getConnection();
            //2.预编译sql语句,返回PreparedStatement的实例
            String sql = "insert into customers(name,email,birth)values(?,?,?)";//?是占位符
            PreparedStatement ps = connection.prepareStatement(sql);
            //3.填充占位符
            ps.setString(1, "梨花");
            ps.setString(2, "lihua@gmail.com");
            //SimpleDateFormat format = new SimpleDateFormat("yyyy-MM-dd");
            //java.util.Date date=format.parse("1000-01-01");
            //ps.setDate(3, (java.sql.Date) new Date(date.getTime()));
            ps.setString(3, "1000-01-01");
            //4.执行操作
            ps.execute();
            //5.资源关闭
            if (ps != null) {
                ps.close();
            }
            if (connection != null) {
                connection.close();
            }
    
        }
        //修改customers表中的一条记录
        public static void testUpdate() throws SQLException, IOException, ClassNotFoundException {
            //1.获取数据库连接
            Connection connection= JDBCUtils.getConnection();
            //2.预编译sql的语句,返回PreparedStatement的实例
            String sql="update customers set name=? where id=?";
            PreparedStatement preparedStatement=connection.prepareStatement(sql);
            //3.填充占位符
            preparedStatement.setString(1,"白皮猪");
            preparedStatement.setInt(2,18);
            //4.执行操作
            preparedStatement.execute();
            //5.资源关闭
            JDBCUtils.closeResource(connection,preparedStatement);
          }
          //删除customers表中的一条记录
        public static void testDelete() throws SQLException, IOException, ClassNotFoundException {
            //1.获取数据库连接
            Connection connection=JDBCUtils.getConnection();
            //2.预编译Sql语句,返回PreparedStatement实例
            String sql="delete from customers where id=?";
            PreparedStatement preparedStatement=connection.prepareStatement(sql);
            //3.填充占位符
            preparedStatement.setInt(1,20);
            //4.执行操作
            preparedStatement.execute();
            //5.关闭资源
            JDBCUtils.closeResource(connection,preparedStatement);
    
        }
       //通用的增删改操作
        public static  void  change(String sql,Object ...args) throws SQLException, IOException, ClassNotFoundException {
             //1.获取数据库连接
            Connection connection = JDBCUtils.getConnection();
            //2.预编译Sql语句,返回PreparedStatement实例
            PreparedStatement preparedStatement = connection.prepareStatement(sql);
            //3.填充占位符
            for(int i=0;i<args.length;i++){
                preparedStatement.setObject(i+1,args[i]);
            }
            //4.执行操作
            preparedStatement.execute();
            //5.关闭资源
            JDBCUtils.closeResource(connection,preparedStatement);
        }
    }
    
    使用PreparedStatement实现查询操作
    
    /// 针对于表的字段名与类的属性名不相同的情况:
    //1.必须声明sql时,使用类的属性名来命名字段的别名
    //2.使用ResultSetMetaData时,需要使用getColumnLabel()来替换getColumnName()
    //获取列的别名。
    //说明。如果sql中没有给字段其别名,getcolumnLabel()获取的就是列名|
        //查询customers中的数据(只针对customers表)
        public static Customer testQueryOfCustomer(String sql, Object... args) throws SQLException, IOException, ClassNotFoundException {
    
            Connection connection = JDBCUtils.getConnection();
            PreparedStatement preparedStatement = connection.prepareStatement(sql);
            for (int i = 0; i < args.length; i++) {
                preparedStatement.setObject(i + 1, args[i]);
            }
            //执行并返回结果集
            ResultSet resultSet = preparedStatement.executeQuery();
            Customer customer = null;
            //处理结果集
            if (resultSet.next()) {//next():判断结果集下一条是否有数据,如果有数据返回true,并指针下移;如果返回false,直接结束
                customer = new Customer();
                //获取当前这条数据的各个字段值
                int id = resultSet.getInt(1);
                String name = resultSet.getString(2);
                String email = resultSet.getString(3);
                Date birth = resultSet.getDate(4);
    //            //方式一:
                System.out.println("id= " + id + " name= " + name + " email= " + email + " birth= " + birth);
    //             //方式二:
    //            Object [] data=new Object[]{id,name,email,birth};
                //推荐方式
    
                customer.setBirth(birth);
                customer.setEmail(email);
                customer.setId(id);
                customer.setName(name);
            }
            JDBCUtils.closeResource(connection, preparedStatement);
            resultSet.close();
    
            return customer;
        }
    
        //查询customers中的数据,利用反射
        public static Customer testQueryOfCustomer2(String sql, Object... args) throws SQLException, IOException, ClassNotFoundException, NoSuchFieldException, IllegalAccessException {
            Connection connection = JDBCUtils.getConnection();
            PreparedStatement preparedStatement = connection.prepareStatement(sql);
            for (int i = 0; i < args.length; i++) {
                preparedStatement.setObject(i + 1, args[i]);
            }
            ResultSet resultSet = preparedStatement.executeQuery();
            //获取结果集的元数据:ResultSetMethod
            ResultSetMetaData resultSetMetaData = resultSet.getMetaData();
            //通过ResultSetMetaData和获取结果集中的列数
            int columnCount = resultSetMetaData.getColumnCount();
            if (resultSet.next()) {
                Customer customer = new Customer();
                for (int i = 0; i < columnCount; i++) {
                    Object columnValue = resultSet.getObject(i + 1);
                    //获取每个列的列名
                    String columnName = resultSetMetaData.getColumnName(i + 1);
                    //给Customer对象columnName属性,赋值位columnValue,通过反射
                    Field field = Customer.class.getDeclaredField(columnName);
                    field.setAccessible(true);
                    field.set(customer, columnValue);
                }
    
                return customer;
            }
            JDBCUtils.closeResource2(connection, preparedStatement, resultSet);
            return null;
        }
    
        查询customers中的数据,利用反射(当表中数据名与类中属性名不一致起别名)
        public static Order testQueryOfOrder() throws SQLException, IOException, ClassNotFoundException, IllegalAccessException, NoSuchFieldException {
            String sql = "select order_id id,order_name name,order_date date from `order` where order_id=?";
            Connection connection = JDBCUtils.getConnection();
            PreparedStatement preparedStatement = connection.prepareStatement(sql);
            preparedStatement.setObject(1, 1);
    
            ResultSet resultSet = preparedStatement.executeQuery();
            //获取结果集的元数据:ResultSetMethod
            ResultSetMetaData resultSetMetaData = resultSet.getMetaData();
            //通过ResultSetMetaData和获取结果集中的列数
            int columnCount = resultSetMetaData.getColumnCount();
            if (resultSet.next()) {
                Order order = new Order();
                for (int i = 0; i < columnCount; i++) {
    
    
                    Object columnValue = resultSet.getObject(i + 1);
    
                    //获取每个列的列名:getColumnName()
                    //获取列的别名:getColumnLabel()
                    // String columnName = resultSetMetaData.getColumnName(i + 1);
                    String columnLabel = resultSetMetaData.getColumnLabel(i + 1);
                    //给Customer对象columnName属性,赋值位columnValue,通过反射
                    Field field = Order.class.getDeclaredField(columnLabel);
                    field.setAccessible(true);
                    field.set(order, columnValue);
    
    
                }
    
    
                return order;
            }
            JDBCUtils.closeResource2(connection, preparedStatement, resultSet);
    
            return null;
    
    
        }
    

    通用的针对于不同表的查询

    //    通用的针对于不同表的查询
        public static  <T>T getInstance(Class<T> clazz,String sql,Object...args) throws SQLException, IOException, ClassNotFoundException, NoSuchFieldException, IllegalAccessException, InstantiationException {
    
            Connection connection = JDBCUtils.getConnection();
            PreparedStatement preparedStatement = connection.prepareStatement(sql);
            preparedStatement.setObject(1, 1);
    
            ResultSet resultSet = preparedStatement.executeQuery();
            //获取结果集的元数据:ResultSetMethod
            ResultSetMetaData resultSetMetaData = resultSet.getMetaData();
            //通过ResultSetMetaData和获取结果集中的列数
            int columnCount = resultSetMetaData.getColumnCount();
            if (resultSet.next()) {
                T t=clazz.newInstance();
                for (int i = 0; i < columnCount; i++) {
    
    
                    Object columnValue = resultSet.getObject(i + 1);
    
                    //获取每个列的列名:getColumnName()
                    //获取列的别名:getColumnLabel()
                    // String columnName = resultSetMetaData.getColumnName(i + 1);
                    String columnLabel = resultSetMetaData.getColumnLabel(i + 1);
                    //给Customer对象columnName属性,赋值位columnValue,通过反射
                    Field field = clazz.getDeclaredField(columnLabel);
                    field.setAccessible(true);
                    field.set(t, columnValue);
    
    
                }
    
    
                return t;
            }
            JDBCUtils.closeResource2(connection, preparedStatement, resultSet);
    
            return null;
    
        }
    

    ResultSet与ResultSetMetaData

    ResultSet
    查询需要调用PreparedStatement 的 executeQuery() 方法,查询结果是一个ResultSet 对象
    ResultSet 对象以逻辑表格的形式封装了执行数据库操作的结果集,ResultSet 接口由数据库厂商提供实现
    ResultSet 返回的实际上就是一张数据表。有一个指针指向数据表的第一条记录的前面。
    ResultSet 对象维护了一个指向当前数据行的游标,初始的时候,游标在第一行之前,可以通过 ResultSet 对象
    的 next() 方法移动到下一行。调用 next()方法检测下一行是否有效。若有效,该方法返回 true,且指针下移。
    相当于Iterator对象的 hasNext() 和 next() 方法的结合体。
    当指针指向一行时, 可以通过调用 getXxx(int index) 或 getXxx(int columnName) 获取每一列的值。
    例如: getInt(1), getString(“name”)
    注意:Java与数据库交互涉及到的相关Java API中的索引都从1开始。
    ResultSet 接口的常用方法:
    boolean next()
    getString
    在这里插入图片描述
    ResultSetMetaData
    可用于获取关于 ResultSet 对象中列的类型和属性信息的对象
    ResultSetMetaData meta = rs.getMetaData();
    getColumnName(int column):获取指定列的名称
    getColumnLabel(int column):获取指定列的别名
    }
    54
    55
    getColumnCount():返回当前 ResultSet 对象中的列数。
    getColumnTypeName(int column):检索指定列的数据库特定的类型名称。
    getColumnDisplaySize(int column):指示指定列的最大标准宽度,以字符为单位。
    isNullable(int column):指示指定列中的值是否可以为 null。
    isAutoIncrement(int column):指示是否自动为指定列进行编号,这样这些列仍然是只读的。
    在这里插入图片描述
    问题1:得到结果集后, 如何知道该结果集中有哪些列 ? 列名是什么?
    需要使用一个描述 ResultSet 的对象, 即 ResultSetMetaData
    问题2:关于ResultSetMetaData

    1. 如何获取 ResultSetMetaData: 调用 ResultSet 的 getMetaData() 方法即可
    2. 获取 ResultSet 中有多少列:调用 ResultSetMetaData 的 getColumnCount() 方法
    3. 获取 ResultSet 每一列的列的别名是什么:调用 ResultSetMetaData 的getColumnLabel() 方法

    在这里插入图片描述

    资源的释放

    释放ResultSet, Statement,Connection。
    数据库连接(Connection)是非常稀有的资源,用完后必须马上释放,如果Connection不能及时正确的关闭将
    导致系统宕机。Connection的使用原则是尽量晚创建,尽量早的释放。
    可以在finally中关闭,保证及时其他代码出现异常,资源也一定能被关闭

    JDBC API小结

    两种思想
    面向接口编程的思想
    ORM思想(object relational mapping)
    一个数据表对应一个java类
    表中的一条记录对应java类的一个对象
    表中的一个字段对应java类的一个属性
    sql是需要结合列名和表的属性名来写。注意起别名。
    两种技术
    JDBC结果集的元数据:ResultSetMetaData
    获取列数:getColumnCount()
    获取列的别名:getColumnLabel()
    通过反射,创建指定类的对象,获取指定的属性并赋值

    操作BLOB类型字段

    1 MySQL BLOB类型
    MySQL中,BLOB是一个二进制大型对象,是一个可以存储大量数据的容器,它能容纳不同大小的数据。
    插入BLOB类型的数据必须使用PreparedStatement,因为BLOB类型的数据无法使用字符串拼接写的。在这里插入图片描述

    MySQL的四种BLOB类型(除了在存储的最大信息量上不同外,他们是等同的)

    实际使用中根据需要存入的数据大小定义不同的BLOB类型。
    需要注意的是:如果存储的文件过大,数据库的性能会下降。

    批量插入

    批量执行SQL语句

    当需要成批插入或者更新记录时,可以采用Java的批量更新机制,这一机制允许多条语句一次性提交给数据库批量处
    理。通常情况下比单独提交处理更有效率
    JDBC的批量处理语句包括下面三个方法:
    addBatch(String):添加需要批量处理的SQL语句或是参数;
    executeBatch():执行批量处理语句;
    clearBatch():清空缓存的数据
    通常我们会遇到两种批量执行SQL语句的情况:
    多条SQL语句的批量处理;
    一个SQL语句的批量传参;
    以批量插入数据为例

    //使用PreparedStatement实现批量数据的插入
    //方式一:使用Statement
    //
    
    import util.JDBCUtils;
    
    import java.io.IOException;
    import java.sql.Connection;
    import java.sql.PreparedStatement;
    import java.sql.SQLException;
    
    public class InsertTest {
    
        public static void main(String[] args) throws SQLException, IOException, ClassNotFoundException {
            insert3();
        }
    
        public static void insert1() throws SQLException, IOException, ClassNotFoundException {
            long startTime = System.currentTimeMillis();
            Connection connection = JDBCUtils.getConnection();
            String sql = "insert into goods(name)values(?)";
            PreparedStatement preparedStatement = connection.prepareStatement(sql);
            for (int i = 0; i < 20000; i++) {
                preparedStatement.setObject(1, "name_" + i);
                preparedStatement.execute();
            }
            long endTime = System.currentTimeMillis();
            System.out.println("花费的时间为:" + (endTime - startTime));//大约80多秒
            JDBCUtils.closeResource(connection, preparedStatement);
        }
        //批量插入的方式三
        //修改1: 使用 addBatch() / executeBatch() / clearBatch()
        //* 修改2:mysql服务器默认是关闭批处理的,我们需要通过一个参数,让mysql开启批处理的支持。
        //* ?rewriteBatchedStatements=true 写在配置文件的url后面
        //* 修改3:使用更新的mysql 驱动:mysql-connector-java-5.1.37-bin.jar
    
    
        public static void insert2() throws SQLException, IOException, ClassNotFoundException {
            long startTime = System.currentTimeMillis();
            Connection connection = JDBCUtils.getConnection();
            String sql = "insert into goods(name)values(?)";
            PreparedStatement preparedStatement = connection.prepareStatement(sql);
            for (int i = 0; i <= 20000; i++) {
                preparedStatement.setObject(1, "name_" + i);
                // preparedStatement.execute();
                //1.攒sql
                preparedStatement.addBatch();
                if (i % 500 == 0) {
                    //2.执行batch()
                    preparedStatement.executeBatch();
                    //3.清空batch()
                    preparedStatement.clearBatch();
                }
            }
            long endTime = System.currentTimeMillis();
            System.out.println("花费的时间为:" + (endTime - startTime));//大约80多秒
            JDBCUtils.closeResource(connection, preparedStatement);
        }
    
        //批量插入的方式四:设置连接不允许自动提交数据
        public static void insert3() throws SQLException, IOException, ClassNotFoundException {
            long startTime = System.currentTimeMillis();
            Connection connection = JDBCUtils.getConnection();
            //设置不允许自动提交数据
            connection.setAutoCommit(false);
            String sql = "insert into goods(name)values(?)";
            PreparedStatement preparedStatement = connection.prepareStatement(sql);
            for (int i = 0; i <= 20000; i++) {
                preparedStatement.setObject(1, "name_" + i);
                // preparedStatement.execute();
                //1.攒sql
                preparedStatement.addBatch();
                if (i % 500 == 0) {
                    //2.执行batch()
                    preparedStatement.executeBatch();
                    //3.清空batch()
                    preparedStatement.clearBatch();
                }
            }
            //提交数据
            connection.commit();
            long endTime = System.currentTimeMillis();
            System.out.println("花费的时间为:" + (endTime - startTime));//大约80多秒
            JDBCUtils.closeResource(connection, preparedStatement);
        }
    
    }
    
    

    前往 JDBC(Java数据库连接) 学习笔记 第三天

    展开全文
  • 首先需要在同一个局域网内的两台机器(当然也可以用台机器...1、先登录主机 A,执行如下命令赋予从机权限,如果有多台丛机,就执行多次:mysql>GRANT REPLICATION SLAVE ON *.* TO 'backup'@'192.168.1.101' I...

    首先需要在同一个局域网内的两台机器(当然也可以用一台机器虚拟两台机器出来),都安装上mysql服务。

    主机A: 192.168.1.100

    从机B: 192.168.1.101

    可以有多台从机。

    1、先登录主机 A,执行如下命令赋予从机权限,如果有多台丛机,就执行多次:

    mysql>GRANT REPLICATION SLAVE ON *.* TO 'backup'@'192.168.1.101' IDENTIFIED BY '123456';

    2、 打开主机A的my.cnf,输入如下配置参数:

    server-id = 1 #主机标示,整数

    log_bin = /var/log/mysql/mysql-bin.log #确保此文件可写

    read-only =0 #主机,读写都可以

    binlog-do-db =test #需要备份数据,多个写多行

    binlog-ignore-db =mysql #不需要备份的数据库,多个写多行

    3、打开从机B的my.cnf,输入如下配置参数:

    server-id = 2

    log_bin = /var/log/mysql/mysql-bin.log

    master-host =192.168.1.100

    master-user =backup

    master-pass =123456

    master-port =3306

    master-connect-retry=60 #如果从服务器发现主服务器断掉,重新连接的时间差(秒)

    replicate-do-db =test #只复制某个库

    replicate-ignore-db=mysql #不复制某个库

    4、同步数据库

    进过以上的配置,分别重启主机A和从机B,即可自动实现同步。

    5、验证

    在主机A中,mysql>show master statusG;

    在从机B中,mysql>show slave statusG;

    能看到大致这些内容

    File: mysql-bin.000001

    Position: 1374

    Binlog_Do_DB: test

    Binlog_Ignore_DB: mysql

    另外可以在主机A中,做一些INSERT, UPDATE, DELETE 操作,看看主机B中,是否已经被修改。

    展开全文
  • 该实现包括两个组件,个服务器和多个客户端(最少100个客户端)。 服务器从客户端接受传入的网络连接,从这些连接接收传入的流量,并通过发送回接收到的消息的哈希码,以接收到的消息的顺序响应客户端。 每个...
  • 疯狂java18

    2020-09-25 11:47:33
    ​ 当调用java命令运行java程序时,该命令将会启动一个java虚拟机进程,不管该java程序有复杂,该程序启动了多少进程,它们都处于该java虚拟机进程里。当系统出现以下情况时,JVM进程会被终止: 程序运行到...
  • Java多线程打辅助的三小伙子 还在用Synchronized?Atomic你了解不? 《求求大厂给Offer》:线程基础面试题 :ski:设计模式 给女朋友讲解什么是代理模式 包装模式就是这么简单啦 单例模式你会几种写法? 工厂...
  • java开源包1

    千次下载 热门讨论 2013-06-28 09:14:34
    MyBatchFramework 是一个开源的轻量级的用以创建可靠的易管理的批量作业的Java包,主要特点是线程、调度、JMX管理和批量执行报表,执行历史等。 SIP协议包 jSIP.tar jSIP这个Java包目标是用Java实现SIP(SIP:...
  • java开源包12

    热门讨论 2013-06-28 10:14:45
    MyBatchFramework 是一个开源的轻量级的用以创建可靠的易管理的批量作业的Java包,主要特点是线程、调度、JMX管理和批量执行报表,执行历史等。 SIP协议包 jSIP.tar jSIP这个Java包目标是用Java实现SIP(SIP:...
  • Java资源包01

    2016-08-31 09:16:25
    MyBatchFramework 是一个开源的轻量级的用以创建可靠的易管理的批量作业的Java包,主要特点是线程、调度、JMX管理和批量执行报表,执行历史等。 SIP协议包 jSIP.tar jSIP这个Java包目标是用Java实现SIP(SIP:...
  • java开源包101

    2016-07-13 10:11:08
    MyBatchFramework 是一个开源的轻量级的用以创建可靠的易管理的批量作业的Java包,主要特点是线程、调度、JMX管理和批量执行报表,执行历史等。 SIP协议包 jSIP.tar jSIP这个Java包目标是用Java实现SIP(SIP:...
  • java开源包11

    热门讨论 2013-06-28 10:10:38
    MyBatchFramework 是一个开源的轻量级的用以创建可靠的易管理的批量作业的Java包,主要特点是线程、调度、JMX管理和批量执行报表,执行历史等。 SIP协议包 jSIP.tar jSIP这个Java包目标是用Java实现SIP(SIP:...
  • java开源包2

    热门讨论 2013-06-28 09:17:39
    MyBatchFramework 是一个开源的轻量级的用以创建可靠的易管理的批量作业的Java包,主要特点是线程、调度、JMX管理和批量执行报表,执行历史等。 SIP协议包 jSIP.tar jSIP这个Java包目标是用Java实现SIP(SIP:...
  • java开源包3

    热门讨论 2013-06-28 09:20:52
    MyBatchFramework 是一个开源的轻量级的用以创建可靠的易管理的批量作业的Java包,主要特点是线程、调度、JMX管理和批量执行报表,执行历史等。 SIP协议包 jSIP.tar jSIP这个Java包目标是用Java实现SIP(SIP:...
  • java开源包6

    热门讨论 2013-06-28 09:48:32
    MyBatchFramework 是一个开源的轻量级的用以创建可靠的易管理的批量作业的Java包,主要特点是线程、调度、JMX管理和批量执行报表,执行历史等。 SIP协议包 jSIP.tar jSIP这个Java包目标是用Java实现SIP(SIP:...
  • java开源包5

    热门讨论 2013-06-28 09:38:46
    MyBatchFramework 是一个开源的轻量级的用以创建可靠的易管理的批量作业的Java包,主要特点是线程、调度、JMX管理和批量执行报表,执行历史等。 SIP协议包 jSIP.tar jSIP这个Java包目标是用Java实现SIP(SIP:...
  • java开源包10

    热门讨论 2013-06-28 10:06:40
    MyBatchFramework 是一个开源的轻量级的用以创建可靠的易管理的批量作业的Java包,主要特点是线程、调度、JMX管理和批量执行报表,执行历史等。 SIP协议包 jSIP.tar jSIP这个Java包目标是用Java实现SIP(SIP:...
  • java开源包4

    热门讨论 2013-06-28 09:26:54
    MyBatchFramework 是一个开源的轻量级的用以创建可靠的易管理的批量作业的Java包,主要特点是线程、调度、JMX管理和批量执行报表,执行历史等。 SIP协议包 jSIP.tar jSIP这个Java包目标是用Java实现SIP(SIP:...
  • java开源包8

    热门讨论 2013-06-28 09:55:26
    MyBatchFramework 是一个开源的轻量级的用以创建可靠的易管理的批量作业的Java包,主要特点是线程、调度、JMX管理和批量执行报表,执行历史等。 SIP协议包 jSIP.tar jSIP这个Java包目标是用Java实现SIP(SIP:...
  • java开源包9

    热门讨论 2013-06-28 09:58:55
    MyBatchFramework 是一个开源的轻量级的用以创建可靠的易管理的批量作业的Java包,主要特点是线程、调度、JMX管理和批量执行报表,执行历史等。 SIP协议包 jSIP.tar jSIP这个Java包目标是用Java实现SIP(SIP:...
  • java开源包7

    热门讨论 2013-06-28 09:52:16
    MyBatchFramework 是一个开源的轻量级的用以创建可靠的易管理的批量作业的Java包,主要特点是线程、调度、JMX管理和批量执行报表,执行历史等。 SIP协议包 jSIP.tar jSIP这个Java包目标是用Java实现SIP(SIP:...
  • JAVA上百实例源码以及开源项目

    千次下载 热门讨论 2016-01-03 17:37:40
     用JAVA开发的一小型的目录监视系统,系统会每5秒自动扫描一次需要监视的目录,可以用来监视目录中文件大小及文件增减数目的变化。 Java日期选择控件完整源代码 14目标文件 内容索引:JAVA源码,系统相关,日历,...
  • java面试题

    2018-04-27 19:55:38
    使用存储过程比单纯的SQL语句要快,可以用一个命令对象来调用存储过程。 索引优缺点? 答:索引可以提高对数据库中数据的检索,缺点是减慢了数据录入速度,同时也增加了数据库的尺寸大小。 什么是事务?什么是事锁? ...
  • Java目录监视器源程序 9目标文件 内容索引:JAVA源码,综合应用,目录监视 用JAVA开发的一小型的目录监视系统,系统会每5秒自动扫描一次需要监视的目录,可以用来监视目录中文件大小及文件增减数目的变化。 Java...
  • MySQL命令大全

    2018-01-15 11:19:17
     此操作使testuser能够在每一个test数据库中的表执行SELECT,INSERT和DELETE以及UPDATE查询操作。现在我们结束操作并退出MySQL客户程序:  mysql> exit  Bye9! 1:使用SHOW语句找出在服务器上当前存在什么...
  • java 面试题 总结

    2009-09-16 08:45:34
    与cgi的区别在于servlet处于服务器进程中,它通过多线程方式运行其service方法,个实例可以服务于多个请求,并且其实例一般不会销毁,而CGI对每个请求都产生新的进程,服务完成后就销毁,所以效率上低于servlet。...
  • 一次jsp接触(win32安装版) 线程程序设计 发邮件时终于可以通过sina的smtp验证了 发主流技术的评价之JSP 分别用java Application 及 JSP调用javaBean的例子 分布式对话服务器的管理 给javabeans的礼物一篇...
  • 批处理,介绍如何一次执行一批sql语句,这些sql语句用插入、更新和删除等相关操作; 提交和回滚transaction.java; 使用PreparedStatement,动态执行sql语句,UsingPreparedStatement.java; 读写二进制数据,Binary...

空空如也

空空如也

1 2 3 4 5 ... 8
收藏数 158
精华内容 63
关键字:

java连一次执行多个命令

java 订阅