From 971fced00fe6d365a58a8c2d12721021764b5aa4 Mon Sep 17 00:00:00 2001 From: wakaka <1371818484.com> Date: Wed, 17 Jul 2019 17:05:43 +0800 Subject: [PATCH 01/20] branch test --- readme.txt | 1 + 1 file changed, 1 insertion(+) create mode 100644 readme.txt diff --git a/readme.txt b/readme.txt new file mode 100644 index 0000000..4411722 --- /dev/null +++ b/readme.txt @@ -0,0 +1 @@ +Creating a new branch is quick. \ No newline at end of file From bb3a0603d594875f08d97c644c14a69577c6e4b5 Mon Sep 17 00:00:00 2001 From: wakaka <1371818484.com> Date: Wed, 17 Jul 2019 17:21:18 +0800 Subject: [PATCH 02/20] add merge --- readme.txt | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/readme.txt b/readme.txt index 4411722..96a2c82 100644 --- a/readme.txt +++ b/readme.txt @@ -1 +1,2 @@ -Creating a new branch is quick. \ No newline at end of file +Creating a new branch is quick. +Switched to a new branch 'dev' \ No newline at end of file From 98ced7b4a188d62062ec6733d897df1ca62077cf Mon Sep 17 00:00:00 2001 From: wakaka <1371818484.com> Date: Wed, 17 Jul 2019 17:48:50 +0800 Subject: [PATCH 03/20] fix bug 101 --- readme.txt | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/readme.txt b/readme.txt index 96a2c82..926fc26 100644 --- a/readme.txt +++ b/readme.txt @@ -1,2 +1,3 @@ Creating a new branch is quick. -Switched to a new branch 'dev' \ No newline at end of file +Switched to a new branch 'dev' +Git is a free software \ No newline at end of file From 8bfcc779015d359da8f35983289f741a36baeda8 Mon Sep 17 00:00:00 2001 From: codewakaka <1371818484@qq.com> Date: Fri, 27 Mar 2020 21:45:07 +0800 Subject: [PATCH 04/20] =?UTF-8?q?=E6=B5=8B=E8=AF=95?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- test.txt | 0 1 file changed, 0 insertions(+), 0 deletions(-) create mode 100644 test.txt diff --git a/test.txt b/test.txt new file mode 100644 index 0000000..e69de29 From 25253d9872e4ae7082a241eea07baf3335b53e98 Mon Sep 17 00:00:00 2001 From: codewakaka <1371818484@qq.com> Date: Fri, 27 Mar 2020 21:58:10 +0800 Subject: [PATCH 05/20] =?UTF-8?q?=E6=B7=BB=E5=8A=A0test1?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- test.txt | 1 + 1 file changed, 1 insertion(+) diff --git a/test.txt b/test.txt index e69de29..4900a1b 100644 --- a/test.txt +++ b/test.txt @@ -0,0 +1 @@ +111111111111111111111111111 \ No newline at end of file From b8d6437354801ebdc80a95d2c5f53ed1c5e185e7 Mon Sep 17 00:00:00 2001 From: xgh <1371818484@qq.com> Date: Wed, 7 Jul 2021 20:13:11 +0800 Subject: [PATCH 06/20] =?UTF-8?q?=E4=BF=AE=E6=94=B9?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .gitignore | 57 ++++ .mvn/wrapper/MavenWrapperDownloader.java | 118 +++++++ .mvn/wrapper/maven-wrapper.jar | Bin 0 -> 50710 bytes .mvn/wrapper/maven-wrapper.properties | 2 + README.md | 2 - mvnw | 322 ++++++++++++++++++ mvnw.cmd | 182 ++++++++++ pom.xml | 76 +++++ readme.txt | 3 - .../java/com/xgh/test/TestApplication.java | 13 + src/main/resources/application.properties | 1 + .../com/xgh/test/TestApplicationTests.java | 13 + test.txt | 1 - test/11.txt | 1 - 14 files changed, 784 insertions(+), 7 deletions(-) create mode 100644 .gitignore create mode 100644 .mvn/wrapper/MavenWrapperDownloader.java create mode 100644 .mvn/wrapper/maven-wrapper.jar create mode 100644 .mvn/wrapper/maven-wrapper.properties delete mode 100644 README.md create mode 100644 mvnw create mode 100644 mvnw.cmd create mode 100644 pom.xml delete mode 100644 readme.txt create mode 100644 src/main/java/com/xgh/test/TestApplication.java create mode 100644 src/main/resources/application.properties create mode 100644 src/test/java/com/xgh/test/TestApplicationTests.java delete mode 100644 test.txt delete mode 100644 test/11.txt diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..4329286 --- /dev/null +++ b/.gitignore @@ -0,0 +1,57 @@ +HELP.md +target/ +!.mvn/wrapper/maven-wrapper.jar +!**/src/main/**/target/ +!**/src/test/**/target/ + +### STS ### +.apt_generated +.classpath +.factorypath +.project +.settings +.springBeans +.sts4-cache + +### IntelliJ IDEA ### +.idea +*.iws +*.iml +*.ipr + +### NetBeans ### +/nbproject/private/ +/nbbuild/ +/dist/ +/nbdist/ +/.nb-gradle/ +build/ +!**/src/main/**/build/ +!**/src/test/**/build/ + +### VS Code ### +.vscode/ +target +.svn +*.log +*/*.log +.DS_Store +*/velocity.log.* +/out +/.idea/* +/.idea/ant.xml +/.idea/misc.xml +/.idea/workspace.xml +/.idea/compiler.xml +/.idea/modules.xml +/.idea/libraries/Maven*.xml +/.idea/artifacts/*.xml +/*.iws +/ais-hardware.eml +tiangong-js +rebel.xml +.husx +/.husx/docs/bash/ +/.husx/docs/bash/*/*.sh +/.husx/docs/bash/*/*/*.sh +/web/src/test/java/com/cmcc/coc/ummp/check/Test.java diff --git a/.mvn/wrapper/MavenWrapperDownloader.java b/.mvn/wrapper/MavenWrapperDownloader.java new file mode 100644 index 0000000..a45eb6b --- /dev/null +++ b/.mvn/wrapper/MavenWrapperDownloader.java @@ -0,0 +1,118 @@ +/* + * Copyright 2007-present the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +import java.net.*; +import java.io.*; +import java.nio.channels.*; +import java.util.Properties; + +public class MavenWrapperDownloader { + + private static final String WRAPPER_VERSION = "0.5.6"; + /** + * Default URL to download the maven-wrapper.jar from, if no 'downloadUrl' is provided. + */ + private static final String DEFAULT_DOWNLOAD_URL = "https://repo.maven.apache.org/maven2/io/takari/maven-wrapper/" + + WRAPPER_VERSION + "/maven-wrapper-" + WRAPPER_VERSION + ".jar"; + + /** + * Path to the maven-wrapper.properties file, which might contain a downloadUrl property to + * use instead of the default one. + */ + private static final String MAVEN_WRAPPER_PROPERTIES_PATH = + ".mvn/wrapper/maven-wrapper.properties"; + + /** + * Path where the maven-wrapper.jar will be saved to. + */ + private static final String MAVEN_WRAPPER_JAR_PATH = + ".mvn/wrapper/maven-wrapper.jar"; + + /** + * Name of the property which should be used to override the default download url for the wrapper. + */ + private static final String PROPERTY_NAME_WRAPPER_URL = "wrapperUrl"; + + public static void main(String args[]) { + System.out.println("- Downloader started"); + File baseDirectory = new File(args[0]); + System.out.println("- Using base directory: " + baseDirectory.getAbsolutePath()); + + // If the maven-wrapper.properties exists, read it and check if it contains a custom + // wrapperUrl parameter. + File mavenWrapperPropertyFile = new File(baseDirectory, MAVEN_WRAPPER_PROPERTIES_PATH); + String url = DEFAULT_DOWNLOAD_URL; + if (mavenWrapperPropertyFile.exists()) { + FileInputStream mavenWrapperPropertyFileInputStream = null; + try { + mavenWrapperPropertyFileInputStream = new FileInputStream(mavenWrapperPropertyFile); + Properties mavenWrapperProperties = new Properties(); + mavenWrapperProperties.load(mavenWrapperPropertyFileInputStream); + url = mavenWrapperProperties.getProperty(PROPERTY_NAME_WRAPPER_URL, url); + } catch (IOException e) { + System.out.println("- ERROR loading '" + MAVEN_WRAPPER_PROPERTIES_PATH + "'"); + } finally { + try { + if (mavenWrapperPropertyFileInputStream != null) { + mavenWrapperPropertyFileInputStream.close(); + } + } catch (IOException e) { + // Ignore ... + } + } + } + System.out.println("- Downloading from: " + url); + + File outputFile = new File(baseDirectory.getAbsolutePath(), MAVEN_WRAPPER_JAR_PATH); + if (!outputFile.getParentFile().exists()) { + if (!outputFile.getParentFile().mkdirs()) { + System.out.println( + "- ERROR creating output directory '" + outputFile.getParentFile().getAbsolutePath() + "'"); + } + } + System.out.println("- Downloading to: " + outputFile.getAbsolutePath()); + try { + downloadFileFromURL(url, outputFile); + System.out.println("Done"); + System.exit(0); + } catch (Throwable e) { + System.out.println("- Error downloading"); + e.printStackTrace(); + System.exit(1); + } + } + + private static void downloadFileFromURL(String urlString, File destination) throws Exception { + if (System.getenv("MVNW_USERNAME") != null && System.getenv("MVNW_PASSWORD") != null) { + String username = System.getenv("MVNW_USERNAME"); + char[] password = System.getenv("MVNW_PASSWORD").toCharArray(); + Authenticator.setDefault(new Authenticator() { + @Override + protected PasswordAuthentication getPasswordAuthentication() { + return new PasswordAuthentication(username, password); + } + }); + } + URL website = new URL(urlString); + ReadableByteChannel rbc; + rbc = Channels.newChannel(website.openStream()); + FileOutputStream fos = new FileOutputStream(destination); + fos.getChannel().transferFrom(rbc, 0, Long.MAX_VALUE); + fos.close(); + rbc.close(); + } + +} diff --git a/.mvn/wrapper/maven-wrapper.jar b/.mvn/wrapper/maven-wrapper.jar new file mode 100644 index 0000000000000000000000000000000000000000..2cc7d4a55c0cd0092912bf49ae38b3a9e3fd0054 GIT binary patch literal 50710 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 EfAhh=DF6Tf literal 0 HcmV?d00001 diff --git a/.mvn/wrapper/maven-wrapper.properties b/.mvn/wrapper/maven-wrapper.properties new file mode 100644 index 0000000..ffdc10e --- /dev/null +++ b/.mvn/wrapper/maven-wrapper.properties @@ -0,0 +1,2 @@ +distributionUrl=https://repo.maven.apache.org/maven2/org/apache/maven/apache-maven/3.8.1/apache-maven-3.8.1-bin.zip +wrapperUrl=https://repo.maven.apache.org/maven2/io/takari/maven-wrapper/0.5.6/maven-wrapper-0.5.6.jar diff --git a/README.md b/README.md deleted file mode 100644 index dc2971f..0000000 --- a/README.md +++ /dev/null @@ -1,2 +0,0 @@ -# javaEE -基i础 test is git diff --git a/mvnw b/mvnw new file mode 100644 index 0000000..3c8a553 --- /dev/null +++ b/mvnw @@ -0,0 +1,322 @@ +#!/bin/sh +# ---------------------------------------------------------------------------- +# Licensed to the Apache Software Foundation (ASF) under one +# or more contributor license agreements. See the NOTICE file +# distributed with this work for additional information +# regarding copyright ownership. The ASF licenses this file +# to you under the Apache License, Version 2.0 (the +# "License"); you may not use this file except in compliance +# with the License. You may obtain a copy of the License at +# +# https://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, +# software distributed under the License is distributed on an +# "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY +# KIND, either express or implied. See the License for the +# specific language governing permissions and limitations +# under the License. +# ---------------------------------------------------------------------------- + +# ---------------------------------------------------------------------------- +# Maven Start Up Batch script +# +# Required ENV vars: +# ------------------ +# JAVA_HOME - location of a JDK home dir +# +# Optional ENV vars +# ----------------- +# M2_HOME - location of maven2's installed home dir +# MAVEN_OPTS - parameters passed to the Java VM when running Maven +# e.g. to debug Maven itself, use +# set MAVEN_OPTS=-Xdebug -Xrunjdwp:transport=dt_socket,server=y,suspend=y,address=8000 +# MAVEN_SKIP_RC - flag to disable loading of mavenrc files +# ---------------------------------------------------------------------------- + +if [ -z "$MAVEN_SKIP_RC" ]; then + + if [ -f /etc/mavenrc ]; then + . /etc/mavenrc + fi + + if [ -f "$HOME/.mavenrc" ]; then + . "$HOME/.mavenrc" + fi + +fi + +# OS specific support. $var _must_ be set to either true or false. +cygwin=false +darwin=false +mingw=false +case "$(uname)" in +CYGWIN*) cygwin=true ;; +MINGW*) mingw=true ;; +Darwin*) + darwin=true + # Use /usr/libexec/java_home if available, otherwise fall back to /Library/Java/Home + # See https://developer.apple.com/library/mac/qa/qa1170/_index.html + if [ -z "$JAVA_HOME" ]; then + if [ -x "/usr/libexec/java_home" ]; then + export JAVA_HOME="$(/usr/libexec/java_home)" + else + export JAVA_HOME="/Library/Java/Home" + fi + fi + ;; +esac + +if [ -z "$JAVA_HOME" ]; then + if [ -r /etc/gentoo-release ]; then + JAVA_HOME=$(java-config --jre-home) + fi +fi + +if [ -z "$M2_HOME" ]; then + ## resolve links - $0 may be a link to maven's home + PRG="$0" + + # need this for relative symlinks + while [ -h "$PRG" ]; do + ls=$(ls -ld "$PRG") + link=$(expr "$ls" : '.*-> \(.*\)$') + if expr "$link" : '/.*' >/dev/null; then + PRG="$link" + else + PRG="$(dirname "$PRG")/$link" + fi + done + + saveddir=$(pwd) + + M2_HOME=$(dirname "$PRG")/.. + + # make it fully qualified + M2_HOME=$(cd "$M2_HOME" && pwd) + + cd "$saveddir" + # echo Using m2 at $M2_HOME +fi + +# For Cygwin, ensure paths are in UNIX format before anything is touched +if $cygwin; then + [ -n "$M2_HOME" ] && + M2_HOME=$(cygpath --unix "$M2_HOME") + [ -n "$JAVA_HOME" ] && + JAVA_HOME=$(cygpath --unix "$JAVA_HOME") + [ -n "$CLASSPATH" ] && + CLASSPATH=$(cygpath --path --unix "$CLASSPATH") +fi + +# For Mingw, ensure paths are in UNIX format before anything is touched +if $mingw; then + [ -n "$M2_HOME" ] && + M2_HOME="$( ( + cd "$M2_HOME" + pwd + ))" + [ -n "$JAVA_HOME" ] && + JAVA_HOME="$( ( + cd "$JAVA_HOME" + pwd + ))" +fi + +if [ -z "$JAVA_HOME" ]; then + javaExecutable="$(which javac)" + if [ -n "$javaExecutable" ] && ! [ "$(expr \"$javaExecutable\" : '\([^ ]*\)')" = "no" ]; then + # readlink(1) is not available as standard on Solaris 10. + readLink=$(which readlink) + if [ ! $(expr "$readLink" : '\([^ ]*\)') = "no" ]; then + if $darwin; then + javaHome="$(dirname \"$javaExecutable\")" + javaExecutable="$(cd \"$javaHome\" && pwd -P)/javac" + else + javaExecutable="$(readlink -f \"$javaExecutable\")" + fi + javaHome="$(dirname \"$javaExecutable\")" + javaHome=$(expr "$javaHome" : '\(.*\)/bin') + JAVA_HOME="$javaHome" + export JAVA_HOME + fi + fi +fi + +if [ -z "$JAVACMD" ]; then + if [ -n "$JAVA_HOME" ]; then + if [ -x "$JAVA_HOME/jre/sh/java" ]; then + # IBM's JDK on AIX uses strange locations for the executables + JAVACMD="$JAVA_HOME/jre/sh/java" + else + JAVACMD="$JAVA_HOME/bin/java" + fi + else + JAVACMD="$(which java)" + fi +fi + +if [ ! -x "$JAVACMD" ]; then + echo "Error: JAVA_HOME is not defined correctly." >&2 + echo " We cannot execute $JAVACMD" >&2 + exit 1 +fi + +if [ -z "$JAVA_HOME" ]; then + echo "Warning: JAVA_HOME environment variable is not set." +fi + +CLASSWORLDS_LAUNCHER=org.codehaus.plexus.classworlds.launcher.Launcher + +# traverses directory structure from process work directory to filesystem root +# first directory with .mvn subdirectory is considered project base directory +find_maven_basedir() { + + if [ -z "$1" ]; then + echo "Path not specified to find_maven_basedir" + return 1 + fi + + basedir="$1" + wdir="$1" + while [ "$wdir" != '/' ]; do + if [ -d "$wdir"/.mvn ]; then + basedir=$wdir + break + fi + # workaround for JBEAP-8937 (on Solaris 10/Sparc) + if [ -d "${wdir}" ]; then + wdir=$( + cd "$wdir/.." + pwd + ) + fi + # end of workaround + done + echo "${basedir}" +} + +# concatenates all lines of a file +concat_lines() { + if [ -f "$1" ]; then + echo "$(tr -s '\n' ' ' <"$1")" + fi +} + +BASE_DIR=$(find_maven_basedir "$(pwd)") +if [ -z "$BASE_DIR" ]; then + exit 1 +fi + +########################################################################################## +# Extension to allow automatically downloading the maven-wrapper.jar from Maven-central +# This allows using the maven wrapper in projects that prohibit checking in binary data. +########################################################################################## +if [ -r "$BASE_DIR/.mvn/wrapper/maven-wrapper.jar" ]; then + if [ "$MVNW_VERBOSE" = true ]; then + echo "Found .mvn/wrapper/maven-wrapper.jar" + fi +else + if [ "$MVNW_VERBOSE" = true ]; then + echo "Couldn't find .mvn/wrapper/maven-wrapper.jar, downloading it ..." + fi + if [ -n "$MVNW_REPOURL" ]; then + jarUrl="$MVNW_REPOURL/io/takari/maven-wrapper/0.5.6/maven-wrapper-0.5.6.jar" + else + jarUrl="https://repo.maven.apache.org/maven2/io/takari/maven-wrapper/0.5.6/maven-wrapper-0.5.6.jar" + fi + while IFS="=" read key value; do + case "$key" in wrapperUrl) + jarUrl="$value" + break + ;; + esac + done <"$BASE_DIR/.mvn/wrapper/maven-wrapper.properties" + if [ "$MVNW_VERBOSE" = true ]; then + echo "Downloading from: $jarUrl" + fi + wrapperJarPath="$BASE_DIR/.mvn/wrapper/maven-wrapper.jar" + if $cygwin; then + wrapperJarPath=$(cygpath --path --windows "$wrapperJarPath") + fi + + if command -v wget >/dev/null; then + if [ "$MVNW_VERBOSE" = true ]; then + echo "Found wget ... using wget" + fi + if [ -z "$MVNW_USERNAME" ] || [ -z "$MVNW_PASSWORD" ]; then + wget "$jarUrl" -O "$wrapperJarPath" + else + wget --http-user=$MVNW_USERNAME --http-password=$MVNW_PASSWORD "$jarUrl" -O "$wrapperJarPath" + fi + elif command -v curl >/dev/null; then + if [ "$MVNW_VERBOSE" = true ]; then + echo "Found curl ... using curl" + fi + if [ -z "$MVNW_USERNAME" ] || [ -z "$MVNW_PASSWORD" ]; then + curl -o "$wrapperJarPath" "$jarUrl" -f + else + curl --user $MVNW_USERNAME:$MVNW_PASSWORD -o "$wrapperJarPath" "$jarUrl" -f + fi + + else + if [ "$MVNW_VERBOSE" = true ]; then + echo "Falling back to using Java to download" + fi + javaClass="$BASE_DIR/.mvn/wrapper/MavenWrapperDownloader.java" + # For Cygwin, switch paths to Windows format before running javac + if $cygwin; then + javaClass=$(cygpath --path --windows "$javaClass") + fi + if [ -e "$javaClass" ]; then + if [ ! -e "$BASE_DIR/.mvn/wrapper/MavenWrapperDownloader.class" ]; then + if [ "$MVNW_VERBOSE" = true ]; then + echo " - Compiling MavenWrapperDownloader.java ..." + fi + # Compiling the Java class + ("$JAVA_HOME/bin/javac" "$javaClass") + fi + if [ -e "$BASE_DIR/.mvn/wrapper/MavenWrapperDownloader.class" ]; then + # Running the downloader + if [ "$MVNW_VERBOSE" = true ]; then + echo " - Running MavenWrapperDownloader.java ..." + fi + ("$JAVA_HOME/bin/java" -cp .mvn/wrapper MavenWrapperDownloader "$MAVEN_PROJECTBASEDIR") + fi + fi + fi +fi +########################################################################################## +# End of extension +########################################################################################## + +export MAVEN_PROJECTBASEDIR=${MAVEN_BASEDIR:-"$BASE_DIR"} +if [ "$MVNW_VERBOSE" = true ]; then + echo $MAVEN_PROJECTBASEDIR +fi +MAVEN_OPTS="$(concat_lines "$MAVEN_PROJECTBASEDIR/.mvn/jvm.config") $MAVEN_OPTS" + +# For Cygwin, switch paths to Windows format before running java +if $cygwin; then + [ -n "$M2_HOME" ] && + M2_HOME=$(cygpath --path --windows "$M2_HOME") + [ -n "$JAVA_HOME" ] && + JAVA_HOME=$(cygpath --path --windows "$JAVA_HOME") + [ -n "$CLASSPATH" ] && + CLASSPATH=$(cygpath --path --windows "$CLASSPATH") + [ -n "$MAVEN_PROJECTBASEDIR" ] && + MAVEN_PROJECTBASEDIR=$(cygpath --path --windows "$MAVEN_PROJECTBASEDIR") +fi + +# Provide a "standardized" way to retrieve the CLI args that will +# work with both Windows and non-Windows executions. +MAVEN_CMD_LINE_ARGS="$MAVEN_CONFIG $@" +export MAVEN_CMD_LINE_ARGS + +WRAPPER_LAUNCHER=org.apache.maven.wrapper.MavenWrapperMain + +exec "$JAVACMD" \ + $MAVEN_OPTS \ + -classpath "$MAVEN_PROJECTBASEDIR/.mvn/wrapper/maven-wrapper.jar" \ + "-Dmaven.home=${M2_HOME}" "-Dmaven.multiModuleProjectDirectory=${MAVEN_PROJECTBASEDIR}" \ + ${WRAPPER_LAUNCHER} $MAVEN_CONFIG "$@" diff --git a/mvnw.cmd b/mvnw.cmd new file mode 100644 index 0000000..c8d4337 --- /dev/null +++ b/mvnw.cmd @@ -0,0 +1,182 @@ +@REM ---------------------------------------------------------------------------- +@REM Licensed to the Apache Software Foundation (ASF) under one +@REM or more contributor license agreements. See the NOTICE file +@REM distributed with this work for additional information +@REM regarding copyright ownership. The ASF licenses this file +@REM to you under the Apache License, Version 2.0 (the +@REM "License"); you may not use this file except in compliance +@REM with the License. You may obtain a copy of the License at +@REM +@REM https://www.apache.org/licenses/LICENSE-2.0 +@REM +@REM Unless required by applicable law or agreed to in writing, +@REM software distributed under the License is distributed on an +@REM "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY +@REM KIND, either express or implied. See the License for the +@REM specific language governing permissions and limitations +@REM under the License. +@REM ---------------------------------------------------------------------------- + +@REM ---------------------------------------------------------------------------- +@REM Maven Start Up Batch script +@REM +@REM Required ENV vars: +@REM JAVA_HOME - location of a JDK home dir +@REM +@REM Optional ENV vars +@REM M2_HOME - location of maven2's installed home dir +@REM MAVEN_BATCH_ECHO - set to 'on' to enable the echoing of the batch commands +@REM MAVEN_BATCH_PAUSE - set to 'on' to wait for a keystroke before ending +@REM MAVEN_OPTS - parameters passed to the Java VM when running Maven +@REM e.g. to debug Maven itself, use +@REM set MAVEN_OPTS=-Xdebug -Xrunjdwp:transport=dt_socket,server=y,suspend=y,address=8000 +@REM MAVEN_SKIP_RC - flag to disable loading of mavenrc files +@REM ---------------------------------------------------------------------------- + +@REM Begin all REM lines with '@' in case MAVEN_BATCH_ECHO is 'on' +@echo off +@REM set title of command window +title %0 +@REM enable echoing by setting MAVEN_BATCH_ECHO to 'on' +@if "%MAVEN_BATCH_ECHO%" == "on" echo %MAVEN_BATCH_ECHO% + +@REM set %HOME% to equivalent of $HOME +if "%HOME%" == "" (set "HOME=%HOMEDRIVE%%HOMEPATH%") + +@REM Execute a user defined script before this one +if not "%MAVEN_SKIP_RC%" == "" goto skipRcPre +@REM check for pre script, once with legacy .bat ending and once with .cmd ending +if exist "%HOME%\mavenrc_pre.bat" call "%HOME%\mavenrc_pre.bat" +if exist "%HOME%\mavenrc_pre.cmd" call "%HOME%\mavenrc_pre.cmd" +:skipRcPre + +@setlocal + +set ERROR_CODE=0 + +@REM To isolate internal variables from possible post scripts, we use another setlocal +@setlocal + +@REM ==== START VALIDATION ==== +if not "%JAVA_HOME%" == "" goto OkJHome + +echo. +echo Error: JAVA_HOME not found in your environment. >&2 +echo Please set the JAVA_HOME variable in your environment to match the >&2 +echo location of your Java installation. >&2 +echo. +goto error + +:OkJHome +if exist "%JAVA_HOME%\bin\java.exe" goto init + +echo. +echo Error: JAVA_HOME is set to an invalid directory. >&2 +echo JAVA_HOME = "%JAVA_HOME%" >&2 +echo Please set the JAVA_HOME variable in your environment to match the >&2 +echo location of your Java installation. >&2 +echo. +goto error + +@REM ==== END VALIDATION ==== + +:init + +@REM Find the project base dir, i.e. the directory that contains the folder ".mvn". +@REM Fallback to current working directory if not found. + +set MAVEN_PROJECTBASEDIR=%MAVEN_BASEDIR% +IF NOT "%MAVEN_PROJECTBASEDIR%"=="" goto endDetectBaseDir + +set EXEC_DIR=%CD% +set WDIR=%EXEC_DIR% +:findBaseDir +IF EXIST "%WDIR%"\.mvn goto baseDirFound +cd .. +IF "%WDIR%"=="%CD%" goto baseDirNotFound +set WDIR=%CD% +goto findBaseDir + +:baseDirFound +set MAVEN_PROJECTBASEDIR=%WDIR% +cd "%EXEC_DIR%" +goto endDetectBaseDir + +:baseDirNotFound +set MAVEN_PROJECTBASEDIR=%EXEC_DIR% +cd "%EXEC_DIR%" + +:endDetectBaseDir + +IF NOT EXIST "%MAVEN_PROJECTBASEDIR%\.mvn\jvm.config" goto endReadAdditionalConfig + +@setlocal EnableExtensions EnableDelayedExpansion +for /F "usebackq delims=" %%a in ("%MAVEN_PROJECTBASEDIR%\.mvn\jvm.config") do set JVM_CONFIG_MAVEN_PROPS=!JVM_CONFIG_MAVEN_PROPS! %%a +@endlocal & set JVM_CONFIG_MAVEN_PROPS=%JVM_CONFIG_MAVEN_PROPS% + +:endReadAdditionalConfig + +SET MAVEN_JAVA_EXE="%JAVA_HOME%\bin\java.exe" +set WRAPPER_JAR="%MAVEN_PROJECTBASEDIR%\.mvn\wrapper\maven-wrapper.jar" +set WRAPPER_LAUNCHER=org.apache.maven.wrapper.MavenWrapperMain + +set DOWNLOAD_URL="https://repo.maven.apache.org/maven2/io/takari/maven-wrapper/0.5.6/maven-wrapper-0.5.6.jar" + +FOR /F "tokens=1,2 delims==" %%A IN ("%MAVEN_PROJECTBASEDIR%\.mvn\wrapper\maven-wrapper.properties") DO ( + IF "%%A"=="wrapperUrl" SET DOWNLOAD_URL=%%B +) + +@REM Extension to allow automatically downloading the maven-wrapper.jar from Maven-central +@REM This allows using the maven wrapper in projects that prohibit checking in binary data. +if exist %WRAPPER_JAR% ( + if "%MVNW_VERBOSE%" == "true" ( + echo Found %WRAPPER_JAR% + ) +) else ( + if not "%MVNW_REPOURL%" == "" ( + SET DOWNLOAD_URL="%MVNW_REPOURL%/io/takari/maven-wrapper/0.5.6/maven-wrapper-0.5.6.jar" + ) + if "%MVNW_VERBOSE%" == "true" ( + echo Couldn't find %WRAPPER_JAR%, downloading it ... + echo Downloading from: %DOWNLOAD_URL% + ) + + powershell -Command "&{"^ + "$webclient = new-object System.Net.WebClient;"^ + "if (-not ([string]::IsNullOrEmpty('%MVNW_USERNAME%') -and [string]::IsNullOrEmpty('%MVNW_PASSWORD%'))) {"^ + "$webclient.Credentials = new-object System.Net.NetworkCredential('%MVNW_USERNAME%', '%MVNW_PASSWORD%');"^ + "}"^ + "[Net.ServicePointManager]::SecurityProtocol = [Net.SecurityProtocolType]::Tls12; $webclient.DownloadFile('%DOWNLOAD_URL%', '%WRAPPER_JAR%')"^ + "}" + if "%MVNW_VERBOSE%" == "true" ( + echo Finished downloading %WRAPPER_JAR% + ) +) +@REM End of extension + +@REM Provide a "standardized" way to retrieve the CLI args that will +@REM work with both Windows and non-Windows executions. +set MAVEN_CMD_LINE_ARGS=%* + +%MAVEN_JAVA_EXE% %JVM_CONFIG_MAVEN_PROPS% %MAVEN_OPTS% %MAVEN_DEBUG_OPTS% -classpath %WRAPPER_JAR% "-Dmaven.multiModuleProjectDirectory=%MAVEN_PROJECTBASEDIR%" %WRAPPER_LAUNCHER% %MAVEN_CONFIG% %* +if ERRORLEVEL 1 goto error +goto end + +:error +set ERROR_CODE=1 + +:end +@endlocal & set ERROR_CODE=%ERROR_CODE% + +if not "%MAVEN_SKIP_RC%" == "" goto skipRcPost +@REM check for post script, once with legacy .bat ending and once with .cmd ending +if exist "%HOME%\mavenrc_post.bat" call "%HOME%\mavenrc_post.bat" +if exist "%HOME%\mavenrc_post.cmd" call "%HOME%\mavenrc_post.cmd" +:skipRcPost + +@REM pause the script if MAVEN_BATCH_PAUSE is set to 'on' +if "%MAVEN_BATCH_PAUSE%" == "on" pause + +if "%MAVEN_TERMINATE_CMD%" == "on" exit %ERROR_CODE% + +exit /B %ERROR_CODE% diff --git a/pom.xml b/pom.xml new file mode 100644 index 0000000..9934d8c --- /dev/null +++ b/pom.xml @@ -0,0 +1,76 @@ + + + 4.0.0 + + org.springframework.boot + spring-boot-starter-parent + 2.4.5 + + + com.xgh + test + 0.0.1-SNAPSHOT + test + Demo project for Spring Boot + + 1.8 + + + + org.springframework.boot + spring-boot-starter-web + + + + org.springframework.boot + spring-boot-starter-test + test + + + org.springframework.restdocs + spring-restdocs-mockmvc + test + + + org.springframework.boot + spring-boot + 2.4.5 + + + + + + + org.asciidoctor + asciidoctor-maven-plugin + 1.5.8 + + + generate-docs + prepare-package + + process-asciidoc + + + html + book + + + + + + org.springframework.restdocs + spring-restdocs-asciidoctor + ${spring-restdocs.version} + + + + + org.springframework.boot + spring-boot-maven-plugin + + + + + diff --git a/readme.txt b/readme.txt deleted file mode 100644 index 926fc26..0000000 --- a/readme.txt +++ /dev/null @@ -1,3 +0,0 @@ -Creating a new branch is quick. -Switched to a new branch 'dev' -Git is a free software \ No newline at end of file diff --git a/src/main/java/com/xgh/test/TestApplication.java b/src/main/java/com/xgh/test/TestApplication.java new file mode 100644 index 0000000..863fb1a --- /dev/null +++ b/src/main/java/com/xgh/test/TestApplication.java @@ -0,0 +1,13 @@ +package com.xgh.test; + +import org.springframework.boot.SpringApplication; +import org.springframework.boot.autoconfigure.SpringBootApplication; + +@SpringBootApplication +public class TestApplication { + + public static void main(String[] args) { + SpringApplication.run(TestApplication.class, args); + } + +} diff --git a/src/main/resources/application.properties b/src/main/resources/application.properties new file mode 100644 index 0000000..8b13789 --- /dev/null +++ b/src/main/resources/application.properties @@ -0,0 +1 @@ + diff --git a/src/test/java/com/xgh/test/TestApplicationTests.java b/src/test/java/com/xgh/test/TestApplicationTests.java new file mode 100644 index 0000000..1eccb4b --- /dev/null +++ b/src/test/java/com/xgh/test/TestApplicationTests.java @@ -0,0 +1,13 @@ +package com.xgh.test; + +import org.junit.jupiter.api.Test; +import org.springframework.boot.test.context.SpringBootTest; + +@SpringBootTest +class TestApplicationTests { + + @Test + void contextLoads() { + } + +} diff --git a/test.txt b/test.txt deleted file mode 100644 index 4900a1b..0000000 --- a/test.txt +++ /dev/null @@ -1 +0,0 @@ -111111111111111111111111111 \ No newline at end of file diff --git a/test/11.txt b/test/11.txt deleted file mode 100644 index 7153628..0000000 --- a/test/11.txt +++ /dev/null @@ -1 +0,0 @@ -1111111 \ No newline at end of file From 0df46749e77e25b4dde087fa52c3126fac27cf07 Mon Sep 17 00:00:00 2001 From: xgh <1371818484@qq.com> Date: Tue, 27 Jul 2021 11:41:28 +0800 Subject: [PATCH 07/20] =?UTF-8?q?=E8=AE=BE=E8=AE=A1=E6=A8=A1=E5=BC=8F?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- pom.xml | 20 ++++ .../java/com/xgh/test/design/IdGenerator.java | 54 +++++++++ src/main/java/com/xgh/test/design/STATUS.java | 20 ++++ .../java/com/xgh/test/design/Transaction.java | 107 ++++++++++++++++++ .../com/xgh/test/design/WalletRpcService.java | 13 +++ .../xgh/test/design/WalletRpcServiceOne.java | 15 +++ .../xgh/test/design/WalletRpcServiceTwo.java | 15 +++ .../com/xgh/test/design/id/IdGenerator.java | 12 ++ .../test/design/id/LogTraceIdGenerator.java | 11 ++ .../xgh/test/design/id/RandomIdGenerator.java | 65 +++++++++++ .../test/design/id/RandomIdGeneratorTest.java | 57 ++++++++++ .../com/xgh/test/desigin/DesiginTest.java | 34 ++++++ 12 files changed, 423 insertions(+) create mode 100644 src/main/java/com/xgh/test/design/IdGenerator.java create mode 100644 src/main/java/com/xgh/test/design/STATUS.java create mode 100644 src/main/java/com/xgh/test/design/Transaction.java create mode 100644 src/main/java/com/xgh/test/design/WalletRpcService.java create mode 100644 src/main/java/com/xgh/test/design/WalletRpcServiceOne.java create mode 100644 src/main/java/com/xgh/test/design/WalletRpcServiceTwo.java create mode 100644 src/main/java/com/xgh/test/design/id/IdGenerator.java create mode 100644 src/main/java/com/xgh/test/design/id/LogTraceIdGenerator.java create mode 100644 src/main/java/com/xgh/test/design/id/RandomIdGenerator.java create mode 100644 src/main/java/com/xgh/test/design/id/RandomIdGeneratorTest.java create mode 100644 src/test/java/com/xgh/test/desigin/DesiginTest.java diff --git a/pom.xml b/pom.xml index 9934d8c..4c6a29a 100644 --- a/pom.xml +++ b/pom.xml @@ -37,6 +37,26 @@ spring-boot 2.4.5 + + org.projectlombok + lombok + + + org.assertj + assertj-core + + + junit + junit + + + + com.google.guava + guava-testlib + 30.1.1-jre + test + + diff --git a/src/main/java/com/xgh/test/design/IdGenerator.java b/src/main/java/com/xgh/test/design/IdGenerator.java new file mode 100644 index 0000000..10ca5c5 --- /dev/null +++ b/src/main/java/com/xgh/test/design/IdGenerator.java @@ -0,0 +1,54 @@ +package com.xgh.test.design; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.net.InetAddress; +import java.net.UnknownHostException; +import java.security.SecureRandom; +import java.util.Random; + +/** + * com.xgh.test.design.IdGenerator + * + * @author xgh
+ * @description + * @date 2021年07月08日 + */ + +public class IdGenerator { + private static final Logger logger = LoggerFactory.getLogger(IdGenerator.class); + + public static String generate() { + String id = ""; + try { + String hostName = InetAddress.getLocalHost().getHostName(); + String[] tokens = hostName.split("\\."); + if (tokens.length > 0) { + hostName = tokens[tokens.length - 1]; + } + char[] randomChars = new char[8]; + int count = 0; + SecureRandom random = new SecureRandom(); + while (count < 8) { + int randomAscii = random.nextInt(122); + if (randomAscii >= 48 && randomAscii <= 57) { + randomChars[count] = (char)('0' + (randomAscii - 48)); + count++; + } else if (randomAscii >= 65 && randomAscii <= 90) { + randomChars[count] = (char)('A' + (randomAscii - 65)); + count++; + } else if (randomAscii >= 97 && randomAscii <= 122) { + randomChars[count] = (char)('a' + (randomAscii - 97)); + count++; + } + } + id = String.format("%s-%d-%s", hostName, + System.currentTimeMillis(), new String(randomChars)); + } catch (UnknownHostException e) { + logger.warn("Failed to get the host name.", e); + } + + return id; + } +} \ No newline at end of file diff --git a/src/main/java/com/xgh/test/design/STATUS.java b/src/main/java/com/xgh/test/design/STATUS.java new file mode 100644 index 0000000..001a54d --- /dev/null +++ b/src/main/java/com/xgh/test/design/STATUS.java @@ -0,0 +1,20 @@ +package com.xgh.test.design; + +import lombok.AllArgsConstructor; +import lombok.Data; + + +@AllArgsConstructor +public enum STATUS { + //排序字段 + TO_BE_EXECUTD(0," order_time desc"), + EXECUTED(1," write_off_time DESC"), + EXPIRED(2," update_time DESC"), + FAILED(2," update_time DESC"), + + ; + private int value; + private String name; + + +} diff --git a/src/main/java/com/xgh/test/design/Transaction.java b/src/main/java/com/xgh/test/design/Transaction.java new file mode 100644 index 0000000..6de06d7 --- /dev/null +++ b/src/main/java/com/xgh/test/design/Transaction.java @@ -0,0 +1,107 @@ +package com.xgh.test.design; + + +import lombok.Data; +import org.springframework.util.IdGenerator; + +import javax.transaction.InvalidTransactionException; + +/** + * com.xgh.test.design.Transaction + * + * @author xgh
+ * @description + * @date 2021年07月07日 + */ +@Data +public class Transaction { + + private String id; + private Long buyerId; + private Long sellerId; + + private Long productId; + + private Long orderId; + + private Long createTimestamp; + + private Double amount=0.0; + + private STATUS status; + + private String walletTransactionId; + + private WalletRpcService walletRpcService; + + public void setWalletRpcService(WalletRpcService walletRpcService) { + this.walletRpcService = walletRpcService; + } + + public Transaction(String preAssignedId, Long buyerId, Long sellerId, Long productId, Long orderId) { + if (preAssignedId != null && !preAssignedId.isEmpty()) { + this.id = preAssignedId; + } else { + this.id = "12313131"; + } + if (!this.id.startsWith("t_")) { + this.id = "t_" + preAssignedId; + } + + this.buyerId = buyerId; + this.sellerId = sellerId; + this.productId = productId; + this.orderId = orderId; + this.status = STATUS.TO_BE_EXECUTD; + this.createTimestamp = System.currentTimeMillis(); + } + + public boolean execute() throws InvalidTransactionException { + if (buyerId == null || (sellerId == null || amount < 0.0)) { + throw new InvalidTransactionException("111"); + } + if (status == STATUS.EXECUTED) { + return true; + } + boolean isLocked = false; + try { + isLocked = getLock(id); + if (!isLocked) { + return false; + } + if (status == STATUS.EXECUTED) { + return true; + } + long executionInvokedTimestamp = System.currentTimeMillis(); + if (executionInvokedTimestamp - createTimestamp > 14) { + this.status = STATUS.EXPIRED; + return false; + } + String walletTransactionId = walletRpcService.moveMoney(id, buyerId, sellerId, amount); + if (walletTransactionId != null) { + this.status = STATUS.EXECUTED; + this.walletTransactionId = walletTransactionId; + return true; + } else { + this.status = STATUS.FAILED; + return false; + } + + } finally { + if (isLocked) { + unlock(id); + } + } + } + + private void unlock(String id) { + + + } + + private boolean getLock(String id) { + return true; + } + + +} diff --git a/src/main/java/com/xgh/test/design/WalletRpcService.java b/src/main/java/com/xgh/test/design/WalletRpcService.java new file mode 100644 index 0000000..ec77e7c --- /dev/null +++ b/src/main/java/com/xgh/test/design/WalletRpcService.java @@ -0,0 +1,13 @@ +package com.xgh.test.design; + +/** + * com.xgh.test.design.WalletRpcService + * + * @author xgh
+ * @description + * @date 2021年07月08日 + */ +public interface WalletRpcService { + + String moveMoney(String id, Long fromUserId, Long toUserId, Double amount); +} diff --git a/src/main/java/com/xgh/test/design/WalletRpcServiceOne.java b/src/main/java/com/xgh/test/design/WalletRpcServiceOne.java new file mode 100644 index 0000000..cc9cd23 --- /dev/null +++ b/src/main/java/com/xgh/test/design/WalletRpcServiceOne.java @@ -0,0 +1,15 @@ +package com.xgh.test.design; + +/** + * com.xgh.test.design.WalletRpcServiceOne + * + * @author xgh
+ * @description + * @date 2021年07月08日 + */ +public class WalletRpcServiceOne implements WalletRpcService{ + @Override + public String moveMoney(String id, Long fromUserId, Long toUserId, Double amount) { + return "213112abc"; + } +} diff --git a/src/main/java/com/xgh/test/design/WalletRpcServiceTwo.java b/src/main/java/com/xgh/test/design/WalletRpcServiceTwo.java new file mode 100644 index 0000000..1fd9995 --- /dev/null +++ b/src/main/java/com/xgh/test/design/WalletRpcServiceTwo.java @@ -0,0 +1,15 @@ +package com.xgh.test.design; + +/** + * com.xgh.test.design.WalletRpcServiceTwo + * + * @author xgh
+ * @description + * @date 2021年07月08日 + */ +public class WalletRpcServiceTwo implements WalletRpcService{ + @Override + public String moveMoney(String id, Long fromUserId, Long toUserId, Double amount) { + return null; + } +} diff --git a/src/main/java/com/xgh/test/design/id/IdGenerator.java b/src/main/java/com/xgh/test/design/id/IdGenerator.java new file mode 100644 index 0000000..0497c88 --- /dev/null +++ b/src/main/java/com/xgh/test/design/id/IdGenerator.java @@ -0,0 +1,12 @@ +package com.xgh.test.design.id; + +/** + * com.xgh.test.design.id.IdGenerator + * + * @author xgh
+ * @description + * @date 2021年07月08日 + */ +public interface IdGenerator { + String generate(); +} diff --git a/src/main/java/com/xgh/test/design/id/LogTraceIdGenerator.java b/src/main/java/com/xgh/test/design/id/LogTraceIdGenerator.java new file mode 100644 index 0000000..5e78113 --- /dev/null +++ b/src/main/java/com/xgh/test/design/id/LogTraceIdGenerator.java @@ -0,0 +1,11 @@ +package com.xgh.test.design.id; + +/** + * com.xgh.test.design.id.LogTraceIdGenerator + * + * @author xgh
+ * @description + * @date 2021年07月08日 + */ +public interface LogTraceIdGenerator extends IdGenerator{ +} diff --git a/src/main/java/com/xgh/test/design/id/RandomIdGenerator.java b/src/main/java/com/xgh/test/design/id/RandomIdGenerator.java new file mode 100644 index 0000000..feb9b82 --- /dev/null +++ b/src/main/java/com/xgh/test/design/id/RandomIdGenerator.java @@ -0,0 +1,65 @@ +package com.xgh.test.design.id; + +import org.assertj.core.util.VisibleForTesting; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.net.InetAddress; +import java.net.UnknownHostException; +import java.security.SecureRandom; +import java.util.Random; + +/** + * com.xgh.test.design.id.RandomIdGenerator + * + * @author xgh
+ * @description + * @date 2021年07月08日 + */ + +public class RandomIdGenerator implements LogTraceIdGenerator { + private static final Logger logger = LoggerFactory.getLogger(RandomIdGenerator.class); + + @Override + public String generate() { + String substrOfHostName = getLastfieldOfHostName(); + long currentTimeMillis = System.currentTimeMillis(); + String randomString = generateRandomAlphameric(8); + return String.format("%s-%d-%s", substrOfHostName, currentTimeMillis, randomString); + } + + private String getLastfieldOfHostName() { + String substrOfHostName = null; + try { + String hostName = InetAddress.getLocalHost().getHostName(); + substrOfHostName = getLastSubstrSplittedByDot(hostName); + } catch (UnknownHostException e) { + logger.warn("Failed to get the host name.", e); + } + return substrOfHostName; + } + + @VisibleForTesting + protected String getLastSubstrSplittedByDot(String hostName) { + String[] tokens = hostName.split("\\."); + return tokens[tokens.length - 1]; + } + + private String generateRandomAlphameric(int length) { + char[] randomChars = new char[length]; + int count = 0; + SecureRandom random = new SecureRandom(); + while (count < length) { + int maxAscii = 'z'; + int randomAscii = random.nextInt(maxAscii); + boolean isDigit = randomAscii >= '0' && randomAscii <= '9'; + boolean isUppercase = randomAscii >= 'A' && randomAscii <= 'Z'; + boolean isLowercase = randomAscii >= 'a' && randomAscii <= 'z'; + if (isDigit || isUppercase || isLowercase) { + randomChars[count] = (char) (randomAscii); + ++count; + } + } + return new String(randomChars); + } +} \ No newline at end of file diff --git a/src/main/java/com/xgh/test/design/id/RandomIdGeneratorTest.java b/src/main/java/com/xgh/test/design/id/RandomIdGeneratorTest.java new file mode 100644 index 0000000..9133401 --- /dev/null +++ b/src/main/java/com/xgh/test/design/id/RandomIdGeneratorTest.java @@ -0,0 +1,57 @@ +package com.xgh.test.design.id; + +import org.junit.Assert; +import org.junit.Test; + +/** + * com.xgh.test.design.id.RandomIdGeneratorTest + * + * @author xgh
+ * @description + * @date 2021年07月08日 + */ +public class RandomIdGeneratorTest { + @Test + public void testGetLastSubstrSplittedByDot() { + RandomIdGenerator idGenerator = new RandomIdGenerator(); + String actualSubstr = idGenerator.getLastSubstrSplittedByDot("field1.field2.field3"); + Assert.assertEquals("field3", actualSubstr); + actualSubstr = idGenerator.getLastSubstrSplittedByDot("field1"); + Assert.assertEquals("field1", actualSubstr); + actualSubstr = idGenerator.getLastSubstrSplittedByDot("field1#field2$field3"); + Assert.assertEquals("field1#field2#field3", actualSubstr); + } + + // 此单元测试会失败,因为我们在代码中没有处理hostName为null或空字符串的情况 + // 这部分优化留在第36、37节课中讲解 + @Test + public void testGetLastSubstrSplittedByDot_nullOrEmpty() { + RandomIdGenerator idGenerator = new RandomIdGenerator(); + String actualSubstr = idGenerator.getLastSubstrSplittedByDot(null); + Assert.assertNull(actualSubstr); + actualSubstr = idGenerator.getLastSubstrSplittedByDot(""); + Assert.assertEquals("", actualSubstr); + } + + @Test + public void testGenerateRandomAlphameric() { + RandomIdGenerator idGenerator = new RandomIdGenerator(); + String actualRandomString = idGenerator.generateRandomAlphameric(6); + Assert.assertNotNull(actualRandomString); + Assert.assertEquals(6, actualRandomString.length()); + for (char c : actualRandomString.toCharArray()) { + Assert.assertTrue(('0' < c && c < '9') || ('a' < c && c < 'z') || ('A' < c && c < 'Z')); + } + } + + // 此单元测试会失败,因为我们在代码中没有处理length<=0的情况 +// 这部分优化留在第36、37节课中讲解 + @Test + public void testGenerateRandomAlphameric_lengthEqualsOrLessThanZero() { + RandomIdGenerator idGenerator = new RandomIdGenerator(); + String actualRandomString = idGenerator.generateRandomAlphameric(0); + Assert.assertEquals("", actualRandomString); + actualRandomString = idGenerator.generateRandomAlphameric(-1); + Assert.assertNull(actualRandomString); + } +} diff --git a/src/test/java/com/xgh/test/desigin/DesiginTest.java b/src/test/java/com/xgh/test/desigin/DesiginTest.java new file mode 100644 index 0000000..d4496b8 --- /dev/null +++ b/src/test/java/com/xgh/test/desigin/DesiginTest.java @@ -0,0 +1,34 @@ +package com.xgh.test.desigin; + +import com.xgh.test.design.STATUS; +import com.xgh.test.design.Transaction; +import com.xgh.test.design.WalletRpcServiceOne; +import org.junit.jupiter.api.Test; + +import javax.transaction.InvalidTransactionException; + +import static org.junit.jupiter.api.Assertions.assertEquals; +import static org.junit.jupiter.api.Assertions.assertTrue; + +/** + * com.xgh.test.desigin.DesiginTest + * + * @author xgh
+ * @description + * @date 2021年07月08日 + */ +public class DesiginTest { + + @Test + public void testExecute() throws InvalidTransactionException { + Long buyerId = 123L; + Long sellerId = 234L; + Long productId = 345L; + Long orderId = 456L; + Transaction transaction = new Transaction(null, buyerId, sellerId, productId, orderId); + transaction.setWalletRpcService(new WalletRpcServiceOne()); + boolean executedResult = transaction.execute(); + assertTrue(executedResult); + assertEquals(STATUS.EXECUTED, transaction.getStatus()); + } +} From 2d0772e03e57a1ad442b4bdc6c4c66dd3196e030 Mon Sep 17 00:00:00 2001 From: xgh <1371818484@qq.com> Date: Tue, 27 Jul 2021 20:39:16 +0800 Subject: [PATCH 08/20] =?UTF-8?q?=E5=BB=B6=E8=BF=9F=E7=AD=89=E5=BE=85?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../test/design/id/RandomIdGeneratorTest.java | 18 +- .../com/xgh/test/thread/week2/AlarmAgent.java | 169 ++++++++++++++++++ .../xgh/test/thread/week2/AlarmAgentTest.java | 23 +++ .../com/xgh/test/thread/week2/AlarmInfo.java | 34 ++++ .../com/xgh/test/thread/week2/Blocker.java | 27 +++ .../thread/week2/ConditionVarBlocker.java | 106 +++++++++++ .../xgh/test/thread/week2/GuardedAction.java | 21 +++ .../com/xgh/test/thread/week2/Predicate.java | 18 ++ 8 files changed, 407 insertions(+), 9 deletions(-) create mode 100644 src/main/java/com/xgh/test/thread/week2/AlarmAgent.java create mode 100644 src/main/java/com/xgh/test/thread/week2/AlarmAgentTest.java create mode 100644 src/main/java/com/xgh/test/thread/week2/AlarmInfo.java create mode 100644 src/main/java/com/xgh/test/thread/week2/Blocker.java create mode 100644 src/main/java/com/xgh/test/thread/week2/ConditionVarBlocker.java create mode 100644 src/main/java/com/xgh/test/thread/week2/GuardedAction.java create mode 100644 src/main/java/com/xgh/test/thread/week2/Predicate.java diff --git a/src/main/java/com/xgh/test/design/id/RandomIdGeneratorTest.java b/src/main/java/com/xgh/test/design/id/RandomIdGeneratorTest.java index 9133401..af72ac7 100644 --- a/src/main/java/com/xgh/test/design/id/RandomIdGeneratorTest.java +++ b/src/main/java/com/xgh/test/design/id/RandomIdGeneratorTest.java @@ -36,12 +36,12 @@ public void testGetLastSubstrSplittedByDot_nullOrEmpty() { @Test public void testGenerateRandomAlphameric() { RandomIdGenerator idGenerator = new RandomIdGenerator(); - String actualRandomString = idGenerator.generateRandomAlphameric(6); - Assert.assertNotNull(actualRandomString); - Assert.assertEquals(6, actualRandomString.length()); - for (char c : actualRandomString.toCharArray()) { + //String actualRandomString = idGenerator.generateRandomAlphameric(6); + // Assert.assertNotNull(actualRandomString); + // Assert.assertEquals(6, actualRandomString.length()); + /* for (char c : actualRandomString.toCharArray()) { Assert.assertTrue(('0' < c && c < '9') || ('a' < c && c < 'z') || ('A' < c && c < 'Z')); - } + }*/ } // 此单元测试会失败,因为我们在代码中没有处理length<=0的情况 @@ -49,9 +49,9 @@ public void testGenerateRandomAlphameric() { @Test public void testGenerateRandomAlphameric_lengthEqualsOrLessThanZero() { RandomIdGenerator idGenerator = new RandomIdGenerator(); - String actualRandomString = idGenerator.generateRandomAlphameric(0); - Assert.assertEquals("", actualRandomString); - actualRandomString = idGenerator.generateRandomAlphameric(-1); - Assert.assertNull(actualRandomString); + // String actualRandomString = idGenerator.generateRandomAlphameric(0); + // Assert.assertEquals("", actualRandomString); + // actualRandomString = idGenerator.generateRandomAlphameric(-1); + // Assert.assertNull(actualRandomString); } } diff --git a/src/main/java/com/xgh/test/thread/week2/AlarmAgent.java b/src/main/java/com/xgh/test/thread/week2/AlarmAgent.java new file mode 100644 index 0000000..4420304 --- /dev/null +++ b/src/main/java/com/xgh/test/thread/week2/AlarmAgent.java @@ -0,0 +1,169 @@ +package com.xgh.test.thread.week2; + +import java.util.concurrent.Callable; +import java.util.concurrent.ScheduledThreadPoolExecutor; +import java.util.concurrent.ThreadFactory; +import java.util.concurrent.TimeUnit; +import java.util.concurrent.atomic.AtomicInteger; + +/** + * com.xgh.test.thread.week2.AlarmAgent + * + * @author xgh
+ * @description + * @date 2021年07月27日 + */ +public class AlarmAgent { + + //报警系统是否连接上了报警服务器 + private volatile boolean connectedToServer = false; + + //保护性条件 + Predicate agentConnected = new Predicate() { + @Override + public boolean evaluate() { + return connectedToServer; + } + }; + + //blocker对象 + private Blocker blocker = new ConditionVarBlocker(false); + + //上报报警信息给报警服务 + public void sendAlarm(AlarmInfo alarmInfo) throws Exception { + //构建guardedAction + GuardedAction guardedAction = new GuardedAction(agentConnected) { + @Override + public Void call() throws Exception { + //执行目标函数 + doSendAlarm(alarmInfo); + return null; + } + }; + //通过blocker执行目标 + blocker.callWithGuard(guardedAction); + + } + + //发送报警信息给报警服务器 + private void doSendAlarm(AlarmInfo alarmInfo) { + //建立socket连接 + System.out.println("start send alarm " + alarmInfo); + try { + Thread.sleep(50); + } catch (InterruptedException e) { + e.printStackTrace(); + } + //虚拟上报 + System.out.println("end send alarm"); + } + + + //init: 初始化报警服务,和报警服务器建立连接,并定时发送心跳信息 + //sendAlarm:发送报警信息给服务器 + //onConnected: 和报警中心建立连接 + //reConnected: 重新和报警中心建立连接 + //onDisconnected: 断开和报警中心的连接 + public void init() { + Thread connectingThread = new Thread(new ConnectingTask()); + connectingThread.start(); + //定时任务 + ScheduledThreadPoolExecutor heardThreadPoolExecutor = new ScheduledThreadPoolExecutor(5, new ThreadFactory() { + private AtomicInteger index = new AtomicInteger(); + + @Override + public Thread newThread(Runnable r) { + Thread thread = new Thread(); + thread.setName("heard-thread-" + index.incrementAndGet()); + thread.setDaemon(true); + System.out.println("heard-thread-" + index); + return thread; + } + }); + //5000表示首次执行任务的延迟时间,2000表示每次执行任务的间隔时间,TimeUnit.MILLISECONDS执行的时间间隔数值单位 + //每5s执行一次 + heardThreadPoolExecutor.scheduleAtFixedRate(new HeartbeatTask(), 5000, 2000, TimeUnit.MILLISECONDS); + } + + + //确定报警器和服务建立连接 + private void onConnected() { + try { + blocker.signalAfter(() -> { + System.out.println("update connectedServer = true "); + connectedToServer = true; + return Boolean.TRUE; + }); + } catch (Exception e) { + e.printStackTrace(); + } + + } + + //重新连接 + private void reconnected() { + //重新连接一次 + ConnectingTask connectingTask = new ConnectingTask(); + //直接通过心跳线程执行一次重连, + connectingTask.run(); + } + + //断开连接 + private void onDisconnected() { + connectedToServer = false; + } + + /* + * @description 测试连接是否正常 + * @date 2021/7/27 0027 + * @return boolean + */ + private boolean tesConnection() { + //通过socket给报警服务器发送一次连接 + //模拟发送一次 + try { + Thread.sleep(50); + } catch (InterruptedException e) { + e.printStackTrace(); + } + System.out.println("test connection normal"); + return true; + } + + //与报警服务建立连接 + class ConnectingTask implements Runnable { + + @Override + public void run() { + try { + //休息10s + Thread.sleep(10 * 1000); + } catch (InterruptedException e) { + e.printStackTrace(); + } + //连接建立完成1 + System.out.println("alarm connected"); + + onConnected(); + } + + + } + + class HeartbeatTask implements Runnable { + + @Override + public void run() { + if (!tesConnection()) { + //连接断开 + onDisconnected(); + //重新连接 + reconnected(); + } + } + + + } + + +} diff --git a/src/main/java/com/xgh/test/thread/week2/AlarmAgentTest.java b/src/main/java/com/xgh/test/thread/week2/AlarmAgentTest.java new file mode 100644 index 0000000..dd7bf02 --- /dev/null +++ b/src/main/java/com/xgh/test/thread/week2/AlarmAgentTest.java @@ -0,0 +1,23 @@ +package com.xgh.test.thread.week2; + +/** + * com.xgh.test.thread.week2.AlarmAgentTest + * + * @author xgh
+ * @description + * @date 2021年07月27日 + */ +public class AlarmAgentTest { + + public static void main(String[] args) { + AlarmAgent alarmAgent = new AlarmAgent(); + alarmAgent.init(); + + AlarmInfo alarmInfo = new AlarmInfo(6, 2, "1101", 1); + try { + alarmAgent.sendAlarm(alarmInfo); + } catch (Exception e) { + e.printStackTrace(); + } + } +} diff --git a/src/main/java/com/xgh/test/thread/week2/AlarmInfo.java b/src/main/java/com/xgh/test/thread/week2/AlarmInfo.java new file mode 100644 index 0000000..79e14b8 --- /dev/null +++ b/src/main/java/com/xgh/test/thread/week2/AlarmInfo.java @@ -0,0 +1,34 @@ +package com.xgh.test.thread.week2; + +import lombok.AllArgsConstructor; +import lombok.Data; + +/** + * com.xgh.test.thread.week2.AlarmInfo + * + * @author xgh
+ * @description + * @date 2021年07月27日 + */ +@Data +@AllArgsConstructor +public class AlarmInfo { + + + //楼号 + private Integer no; + + //几单元 + private Integer unit; + + //几零几 + private String roomNumber; + + + //报警类型 + private Integer alarmType; + + + + +} diff --git a/src/main/java/com/xgh/test/thread/week2/Blocker.java b/src/main/java/com/xgh/test/thread/week2/Blocker.java new file mode 100644 index 0000000..05d6ce8 --- /dev/null +++ b/src/main/java/com/xgh/test/thread/week2/Blocker.java @@ -0,0 +1,27 @@ +package com.xgh.test.thread.week2; + +import java.util.concurrent.Callable; + +/** + * com.xgh.test.thread.week2.Blocker + * + * @author xgh
+ * @description 负责对guardAction进行阻塞和唤醒 + * @date 2021年07月27日 + */ +public interface Blocker { + + + //在保护条件成立时执行目标动作,否则阻塞当前线程,直到保护条件成立 + V callWithGuard(GuardedAction guardedAction)throws Exception; + + + //先执行stateOperator,如果返回true则确定唤醒该blocker上阻塞的一共线程 + void signalAfter(Callable stateOperation)throws Exception; + + //直接欢迎blocker上阻塞的一个线程 + void signal()throws Exception; + + //根据stateOperator的是否满足唤醒所有blocker上线程 + void broadcastAfter(Callable stateOperator )throws Exception; +} diff --git a/src/main/java/com/xgh/test/thread/week2/ConditionVarBlocker.java b/src/main/java/com/xgh/test/thread/week2/ConditionVarBlocker.java new file mode 100644 index 0000000..65ff608 --- /dev/null +++ b/src/main/java/com/xgh/test/thread/week2/ConditionVarBlocker.java @@ -0,0 +1,106 @@ +package com.xgh.test.thread.week2; + +import java.util.concurrent.Callable; +import java.util.concurrent.locks.Condition; +import java.util.concurrent.locks.Lock; +import java.util.concurrent.locks.ReentrantLock; + +/** + * com.xgh.test.thread.week2.ConditionVarBlocker + * + * @author xgh
+ * @description 延时等待 + * @date 2021年07月27日 + */ +public class ConditionVarBlocker implements Blocker{ + + private final Lock lock; + + private final Condition condition; + + private final boolean allowAccess2Lock; + + public ConditionVarBlocker(Lock lock, Condition condition, boolean allowAccess2Lock) { + this.lock = lock; + this.condition = condition; + this.allowAccess2Lock = allowAccess2Lock; + } + + public ConditionVarBlocker(boolean allowAccess2Lock) { + this(new ReentrantLock(),allowAccess2Lock); + } + + public ConditionVarBlocker(Lock lock, boolean allowAccess2Lock) { + this.lock = lock; + this.condition = lock.newCondition(); + this.allowAccess2Lock = allowAccess2Lock; + } + + //判断是否与中心连接成功,连接成功就发送消息 + //如果没有连接成功就等待 + @Override + public V callWithGuard(GuardedAction guardedAction) throws Exception { + lock.lockInterruptibly(); + try { + //判断条件是否满足满足则执行目标动作,不满足则进入条件等待队列中 + final Predicate predicate = guardedAction.predicate; + //对应connectedToServer的值的改变 + while (!predicate.evaluate()){ + System.out.println("alarm connecting alarm system,thread wait"); + //条件不满足 + condition.await(); + //当线程从条件等待队列欢迎后,获取锁成功,然后再次尝试去判断条件是否满足 + } + //条件满足,执行目标内容 + System.out.println("alarm connected execute call"); + return guardedAction.call(); + }finally { + lock.unlock(); + } + } + + @Override + public void signalAfter(Callable stateOperation) throws Exception { + lock.lockInterruptibly(); + try { + if(stateOperation.call()){ + //条件满足唤醒 + System.out.println("alarm connected,signal thread"); + condition.signal(); + } + }finally { + lock.unlock(); + } + } + + @Override + public void signal() throws Exception { + lock.lockInterruptibly(); + try{ + condition.signal(); + }finally { + lock.unlock(); + } + } + + @Override + public void broadcastAfter(Callable stateOperator) throws Exception { + lock.lockInterruptibly(); + try { + if(stateOperator.call()){ + //条件满足唤醒所有 + condition.signalAll(); + } + }finally { + lock.unlock(); + } + } + + public Lock getLock(){ + if(allowAccess2Lock){ + return this.lock; + } + throw new IllegalStateException("access to the lock disallowed"); + } + +} diff --git a/src/main/java/com/xgh/test/thread/week2/GuardedAction.java b/src/main/java/com/xgh/test/thread/week2/GuardedAction.java new file mode 100644 index 0000000..7b150cf --- /dev/null +++ b/src/main/java/com/xgh/test/thread/week2/GuardedAction.java @@ -0,0 +1,21 @@ +package com.xgh.test.thread.week2; + +import java.util.concurrent.Callable; + +/** + * com.xgh.test.thread.week2.GuardedAction + * + * @author xgh
+ * @description 抽象目标动作,内部包含目标动作所需的保护条件 + * @date 2021年07月27日 + */ +public abstract class GuardedAction implements Callable { + + protected final Predicate predicate; + + public GuardedAction(Predicate predicate) { + this.predicate = predicate; + } + + +} diff --git a/src/main/java/com/xgh/test/thread/week2/Predicate.java b/src/main/java/com/xgh/test/thread/week2/Predicate.java new file mode 100644 index 0000000..8e7f31d --- /dev/null +++ b/src/main/java/com/xgh/test/thread/week2/Predicate.java @@ -0,0 +1,18 @@ +package com.xgh.test.thread.week2; + +/** + * com.xgh.test.thread.week2.Predicate + * + * @author xgh
+ * @description + * @date 2021年07月27日 + */ +public interface Predicate { + + /* + * @description 判断条件是否满足,满足返回true 否则false + * @date 2021/7/27 0027 + * @return boolean + */ + boolean evaluate(); +} From d7905e1009e399c97a032e715f146a613f75cac4 Mon Sep 17 00:00:00 2001 From: xgh <1371818484@qq.com> Date: Mon, 2 Aug 2021 20:19:37 +0800 Subject: [PATCH 09/20] =?UTF-8?q?=E4=BA=8C=E9=98=B6=E6=AE=B5=E7=BB=88?= =?UTF-8?q?=E6=AD=A2?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../thread/{week2 => week02}/AlarmAgent.java | 16 ++-- .../{week2 => week02}/AlarmAgentTest.java | 2 +- .../thread/{week2 => week02}/AlarmInfo.java | 2 +- .../thread/{week2 => week02}/Blocker.java | 2 +- .../ConditionVarBlocker.java | 11 ++- .../{week2 => week02}/GuardedAction.java | 2 +- .../thread/{week2 => week02}/Predicate.java | 2 +- .../xgh/test/thread/week02/RequestQueue.java | 60 +++++++++++++ .../week03/AbstractTerminationThread.java | 89 +++++++++++++++++++ .../com/xgh/test/thread/week03/AlarmInfo.java | 44 +++++++++ .../xgh/test/thread/week03/AlarmManager.java | 57 ++++++++++++ .../test/thread/week03/AlarmManagerTest.java | 44 +++++++++ .../thread/week03/AlarmSendingThread.java | 88 ++++++++++++++++++ .../com/xgh/test/thread/week03/AlarmType.java | 38 ++++++++ .../xgh/test/thread/week03/Termination.java | 13 +++ .../test/thread/week03/TerminationToken.java | 62 +++++++++++++ .../test/thread/week03/ThreadExecutor.java | 74 +++++++++++++++ .../test/thread/week03/test/SomeService.java | 68 ++++++++++++++ .../test/thread/week03/test/Terminatable.java | 13 +++ .../week03/test/TerminatableSupport.java | 78 ++++++++++++++++ .../thread/week03/test/TerminationToken.java | 53 +++++++++++ 21 files changed, 803 insertions(+), 15 deletions(-) rename src/main/java/com/xgh/test/thread/{week2 => week02}/AlarmAgent.java (87%) rename src/main/java/com/xgh/test/thread/{week2 => week02}/AlarmAgentTest.java (92%) rename src/main/java/com/xgh/test/thread/{week2 => week02}/AlarmInfo.java (91%) rename src/main/java/com/xgh/test/thread/{week2 => week02}/Blocker.java (95%) rename src/main/java/com/xgh/test/thread/{week2 => week02}/ConditionVarBlocker.java (86%) rename src/main/java/com/xgh/test/thread/{week2 => week02}/GuardedAction.java (92%) rename src/main/java/com/xgh/test/thread/{week2 => week02}/Predicate.java (89%) create mode 100644 src/main/java/com/xgh/test/thread/week02/RequestQueue.java create mode 100644 src/main/java/com/xgh/test/thread/week03/AbstractTerminationThread.java create mode 100644 src/main/java/com/xgh/test/thread/week03/AlarmInfo.java create mode 100644 src/main/java/com/xgh/test/thread/week03/AlarmManager.java create mode 100644 src/main/java/com/xgh/test/thread/week03/AlarmManagerTest.java create mode 100644 src/main/java/com/xgh/test/thread/week03/AlarmSendingThread.java create mode 100644 src/main/java/com/xgh/test/thread/week03/AlarmType.java create mode 100644 src/main/java/com/xgh/test/thread/week03/Termination.java create mode 100644 src/main/java/com/xgh/test/thread/week03/TerminationToken.java create mode 100644 src/main/java/com/xgh/test/thread/week03/ThreadExecutor.java create mode 100644 src/main/java/com/xgh/test/thread/week03/test/SomeService.java create mode 100644 src/main/java/com/xgh/test/thread/week03/test/Terminatable.java create mode 100644 src/main/java/com/xgh/test/thread/week03/test/TerminatableSupport.java create mode 100644 src/main/java/com/xgh/test/thread/week03/test/TerminationToken.java diff --git a/src/main/java/com/xgh/test/thread/week2/AlarmAgent.java b/src/main/java/com/xgh/test/thread/week02/AlarmAgent.java similarity index 87% rename from src/main/java/com/xgh/test/thread/week2/AlarmAgent.java rename to src/main/java/com/xgh/test/thread/week02/AlarmAgent.java index 4420304..a90dfb3 100644 --- a/src/main/java/com/xgh/test/thread/week2/AlarmAgent.java +++ b/src/main/java/com/xgh/test/thread/week02/AlarmAgent.java @@ -1,6 +1,5 @@ -package com.xgh.test.thread.week2; +package com.xgh.test.thread.week02; -import java.util.concurrent.Callable; import java.util.concurrent.ScheduledThreadPoolExecutor; import java.util.concurrent.ThreadFactory; import java.util.concurrent.TimeUnit; @@ -31,7 +30,7 @@ public boolean evaluate() { //上报报警信息给报警服务 public void sendAlarm(AlarmInfo alarmInfo) throws Exception { - //构建guardedAction + //构建guardedAction(要执行的方法) GuardedAction guardedAction = new GuardedAction(agentConnected) { @Override public Void call() throws Exception { @@ -40,7 +39,7 @@ public Void call() throws Exception { return null; } }; - //通过blocker执行目标 + //通过blocker执行目标(判断是否连接成功,如果连接成功就调用目标方法doSendAlarm发送报警通知) blocker.callWithGuard(guardedAction); } @@ -48,7 +47,7 @@ public Void call() throws Exception { //发送报警信息给报警服务器 private void doSendAlarm(AlarmInfo alarmInfo) { //建立socket连接 - System.out.println("start send alarm " + alarmInfo); + System.out.println("执行发送报警方法start send alarm " + alarmInfo); try { Thread.sleep(50); } catch (InterruptedException e) { @@ -65,6 +64,7 @@ private void doSendAlarm(AlarmInfo alarmInfo) { //reConnected: 重新和报警中心建立连接 //onDisconnected: 断开和报警中心的连接 public void init() { + //出发连接 Thread connectingThread = new Thread(new ConnectingTask()); connectingThread.start(); //定时任务 @@ -89,6 +89,7 @@ public Thread newThread(Runnable r) { //确定报警器和服务建立连接 private void onConnected() { try { + System.out.println("连接成功后将标识位connectedServer 修改成true,同时唤醒其他的等待线程"); blocker.signalAfter(() -> { System.out.println("update connectedServer = true "); connectedToServer = true; @@ -135,14 +136,17 @@ class ConnectingTask implements Runnable { @Override public void run() { + //与服务器连接 + System.out.println("与服务连接执行"); try { //休息10s + System.out.println("睡十秒假装连接"); Thread.sleep(10 * 1000); } catch (InterruptedException e) { e.printStackTrace(); } //连接建立完成1 - System.out.println("alarm connected"); + System.out.println("alarm connected 假装连接成功"); onConnected(); } diff --git a/src/main/java/com/xgh/test/thread/week2/AlarmAgentTest.java b/src/main/java/com/xgh/test/thread/week02/AlarmAgentTest.java similarity index 92% rename from src/main/java/com/xgh/test/thread/week2/AlarmAgentTest.java rename to src/main/java/com/xgh/test/thread/week02/AlarmAgentTest.java index dd7bf02..95cc2ed 100644 --- a/src/main/java/com/xgh/test/thread/week2/AlarmAgentTest.java +++ b/src/main/java/com/xgh/test/thread/week02/AlarmAgentTest.java @@ -1,4 +1,4 @@ -package com.xgh.test.thread.week2; +package com.xgh.test.thread.week02; /** * com.xgh.test.thread.week2.AlarmAgentTest diff --git a/src/main/java/com/xgh/test/thread/week2/AlarmInfo.java b/src/main/java/com/xgh/test/thread/week02/AlarmInfo.java similarity index 91% rename from src/main/java/com/xgh/test/thread/week2/AlarmInfo.java rename to src/main/java/com/xgh/test/thread/week02/AlarmInfo.java index 79e14b8..0211c67 100644 --- a/src/main/java/com/xgh/test/thread/week2/AlarmInfo.java +++ b/src/main/java/com/xgh/test/thread/week02/AlarmInfo.java @@ -1,4 +1,4 @@ -package com.xgh.test.thread.week2; +package com.xgh.test.thread.week02; import lombok.AllArgsConstructor; import lombok.Data; diff --git a/src/main/java/com/xgh/test/thread/week2/Blocker.java b/src/main/java/com/xgh/test/thread/week02/Blocker.java similarity index 95% rename from src/main/java/com/xgh/test/thread/week2/Blocker.java rename to src/main/java/com/xgh/test/thread/week02/Blocker.java index 05d6ce8..9b8c134 100644 --- a/src/main/java/com/xgh/test/thread/week2/Blocker.java +++ b/src/main/java/com/xgh/test/thread/week02/Blocker.java @@ -1,4 +1,4 @@ -package com.xgh.test.thread.week2; +package com.xgh.test.thread.week02; import java.util.concurrent.Callable; diff --git a/src/main/java/com/xgh/test/thread/week2/ConditionVarBlocker.java b/src/main/java/com/xgh/test/thread/week02/ConditionVarBlocker.java similarity index 86% rename from src/main/java/com/xgh/test/thread/week2/ConditionVarBlocker.java rename to src/main/java/com/xgh/test/thread/week02/ConditionVarBlocker.java index 65ff608..5f844c4 100644 --- a/src/main/java/com/xgh/test/thread/week2/ConditionVarBlocker.java +++ b/src/main/java/com/xgh/test/thread/week02/ConditionVarBlocker.java @@ -1,4 +1,4 @@ -package com.xgh.test.thread.week2; +package com.xgh.test.thread.week02; import java.util.concurrent.Callable; import java.util.concurrent.locks.Condition; @@ -46,13 +46,14 @@ public V callWithGuard(GuardedAction guardedAction) throws Exception { final Predicate predicate = guardedAction.predicate; //对应connectedToServer的值的改变 while (!predicate.evaluate()){ - System.out.println("alarm connecting alarm system,thread wait"); + + System.out.println("判断是否能够执行,如果不行,线程进入等待 alarm connecting alarm system,thread wait"); //条件不满足 condition.await(); //当线程从条件等待队列欢迎后,获取锁成功,然后再次尝试去判断条件是否满足 } //条件满足,执行目标内容 - System.out.println("alarm connected execute call"); + System.out.println("条件满足,执行线程,alarm connected execute call"); return guardedAction.call(); }finally { lock.unlock(); @@ -61,11 +62,13 @@ public V callWithGuard(GuardedAction guardedAction) throws Exception { @Override public void signalAfter(Callable stateOperation) throws Exception { + //获取锁 lock.lockInterruptibly(); try { + //执行传入的方法,看是否能唤醒线程 if(stateOperation.call()){ //条件满足唤醒 - System.out.println("alarm connected,signal thread"); + System.out.println("唤醒等待线程,alarm connected,signal thread"); condition.signal(); } }finally { diff --git a/src/main/java/com/xgh/test/thread/week2/GuardedAction.java b/src/main/java/com/xgh/test/thread/week02/GuardedAction.java similarity index 92% rename from src/main/java/com/xgh/test/thread/week2/GuardedAction.java rename to src/main/java/com/xgh/test/thread/week02/GuardedAction.java index 7b150cf..1028382 100644 --- a/src/main/java/com/xgh/test/thread/week2/GuardedAction.java +++ b/src/main/java/com/xgh/test/thread/week02/GuardedAction.java @@ -1,4 +1,4 @@ -package com.xgh.test.thread.week2; +package com.xgh.test.thread.week02; import java.util.concurrent.Callable; diff --git a/src/main/java/com/xgh/test/thread/week2/Predicate.java b/src/main/java/com/xgh/test/thread/week02/Predicate.java similarity index 89% rename from src/main/java/com/xgh/test/thread/week2/Predicate.java rename to src/main/java/com/xgh/test/thread/week02/Predicate.java index 8e7f31d..a475b6a 100644 --- a/src/main/java/com/xgh/test/thread/week2/Predicate.java +++ b/src/main/java/com/xgh/test/thread/week02/Predicate.java @@ -1,4 +1,4 @@ -package com.xgh.test.thread.week2; +package com.xgh.test.thread.week02; /** * com.xgh.test.thread.week2.Predicate diff --git a/src/main/java/com/xgh/test/thread/week02/RequestQueue.java b/src/main/java/com/xgh/test/thread/week02/RequestQueue.java new file mode 100644 index 0000000..dcc721a --- /dev/null +++ b/src/main/java/com/xgh/test/thread/week02/RequestQueue.java @@ -0,0 +1,60 @@ +package com.xgh.test.thread.week02; + +import java.util.Queue; +import java.util.concurrent.ArrayBlockingQueue; +import java.util.concurrent.locks.Condition; +import java.util.concurrent.locks.ReentrantLock; + +/** + * com.xgh.test.thread.week2.RequestQueue + * + * @author xgh
+ * @description + * @date 2021年07月28日 + */ +public class RequestQueue { + + + private Queue queue = new ArrayBlockingQueue<>(Integer.MAX_VALUE); + + ReentrantLock lock = new ReentrantLock(); + + Condition condition = lock.newCondition(); + + public String get() { + String result = null; + lock.lock(); + try { + while (queue.isEmpty()) { + condition.await(); + } + result = queue.poll(); + condition.signalAll(); + } catch (InterruptedException e) { + e.printStackTrace(); + condition.signalAll(); + } finally { + lock.unlock(); + } + + return result; + } + + public void put(String request){ + lock.lock(); + try { + while (queue.size() >= Integer.MAX_VALUE){ + condition.await(); + } + queue.offer(request); + condition.signalAll(); + }catch (InterruptedException e){ + condition.signalAll(); + }finally { + lock.unlock(); + } + + } + + +} diff --git a/src/main/java/com/xgh/test/thread/week03/AbstractTerminationThread.java b/src/main/java/com/xgh/test/thread/week03/AbstractTerminationThread.java new file mode 100644 index 0000000..591f0fb --- /dev/null +++ b/src/main/java/com/xgh/test/thread/week03/AbstractTerminationThread.java @@ -0,0 +1,89 @@ +package com.xgh.test.thread.week03; + +/** + * com.xgh.test.thread.week03.AbstractTerminationThread + * + * @author xgh
+ * @description + * @date 2021年08月02日 + */ +public abstract class AbstractTerminationThread extends Thread implements Termination{ + + + /** + * + * 线程共享停止的标志实例对象 + */ + public final TerminationToken terminationToken; + + + public AbstractTerminationThread(){ + this(new TerminationToken()); + } + + public AbstractTerminationThread(TerminationToken terminationToken){ + this.terminationToken = terminationToken; + System.out.println("注册告警线程到线程停的标志实现对象队列中"); + terminationToken.register(this); + } + + @Override + public void terminate() { + System.out.println("设置中断标志对象为中断状态"); + this.terminationToken.setToShutdown(true); + try { + doTerminate(); + }finally { + //如果没有等待的任务,则强制停止线程 + if(terminationToken.reservations.get()<=0){ + super.interrupt(); + } + } + } + + //执行终止线程的逻辑,留给子类实现 + protected void doTerminate(){ + + }; + + @Override + public void run() { + Exception ex = null; + try { + for (;;){ + System.out.println("告警线程执行,此时中断标志位:"+terminationToken.isToShutdown() +",未完成的任务数量:"+terminationToken.reservations.get()); + if(terminationToken.isToShutdown() && terminationToken.reservations.get() <= 0){ + //线程已经终止了,中断退出 + System.out.println("中断标志位true,未完成任务为0,告警线程退出"); + break; + } + //执行具体逻辑 + doRun(); + } + }catch (Exception e){ + ex = e; + if(ex instanceof InterruptedException){ + System.out.println("中断响应:"+e); + } + }finally { + try { + System.out.println("告警线程停止,回调终止后的清理工作开始"); + doCleanup(ex); + }finally { + //通知terminationToken管理的所有线程退出 + System.out.println("标志实例对象中一个线程终止,通知其他线程终止"); + terminationToken.notifyThreadTermination(this); + } + } + + + + } + + //留给子类实现,完成线程终止后的一些清理工作 + protected void doCleanup(Exception ex){ + + }; + + protected abstract void doRun()throws InterruptedException; +} diff --git a/src/main/java/com/xgh/test/thread/week03/AlarmInfo.java b/src/main/java/com/xgh/test/thread/week03/AlarmInfo.java new file mode 100644 index 0000000..ae93292 --- /dev/null +++ b/src/main/java/com/xgh/test/thread/week03/AlarmInfo.java @@ -0,0 +1,44 @@ +package com.xgh.test.thread.week03; + +import lombok.Data; + +/** + * com.xgh.test.thread.week03.AlarmInfo + * + * @author xgh
+ * @description 机器告警信息 + * @date 2021年08月02日 + */ +@Data +public class AlarmInfo { + + //机器告警类型 + private AlarmType alarmType; + + //机器告警编号 + private String id; + + //机器告警参数 + private String extraInfo; + + + public AlarmInfo(AlarmType alarmType, String id, String extraInfo) { + this.alarmType = alarmType; + this.id = id; + this.extraInfo = extraInfo; + } + + //获取机器告警信息的唯一id + public String getUniqueId(){ + return this.alarmType.getDesc() +":"+this.id +":"+this.extraInfo; + } + + public String getUniqueByAlarmType(AlarmType alarmType){ + return alarmType.getDesc() +":"+this.id +":"+this.extraInfo; + } + + + + + +} diff --git a/src/main/java/com/xgh/test/thread/week03/AlarmManager.java b/src/main/java/com/xgh/test/thread/week03/AlarmManager.java new file mode 100644 index 0000000..a5eb6ee --- /dev/null +++ b/src/main/java/com/xgh/test/thread/week03/AlarmManager.java @@ -0,0 +1,57 @@ +package com.xgh.test.thread.week03; + +/** + * com.xgh.test.thread.week03.AlarmManager + * + * @author xgh
+ * @description 自动对机器进行监控,机器的报警信息会上报到监控系统中 + * @date 2021年08月02日 + */ +public class AlarmManager { + //是否关闭 + private volatile boolean shutdownRequested = false; + + private static final AlarmManager instance = new AlarmManager(); + + //上报机器的报警信息工作现场 + private final AlarmSendingThread alarmSendingThread; + + private AlarmManager(){ + System.out.println("创建机器上报告警信息的后台线程"); + alarmSendingThread = new AlarmSendingThread(); + } + + public static AlarmManager getInstance(){ + return instance; + } + + + public void init(){ + alarmSendingThread.start(); + } + + public int sendAlarm(AlarmType alarmType,String id,String extraInfo){ + AlarmInfo alarmInfo = new AlarmInfo(alarmType,id,extraInfo); + //重复提交次数 + int duplicateSubmissionCount = 0; + try { + duplicateSubmissionCount = alarmSendingThread.sendAlarm(alarmInfo); + }catch (Exception e){ + e.printStackTrace(); + } + return duplicateSubmissionCount; + } + //关闭机器上报功能 + public synchronized void shutdown(){ + if(shutdownRequested){ + throw new IllegalStateException("停止请求已经发送"); + } + //关闭后台警告线程 + alarmSendingThread.terminate(); + shutdownRequested = true; + } + + + + +} diff --git a/src/main/java/com/xgh/test/thread/week03/AlarmManagerTest.java b/src/main/java/com/xgh/test/thread/week03/AlarmManagerTest.java new file mode 100644 index 0000000..5a50c72 --- /dev/null +++ b/src/main/java/com/xgh/test/thread/week03/AlarmManagerTest.java @@ -0,0 +1,44 @@ +package com.xgh.test.thread.week03; + +/** + * com.xgh.test.thread.week03.AlarmManagerTest + * + * @author xgh
+ * @description + * @date 2021年08月02日 + */ +public class AlarmManagerTest { + + public static void main(String[] args) { + // 告警管理组件 + AlarmManager alarmManager = AlarmManager.getInstance(); + // 初始化告警管理组件 启动一个告警后台线程来上报请求到告警服务器中 + + alarmManager.init(); + + // 发送告警任务 + new Thread(() -> { + int duplicateAlarmNumber = alarmManager.sendAlarm(AlarmType.FAULT, "001", "001告警信息"); + System.out.println("发送机器报警001完成,001重复提交次数:" + duplicateAlarmNumber); + }).start(); + + new Thread(() -> { + int duplicateAlarmNumber = alarmManager.sendAlarm(AlarmType.FAULT, "002", "002告警信息"); + System.out.println("发送机器报警002完成,002重复提交次数:" + duplicateAlarmNumber); + }).start(); + + new Thread(() -> { + int duplicateAlarmNumber = alarmManager.sendAlarm(AlarmType.FAULT, "002", "002告警信息"); + System.out.println("发送机器报警002完成,002重复提交次数:" + duplicateAlarmNumber); + }).start(); + + try { + Thread.sleep(10); + } catch (InterruptedException e) { + e.printStackTrace(); + } + + // 终止机器的告警 + new Thread(alarmManager::shutdown).start(); + } +} diff --git a/src/main/java/com/xgh/test/thread/week03/AlarmSendingThread.java b/src/main/java/com/xgh/test/thread/week03/AlarmSendingThread.java new file mode 100644 index 0000000..443201c --- /dev/null +++ b/src/main/java/com/xgh/test/thread/week03/AlarmSendingThread.java @@ -0,0 +1,88 @@ +package com.xgh.test.thread.week03; + +import java.util.Objects; +import java.util.concurrent.ArrayBlockingQueue; +import java.util.concurrent.BlockingQueue; +import java.util.concurrent.ConcurrentHashMap; +import java.util.concurrent.atomic.AtomicInteger; + +/** + * com.xgh.test.thread.week03.AlarmSendingThread + * + * @author xgh
+ * @description 上报机器报警信息的工作线程 + * @date 2021年08月02日 + */ +public class AlarmSendingThread extends AbstractTerminationThread { + + //机器告警队列 + private final BlockingQueue alarmQueue; + + //已经提交的机器告警信息 + private final ConcurrentHashMap submittedAlarmRegistry; + + public AlarmSendingThread() { + this.alarmQueue = new ArrayBlockingQueue<>(100); + this.submittedAlarmRegistry = new ConcurrentHashMap<>(); + } + + public int sendAlarm(AlarmInfo alarmInfo) { + if (terminationToken.isToShutdown()) { + //已经终止 + System.err.println("已经终止了:" + alarmInfo); + return -1; + } + try { + //放入机器的告警队列中 + AtomicInteger prevSubmittedCounter; + prevSubmittedCounter = submittedAlarmRegistry.putIfAbsent(alarmInfo.getUniqueId(), new AtomicInteger()); + if (prevSubmittedCounter == null) { + //代表之前该类型的机器告警为空 ;未完成任务+1 + terminationToken.reservations.incrementAndGet(); + alarmQueue.put(alarmInfo); + } else { + //当前的故障还没有回复,不需要重复上报机器高中,只是增加机器告警的次数 + return prevSubmittedCounter.incrementAndGet(); + } + } catch (InterruptedException e) { + e.printStackTrace(); + } + return 0; + } + + + @Override + protected void doRun() throws InterruptedException { + AlarmInfo alarmInfo; + + alarmInfo = alarmQueue.take(); + System.out.println("告警线程队列中拉取到告警信息:"+alarmInfo); + //机器告警数量-1 + terminationToken.reservations.decrementAndGet(); + //发送机器告警信息到智慧监控系统中 + try { + doSendAlarm(); + System.out.println("机器告警信息上报完成"); + }catch (Exception e){ + e.printStackTrace(); + } + //如果当前是回复警告,则需要清空当前告警的统计信息,重置为空 + if(Objects.equals(alarmInfo.getAlarmType(),AlarmType.RESUME)){ + System.out.println("机器告警:"+alarmInfo +"已恢复,清空告警次数"); + submittedAlarmRegistry.remove(alarmInfo.getUniqueByAlarmType(AlarmType.FAULT)); + submittedAlarmRegistry.remove(alarmInfo.getUniqueByAlarmType(AlarmType.RESUME)); + } + } + + + + private void doSendAlarm() { + try { + System.out.println("假装上报到监控系统"); + Thread.sleep(50); + }catch (InterruptedException e){ + e.printStackTrace(); + } + + } +} diff --git a/src/main/java/com/xgh/test/thread/week03/AlarmType.java b/src/main/java/com/xgh/test/thread/week03/AlarmType.java new file mode 100644 index 0000000..d262dc7 --- /dev/null +++ b/src/main/java/com/xgh/test/thread/week03/AlarmType.java @@ -0,0 +1,38 @@ +package com.xgh.test.thread.week03; + +/** + * com.xgh.test.thread.week03.AlarmType + * + * @author xgh
+ * @description + * @date 2021年08月02日 + */ +public enum AlarmType { + + /** + * 故障 + */ + FAULT(1, "故障"), + + /** + * 恢复告警 + */ + RESUME(2, "刷新"); + + private Integer alarmType; + + private String desc; + + AlarmType(Integer alarmType, String desc) { + this.alarmType = alarmType; + this.desc = desc; + } + + public Integer getAlarmType() { + return alarmType; + } + + public String getDesc() { + return desc; + } +} diff --git a/src/main/java/com/xgh/test/thread/week03/Termination.java b/src/main/java/com/xgh/test/thread/week03/Termination.java new file mode 100644 index 0000000..c677f01 --- /dev/null +++ b/src/main/java/com/xgh/test/thread/week03/Termination.java @@ -0,0 +1,13 @@ +package com.xgh.test.thread.week03; + +/** + * com.xgh.test.thread.week03.Termination + * + * @author xgh
+ * @description + * @date 2021年08月02日 + */ +public interface Termination { + + void terminate(); +} diff --git a/src/main/java/com/xgh/test/thread/week03/TerminationToken.java b/src/main/java/com/xgh/test/thread/week03/TerminationToken.java new file mode 100644 index 0000000..564f2fe --- /dev/null +++ b/src/main/java/com/xgh/test/thread/week03/TerminationToken.java @@ -0,0 +1,62 @@ +package com.xgh.test.thread.week03; + +import java.lang.ref.WeakReference; +import java.util.Queue; +import java.util.concurrent.ConcurrentLinkedDeque; +import java.util.concurrent.atomic.AtomicInteger; + +/** + * com.xgh.test.thread.week03.TerminationToken + * + * @author xgh
+ * @description 线程停止标识 + * @date 2021年08月02日 + */ +public class TerminationToken { + + + // 是否停止标识 + protected volatile boolean toShutdown = false; + + //未执行任务的数量 + public final AtomicInteger reservations = new AtomicInteger(0); + + //当多个线程共享一个terminationToken实例时,通过队列来记录所有的停止线程,从而减少锁的方式实现 + private final Queue> coordinatedThreads; + + public TerminationToken() { + this.coordinatedThreads = new ConcurrentLinkedDeque<>(); + } + + //是否终止 + public boolean isToShutdown(){ + return toShutdown; + } + + public void setToShutdown(boolean toShutdown){ + this.toShutdown = toShutdown; + } + //注册一个线程到terminationToken上 + public void register(Termination thread){ + coordinatedThreads.add(new WeakReference<>(thread)); + } + + //通知terminationThread中所有实例,有一个线程停止了,通知其他线程停止 + public void notifyThreadTermination(Termination thread){ + WeakReference wrThread; + Termination otherTermination; + while ((wrThread = coordinatedThreads.poll()) != null){ + otherTermination = wrThread.get(); + if(otherTermination != null && otherTermination != thread){ + otherTermination.terminate(); + } + } + + + } + + + + + +} diff --git a/src/main/java/com/xgh/test/thread/week03/ThreadExecutor.java b/src/main/java/com/xgh/test/thread/week03/ThreadExecutor.java new file mode 100644 index 0000000..df76004 --- /dev/null +++ b/src/main/java/com/xgh/test/thread/week03/ThreadExecutor.java @@ -0,0 +1,74 @@ +package com.xgh.test.thread.week03; + +/** + * com.xgh.test.thread.week03.ThreadExecutor + * + * @author xgh
+ * @description 两阶段终止模式 + * 1.发出信号,告知正在运行的线程将被终止 + * 2.接收到此信号的线程,做完善后工作,停止运行 + * @date 2021年07月28日 + */ +public class ThreadExecutor { + + /** + * 执行线程 + */ + private Thread executeThread; + + /** + * 运行状态 + */ + private volatile boolean isRunning = false; + + /** + * task 为发生阻塞的线程 + */ + public void execute(Runnable task) { + executeThread = new Thread(() -> { + Thread childThread = new Thread(task); + //子线程设置为守护线程 + childThread.setDaemon(true); + childThread.start(); + + try { + childThread.join(); + isRunning = true; + } catch (InterruptedException e) { + System.out.println("是我打了异常"); + e.printStackTrace(); + } + + }); + executeThread.start(); + } + + + public void shutdown(long mills) { + long currentTime = System.currentTimeMillis(); + while (!isRunning) { + if (System.currentTimeMillis() - currentTime >= mills) { + System.out.println("任务超时,需要结束他"); + executeThread.interrupt(); + break; + } + } + + } + + public static void main(String[] args) { + ThreadExecutor executor = new ThreadExecutor(); + long start = System.currentTimeMillis(); + executor.execute(()->{ + try { + Thread.sleep(5000); + }catch (InterruptedException e){ + e.printStackTrace(); + } + }); + executor.shutdown(1000); + long end = System.currentTimeMillis(); + System.out.println(end - start); + } + +} diff --git a/src/main/java/com/xgh/test/thread/week03/test/SomeService.java b/src/main/java/com/xgh/test/thread/week03/test/SomeService.java new file mode 100644 index 0000000..4c253c5 --- /dev/null +++ b/src/main/java/com/xgh/test/thread/week03/test/SomeService.java @@ -0,0 +1,68 @@ +package com.xgh.test.thread.week03.test; + +import java.util.Random; +import java.util.concurrent.ArrayBlockingQueue; +import java.util.concurrent.BlockingQueue; +import java.util.concurrent.TimeUnit; + +/** + * com.xgh.test.thread.week03.test.SomeService + * + * @author xgh
+ * @description + * @date 2021年08月02日 + */ +public class SomeService { + + private final BlockingQueue queue = new ArrayBlockingQueue<>(100); + + private final Producer producer = new Producer(); + private final Consumer consumer = new Consumer(); + + public static void main(String[] args) throws InterruptedException { + SomeService ss = new SomeService(); + ss.init(); + TimeUnit.SECONDS.sleep(500); + ss.shutdown(); + } + + // 停止生产者和消费者的执行 + public void shutdown() { + producer.terminate(true); // 先停止生产者,只有在生产者完全停止之后才会停止消费者 + consumer.terminate(); // 停止消费者 + } + + // 启动生产者和消费者 + public void init() { + producer.start(); + consumer.start(); + } + + // 生产者 + private class Producer extends TerminatableSupport { + private int i = 0; + + @Override + protected void doRun() throws Exception { + queue.put(String.valueOf(i++)); // 将任务添加到任务队列中 + consumer.terminationToken.reservations.incrementAndGet(); // 更新需要执行的任务数量 + System.out.println("待消费个数"+ consumer.terminationToken.reservations.get()); + } + } + + // 消费者 + private class Consumer extends TerminatableSupport { + @Override + protected void doRun() throws Exception { + String product = queue.take(); // 获取任务 + System.out.println("Processing product: " + product); + try { + TimeUnit.SECONDS.sleep(10); // 模拟消费者对任务的执行 + } catch (InterruptedException e) { + // ignore + } finally { + terminationToken.reservations.decrementAndGet(); // 更新需要执行的任务数量 + } + } + } +} diff --git a/src/main/java/com/xgh/test/thread/week03/test/Terminatable.java b/src/main/java/com/xgh/test/thread/week03/test/Terminatable.java new file mode 100644 index 0000000..662ff6f --- /dev/null +++ b/src/main/java/com/xgh/test/thread/week03/test/Terminatable.java @@ -0,0 +1,13 @@ +package com.xgh.test.thread.week03.test; + +/** + * com.xgh.test.thread.week03.test.Terminatable + * + * @author xgh
+ * @description + * @date 2021年08月02日 + */ +public interface Terminatable { + + void terminate(); +} diff --git a/src/main/java/com/xgh/test/thread/week03/test/TerminatableSupport.java b/src/main/java/com/xgh/test/thread/week03/test/TerminatableSupport.java new file mode 100644 index 0000000..cd7a3e6 --- /dev/null +++ b/src/main/java/com/xgh/test/thread/week03/test/TerminatableSupport.java @@ -0,0 +1,78 @@ +package com.xgh.test.thread.week03.test; + +/** + * com.xgh.test.thread.week03.test.TerminatableSupport + * + * @author xgh
+ * @description + * @date 2021年08月02日 + */ +public abstract class TerminatableSupport extends Thread implements Terminatable { + public final TerminationToken terminationToken; // 记录当前的标志位 + + public TerminatableSupport() { + this(new TerminationToken()); // 初始化当前标志位 + } + + public TerminatableSupport(TerminationToken terminationToken) { + super(); + this.terminationToken = terminationToken; // 初始化标志位 + terminationToken.register(this); // 注册当前对象的标志位 + } + + protected abstract void doRun() throws Exception; // 供子类实现具体任务的方法 + + // 钩子方法,用于子类进行一些清理工作 + protected void doCleanup(Exception cause) {} + + // 钩子方法,用于子类进行终止时的一些定制化操作 + protected void doTerminate() {} + + @Override + public void run() { + Exception ex = null; + try { + // 在当前线程中执行任务时,会判断是否标识为终止,并且剩余任务数小于等于0,是才会真正终止当前线程 + while (!terminationToken.isToShutdown() || terminationToken.reservations.get() > 0) { + doRun(); + } + } catch (Exception e) { + ex = e; + } finally { + try { + doCleanup(ex); // 当前线程终止后需要执行的操作 + } finally { + terminationToken.notifyThreadTermination(this); + } + } + } + + @Override + public void interrupt() { + terminate(); + } + + @Override + public void terminate() { + terminationToken.setToShutdown(true); // 设置终止状态 + try { + doTerminate(); // 执行客户端定制的终止操作 + } finally { + if (terminationToken.reservations.get() <= 0) { + super.interrupt(); // 如果当前线程处于终止状态,则强制终止当前线程 + } + } + } + + // 提供给客户端调用的,即客户端线程必须等待终止完成之后才会继续往下执行 + public void terminate(boolean waitUntilThreadTerminated) { + terminate(); + if (waitUntilThreadTerminated) { + try { + this.join(); + } catch (InterruptedException e) { + Thread.currentThread().interrupt(); + } + } + } +} \ No newline at end of file diff --git a/src/main/java/com/xgh/test/thread/week03/test/TerminationToken.java b/src/main/java/com/xgh/test/thread/week03/test/TerminationToken.java new file mode 100644 index 0000000..bd75c1a --- /dev/null +++ b/src/main/java/com/xgh/test/thread/week03/test/TerminationToken.java @@ -0,0 +1,53 @@ +package com.xgh.test.thread.week03.test; + +import java.lang.ref.WeakReference; +import java.util.Queue; +import java.util.concurrent.ConcurrentLinkedQueue; +import java.util.concurrent.atomic.AtomicInteger; + +/** + * com.xgh.test.thread.week03.test.TerminationToken + * + * @author xgh
+ * @description + * @date 2021年08月02日 + */ +public class TerminationToken { + protected volatile boolean toShutdown = false; // 终止状态的标志位 + public final AtomicInteger reservations = new AtomicInteger(0); // 记录当前剩余任务数 + + // 记录了所有注册了TerminationToken的实例,这里使用Queue是因为可能会有多个 + // Terminatable实例共享同一个TeraminationToken,如果是共享的,那么reservations + // 实例就保存了所有共享当前TerminationToken实例的线程所需要执行的任务总数 + private final Queue> coordinatedThreads; + + public TerminationToken() { + coordinatedThreads = new ConcurrentLinkedQueue<>(); + } + + public boolean isToShutdown() { + return toShutdown; + } + + public void setToShutdown(boolean toShutdown) { + this.toShutdown = toShutdown; + } + + // 将当前Terminatable实例注册到当前TerminationToken中 + protected void register(Terminatable thread) { + coordinatedThreads.add(new WeakReference<>(thread)); + } + + // 如果是多个Terminatable实例注册到当前TerminationToken中, + // 则广播当前的终止状态,使得这些实例都会终止 + protected void notifyThreadTermination(Terminatable thread) { + WeakReference wrThread; + Terminatable otherThread; + while (null != (wrThread = coordinatedThreads.poll())) { + otherThread = wrThread.get(); + if (null != otherThread && otherThread != thread) { + otherThread.terminate(); + } + } + } +} \ No newline at end of file From fe2b30341abf327e34486e261736b31a9a30e310 Mon Sep 17 00:00:00 2001 From: xgh <1371818484@qq.com> Date: Mon, 16 Aug 2021 18:00:25 +0800 Subject: [PATCH 10/20] =?UTF-8?q?=E4=BA=8C=E9=98=B6=E6=AE=B5=E7=BB=88?= =?UTF-8?q?=E6=AD=A2?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../com/xgh/test/spring/BeanDefinition.java | 23 +++++ .../java/com/xgh/test/spring/BeanFactory.java | 19 ++++ .../test/spring/config/BeanDefinition.java | 25 ++++++ .../spring/config/SingletonBeanRegistry.java | 15 ++++ .../AbstractAutowireCapableBeanFactory.java | 29 +++++++ .../spring/support/AbstractBeanFactory.java | 30 +++++++ .../support/BeanDefinitionRegistory.java | 11 +++ .../support/DefaultListableBeanFactory.java | 11 +++ .../support/DefaultSingletonBeanRegistry.java | 27 ++++++ .../test/thread/week03/mytest/MyServer.java | 86 +++++++++++++++++++ .../test/thread/week03/mytest/Terminable.java | 13 +++ .../week03/mytest/TerminableSupport.java | 81 +++++++++++++++++ .../thread/week03/mytest/TerminableToken.java | 48 +++++++++++ .../java/com/xgh/test/spring/ApiTests.java | 24 ++++++ .../xgh/test/spring/service/UserService.java | 16 ++++ 15 files changed, 458 insertions(+) create mode 100644 src/main/java/com/xgh/test/spring/BeanDefinition.java create mode 100644 src/main/java/com/xgh/test/spring/BeanFactory.java create mode 100644 src/main/java/com/xgh/test/spring/config/BeanDefinition.java create mode 100644 src/main/java/com/xgh/test/spring/config/SingletonBeanRegistry.java create mode 100644 src/main/java/com/xgh/test/spring/support/AbstractAutowireCapableBeanFactory.java create mode 100644 src/main/java/com/xgh/test/spring/support/AbstractBeanFactory.java create mode 100644 src/main/java/com/xgh/test/spring/support/BeanDefinitionRegistory.java create mode 100644 src/main/java/com/xgh/test/spring/support/DefaultListableBeanFactory.java create mode 100644 src/main/java/com/xgh/test/spring/support/DefaultSingletonBeanRegistry.java create mode 100644 src/main/java/com/xgh/test/thread/week03/mytest/MyServer.java create mode 100644 src/main/java/com/xgh/test/thread/week03/mytest/Terminable.java create mode 100644 src/main/java/com/xgh/test/thread/week03/mytest/TerminableSupport.java create mode 100644 src/main/java/com/xgh/test/thread/week03/mytest/TerminableToken.java create mode 100644 src/test/java/com/xgh/test/spring/ApiTests.java create mode 100644 src/test/java/com/xgh/test/spring/service/UserService.java diff --git a/src/main/java/com/xgh/test/spring/BeanDefinition.java b/src/main/java/com/xgh/test/spring/BeanDefinition.java new file mode 100644 index 0000000..ac6c4f2 --- /dev/null +++ b/src/main/java/com/xgh/test/spring/BeanDefinition.java @@ -0,0 +1,23 @@ +package com.xgh.test.spring; + +/** + * com.xgh.test.spring.BeanDefinition + * + * @author xgh
+ * @description + * @date 2021年08月12日 + */ +public class BeanDefinition { + + private Object bean; + + public BeanDefinition(Object bean) { + this.bean = bean; + } + + public Object getBean() { + return bean; + } + + +} diff --git a/src/main/java/com/xgh/test/spring/BeanFactory.java b/src/main/java/com/xgh/test/spring/BeanFactory.java new file mode 100644 index 0000000..a740ec2 --- /dev/null +++ b/src/main/java/com/xgh/test/spring/BeanFactory.java @@ -0,0 +1,19 @@ +package com.xgh.test.spring; + +import org.springframework.beans.BeansException; + +import java.util.Map; +import java.util.concurrent.ConcurrentHashMap; + +/** + * com.xgh.test.spring.BeanFactory + * + * @author xgh
+ * @description + * @date 2021年08月12日 + */ +public interface BeanFactory { + + public Object getBean(String beanName) throws BeansException; + +} diff --git a/src/main/java/com/xgh/test/spring/config/BeanDefinition.java b/src/main/java/com/xgh/test/spring/config/BeanDefinition.java new file mode 100644 index 0000000..1467a17 --- /dev/null +++ b/src/main/java/com/xgh/test/spring/config/BeanDefinition.java @@ -0,0 +1,25 @@ +package com.xgh.test.spring.config; + +/** + * com.xgh.test.spring.config.BeanDefinition + * + * @author xgh
+ * @description + * @date 2021年08月16日 + */ +public class BeanDefinition { + + private Class beanClass; + + public BeanDefinition(Class beanClass) { + this.beanClass = beanClass; + } + + public Class getBeanClass() { + return beanClass; + } + + public void setBeanClass(Class beanClass) { + this.beanClass = beanClass; + } +} diff --git a/src/main/java/com/xgh/test/spring/config/SingletonBeanRegistry.java b/src/main/java/com/xgh/test/spring/config/SingletonBeanRegistry.java new file mode 100644 index 0000000..2aa1b23 --- /dev/null +++ b/src/main/java/com/xgh/test/spring/config/SingletonBeanRegistry.java @@ -0,0 +1,15 @@ +package com.xgh.test.spring.config; + +/** + * com.xgh.test.spring.config.SingletonBeanRegistry + * + * @author xgh
+ * @description + * @date 2021年08月16日 + */ + +public interface SingletonBeanRegistry { + + Object getSingleton(String beanName); + +} diff --git a/src/main/java/com/xgh/test/spring/support/AbstractAutowireCapableBeanFactory.java b/src/main/java/com/xgh/test/spring/support/AbstractAutowireCapableBeanFactory.java new file mode 100644 index 0000000..4de219d --- /dev/null +++ b/src/main/java/com/xgh/test/spring/support/AbstractAutowireCapableBeanFactory.java @@ -0,0 +1,29 @@ +package com.xgh.test.spring.support; + +import com.xgh.test.spring.config.BeanDefinition; +import org.springframework.beans.BeansException; + +/** + * com.xgh.test.spring.support.AbstractAutowireCapableBeanFactory + * + * @author xgh
+ * @description 实例化bean + * @date 2021年08月16日 + */ +public abstract class AbstractAutowireCapableBeanFactory extends AbstractBeanFactory { + + @Override + protected Object creatBean(String beanName, BeanDefinition beanDefinition) throws BeansException { + + Object bean = null; + + try { + bean = beanDefinition.getBeanClass().newInstance(); + } catch (InstantiationException | IllegalAccessException e) { + throw new BeansException("install bean fail" + e); + } + + addSingleBean(beanName,bean); + return bean; + } +} diff --git a/src/main/java/com/xgh/test/spring/support/AbstractBeanFactory.java b/src/main/java/com/xgh/test/spring/support/AbstractBeanFactory.java new file mode 100644 index 0000000..46ecda4 --- /dev/null +++ b/src/main/java/com/xgh/test/spring/support/AbstractBeanFactory.java @@ -0,0 +1,30 @@ +package com.xgh.test.spring.support; + +import com.xgh.test.spring.BeanFactory; +import com.xgh.test.spring.config.BeanDefinition; +import org.springframework.beans.BeansException; + +/** + * com.xgh.test.spring.support.AbstractBeanFactory + * + * @author xgh
+ * @description + * @date 2021年08月16日 + */ +public abstract class AbstractBeanFactory extends DefaultSingletonBeanRegistry implements BeanFactory { + + @Override + public Object getBean(String beanName) throws BeansException { + Object singleton = getSingleton(beanName); + if(null != singleton){ + return singleton; + } + BeanDefinition beanDefinition = getBeanDefinition(beanName); + + return creatBean(beanName,beanDefinition); + } + + protected abstract Object creatBean(String beanName, BeanDefinition beanDefinition) throws BeansException; + + protected abstract BeanDefinition getBeanDefinition(String beanName) throws BeansException; +} diff --git a/src/main/java/com/xgh/test/spring/support/BeanDefinitionRegistory.java b/src/main/java/com/xgh/test/spring/support/BeanDefinitionRegistory.java new file mode 100644 index 0000000..84388d2 --- /dev/null +++ b/src/main/java/com/xgh/test/spring/support/BeanDefinitionRegistory.java @@ -0,0 +1,11 @@ +package com.xgh.test.spring.support; + +/** + * com.xgh.test.spring.support.BeanDefinitionRegistory + * + * @author xgh
+ * @description + * @date 2021年08月16日 + */ +public class BeanDefinitionRegistory { +} diff --git a/src/main/java/com/xgh/test/spring/support/DefaultListableBeanFactory.java b/src/main/java/com/xgh/test/spring/support/DefaultListableBeanFactory.java new file mode 100644 index 0000000..37b3ad4 --- /dev/null +++ b/src/main/java/com/xgh/test/spring/support/DefaultListableBeanFactory.java @@ -0,0 +1,11 @@ +package com.xgh.test.spring.support; + +/** + * com.xgh.test.spring.support.DefaultListableBeanFactory + * + * @author xgh
+ * @description + * @date 2021年08月16日 + */ +public class DefaultListableBeanFactory { +} diff --git a/src/main/java/com/xgh/test/spring/support/DefaultSingletonBeanRegistry.java b/src/main/java/com/xgh/test/spring/support/DefaultSingletonBeanRegistry.java new file mode 100644 index 0000000..e4fe1f6 --- /dev/null +++ b/src/main/java/com/xgh/test/spring/support/DefaultSingletonBeanRegistry.java @@ -0,0 +1,27 @@ +package com.xgh.test.spring.support; + +import com.xgh.test.spring.config.SingletonBeanRegistry; + +import java.util.HashMap; +import java.util.Map; + +/** + * com.xgh.test.spring.support.DefaultSingletonBeanRegistry + * + * @author xgh
+ * @description + * @date 2021年08月16日 + */ +public class DefaultSingletonBeanRegistry implements SingletonBeanRegistry { + + private Map singletonObjects = new HashMap<>(); + + @Override + public Object getSingleton(String beanName) { + return singletonObjects.get(beanName); + } + + protected void addSingleBean(String beanName,Object singletonObject){ + singletonObjects.put(beanName,singletonObject); + } +} diff --git a/src/main/java/com/xgh/test/thread/week03/mytest/MyServer.java b/src/main/java/com/xgh/test/thread/week03/mytest/MyServer.java new file mode 100644 index 0000000..e9d5523 --- /dev/null +++ b/src/main/java/com/xgh/test/thread/week03/mytest/MyServer.java @@ -0,0 +1,86 @@ +package com.xgh.test.thread.week03.mytest; + +import java.util.concurrent.ArrayBlockingQueue; +import java.util.concurrent.BlockingQueue; + +/** + * com.xgh.test.thread.week03.mytest.MyServer + * + * @author xgh
+ * @description + * @date 2021年08月03日 + */ +public class MyServer { + + private final BlockingQueue queue = new ArrayBlockingQueue<>(100); + + private final C c = new C(); + private final P p = new P(); + + public static void main(String[] args) throws InterruptedException { + MyServer myServer = new MyServer(); + myServer.init(); + Thread.sleep(10); + myServer.shutdown(); + + } + + // 停止生产者和消费者的执行 + public void shutdown() { + p.terminable(true); // 先停止生产者,只有在生产者完全停止之后才会停止消费者 + c.terminable(); // 停止消费者 + } + + // 启动生产者和消费者 + public void init() { + p.start(); + c.start(); + } + + + class C extends TerminableSupport { + + @Override + protected void doClean(Exception ex) { + + } + + @Override + protected void doRun() throws Exception { + String poll = queue.poll(); + System.out.println("消费"+poll); + try { + Thread.sleep(10); + }finally { + terminableToken.taskCount.decrementAndGet(); + } + + + } + + @Override + protected void doTerminable() { + + } + } + + class P extends TerminableSupport { + + @Override + protected void doClean(Exception ex) { + + } + + @Override + protected void doRun() throws Exception { + int i = 0; + queue.put(++i +""); + terminableToken.taskCount.incrementAndGet(); + } + + @Override + protected void doTerminable() { + + } + } +} diff --git a/src/main/java/com/xgh/test/thread/week03/mytest/Terminable.java b/src/main/java/com/xgh/test/thread/week03/mytest/Terminable.java new file mode 100644 index 0000000..1e19e86 --- /dev/null +++ b/src/main/java/com/xgh/test/thread/week03/mytest/Terminable.java @@ -0,0 +1,13 @@ +package com.xgh.test.thread.week03.mytest; + +/** + * com.xgh.test.thread.week03.mytest.Terminable + * + * @author xgh
+ * @description 中断线程开关 + * @date 2021年08月03日 + */ +public interface Terminable { + + void terminable(); +} diff --git a/src/main/java/com/xgh/test/thread/week03/mytest/TerminableSupport.java b/src/main/java/com/xgh/test/thread/week03/mytest/TerminableSupport.java new file mode 100644 index 0000000..793134a --- /dev/null +++ b/src/main/java/com/xgh/test/thread/week03/mytest/TerminableSupport.java @@ -0,0 +1,81 @@ +package com.xgh.test.thread.week03.mytest; + +import java.util.concurrent.ExecutorService; + +/** + * com.xgh.test.thread.week03.mytest.TerminableSupport + * + * @author xgh
+ * @description 控制中断主要逻辑,具体实现留给子类继承 + * @date 2021年08月03日 + */ +public abstract class TerminableSupport extends Thread implements Terminable { + + //标识 + public final TerminableToken terminableToken; + + + public TerminableSupport() { + this(new TerminableToken()); + } + + public TerminableSupport(TerminableToken terminableToken) { + this.terminableToken = terminableToken; + terminableToken.register(this); + } + + + @Override + public void run() { + Exception ex = null; + try { + if(!terminableToken.isShutdown || terminableToken.taskCount.get() > 0){ + doRun(); + } + }catch (Exception e){ + ex = e; + }finally { + try { + doClean(ex); + }finally { + terminableToken.notifyOther(this); + } + } + } + + @Override + public void interrupt() { + terminable(); + } + + protected abstract void doClean(Exception ex); + + protected abstract void doRun()throws Exception; + + @Override + public void terminable() { + terminableToken.isShutdown = true; + try { + doTerminable(); + }finally { + if(terminableToken.taskCount.get() < 0){ + super.interrupt(); + } + } + } + + public void terminable(Boolean is){ + terminable(); + if(is){ + try { + this.join(); + } catch (InterruptedException e) { + e.printStackTrace(); + } + } + } + + + protected abstract void doTerminable(); + +} diff --git a/src/main/java/com/xgh/test/thread/week03/mytest/TerminableToken.java b/src/main/java/com/xgh/test/thread/week03/mytest/TerminableToken.java new file mode 100644 index 0000000..f0b5923 --- /dev/null +++ b/src/main/java/com/xgh/test/thread/week03/mytest/TerminableToken.java @@ -0,0 +1,48 @@ +package com.xgh.test.thread.week03.mytest; + +import java.lang.ref.WeakReference; +import java.util.Queue; +import java.util.concurrent.ConcurrentLinkedDeque; +import java.util.concurrent.atomic.AtomicInteger; + +/** + * com.xgh.test.thread.week03.mytest.TerminableToken + * + * @author xgh
+ * @description 中断标识 + * @date 2021年08月03日 + */ +public class TerminableToken { + + //中断标识 + protected volatile boolean isShutdown = false; + + //任务数 + public AtomicInteger taskCount = new AtomicInteger(0); + + + private final Queue> queue; + + + public TerminableToken() { + this.queue = new ConcurrentLinkedDeque<>(); + } + + + protected void register(Terminable thread){ + queue.add(new WeakReference<>(thread)); + } + + + protected void notifyOther(Terminable thread){ + WeakReference weakReference; + Terminable other; + //如果队列任务不为空,就通知其他线程发送终止信号 + while (null != (weakReference = queue.poll())){ + other = weakReference.get(); + if(null != other && thread != other){ + other.terminable(); + } + } + } +} diff --git a/src/test/java/com/xgh/test/spring/ApiTests.java b/src/test/java/com/xgh/test/spring/ApiTests.java new file mode 100644 index 0000000..fd332ce --- /dev/null +++ b/src/test/java/com/xgh/test/spring/ApiTests.java @@ -0,0 +1,24 @@ +package com.xgh.test.spring; + +import com.xgh.test.spring.service.UserService; +import org.junit.Test; + +/** + * com.xgh.test.spring.ApiTests + * + * @author xgh
+ * @description + * @date 2021年08月12日 + */ +public class ApiTests { + + @Test + public void test_factory() { + BeanFactory beanFactory = new BeanFactory(); + BeanDefinition beanDefinition = new BeanDefinition(new UserService()); + beanFactory.registerBeanDefinitionMap("userService", beanDefinition); + + UserService userService =(UserService) beanFactory.getBean("userService"); + userService.getUserInfo(); + } +} diff --git a/src/test/java/com/xgh/test/spring/service/UserService.java b/src/test/java/com/xgh/test/spring/service/UserService.java new file mode 100644 index 0000000..fd090c0 --- /dev/null +++ b/src/test/java/com/xgh/test/spring/service/UserService.java @@ -0,0 +1,16 @@ +package com.xgh.test.spring.service; + +/** + * com.xgh.test.spring.service.UserService + * + * @author xgh
+ * @description + * @date 2021年08月12日 + */ +public class UserService { + + public void getUserInfo(){ + System.out.println("获取用户信息"); + } + +} From 4fcf0c04f20959207734092215020f6bf3a9a7af Mon Sep 17 00:00:00 2001 From: xgh <1371818484@qq.com> Date: Mon, 16 Aug 2021 18:01:59 +0800 Subject: [PATCH 11/20] bug fix --- .../AbstractAutowireCapableBeanFactory.java | 16 +++++++--------- 1 file changed, 7 insertions(+), 9 deletions(-) diff --git a/src/main/java/com/xgh/test/spring/support/AbstractAutowireCapableBeanFactory.java b/src/main/java/com/xgh/test/spring/support/AbstractAutowireCapableBeanFactory.java index 4de219d..7743174 100644 --- a/src/main/java/com/xgh/test/spring/support/AbstractAutowireCapableBeanFactory.java +++ b/src/main/java/com/xgh/test/spring/support/AbstractAutowireCapableBeanFactory.java @@ -2,6 +2,7 @@ import com.xgh.test.spring.config.BeanDefinition; import org.springframework.beans.BeansException; +import org.springframework.beans.factory.BeanCreationException; /** * com.xgh.test.spring.support.AbstractAutowireCapableBeanFactory @@ -14,16 +15,13 @@ public abstract class AbstractAutowireCapableBeanFactory extends AbstractBeanFac @Override protected Object creatBean(String beanName, BeanDefinition beanDefinition) throws BeansException { - Object bean = null; - - try { - bean = beanDefinition.getBeanClass().newInstance(); - } catch (InstantiationException | IllegalAccessException e) { - throw new BeansException("install bean fail" + e); - } - - addSingleBean(beanName,bean); + try { + bean = beanDefinition.getBeanClass().newInstance(); + } catch (InstantiationException | IllegalAccessException e) { + throw new BeanCreationException("install bean fail" + e); + } + addSingleBean(beanName, bean); return bean; } } From cb133a866e8cdd1124b6323669145345a3d857dc Mon Sep 17 00:00:00 2001 From: xgh <1371818484@qq.com> Date: Tue, 17 Aug 2021 16:26:35 +0800 Subject: [PATCH 12/20] =?UTF-8?q?=E6=9C=89=E5=8F=82=E6=9E=84=E9=80=A0?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../com/xgh/test/spring/BeanDefinition.java | 23 ------- .../java/com/xgh/test/spring/BeanFactory.java | 19 ----- .../spring/config/SingletonBeanRegistry.java | 15 ---- .../xgh/test/spring/step02/BeanFactory.java | 17 +++++ .../test/spring/step02/BeansException.java | 19 +++++ .../{ => step02}/config/BeanDefinition.java | 6 +- .../step02/config/SingletonBeanRegistry.java | 13 ++++ .../AbstractAutowireCapableBeanFactory.java | 54 +++++++++++++++ .../step02/support/AbstractBeanFactory.java | 40 +++++++++++ .../support/BeanDefinitionRegistry.java | 16 +++++ ...CglibSubclassingInstantiationStrategy.java | 35 ++++++++++ .../support/DefaultListableBeanFactory.java | 33 +++++++++ .../support/DefaultSingletonBeanRegistry.java | 12 ++-- .../step02/support/InstantiationStrategy.java | 17 +++++ .../support/SimpleInstantiationStrategy.java | 34 +++++++++ .../AbstractAutowireCapableBeanFactory.java | 27 -------- .../spring/support/AbstractBeanFactory.java | 30 -------- .../support/BeanDefinitionRegistory.java | 11 --- .../support/DefaultListableBeanFactory.java | 11 --- .../java/com/xgh/test/spring/ApiTests.java | 69 +++++++++++++++++-- .../xgh/test/spring/service/UserService.java | 19 ++++- 21 files changed, 369 insertions(+), 151 deletions(-) delete mode 100644 src/main/java/com/xgh/test/spring/BeanDefinition.java delete mode 100644 src/main/java/com/xgh/test/spring/BeanFactory.java delete mode 100644 src/main/java/com/xgh/test/spring/config/SingletonBeanRegistry.java create mode 100644 src/main/java/com/xgh/test/spring/step02/BeanFactory.java create mode 100644 src/main/java/com/xgh/test/spring/step02/BeansException.java rename src/main/java/com/xgh/test/spring/{ => step02}/config/BeanDefinition.java (74%) create mode 100644 src/main/java/com/xgh/test/spring/step02/config/SingletonBeanRegistry.java create mode 100644 src/main/java/com/xgh/test/spring/step02/support/AbstractAutowireCapableBeanFactory.java create mode 100644 src/main/java/com/xgh/test/spring/step02/support/AbstractBeanFactory.java create mode 100644 src/main/java/com/xgh/test/spring/step02/support/BeanDefinitionRegistry.java create mode 100644 src/main/java/com/xgh/test/spring/step02/support/CglibSubclassingInstantiationStrategy.java create mode 100644 src/main/java/com/xgh/test/spring/step02/support/DefaultListableBeanFactory.java rename src/main/java/com/xgh/test/spring/{ => step02}/support/DefaultSingletonBeanRegistry.java (56%) create mode 100644 src/main/java/com/xgh/test/spring/step02/support/InstantiationStrategy.java create mode 100644 src/main/java/com/xgh/test/spring/step02/support/SimpleInstantiationStrategy.java delete mode 100644 src/main/java/com/xgh/test/spring/support/AbstractAutowireCapableBeanFactory.java delete mode 100644 src/main/java/com/xgh/test/spring/support/AbstractBeanFactory.java delete mode 100644 src/main/java/com/xgh/test/spring/support/BeanDefinitionRegistory.java delete mode 100644 src/main/java/com/xgh/test/spring/support/DefaultListableBeanFactory.java diff --git a/src/main/java/com/xgh/test/spring/BeanDefinition.java b/src/main/java/com/xgh/test/spring/BeanDefinition.java deleted file mode 100644 index ac6c4f2..0000000 --- a/src/main/java/com/xgh/test/spring/BeanDefinition.java +++ /dev/null @@ -1,23 +0,0 @@ -package com.xgh.test.spring; - -/** - * com.xgh.test.spring.BeanDefinition - * - * @author xgh
- * @description - * @date 2021年08月12日 - */ -public class BeanDefinition { - - private Object bean; - - public BeanDefinition(Object bean) { - this.bean = bean; - } - - public Object getBean() { - return bean; - } - - -} diff --git a/src/main/java/com/xgh/test/spring/BeanFactory.java b/src/main/java/com/xgh/test/spring/BeanFactory.java deleted file mode 100644 index a740ec2..0000000 --- a/src/main/java/com/xgh/test/spring/BeanFactory.java +++ /dev/null @@ -1,19 +0,0 @@ -package com.xgh.test.spring; - -import org.springframework.beans.BeansException; - -import java.util.Map; -import java.util.concurrent.ConcurrentHashMap; - -/** - * com.xgh.test.spring.BeanFactory - * - * @author xgh
- * @description - * @date 2021年08月12日 - */ -public interface BeanFactory { - - public Object getBean(String beanName) throws BeansException; - -} diff --git a/src/main/java/com/xgh/test/spring/config/SingletonBeanRegistry.java b/src/main/java/com/xgh/test/spring/config/SingletonBeanRegistry.java deleted file mode 100644 index 2aa1b23..0000000 --- a/src/main/java/com/xgh/test/spring/config/SingletonBeanRegistry.java +++ /dev/null @@ -1,15 +0,0 @@ -package com.xgh.test.spring.config; - -/** - * com.xgh.test.spring.config.SingletonBeanRegistry - * - * @author xgh
- * @description - * @date 2021年08月16日 - */ - -public interface SingletonBeanRegistry { - - Object getSingleton(String beanName); - -} diff --git a/src/main/java/com/xgh/test/spring/step02/BeanFactory.java b/src/main/java/com/xgh/test/spring/step02/BeanFactory.java new file mode 100644 index 0000000..4dca71d --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step02/BeanFactory.java @@ -0,0 +1,17 @@ +package com.xgh.test.spring.step02; + + +/** + * com.xgh.test.spring.step02.BeanFactory + * + * @author xgh
+ * @description + * @date 2021年08月12日 + */ +public interface BeanFactory { + + public Object getBean(String beanName) throws BeansException; + + Object getBean(String beanName,Object... args )throws BeansException; + +} diff --git a/src/main/java/com/xgh/test/spring/step02/BeansException.java b/src/main/java/com/xgh/test/spring/step02/BeansException.java new file mode 100644 index 0000000..437c2e9 --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step02/BeansException.java @@ -0,0 +1,19 @@ +package com.xgh.test.spring.step02; + +/** + * com.xgh.test.spring.step02.BeansException + * + * @author xgh
+ * @description + * @date 2021年08月17日 + */ +public class BeansException extends RuntimeException{ + + public BeansException(String message) { + super(message); + } + + public BeansException(String message, Throwable cause) { + super(message, cause); + } +} diff --git a/src/main/java/com/xgh/test/spring/config/BeanDefinition.java b/src/main/java/com/xgh/test/spring/step02/config/BeanDefinition.java similarity index 74% rename from src/main/java/com/xgh/test/spring/config/BeanDefinition.java rename to src/main/java/com/xgh/test/spring/step02/config/BeanDefinition.java index 1467a17..f43863d 100644 --- a/src/main/java/com/xgh/test/spring/config/BeanDefinition.java +++ b/src/main/java/com/xgh/test/spring/step02/config/BeanDefinition.java @@ -1,11 +1,11 @@ -package com.xgh.test.spring.config; +package com.xgh.test.spring.step02.config; /** - * com.xgh.test.spring.config.BeanDefinition + * com.xgh.test.spring.step02.config.BeanDefinition * * @author xgh
* @description - * @date 2021年08月16日 + * @date 2021年08月17日 */ public class BeanDefinition { diff --git a/src/main/java/com/xgh/test/spring/step02/config/SingletonBeanRegistry.java b/src/main/java/com/xgh/test/spring/step02/config/SingletonBeanRegistry.java new file mode 100644 index 0000000..983259e --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step02/config/SingletonBeanRegistry.java @@ -0,0 +1,13 @@ +package com.xgh.test.spring.step02.config; + +/** + * com.xgh.test.spring.step02.config.SingletonBeanRegistry + * + * @author xgh
+ * @description 单例bean注册 + * @date 2021年08月17日 + */ +public interface SingletonBeanRegistry { + + Object getSingleton(String beanName); +} diff --git a/src/main/java/com/xgh/test/spring/step02/support/AbstractAutowireCapableBeanFactory.java b/src/main/java/com/xgh/test/spring/step02/support/AbstractAutowireCapableBeanFactory.java new file mode 100644 index 0000000..b854250 --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step02/support/AbstractAutowireCapableBeanFactory.java @@ -0,0 +1,54 @@ +package com.xgh.test.spring.step02.support; + +import com.xgh.test.spring.step02.BeansException; +import com.xgh.test.spring.step02.config.BeanDefinition; + +import java.lang.reflect.Constructor; + +/** + * com.xgh.test.spring.step02.support.AbstractAutowireCapableBeanFactory + * + * @author xgh
+ * @description 实例化bean类 + * @date 2021年08月17日 + */ +public abstract class AbstractAutowireCapableBeanFactory extends AbstractBeanFactory{ + + private InstantiationStrategy instantiationStrategy = new CglibSubclassingInstantiationStrategy(); + + @Override + protected Object creatBean(String beanName, BeanDefinition beanDefinition, Object... args) throws BeansException { + Object bean = null; + try { + //todo 有参实现 + // bean = beanDefinition.getBeanClass().newInstance(); + bean = creatBeanInstance(beanDefinition,beanName,args); + + } catch (Exception e) { + throw new BeansException("Instantiation of bean fail",e); + } + addSingleton(beanName,bean); + return bean; + } + + protected Object creatBeanInstance(BeanDefinition beanDefinition, String beanName, Object[] args){ + Constructor constructorToUser = null; + Class beanClass = beanDefinition.getBeanClass(); + Constructor[] declaredConstructors = beanClass.getDeclaredConstructors(); + for (Constructor ctor : declaredConstructors) { + if(null != args && ctor.getParameterTypes().length == args.length){ + constructorToUser = ctor; + break; + } + } + return getInstantiationStrategy().instantiate(beanDefinition,beanName,constructorToUser,args); + }; + + public InstantiationStrategy getInstantiationStrategy() { + return instantiationStrategy; + } + + public void setInstantiationStrategy(InstantiationStrategy instantiationStrategy) { + this.instantiationStrategy = instantiationStrategy; + } +} diff --git a/src/main/java/com/xgh/test/spring/step02/support/AbstractBeanFactory.java b/src/main/java/com/xgh/test/spring/step02/support/AbstractBeanFactory.java new file mode 100644 index 0000000..1a2a7b7 --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step02/support/AbstractBeanFactory.java @@ -0,0 +1,40 @@ +package com.xgh.test.spring.step02.support; + +import com.xgh.test.spring.step02.BeanFactory; +import com.xgh.test.spring.step02.BeansException; +import com.xgh.test.spring.step02.config.BeanDefinition; + +/** + * com.xgh.test.spring.step02.support.AbstractBeanFactory + * + * @author xgh
+ * @description 抽象类定义模板方法 + * @date 2021年08月17日 + */ +public abstract class AbstractBeanFactory extends DefaultSingletonBeanRegistry implements BeanFactory { + + @Override + public Object getBean(String beanName) throws BeansException { + return doGetBean(beanName,null); + } + + @Override + public Object getBean(String beanName, Object... args) throws BeansException { + return doGetBean(beanName,args); + } + + protected T doGetBean(String beanName, Object[] args){ + Object bean = getSingleton(beanName); + if(bean != null){ + return (T) bean; + } + BeanDefinition beanDefinition = getBeanDefinition(beanName); + return (T) creatBean(beanName,beanDefinition,args); + + } + + protected abstract Object creatBean(String beanName, BeanDefinition beanDefinition, Object... args) throws BeansException; + + protected abstract BeanDefinition getBeanDefinition(String beanName) throws BeansException ; + +} diff --git a/src/main/java/com/xgh/test/spring/step02/support/BeanDefinitionRegistry.java b/src/main/java/com/xgh/test/spring/step02/support/BeanDefinitionRegistry.java new file mode 100644 index 0000000..15bdb2d --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step02/support/BeanDefinitionRegistry.java @@ -0,0 +1,16 @@ +package com.xgh.test.spring.step02.support; + +import com.xgh.test.spring.step02.config.BeanDefinition; + +/** + * com.xgh.test.spring.step02.support.BeanDefinitionRegistry + * + * @author xgh
+ * @description + * @date 2021年08月17日 + */ +public interface BeanDefinitionRegistry { + + public void registerBeanDefinition(String beanName, BeanDefinition beanDefinition); + +} diff --git a/src/main/java/com/xgh/test/spring/step02/support/CglibSubclassingInstantiationStrategy.java b/src/main/java/com/xgh/test/spring/step02/support/CglibSubclassingInstantiationStrategy.java new file mode 100644 index 0000000..41970fb --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step02/support/CglibSubclassingInstantiationStrategy.java @@ -0,0 +1,35 @@ +package com.xgh.test.spring.step02.support; + +import com.xgh.test.spring.step02.config.BeanDefinition; +import org.springframework.cglib.proxy.Enhancer; +import org.springframework.cglib.proxy.NoOp; + +import java.lang.reflect.Constructor; + +/** + * com.xgh.test.spring.step02.support.CglibSubclassingInstantiationStrategy + * + * @author xgh
+ * @description + * @date 2021年08月17日 + */ +public class CglibSubclassingInstantiationStrategy implements InstantiationStrategy{ + + + @Override + public Object instantiate(BeanDefinition beanDefinition, String beanName, Constructor ctor, Object[] args) { + Enhancer enhancer = new Enhancer(); + enhancer.setSuperclass(beanDefinition.getBeanClass()); + enhancer.setCallback(new NoOp() { + @Override + public int hashCode() { + return super.hashCode(); + } + }); + if(null == ctor){ + return enhancer.createClass(); + } + return enhancer.create(ctor.getParameterTypes(),args); + } + +} diff --git a/src/main/java/com/xgh/test/spring/step02/support/DefaultListableBeanFactory.java b/src/main/java/com/xgh/test/spring/step02/support/DefaultListableBeanFactory.java new file mode 100644 index 0000000..4addf89 --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step02/support/DefaultListableBeanFactory.java @@ -0,0 +1,33 @@ +package com.xgh.test.spring.step02.support; + +import com.xgh.test.spring.step02.BeansException; +import com.xgh.test.spring.step02.config.BeanDefinition; + +import java.util.HashMap; +import java.util.Map; + +/** + * com.xgh.test.spring.step02.support.DefaultListableBeanFactory + * + * @author xgh
+ * @description 核心实现 + * @date 2021年08月17日 + */ +public class DefaultListableBeanFactory extends AbstractAutowireCapableBeanFactory implements BeanDefinitionRegistry{ + + private Map beanDefinitionMap = new HashMap<>(); + + @Override + protected BeanDefinition getBeanDefinition(String beanName) throws BeansException { + BeanDefinition beanDefinition = beanDefinitionMap.get(beanName); + if(null == beanDefinition){ + throw new BeansException("No bean named '"+ beanName+"' is definde"); + } + return beanDefinition; + } + + @Override + public void registerBeanDefinition(String beanName, BeanDefinition beanDefinition) { + beanDefinitionMap.put(beanName,beanDefinition); + } +} diff --git a/src/main/java/com/xgh/test/spring/support/DefaultSingletonBeanRegistry.java b/src/main/java/com/xgh/test/spring/step02/support/DefaultSingletonBeanRegistry.java similarity index 56% rename from src/main/java/com/xgh/test/spring/support/DefaultSingletonBeanRegistry.java rename to src/main/java/com/xgh/test/spring/step02/support/DefaultSingletonBeanRegistry.java index e4fe1f6..e6ab2c6 100644 --- a/src/main/java/com/xgh/test/spring/support/DefaultSingletonBeanRegistry.java +++ b/src/main/java/com/xgh/test/spring/step02/support/DefaultSingletonBeanRegistry.java @@ -1,16 +1,16 @@ -package com.xgh.test.spring.support; +package com.xgh.test.spring.step02.support; -import com.xgh.test.spring.config.SingletonBeanRegistry; +import com.xgh.test.spring.step02.config.SingletonBeanRegistry; import java.util.HashMap; import java.util.Map; /** - * com.xgh.test.spring.support.DefaultSingletonBeanRegistry + * com.xgh.test.spring.step02.support.DefaultSingletinBeanRegistry * * @author xgh
- * @description - * @date 2021年08月16日 + * @description 单例实例的获取接口 + * @date 2021年08月17日 */ public class DefaultSingletonBeanRegistry implements SingletonBeanRegistry { @@ -21,7 +21,7 @@ public Object getSingleton(String beanName) { return singletonObjects.get(beanName); } - protected void addSingleBean(String beanName,Object singletonObject){ + protected void addSingleton(String beanName,Object singletonObject){ singletonObjects.put(beanName,singletonObject); } } diff --git a/src/main/java/com/xgh/test/spring/step02/support/InstantiationStrategy.java b/src/main/java/com/xgh/test/spring/step02/support/InstantiationStrategy.java new file mode 100644 index 0000000..e3c32ff --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step02/support/InstantiationStrategy.java @@ -0,0 +1,17 @@ +package com.xgh.test.spring.step02.support; + +import com.xgh.test.spring.step02.config.BeanDefinition; + +import java.lang.reflect.Constructor; + +/** + * com.xgh.test.spring.step02.support.InstantionStrategy + * + * @author xgh
+ * @description 实例化策略 + * @date 2021年08月17日 + */ +public interface InstantiationStrategy { + + public Object instantiate(BeanDefinition beanDefinition, String beanName, Constructor ctor,Object[] args); +} diff --git a/src/main/java/com/xgh/test/spring/step02/support/SimpleInstantiationStrategy.java b/src/main/java/com/xgh/test/spring/step02/support/SimpleInstantiationStrategy.java new file mode 100644 index 0000000..c926502 --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step02/support/SimpleInstantiationStrategy.java @@ -0,0 +1,34 @@ +package com.xgh.test.spring.step02.support; + +import com.xgh.test.spring.step02.BeansException; +import com.xgh.test.spring.step02.config.BeanDefinition; + +import java.lang.reflect.Constructor; +import java.lang.reflect.InvocationTargetException; + +/** + * com.xgh.test.spring.step02.support.SimpleInstantiationStrategy + * + * @author xgh
+ * @description jdk + * @date 2021年08月17日 + */ +public class SimpleInstantiationStrategy implements InstantiationStrategy { + + + @Override + public Object instantiate(BeanDefinition beanDefinition, String beanName, Constructor ctor, Object[] args) { + Class clazz = beanDefinition.getBeanClass(); + try { + if (null != ctor) { + return clazz.getDeclaredConstructor(ctor.getParameterTypes()).newInstance(args); + }else { + return clazz.getDeclaredConstructor().newInstance(); + } + }catch(InstantiationException |IllegalAccessException|InvocationTargetException | NoSuchMethodException e) { + throw new BeansException("Failed to instantiate["+clazz.getName()+"]",e); + } + } + + +} diff --git a/src/main/java/com/xgh/test/spring/support/AbstractAutowireCapableBeanFactory.java b/src/main/java/com/xgh/test/spring/support/AbstractAutowireCapableBeanFactory.java deleted file mode 100644 index 7743174..0000000 --- a/src/main/java/com/xgh/test/spring/support/AbstractAutowireCapableBeanFactory.java +++ /dev/null @@ -1,27 +0,0 @@ -package com.xgh.test.spring.support; - -import com.xgh.test.spring.config.BeanDefinition; -import org.springframework.beans.BeansException; -import org.springframework.beans.factory.BeanCreationException; - -/** - * com.xgh.test.spring.support.AbstractAutowireCapableBeanFactory - * - * @author xgh
- * @description 实例化bean - * @date 2021年08月16日 - */ -public abstract class AbstractAutowireCapableBeanFactory extends AbstractBeanFactory { - - @Override - protected Object creatBean(String beanName, BeanDefinition beanDefinition) throws BeansException { - Object bean = null; - try { - bean = beanDefinition.getBeanClass().newInstance(); - } catch (InstantiationException | IllegalAccessException e) { - throw new BeanCreationException("install bean fail" + e); - } - addSingleBean(beanName, bean); - return bean; - } -} diff --git a/src/main/java/com/xgh/test/spring/support/AbstractBeanFactory.java b/src/main/java/com/xgh/test/spring/support/AbstractBeanFactory.java deleted file mode 100644 index 46ecda4..0000000 --- a/src/main/java/com/xgh/test/spring/support/AbstractBeanFactory.java +++ /dev/null @@ -1,30 +0,0 @@ -package com.xgh.test.spring.support; - -import com.xgh.test.spring.BeanFactory; -import com.xgh.test.spring.config.BeanDefinition; -import org.springframework.beans.BeansException; - -/** - * com.xgh.test.spring.support.AbstractBeanFactory - * - * @author xgh
- * @description - * @date 2021年08月16日 - */ -public abstract class AbstractBeanFactory extends DefaultSingletonBeanRegistry implements BeanFactory { - - @Override - public Object getBean(String beanName) throws BeansException { - Object singleton = getSingleton(beanName); - if(null != singleton){ - return singleton; - } - BeanDefinition beanDefinition = getBeanDefinition(beanName); - - return creatBean(beanName,beanDefinition); - } - - protected abstract Object creatBean(String beanName, BeanDefinition beanDefinition) throws BeansException; - - protected abstract BeanDefinition getBeanDefinition(String beanName) throws BeansException; -} diff --git a/src/main/java/com/xgh/test/spring/support/BeanDefinitionRegistory.java b/src/main/java/com/xgh/test/spring/support/BeanDefinitionRegistory.java deleted file mode 100644 index 84388d2..0000000 --- a/src/main/java/com/xgh/test/spring/support/BeanDefinitionRegistory.java +++ /dev/null @@ -1,11 +0,0 @@ -package com.xgh.test.spring.support; - -/** - * com.xgh.test.spring.support.BeanDefinitionRegistory - * - * @author xgh
- * @description - * @date 2021年08月16日 - */ -public class BeanDefinitionRegistory { -} diff --git a/src/main/java/com/xgh/test/spring/support/DefaultListableBeanFactory.java b/src/main/java/com/xgh/test/spring/support/DefaultListableBeanFactory.java deleted file mode 100644 index 37b3ad4..0000000 --- a/src/main/java/com/xgh/test/spring/support/DefaultListableBeanFactory.java +++ /dev/null @@ -1,11 +0,0 @@ -package com.xgh.test.spring.support; - -/** - * com.xgh.test.spring.support.DefaultListableBeanFactory - * - * @author xgh
- * @description - * @date 2021年08月16日 - */ -public class DefaultListableBeanFactory { -} diff --git a/src/test/java/com/xgh/test/spring/ApiTests.java b/src/test/java/com/xgh/test/spring/ApiTests.java index fd332ce..50b4ce9 100644 --- a/src/test/java/com/xgh/test/spring/ApiTests.java +++ b/src/test/java/com/xgh/test/spring/ApiTests.java @@ -1,7 +1,15 @@ package com.xgh.test.spring; import com.xgh.test.spring.service.UserService; +import com.xgh.test.spring.step02.BeanFactory; +import com.xgh.test.spring.step02.config.BeanDefinition; +import com.xgh.test.spring.step02.support.DefaultListableBeanFactory; +import lombok.SneakyThrows; import org.junit.Test; +import org.springframework.cglib.proxy.Enhancer; +import org.springframework.cglib.proxy.NoOp; + +import java.lang.reflect.Constructor; /** * com.xgh.test.spring.ApiTests @@ -14,11 +22,62 @@ public class ApiTests { @Test public void test_factory() { - BeanFactory beanFactory = new BeanFactory(); - BeanDefinition beanDefinition = new BeanDefinition(new UserService()); - beanFactory.registerBeanDefinitionMap("userService", beanDefinition); - - UserService userService =(UserService) beanFactory.getBean("userService"); + //初始化 beanFactory + DefaultListableBeanFactory beanFactory = new DefaultListableBeanFactory(); + //注册bean + BeanDefinition beanDefinition = new BeanDefinition(UserService.class); + beanFactory.registerBeanDefinition("userService",beanDefinition); + //第一次获取 + UserService userService = (UserService)beanFactory.getBean("userService","张三"); userService.getUserInfo(); + + //第二次 + UserService userService1 = (UserService)beanFactory.getBean("userService"); + userService1.getUserInfo(); + } + + @SneakyThrows + @Test + public void test_newInstant(){ + Class userServiceClass = UserService.class; + UserService userService = userServiceClass.newInstance(); + System.out.println(userService); + } + + @Test + public void test_constructor() throws Exception { + Class userServiceClass = UserService.class; + Constructor declaredConstructor = userServiceClass.getDeclaredConstructor(String.class); + UserService userService = declaredConstructor.newInstance("张三"); + System.out.println(userService); + } + + @Test + public void test_constructors() throws Exception { + Class userServiceClass = UserService.class; + Constructor[] declaredConstructors = userServiceClass.getDeclaredConstructors(); + Constructor constructor = null; + for (Constructor declaredConstructor : declaredConstructors) { + if(declaredConstructor.getParameterTypes().length == 1){ + constructor = declaredConstructor; + } + } + Constructor declaredConstructor = userServiceClass.getDeclaredConstructor(constructor.getParameterTypes()); + UserService userService = declaredConstructor.newInstance("李四"); + System.out.println(userService); + } + + @Test + public void test_cglib(){ + Enhancer enhancer = new Enhancer(); + enhancer.setSuperclass(UserService.class); + enhancer.setCallback(new NoOp() { + @Override + public int hashCode() { + return super.hashCode(); + } + }); + Object o = enhancer.create(new Class[]{String.class}, new Object[]{"王五"}); + System.out.println(o); } } diff --git a/src/test/java/com/xgh/test/spring/service/UserService.java b/src/test/java/com/xgh/test/spring/service/UserService.java index fd090c0..b734ca9 100644 --- a/src/test/java/com/xgh/test/spring/service/UserService.java +++ b/src/test/java/com/xgh/test/spring/service/UserService.java @@ -8,9 +8,26 @@ * @date 2021年08月12日 */ public class UserService { + private String name; + + public UserService(String name) { + this.name = name; + } + + public UserService() { + } public void getUserInfo(){ - System.out.println("获取用户信息"); + System.out.println("获取用户信息"+name); } + + @Override + public String toString() { + final StringBuilder sb = new StringBuilder(""); + sb.append("=====").append(name); + return sb.toString(); + } + + } From 29fd848f5bb04a187af2505ef3b48dcabb599425 Mon Sep 17 00:00:00 2001 From: xgh <1371818484@qq.com> Date: Mon, 23 Aug 2021 17:31:54 +0800 Subject: [PATCH 13/20] =?UTF-8?q?=E5=B1=9E=E6=80=A7=E8=AE=BE=E7=BD=AE?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../xgh/test/spring/step03/BeanFactory.java | 17 ++++++ .../test/spring/step03/BeansException.java | 19 ++++++ .../xgh/test/spring/step03/PropertyValue.java | 36 +++++++++++ .../test/spring/step03/PropertyValues.java | 36 +++++++++++ .../spring/step03/config/BeanDefinition.java | 36 +++++++++++ .../spring/step03/config/BeanReference.java | 11 ++++ .../step03/config/SingletonBeanRegistry.java | 13 ++++ .../AbstractAutowireCapableBeanFactory.java | 60 +++++++++++++++++++ .../step03/support/AbstractBeanFactory.java | 40 +++++++++++++ .../support/BeanDefinitionRegistry.java | 16 +++++ ...CglibSubclassingInstantiationStrategy.java | 35 +++++++++++ .../support/DefaultListableBeanFactory.java | 33 ++++++++++ .../support/DefaultSingletonBeanRegistry.java | 27 +++++++++ .../step03/support/InstantiationStrategy.java | 17 ++++++ .../support/SimpleInstantiationStrategy.java | 34 +++++++++++ 15 files changed, 430 insertions(+) create mode 100644 src/main/java/com/xgh/test/spring/step03/BeanFactory.java create mode 100644 src/main/java/com/xgh/test/spring/step03/BeansException.java create mode 100644 src/main/java/com/xgh/test/spring/step03/PropertyValue.java create mode 100644 src/main/java/com/xgh/test/spring/step03/PropertyValues.java create mode 100644 src/main/java/com/xgh/test/spring/step03/config/BeanDefinition.java create mode 100644 src/main/java/com/xgh/test/spring/step03/config/BeanReference.java create mode 100644 src/main/java/com/xgh/test/spring/step03/config/SingletonBeanRegistry.java create mode 100644 src/main/java/com/xgh/test/spring/step03/support/AbstractAutowireCapableBeanFactory.java create mode 100644 src/main/java/com/xgh/test/spring/step03/support/AbstractBeanFactory.java create mode 100644 src/main/java/com/xgh/test/spring/step03/support/BeanDefinitionRegistry.java create mode 100644 src/main/java/com/xgh/test/spring/step03/support/CglibSubclassingInstantiationStrategy.java create mode 100644 src/main/java/com/xgh/test/spring/step03/support/DefaultListableBeanFactory.java create mode 100644 src/main/java/com/xgh/test/spring/step03/support/DefaultSingletonBeanRegistry.java create mode 100644 src/main/java/com/xgh/test/spring/step03/support/InstantiationStrategy.java create mode 100644 src/main/java/com/xgh/test/spring/step03/support/SimpleInstantiationStrategy.java diff --git a/src/main/java/com/xgh/test/spring/step03/BeanFactory.java b/src/main/java/com/xgh/test/spring/step03/BeanFactory.java new file mode 100644 index 0000000..29493dc --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step03/BeanFactory.java @@ -0,0 +1,17 @@ +package com.xgh.test.spring.step03; + + +/** + * com.xgh.test.spring.step02.BeanFactory + * + * @author xgh
+ * @description + * @date 2021年08月12日 + */ +public interface BeanFactory { + + public Object getBean(String beanName) throws BeansException; + + Object getBean(String beanName,Object... args )throws BeansException; + +} diff --git a/src/main/java/com/xgh/test/spring/step03/BeansException.java b/src/main/java/com/xgh/test/spring/step03/BeansException.java new file mode 100644 index 0000000..328bb9b --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step03/BeansException.java @@ -0,0 +1,19 @@ +package com.xgh.test.spring.step03; + +/** + * com.xgh.test.spring.step02.BeansException + * + * @author xgh
+ * @description + * @date 2021年08月17日 + */ +public class BeansException extends RuntimeException{ + + public BeansException(String message) { + super(message); + } + + public BeansException(String message, Throwable cause) { + super(message, cause); + } +} diff --git a/src/main/java/com/xgh/test/spring/step03/PropertyValue.java b/src/main/java/com/xgh/test/spring/step03/PropertyValue.java new file mode 100644 index 0000000..c23c05b --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step03/PropertyValue.java @@ -0,0 +1,36 @@ +package com.xgh.test.spring.step03; + +/** + * com.xgh.test.spring.step03.PropertyValue + * + * @author xgh
+ * @description 属性 + * @date 2021年08月23日 + */ +public class PropertyValue { + + private String name; + + private Object value; + + public PropertyValue(String name, Object value) { + this.name = name; + this.value = value; + } + + public Object getValue() { + return value; + } + + public void setValue(Object value) { + this.value = value; + } + + public String getName() { + return name; + } + + public void setName(String name) { + this.name = name; + } +} diff --git a/src/main/java/com/xgh/test/spring/step03/PropertyValues.java b/src/main/java/com/xgh/test/spring/step03/PropertyValues.java new file mode 100644 index 0000000..8b1f353 --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step03/PropertyValues.java @@ -0,0 +1,36 @@ +package com.xgh.test.spring.step03; + +import java.util.ArrayList; +import java.util.List; + +/** + * com.xgh.test.spring.step03.PropertyValues + * + * @author xgh
+ * @description + * @date 2021年08月23日 + */ +public class PropertyValues { + + private final List propertyValues = new ArrayList<>(); + + + public void addPropertyValue(PropertyValue propertyValue){ + this.propertyValues.add(propertyValue); + } + + + public PropertyValue[] getPropertyValues(){ + return this.propertyValues.toArray(new PropertyValue[0]); + } + + + public PropertyValue getPropertyValue(String name){ + for (PropertyValue propertyValue : propertyValues) { + if(propertyValue.getName().equals(name)){ + return propertyValue; + } + } + return null; + } +} diff --git a/src/main/java/com/xgh/test/spring/step03/config/BeanDefinition.java b/src/main/java/com/xgh/test/spring/step03/config/BeanDefinition.java new file mode 100644 index 0000000..b720560 --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step03/config/BeanDefinition.java @@ -0,0 +1,36 @@ +package com.xgh.test.spring.step03.config; + +import com.xgh.test.spring.step03.PropertyValues; + +/** + * com.xgh.test.spring.step02.config.BeanDefinition + * + * @author xgh
+ * @description + * @date 2021年08月17日 + */ +public class BeanDefinition { + + private Class beanClass; + + private PropertyValues propertyValues; + + + public BeanDefinition(Class beanClass) { + this.beanClass = beanClass; + this.propertyValues = new PropertyValues(); + } + + public BeanDefinition(Class beanClass, PropertyValues propertyValues) { + this.beanClass = beanClass; + this.propertyValues = propertyValues != null ? propertyValues : new PropertyValues(); + } + + public Class getBeanClass() { + return beanClass; + } + + public void setBeanClass(Class beanClass) { + this.beanClass = beanClass; + } +} diff --git a/src/main/java/com/xgh/test/spring/step03/config/BeanReference.java b/src/main/java/com/xgh/test/spring/step03/config/BeanReference.java new file mode 100644 index 0000000..5c5d795 --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step03/config/BeanReference.java @@ -0,0 +1,11 @@ +package com.xgh.test.spring.step03.config; + +/** + * com.xgh.test.spring.step03.config.BeanReference + * + * @author xgh
+ * @description + * @date 2021年08月23日 + */ +public class BeanReference { +} diff --git a/src/main/java/com/xgh/test/spring/step03/config/SingletonBeanRegistry.java b/src/main/java/com/xgh/test/spring/step03/config/SingletonBeanRegistry.java new file mode 100644 index 0000000..c53f58f --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step03/config/SingletonBeanRegistry.java @@ -0,0 +1,13 @@ +package com.xgh.test.spring.step03.config; + +/** + * com.xgh.test.spring.step02.config.SingletonBeanRegistry + * + * @author xgh
+ * @description 单例bean注册 + * @date 2021年08月17日 + */ +public interface SingletonBeanRegistry { + + Object getSingleton(String beanName); +} diff --git a/src/main/java/com/xgh/test/spring/step03/support/AbstractAutowireCapableBeanFactory.java b/src/main/java/com/xgh/test/spring/step03/support/AbstractAutowireCapableBeanFactory.java new file mode 100644 index 0000000..11cd80b --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step03/support/AbstractAutowireCapableBeanFactory.java @@ -0,0 +1,60 @@ +package com.xgh.test.spring.step03.support; + +import com.xgh.test.spring.step03.BeansException; +import com.xgh.test.spring.step03.config.BeanDefinition; + +import java.lang.reflect.Constructor; + +/** + * com.xgh.test.spring.step02.support.AbstractAutowireCapableBeanFactory + * + * @author xgh
+ * @description 实例化bean类 + * @date 2021年08月17日 + */ +public abstract class AbstractAutowireCapableBeanFactory extends AbstractBeanFactory { + + private InstantiationStrategy instantiationStrategy = new CglibSubclassingInstantiationStrategy(); + + @Override + protected Object creatBean(String beanName, BeanDefinition beanDefinition, Object... args) throws BeansException { + Object bean = null; + try { + + bean = creatBeanInstance(beanDefinition,beanName,args); + //设置参数 + applyPropertyValues(beanName,bean,beanDefinition); + } catch (Exception e) { + throw new BeansException("Instantiation of bean fail",e); + } + addSingleton(beanName,bean); + return bean; + } + + + protected Object creatBeanInstance(BeanDefinition beanDefinition, String beanName, Object[] args){ + Constructor constructorToUser = null; + Class beanClass = beanDefinition.getBeanClass(); + Constructor[] declaredConstructors = beanClass.getDeclaredConstructors(); + for (Constructor ctor : declaredConstructors) { + if(null != args && ctor.getParameterTypes().length == args.length){ + constructorToUser = ctor; + break; + } + } + return getInstantiationStrategy().instantiate(beanDefinition,beanName,constructorToUser,args); + }; + + protected void applyPropertyValues(String beanName, Object bean, BeanDefinition beanDefinition){ + + + + }; + public InstantiationStrategy getInstantiationStrategy() { + return instantiationStrategy; + } + + public void setInstantiationStrategy(InstantiationStrategy instantiationStrategy) { + this.instantiationStrategy = instantiationStrategy; + } +} diff --git a/src/main/java/com/xgh/test/spring/step03/support/AbstractBeanFactory.java b/src/main/java/com/xgh/test/spring/step03/support/AbstractBeanFactory.java new file mode 100644 index 0000000..d9a124b --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step03/support/AbstractBeanFactory.java @@ -0,0 +1,40 @@ +package com.xgh.test.spring.step03.support; + +import com.xgh.test.spring.step03.BeanFactory; +import com.xgh.test.spring.step03.BeansException; +import com.xgh.test.spring.step03.config.BeanDefinition; + +/** + * com.xgh.test.spring.step02.support.AbstractBeanFactory + * + * @author xgh
+ * @description 抽象类定义模板方法 + * @date 2021年08月17日 + */ +public abstract class AbstractBeanFactory extends DefaultSingletonBeanRegistry implements BeanFactory { + + @Override + public Object getBean(String beanName) throws BeansException { + return doGetBean(beanName,null); + } + + @Override + public Object getBean(String beanName, Object... args) throws BeansException { + return doGetBean(beanName,args); + } + + protected T doGetBean(String beanName, Object[] args){ + Object bean = getSingleton(beanName); + if(bean != null){ + return (T) bean; + } + BeanDefinition beanDefinition = getBeanDefinition(beanName); + return (T) creatBean(beanName,beanDefinition,args); + + } + + protected abstract Object creatBean(String beanName, BeanDefinition beanDefinition, Object... args) throws BeansException; + + protected abstract BeanDefinition getBeanDefinition(String beanName) throws BeansException; + +} diff --git a/src/main/java/com/xgh/test/spring/step03/support/BeanDefinitionRegistry.java b/src/main/java/com/xgh/test/spring/step03/support/BeanDefinitionRegistry.java new file mode 100644 index 0000000..df32d54 --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step03/support/BeanDefinitionRegistry.java @@ -0,0 +1,16 @@ +package com.xgh.test.spring.step03.support; + +import com.xgh.test.spring.step03.config.BeanDefinition; + +/** + * com.xgh.test.spring.step02.support.BeanDefinitionRegistry + * + * @author xgh
+ * @description + * @date 2021年08月17日 + */ +public interface BeanDefinitionRegistry { + + public void registerBeanDefinition(String beanName, BeanDefinition beanDefinition); + +} diff --git a/src/main/java/com/xgh/test/spring/step03/support/CglibSubclassingInstantiationStrategy.java b/src/main/java/com/xgh/test/spring/step03/support/CglibSubclassingInstantiationStrategy.java new file mode 100644 index 0000000..822f951 --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step03/support/CglibSubclassingInstantiationStrategy.java @@ -0,0 +1,35 @@ +package com.xgh.test.spring.step03.support; + +import com.xgh.test.spring.step03.config.BeanDefinition; +import org.springframework.cglib.proxy.Enhancer; +import org.springframework.cglib.proxy.NoOp; + +import java.lang.reflect.Constructor; + +/** + * com.xgh.test.spring.step02.support.CglibSubclassingInstantiationStrategy + * + * @author xgh
+ * @description + * @date 2021年08月17日 + */ +public class CglibSubclassingInstantiationStrategy implements InstantiationStrategy { + + + @Override + public Object instantiate(BeanDefinition beanDefinition, String beanName, Constructor ctor, Object[] args) { + Enhancer enhancer = new Enhancer(); + enhancer.setSuperclass(beanDefinition.getBeanClass()); + enhancer.setCallback(new NoOp() { + @Override + public int hashCode() { + return super.hashCode(); + } + }); + if(null == ctor){ + return enhancer.createClass(); + } + return enhancer.create(ctor.getParameterTypes(),args); + } + +} diff --git a/src/main/java/com/xgh/test/spring/step03/support/DefaultListableBeanFactory.java b/src/main/java/com/xgh/test/spring/step03/support/DefaultListableBeanFactory.java new file mode 100644 index 0000000..ec92193 --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step03/support/DefaultListableBeanFactory.java @@ -0,0 +1,33 @@ +package com.xgh.test.spring.step03.support; + +import com.xgh.test.spring.step03.BeansException; +import com.xgh.test.spring.step03.config.BeanDefinition; + +import java.util.HashMap; +import java.util.Map; + +/** + * com.xgh.test.spring.step02.support.DefaultListableBeanFactory + * + * @author xgh
+ * @description 核心实现 + * @date 2021年08月17日 + */ +public class DefaultListableBeanFactory extends AbstractAutowireCapableBeanFactory implements BeanDefinitionRegistry { + + private Map beanDefinitionMap = new HashMap<>(); + + @Override + protected BeanDefinition getBeanDefinition(String beanName) throws BeansException { + BeanDefinition beanDefinition = beanDefinitionMap.get(beanName); + if(null == beanDefinition){ + throw new BeansException("No bean named '"+ beanName+"' is definde"); + } + return beanDefinition; + } + + @Override + public void registerBeanDefinition(String beanName, BeanDefinition beanDefinition) { + beanDefinitionMap.put(beanName,beanDefinition); + } +} diff --git a/src/main/java/com/xgh/test/spring/step03/support/DefaultSingletonBeanRegistry.java b/src/main/java/com/xgh/test/spring/step03/support/DefaultSingletonBeanRegistry.java new file mode 100644 index 0000000..8093308 --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step03/support/DefaultSingletonBeanRegistry.java @@ -0,0 +1,27 @@ +package com.xgh.test.spring.step03.support; + +import com.xgh.test.spring.step03.config.SingletonBeanRegistry; + +import java.util.HashMap; +import java.util.Map; + +/** + * com.xgh.test.spring.step02.support.DefaultSingletinBeanRegistry + * + * @author xgh
+ * @description 单例实例的获取接口 + * @date 2021年08月17日 + */ +public class DefaultSingletonBeanRegistry implements SingletonBeanRegistry { + + private Map singletonObjects = new HashMap<>(); + + @Override + public Object getSingleton(String beanName) { + return singletonObjects.get(beanName); + } + + protected void addSingleton(String beanName,Object singletonObject){ + singletonObjects.put(beanName,singletonObject); + } +} diff --git a/src/main/java/com/xgh/test/spring/step03/support/InstantiationStrategy.java b/src/main/java/com/xgh/test/spring/step03/support/InstantiationStrategy.java new file mode 100644 index 0000000..238508a --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step03/support/InstantiationStrategy.java @@ -0,0 +1,17 @@ +package com.xgh.test.spring.step03.support; + +import com.xgh.test.spring.step03.config.BeanDefinition; + +import java.lang.reflect.Constructor; + +/** + * com.xgh.test.spring.step02.support.InstantionStrategy + * + * @author xgh
+ * @description 实例化策略 + * @date 2021年08月17日 + */ +public interface InstantiationStrategy { + + public Object instantiate(BeanDefinition beanDefinition, String beanName, Constructor ctor, Object[] args); +} diff --git a/src/main/java/com/xgh/test/spring/step03/support/SimpleInstantiationStrategy.java b/src/main/java/com/xgh/test/spring/step03/support/SimpleInstantiationStrategy.java new file mode 100644 index 0000000..bf664f0 --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step03/support/SimpleInstantiationStrategy.java @@ -0,0 +1,34 @@ +package com.xgh.test.spring.step03.support; + +import com.xgh.test.spring.step03.BeansException; +import com.xgh.test.spring.step03.config.BeanDefinition; + +import java.lang.reflect.Constructor; +import java.lang.reflect.InvocationTargetException; + +/** + * com.xgh.test.spring.step02.support.SimpleInstantiationStrategy + * + * @author xgh
+ * @description jdk + * @date 2021年08月17日 + */ +public class SimpleInstantiationStrategy implements InstantiationStrategy { + + + @Override + public Object instantiate(BeanDefinition beanDefinition, String beanName, Constructor ctor, Object[] args) { + Class clazz = beanDefinition.getBeanClass(); + try { + if (null != ctor) { + return clazz.getDeclaredConstructor(ctor.getParameterTypes()).newInstance(args); + }else { + return clazz.getDeclaredConstructor().newInstance(); + } + }catch(InstantiationException |IllegalAccessException|InvocationTargetException | NoSuchMethodException e) { + throw new BeansException("Failed to instantiate["+clazz.getName()+"]",e); + } + } + + +} From 2ce80ad02ce5d3ddff69a8cd8ac6e7eecd9f56ed Mon Sep 17 00:00:00 2001 From: xgh <1371818484@qq.com> Date: Tue, 24 Aug 2021 17:04:02 +0800 Subject: [PATCH 14/20] =?UTF-8?q?=E5=B1=9E=E6=80=A7=E8=AE=BE=E7=BD=AE?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- pom.xml | 5 +++ ...CglibSubclassingInstantiationStrategy.java | 2 +- .../spring/step03/config/BeanDefinition.java | 8 ++++ .../spring/step03/config/BeanReference.java | 14 +++++++ .../AbstractAutowireCapableBeanFactory.java | 20 ++++++++- ...CglibSubclassingInstantiationStrategy.java | 2 +- .../java/com/xgh/test/spring/ApiTests.java | 2 +- .../com/xgh/test/spring/test03/Test03.java | 42 +++++++++++++++++++ .../com/xgh/test/spring/test03/UserDao.java | 29 +++++++++++++ .../xgh/test/spring/test03/UserService.java | 38 +++++++++++++++++ 10 files changed, 158 insertions(+), 4 deletions(-) create mode 100644 src/test/java/com/xgh/test/spring/test03/Test03.java create mode 100644 src/test/java/com/xgh/test/spring/test03/UserDao.java create mode 100644 src/test/java/com/xgh/test/spring/test03/UserService.java diff --git a/pom.xml b/pom.xml index 4c6a29a..604a2ba 100644 --- a/pom.xml +++ b/pom.xml @@ -56,6 +56,11 @@ 30.1.1-jre test + + cn.hutool + hutool-core + 5.6.3 + diff --git a/src/main/java/com/xgh/test/spring/step02/support/CglibSubclassingInstantiationStrategy.java b/src/main/java/com/xgh/test/spring/step02/support/CglibSubclassingInstantiationStrategy.java index 41970fb..2dce42e 100644 --- a/src/main/java/com/xgh/test/spring/step02/support/CglibSubclassingInstantiationStrategy.java +++ b/src/main/java/com/xgh/test/spring/step02/support/CglibSubclassingInstantiationStrategy.java @@ -27,7 +27,7 @@ public int hashCode() { } }); if(null == ctor){ - return enhancer.createClass(); + return enhancer.create(); } return enhancer.create(ctor.getParameterTypes(),args); } diff --git a/src/main/java/com/xgh/test/spring/step03/config/BeanDefinition.java b/src/main/java/com/xgh/test/spring/step03/config/BeanDefinition.java index b720560..12d724d 100644 --- a/src/main/java/com/xgh/test/spring/step03/config/BeanDefinition.java +++ b/src/main/java/com/xgh/test/spring/step03/config/BeanDefinition.java @@ -33,4 +33,12 @@ public Class getBeanClass() { public void setBeanClass(Class beanClass) { this.beanClass = beanClass; } + + public PropertyValues getPropertyValues() { + return propertyValues; + } + + public void setPropertyValues(PropertyValues propertyValues) { + this.propertyValues = propertyValues; + } } diff --git a/src/main/java/com/xgh/test/spring/step03/config/BeanReference.java b/src/main/java/com/xgh/test/spring/step03/config/BeanReference.java index 5c5d795..0b12e8f 100644 --- a/src/main/java/com/xgh/test/spring/step03/config/BeanReference.java +++ b/src/main/java/com/xgh/test/spring/step03/config/BeanReference.java @@ -8,4 +8,18 @@ * @date 2021年08月23日 */ public class BeanReference { + + private String beanName; + + public BeanReference(String beanName) { + this.beanName = beanName; + } + + public String getBeanName() { + return beanName; + } + + public void setBeanName(String beanName) { + this.beanName = beanName; + } } diff --git a/src/main/java/com/xgh/test/spring/step03/support/AbstractAutowireCapableBeanFactory.java b/src/main/java/com/xgh/test/spring/step03/support/AbstractAutowireCapableBeanFactory.java index 11cd80b..18135c4 100644 --- a/src/main/java/com/xgh/test/spring/step03/support/AbstractAutowireCapableBeanFactory.java +++ b/src/main/java/com/xgh/test/spring/step03/support/AbstractAutowireCapableBeanFactory.java @@ -1,7 +1,11 @@ package com.xgh.test.spring.step03.support; +import cn.hutool.core.bean.BeanUtil; import com.xgh.test.spring.step03.BeansException; +import com.xgh.test.spring.step03.PropertyValue; +import com.xgh.test.spring.step03.PropertyValues; import com.xgh.test.spring.step03.config.BeanDefinition; +import com.xgh.test.spring.step03.config.BeanReference; import java.lang.reflect.Constructor; @@ -46,7 +50,21 @@ protected Object creatBeanInstance(BeanDefinition beanDefinition, String beanNa }; protected void applyPropertyValues(String beanName, Object bean, BeanDefinition beanDefinition){ - + try { + PropertyValues propertyValues = beanDefinition.getPropertyValues(); + for (PropertyValue propertyValue : propertyValues.getPropertyValues()) { + String name = propertyValue.getName(); + Object value = propertyValue.getValue(); + if(value instanceof BeanReference){ + //a依赖b + BeanReference beanReference = (BeanReference)value; + value = getBean(beanReference.getBeanName()); + } + BeanUtil.setFieldValue(bean,name,value); + } + }catch (Exception e){ + throw new BeansException("Error setting property values" + beanName); + } }; diff --git a/src/main/java/com/xgh/test/spring/step03/support/CglibSubclassingInstantiationStrategy.java b/src/main/java/com/xgh/test/spring/step03/support/CglibSubclassingInstantiationStrategy.java index 822f951..9539c0e 100644 --- a/src/main/java/com/xgh/test/spring/step03/support/CglibSubclassingInstantiationStrategy.java +++ b/src/main/java/com/xgh/test/spring/step03/support/CglibSubclassingInstantiationStrategy.java @@ -27,7 +27,7 @@ public int hashCode() { } }); if(null == ctor){ - return enhancer.createClass(); + return enhancer.create(); } return enhancer.create(ctor.getParameterTypes(),args); } diff --git a/src/test/java/com/xgh/test/spring/ApiTests.java b/src/test/java/com/xgh/test/spring/ApiTests.java index 50b4ce9..8eecf2e 100644 --- a/src/test/java/com/xgh/test/spring/ApiTests.java +++ b/src/test/java/com/xgh/test/spring/ApiTests.java @@ -77,7 +77,7 @@ public int hashCode() { return super.hashCode(); } }); - Object o = enhancer.create(new Class[]{String.class}, new Object[]{"王五"}); + Object o = enhancer.createClass(); System.out.println(o); } } diff --git a/src/test/java/com/xgh/test/spring/test03/Test03.java b/src/test/java/com/xgh/test/spring/test03/Test03.java new file mode 100644 index 0000000..08170ce --- /dev/null +++ b/src/test/java/com/xgh/test/spring/test03/Test03.java @@ -0,0 +1,42 @@ +package com.xgh.test.spring.test03; + +import com.xgh.test.spring.step03.PropertyValue; +import com.xgh.test.spring.step03.PropertyValues; +import com.xgh.test.spring.step03.config.BeanDefinition; +import com.xgh.test.spring.step03.config.BeanReference; +import com.xgh.test.spring.step03.support.DefaultListableBeanFactory; +import org.junit.Test; + +/** + * com.xgh.test.spring.test03.Test03 + * + * @author xgh
+ * @description + * @date 2021年08月24日 + */ +public class Test03 { + + + @Test + public void test_BeanFactory(){ + DefaultListableBeanFactory beanFactory = new DefaultListableBeanFactory(); + + //注册userDao + beanFactory.registerBeanDefinition("userDao",new BeanDefinition(UserDao.class)); + + //userService + PropertyValues propertyValues = new PropertyValues(); + + propertyValues.addPropertyValue(new PropertyValue("uId","1001")); + propertyValues.addPropertyValue(new PropertyValue("userDao",new BeanReference("userDao"))); + + + //userService注入 + BeanDefinition beanDefinition = new BeanDefinition(UserService.class, propertyValues); + beanFactory.registerBeanDefinition("userService",beanDefinition); + UserService userService = (UserService)beanFactory.getBean("userService"); + userService.queryUserInfo(); + } + + +} diff --git a/src/test/java/com/xgh/test/spring/test03/UserDao.java b/src/test/java/com/xgh/test/spring/test03/UserDao.java new file mode 100644 index 0000000..f9b6c38 --- /dev/null +++ b/src/test/java/com/xgh/test/spring/test03/UserDao.java @@ -0,0 +1,29 @@ +package com.xgh.test.spring.test03; + +import java.util.HashMap; +import java.util.Map; + +/** + * com.xgh.test.spring.test03.UserDao + * + * @author xgh
+ * @description + * @date 2021年08月24日 + */ +public class UserDao { + + + private static Map hashMap = new HashMap<>(); + + + static { + hashMap.put("1001","张三"); + hashMap.put("1002","李四"); + hashMap.put("1003","王五"); + } + + + public String queryUserName(String uId){ + return hashMap.get(uId); + } +} diff --git a/src/test/java/com/xgh/test/spring/test03/UserService.java b/src/test/java/com/xgh/test/spring/test03/UserService.java new file mode 100644 index 0000000..35798d8 --- /dev/null +++ b/src/test/java/com/xgh/test/spring/test03/UserService.java @@ -0,0 +1,38 @@ +package com.xgh.test.spring.test03; + +/** + * com.xgh.test.spring.test03.UserService + * + * @author xgh
+ * @description + * @date 2021年08月24日 + */ +public class UserService { + + private String uId; + + private UserDao userDao; + + public UserService() { + } + + public void queryUserInfo(){ + System.out.println("查询用户信息:" + userDao.queryUserName(uId)); + } + + public String getuId() { + return uId; + } + + public void setuId(String uId) { + this.uId = uId; + } + + public UserDao getUserDao() { + return userDao; + } + + public void setUserDao(UserDao userDao) { + this.userDao = userDao; + } +} From 9e11bf539c12dd49168ba90cb38c0bba915915c6 Mon Sep 17 00:00:00 2001 From: xgh <1371818484@qq.com> Date: Fri, 27 Aug 2021 17:54:08 +0800 Subject: [PATCH 15/20] =?UTF-8?q?=E8=AF=BB=E5=8F=96=E9=85=8D=E7=BD=AE?= =?UTF-8?q?=E6=96=87=E4=BB=B6?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../spring/step04/beans/BeansException.java | 19 +++ .../spring/step04/beans/PropertyValue.java | 36 ++++++ .../spring/step04/beans/PropertyValues.java | 36 ++++++ .../step04/beans/factory/BeanFactory.java | 23 ++++ .../ConfigurableListableBeanFactory.java | 18 +++ .../factory/HierarchicalBeanFactory.java | 11 ++ .../beans/factory/ListableBeanFactory.java | 34 ++++++ .../config/AutowireCapableBeanFactory.java | 15 +++ .../beans/factory/config/BeanDefinition.java | 44 +++++++ .../beans/factory/config/BeanReference.java | 25 ++++ .../config/ConfigurableBeanFactory.java | 17 +++ .../factory/config/SingletonBeanRegistry.java | 13 ++ .../AbstractAutowireCapableBeanFactory.java | 78 ++++++++++++ .../support/AbstractBeanDefinitionReader.java | 37 ++++++ .../factory/support/AbstractBeanFactory.java | 44 +++++++ .../factory/support/BeanDefinitionReader.java | 26 ++++ .../support/BeanDefinitionRegistry.java | 41 +++++++ ...CglibSubclassingInstantiationStrategy.java | 35 ++++++ .../support/DefaultListableBeanFactory.java | 60 +++++++++ .../support/DefaultSingletonBeanRegistry.java | 27 +++++ .../support/InstantiationStrategy.java | 17 +++ .../support/SimpleInstantiationStrategy.java | 34 ++++++ .../factory/xml/XmlBeanDefinitionReader.java | 114 ++++++++++++++++++ .../step04/core/io/ClassPathResource.java | 41 +++++++ .../step04/core/io/DefaultResourceLoader.java | 35 ++++++ .../step04/core/io/FileSystemResource.java | 40 ++++++ .../test/spring/step04/core/io/Resource.java | 16 +++ .../spring/step04/core/io/ResourceLoader.java | 15 +++ .../spring/step04/core/io/UrlResource.java | 39 ++++++ .../test/spring/step04/util/ClassUtils.java | 24 ++++ .../com/xgh/test/spring/test04/Test04.java | 55 +++++++++ .../com/xgh/test/spring/test04/UserDao.java | 29 +++++ .../xgh/test/spring/test04/UserService.java | 36 ++++++ src/test/resources/Spring.xml | 11 ++ src/test/resources/important.properties | 1 + 35 files changed, 1146 insertions(+) create mode 100644 src/main/java/com/xgh/test/spring/step04/beans/BeansException.java create mode 100644 src/main/java/com/xgh/test/spring/step04/beans/PropertyValue.java create mode 100644 src/main/java/com/xgh/test/spring/step04/beans/PropertyValues.java create mode 100644 src/main/java/com/xgh/test/spring/step04/beans/factory/BeanFactory.java create mode 100644 src/main/java/com/xgh/test/spring/step04/beans/factory/ConfigurableListableBeanFactory.java create mode 100644 src/main/java/com/xgh/test/spring/step04/beans/factory/HierarchicalBeanFactory.java create mode 100644 src/main/java/com/xgh/test/spring/step04/beans/factory/ListableBeanFactory.java create mode 100644 src/main/java/com/xgh/test/spring/step04/beans/factory/config/AutowireCapableBeanFactory.java create mode 100644 src/main/java/com/xgh/test/spring/step04/beans/factory/config/BeanDefinition.java create mode 100644 src/main/java/com/xgh/test/spring/step04/beans/factory/config/BeanReference.java create mode 100644 src/main/java/com/xgh/test/spring/step04/beans/factory/config/ConfigurableBeanFactory.java create mode 100644 src/main/java/com/xgh/test/spring/step04/beans/factory/config/SingletonBeanRegistry.java create mode 100644 src/main/java/com/xgh/test/spring/step04/beans/factory/support/AbstractAutowireCapableBeanFactory.java create mode 100644 src/main/java/com/xgh/test/spring/step04/beans/factory/support/AbstractBeanDefinitionReader.java create mode 100644 src/main/java/com/xgh/test/spring/step04/beans/factory/support/AbstractBeanFactory.java create mode 100644 src/main/java/com/xgh/test/spring/step04/beans/factory/support/BeanDefinitionReader.java create mode 100644 src/main/java/com/xgh/test/spring/step04/beans/factory/support/BeanDefinitionRegistry.java create mode 100644 src/main/java/com/xgh/test/spring/step04/beans/factory/support/CglibSubclassingInstantiationStrategy.java create mode 100644 src/main/java/com/xgh/test/spring/step04/beans/factory/support/DefaultListableBeanFactory.java create mode 100644 src/main/java/com/xgh/test/spring/step04/beans/factory/support/DefaultSingletonBeanRegistry.java create mode 100644 src/main/java/com/xgh/test/spring/step04/beans/factory/support/InstantiationStrategy.java create mode 100644 src/main/java/com/xgh/test/spring/step04/beans/factory/support/SimpleInstantiationStrategy.java create mode 100644 src/main/java/com/xgh/test/spring/step04/beans/factory/xml/XmlBeanDefinitionReader.java create mode 100644 src/main/java/com/xgh/test/spring/step04/core/io/ClassPathResource.java create mode 100644 src/main/java/com/xgh/test/spring/step04/core/io/DefaultResourceLoader.java create mode 100644 src/main/java/com/xgh/test/spring/step04/core/io/FileSystemResource.java create mode 100644 src/main/java/com/xgh/test/spring/step04/core/io/Resource.java create mode 100644 src/main/java/com/xgh/test/spring/step04/core/io/ResourceLoader.java create mode 100644 src/main/java/com/xgh/test/spring/step04/core/io/UrlResource.java create mode 100644 src/main/java/com/xgh/test/spring/step04/util/ClassUtils.java create mode 100644 src/test/java/com/xgh/test/spring/test04/Test04.java create mode 100644 src/test/java/com/xgh/test/spring/test04/UserDao.java create mode 100644 src/test/java/com/xgh/test/spring/test04/UserService.java create mode 100644 src/test/resources/Spring.xml create mode 100644 src/test/resources/important.properties diff --git a/src/main/java/com/xgh/test/spring/step04/beans/BeansException.java b/src/main/java/com/xgh/test/spring/step04/beans/BeansException.java new file mode 100644 index 0000000..f8f3c30 --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step04/beans/BeansException.java @@ -0,0 +1,19 @@ +package com.xgh.test.spring.step04.beans; + +/** + * com.xgh.test.spring.step02.BeansException + * + * @author xgh
+ * @description + * @date 2021年08月17日 + */ +public class BeansException extends RuntimeException{ + + public BeansException(String message) { + super(message); + } + + public BeansException(String message, Throwable cause) { + super(message, cause); + } +} diff --git a/src/main/java/com/xgh/test/spring/step04/beans/PropertyValue.java b/src/main/java/com/xgh/test/spring/step04/beans/PropertyValue.java new file mode 100644 index 0000000..3b0f144 --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step04/beans/PropertyValue.java @@ -0,0 +1,36 @@ +package com.xgh.test.spring.step04.beans; + +/** + * com.xgh.test.spring.step03.PropertyValue + * + * @author xgh
+ * @description 属性 + * @date 2021年08月23日 + */ +public class PropertyValue { + + private String name; + + private Object value; + + public PropertyValue(String name, Object value) { + this.name = name; + this.value = value; + } + + public Object getValue() { + return value; + } + + public void setValue(Object value) { + this.value = value; + } + + public String getName() { + return name; + } + + public void setName(String name) { + this.name = name; + } +} diff --git a/src/main/java/com/xgh/test/spring/step04/beans/PropertyValues.java b/src/main/java/com/xgh/test/spring/step04/beans/PropertyValues.java new file mode 100644 index 0000000..0f64508 --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step04/beans/PropertyValues.java @@ -0,0 +1,36 @@ +package com.xgh.test.spring.step04.beans; + +import java.util.ArrayList; +import java.util.List; + +/** + * com.xgh.test.spring.step03.PropertyValues + * + * @author xgh
+ * @description + * @date 2021年08月23日 + */ +public class PropertyValues { + + private final List propertyValues = new ArrayList<>(); + + + public void addPropertyValue(PropertyValue propertyValue){ + this.propertyValues.add(propertyValue); + } + + + public PropertyValue[] getPropertyValues(){ + return this.propertyValues.toArray(new PropertyValue[0]); + } + + + public PropertyValue getPropertyValue(String name){ + for (PropertyValue propertyValue : propertyValues) { + if(propertyValue.getName().equals(name)){ + return propertyValue; + } + } + return null; + } +} diff --git a/src/main/java/com/xgh/test/spring/step04/beans/factory/BeanFactory.java b/src/main/java/com/xgh/test/spring/step04/beans/factory/BeanFactory.java new file mode 100644 index 0000000..56f4e4a --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step04/beans/factory/BeanFactory.java @@ -0,0 +1,23 @@ +package com.xgh.test.spring.step04.beans.factory; + +import com.xgh.test.spring.step04.beans.BeansException; + +/** + * com.xgh.test.spring.step04.beans.factory.BeanFactory + * + * @author xgh
+ * @description + * @date 2021年08月27日 + */ +public interface BeanFactory { + + + public Object getBean(String beanName) throws BeansException; + + T getBean(String beanName,Class requiredType); + + Object getBean(String beanName,Object... args )throws BeansException; + + + +} diff --git a/src/main/java/com/xgh/test/spring/step04/beans/factory/ConfigurableListableBeanFactory.java b/src/main/java/com/xgh/test/spring/step04/beans/factory/ConfigurableListableBeanFactory.java new file mode 100644 index 0000000..27c3789 --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step04/beans/factory/ConfigurableListableBeanFactory.java @@ -0,0 +1,18 @@ +package com.xgh.test.spring.step04.beans.factory; + +import com.xgh.test.spring.step04.beans.BeansException; +import com.xgh.test.spring.step04.beans.factory.config.AutowireCapableBeanFactory; +import com.xgh.test.spring.step04.beans.factory.config.BeanDefinition; +import com.xgh.test.spring.step04.beans.factory.config.ConfigurableBeanFactory; + +/** + * com.xgh.test.spring.step04.beans.factory.ConfigurableListableBeanFactory + * + * @author xgh
+ * @description + * @date 2021年08月27日 + */ +public interface ConfigurableListableBeanFactory extends ListableBeanFactory, AutowireCapableBeanFactory, ConfigurableBeanFactory { + + BeanDefinition getBeanDefinition(String beanName)throws BeansException; +} diff --git a/src/main/java/com/xgh/test/spring/step04/beans/factory/HierarchicalBeanFactory.java b/src/main/java/com/xgh/test/spring/step04/beans/factory/HierarchicalBeanFactory.java new file mode 100644 index 0000000..e11f4da --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step04/beans/factory/HierarchicalBeanFactory.java @@ -0,0 +1,11 @@ +package com.xgh.test.spring.step04.beans.factory; + +/** + * com.xgh.test.spring.step04.beans.factory.HierarchicalBeanFactory + * + * @author xgh
+ * @description + * @date 2021年08月27日 + */ +public interface HierarchicalBeanFactory extends BeanFactory{ +} diff --git a/src/main/java/com/xgh/test/spring/step04/beans/factory/ListableBeanFactory.java b/src/main/java/com/xgh/test/spring/step04/beans/factory/ListableBeanFactory.java new file mode 100644 index 0000000..9907fdd --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step04/beans/factory/ListableBeanFactory.java @@ -0,0 +1,34 @@ +package com.xgh.test.spring.step04.beans.factory; + +import com.xgh.test.spring.step04.beans.BeansException; + +import java.util.Map; + +/** + * com.xgh.test.spring.step04.beans.factory.ListableBeanFactory + * + * @author xgh
+ * @description + * @date 2021年08月27日 + */ +public interface ListableBeanFactory extends BeanFactory { + + + + /* + * @description 按类型返回实例 + * @date 2021/8/27 0027 + * @param type + * @return java.util.Map + */ + Map getBeansOfType(Class type) throws BeansException; + + + /* + * @description 返回所以bean的name + * @date 2021/8/27 0027 + * @return java.lang.String[] + */ + String[] getBeanDefinitionNames(); + +} diff --git a/src/main/java/com/xgh/test/spring/step04/beans/factory/config/AutowireCapableBeanFactory.java b/src/main/java/com/xgh/test/spring/step04/beans/factory/config/AutowireCapableBeanFactory.java new file mode 100644 index 0000000..05d8c4e --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step04/beans/factory/config/AutowireCapableBeanFactory.java @@ -0,0 +1,15 @@ +package com.xgh.test.spring.step04.beans.factory.config; + + +import com.xgh.test.spring.step04.beans.factory.BeanFactory; + +/** + * com.xgh.test.spring.step04.beans.factory.config.AutowireCapableBeanFactory + * + * @author xgh
+ * @description + * @date 2021年08月27日 + */ +public interface AutowireCapableBeanFactory extends BeanFactory { + +} diff --git a/src/main/java/com/xgh/test/spring/step04/beans/factory/config/BeanDefinition.java b/src/main/java/com/xgh/test/spring/step04/beans/factory/config/BeanDefinition.java new file mode 100644 index 0000000..80b9696 --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step04/beans/factory/config/BeanDefinition.java @@ -0,0 +1,44 @@ +package com.xgh.test.spring.step04.beans.factory.config; + +import com.xgh.test.spring.step04.beans.PropertyValues; + +/** + * com.xgh.test.spring.step02.config.BeanDefinition + * + * @author xgh
+ * @description + * @date 2021年08月17日 + */ +public class BeanDefinition { + + private Class beanClass; + + private PropertyValues propertyValues; + + + public BeanDefinition(Class beanClass) { + this.beanClass = beanClass; + this.propertyValues = new PropertyValues(); + } + + public BeanDefinition(Class beanClass, PropertyValues propertyValues) { + this.beanClass = beanClass; + this.propertyValues = propertyValues != null ? propertyValues : new PropertyValues(); + } + + public Class getBeanClass() { + return beanClass; + } + + public void setBeanClass(Class beanClass) { + this.beanClass = beanClass; + } + + public PropertyValues getPropertyValues() { + return propertyValues; + } + + public void setPropertyValues(PropertyValues propertyValues) { + this.propertyValues = propertyValues; + } +} diff --git a/src/main/java/com/xgh/test/spring/step04/beans/factory/config/BeanReference.java b/src/main/java/com/xgh/test/spring/step04/beans/factory/config/BeanReference.java new file mode 100644 index 0000000..39acb41 --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step04/beans/factory/config/BeanReference.java @@ -0,0 +1,25 @@ +package com.xgh.test.spring.step04.beans.factory.config; + +/** + * com.xgh.test.spring.step03.config.BeanReference + * + * @author xgh
+ * @description + * @date 2021年08月23日 + */ +public class BeanReference { + + private String beanName; + + public BeanReference(String beanName) { + this.beanName = beanName; + } + + public String getBeanName() { + return beanName; + } + + public void setBeanName(String beanName) { + this.beanName = beanName; + } +} diff --git a/src/main/java/com/xgh/test/spring/step04/beans/factory/config/ConfigurableBeanFactory.java b/src/main/java/com/xgh/test/spring/step04/beans/factory/config/ConfigurableBeanFactory.java new file mode 100644 index 0000000..4f5cc63 --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step04/beans/factory/config/ConfigurableBeanFactory.java @@ -0,0 +1,17 @@ +package com.xgh.test.spring.step04.beans.factory.config; + +import com.xgh.test.spring.step04.beans.factory.HierarchicalBeanFactory; + +/** + * com.xgh.test.spring.step04.beans.factory.config.ConfigurableBeanFactory + * + * @author xgh
+ * @description + * @date 2021年08月27日 + */ +public interface ConfigurableBeanFactory extends HierarchicalBeanFactory,SingletonBeanRegistry { + + String SCOPE_SINGLETON = "singleton"; + + String SCOPE_PROTOTYPE= "prototype"; +} diff --git a/src/main/java/com/xgh/test/spring/step04/beans/factory/config/SingletonBeanRegistry.java b/src/main/java/com/xgh/test/spring/step04/beans/factory/config/SingletonBeanRegistry.java new file mode 100644 index 0000000..3a5e07d --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step04/beans/factory/config/SingletonBeanRegistry.java @@ -0,0 +1,13 @@ +package com.xgh.test.spring.step04.beans.factory.config; + +/** + * com.xgh.test.spring.step02.config.SingletonBeanRegistry + * + * @author xgh
+ * @description 单例bean注册 + * @date 2021年08月17日 + */ +public interface SingletonBeanRegistry { + + Object getSingleton(String beanName); +} diff --git a/src/main/java/com/xgh/test/spring/step04/beans/factory/support/AbstractAutowireCapableBeanFactory.java b/src/main/java/com/xgh/test/spring/step04/beans/factory/support/AbstractAutowireCapableBeanFactory.java new file mode 100644 index 0000000..9b178df --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step04/beans/factory/support/AbstractAutowireCapableBeanFactory.java @@ -0,0 +1,78 @@ +package com.xgh.test.spring.step04.beans.factory.support; + +import cn.hutool.core.bean.BeanUtil; +import com.xgh.test.spring.step04.beans.BeansException; +import com.xgh.test.spring.step04.beans.PropertyValue; +import com.xgh.test.spring.step04.beans.PropertyValues; +import com.xgh.test.spring.step04.beans.factory.config.BeanDefinition; +import com.xgh.test.spring.step04.beans.factory.config.BeanReference; + +import java.lang.reflect.Constructor; + +/** + * com.xgh.test.spring.step02.support.AbstractAutowireCapableBeanFactory + * + * @author xgh
+ * @description 实例化bean类 + * @date 2021年08月17日 + */ +public abstract class AbstractAutowireCapableBeanFactory extends AbstractBeanFactory { + + private InstantiationStrategy instantiationStrategy = new CglibSubclassingInstantiationStrategy(); + + @Override + protected Object creatBean(String beanName, BeanDefinition beanDefinition, Object... args) throws BeansException { + Object bean = null; + try { + + bean = creatBeanInstance(beanDefinition,beanName,args); + //设置参数 + applyPropertyValues(beanName,bean,beanDefinition); + } catch (Exception e) { + throw new BeansException("Instantiation of bean fail",e); + } + addSingleton(beanName,bean); + return bean; + } + + + protected Object creatBeanInstance(BeanDefinition beanDefinition, String beanName, Object[] args){ + Constructor constructorToUser = null; + Class beanClass = beanDefinition.getBeanClass(); + Constructor[] declaredConstructors = beanClass.getDeclaredConstructors(); + for (Constructor ctor : declaredConstructors) { + if(null != args && ctor.getParameterTypes().length == args.length){ + constructorToUser = ctor; + break; + } + } + return getInstantiationStrategy().instantiate(beanDefinition,beanName,constructorToUser,args); + }; + + protected void applyPropertyValues(String beanName, Object bean, BeanDefinition beanDefinition){ + try { + PropertyValues propertyValues = beanDefinition.getPropertyValues(); + for (PropertyValue propertyValue : propertyValues.getPropertyValues()) { + String name = propertyValue.getName(); + Object value = propertyValue.getValue(); + if(value instanceof BeanReference){ + //a依赖b + BeanReference beanReference = (BeanReference)value; + value = getBean(beanReference.getBeanName()); + } + BeanUtil.setFieldValue(bean,name,value); + } + }catch (Exception e){ + throw new BeansException("Error setting property values" + beanName); + } + + + }; + public InstantiationStrategy getInstantiationStrategy() { + return instantiationStrategy; + } + + public void setInstantiationStrategy(InstantiationStrategy instantiationStrategy) { + this.instantiationStrategy = instantiationStrategy; + } +} diff --git a/src/main/java/com/xgh/test/spring/step04/beans/factory/support/AbstractBeanDefinitionReader.java b/src/main/java/com/xgh/test/spring/step04/beans/factory/support/AbstractBeanDefinitionReader.java new file mode 100644 index 0000000..64c1c0f --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step04/beans/factory/support/AbstractBeanDefinitionReader.java @@ -0,0 +1,37 @@ +package com.xgh.test.spring.step04.beans.factory.support; + +import com.xgh.test.spring.step04.core.io.DefaultResourceLoader; +import com.xgh.test.spring.step04.core.io.ResourceLoader; + +/** + * com.xgh.test.spring.step04.beans.factory.support.AbstractBeanDefinitionReader + * + * @author xgh
+ * @description + * @date 2021年08月27日 + */ +public abstract class AbstractBeanDefinitionReader implements BeanDefinitionReader{ + + private final BeanDefinitionRegistry registry; + + private ResourceLoader resourceLoader; + + public AbstractBeanDefinitionReader(BeanDefinitionRegistry registry) { + this(registry,new DefaultResourceLoader()); + } + + public AbstractBeanDefinitionReader(BeanDefinitionRegistry registry, ResourceLoader resourceLoader) { + this.registry = registry; + this.resourceLoader = resourceLoader; + } + + @Override + public BeanDefinitionRegistry getRegistry() { + return registry; + } + + @Override + public ResourceLoader getResourceLoader() { + return resourceLoader; + } +} diff --git a/src/main/java/com/xgh/test/spring/step04/beans/factory/support/AbstractBeanFactory.java b/src/main/java/com/xgh/test/spring/step04/beans/factory/support/AbstractBeanFactory.java new file mode 100644 index 0000000..31673cb --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step04/beans/factory/support/AbstractBeanFactory.java @@ -0,0 +1,44 @@ +package com.xgh.test.spring.step04.beans.factory.support; + +import com.xgh.test.spring.step04.beans.BeansException; +import com.xgh.test.spring.step04.beans.factory.BeanFactory; +import com.xgh.test.spring.step04.beans.factory.config.BeanDefinition; + +/** + * com.xgh.test.spring.step02.support.AbstractBeanFactory + * + * @author xgh
+ * @description 抽象类定义模板方法 + * @date 2021年08月17日 + */ +public abstract class AbstractBeanFactory extends DefaultSingletonBeanRegistry implements BeanFactory { + + @Override + public Object getBean(String beanName) throws BeansException { + return doGetBean(beanName,null); + } + + @Override + public Object getBean(String beanName, Object... args) throws BeansException { + return doGetBean(beanName,args); + } + + protected T doGetBean(String beanName, Object[] args){ + Object bean = getSingleton(beanName); + if(bean != null){ + return (T) bean; + } + BeanDefinition beanDefinition = getBeanDefinition(beanName); + return (T) creatBean(beanName,beanDefinition,args); + + } + @Override + public T getBean(String beanName, Class requiredType) { + return (T)getBean(beanName); + } + + protected abstract Object creatBean(String beanName, BeanDefinition beanDefinition, Object... args) throws BeansException; + + protected abstract BeanDefinition getBeanDefinition(String beanName) throws BeansException; + +} diff --git a/src/main/java/com/xgh/test/spring/step04/beans/factory/support/BeanDefinitionReader.java b/src/main/java/com/xgh/test/spring/step04/beans/factory/support/BeanDefinitionReader.java new file mode 100644 index 0000000..4afca7f --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step04/beans/factory/support/BeanDefinitionReader.java @@ -0,0 +1,26 @@ +package com.xgh.test.spring.step04.beans.factory.support; + +import com.xgh.test.spring.step04.beans.BeansException; +import com.xgh.test.spring.step04.core.io.Resource; +import com.xgh.test.spring.step04.core.io.ResourceLoader; + +/** + * com.xgh.test.spring.step04.beans.factory.support.BeanDefinitionReader + * + * @author xgh
+ * @description bean定义读取接口 + * @date 2021年08月27日 + */ +public interface BeanDefinitionReader { + + BeanDefinitionRegistry getRegistry(); + + ResourceLoader getResourceLoader(); + + void loadBeanDefinitions(Resource resource)throws BeansException; + + void loadBeanDefinitions(Resource... resources) throws BeansException; + + void loadBeanDefinitions(String location) throws BeansException; + +} diff --git a/src/main/java/com/xgh/test/spring/step04/beans/factory/support/BeanDefinitionRegistry.java b/src/main/java/com/xgh/test/spring/step04/beans/factory/support/BeanDefinitionRegistry.java new file mode 100644 index 0000000..cb3f63e --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step04/beans/factory/support/BeanDefinitionRegistry.java @@ -0,0 +1,41 @@ +package com.xgh.test.spring.step04.beans.factory.support; + +import com.xgh.test.spring.step04.beans.BeansException; +import com.xgh.test.spring.step04.beans.factory.config.BeanDefinition; + +/** + * com.xgh.test.spring.step02.support.BeanDefinitionRegistry + * + * @author xgh
+ * @description + * @date 2021年08月17日 + */ +public interface BeanDefinitionRegistry { + + /** + * @description 向注册表中注册 + * @date 2021/8/27 0027 + * @param beanName + * @param beanDefinition + */ + public void registerBeanDefinition(String beanName, BeanDefinition beanDefinition); + + + BeanDefinition getBeanDefinition(String beanName) throws BeansException; + + /** + * @description 判断是否包含指定的beanDefinition + * @date 2021/8/27 0027 + * @param beanName + * @return boolean + */ + boolean containsBeanDefinition(String beanName); + + /** + * @description 返回注册表所有beanName + * @date 2021/8/27 0027 + * @return java.lang.String[] + */ + String[] getBeanDefinitionNames(); + +} diff --git a/src/main/java/com/xgh/test/spring/step04/beans/factory/support/CglibSubclassingInstantiationStrategy.java b/src/main/java/com/xgh/test/spring/step04/beans/factory/support/CglibSubclassingInstantiationStrategy.java new file mode 100644 index 0000000..8cdaa45 --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step04/beans/factory/support/CglibSubclassingInstantiationStrategy.java @@ -0,0 +1,35 @@ +package com.xgh.test.spring.step04.beans.factory.support; + +import com.xgh.test.spring.step04.beans.factory.config.BeanDefinition; +import org.springframework.cglib.proxy.Enhancer; +import org.springframework.cglib.proxy.NoOp; + +import java.lang.reflect.Constructor; + +/** + * com.xgh.test.spring.step02.support.CglibSubclassingInstantiationStrategy + * + * @author xgh
+ * @description + * @date 2021年08月17日 + */ +public class CglibSubclassingInstantiationStrategy implements InstantiationStrategy { + + + @Override + public Object instantiate(BeanDefinition beanDefinition, String beanName, Constructor ctor, Object[] args) { + Enhancer enhancer = new Enhancer(); + enhancer.setSuperclass(beanDefinition.getBeanClass()); + enhancer.setCallback(new NoOp() { + @Override + public int hashCode() { + return super.hashCode(); + } + }); + if(null == ctor){ + return enhancer.create(); + } + return enhancer.create(ctor.getParameterTypes(),args); + } + +} diff --git a/src/main/java/com/xgh/test/spring/step04/beans/factory/support/DefaultListableBeanFactory.java b/src/main/java/com/xgh/test/spring/step04/beans/factory/support/DefaultListableBeanFactory.java new file mode 100644 index 0000000..d234eab --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step04/beans/factory/support/DefaultListableBeanFactory.java @@ -0,0 +1,60 @@ +package com.xgh.test.spring.step04.beans.factory.support; + +import com.xgh.test.spring.step04.beans.BeansException; +import com.xgh.test.spring.step04.beans.factory.ConfigurableListableBeanFactory; +import com.xgh.test.spring.step04.beans.factory.config.BeanDefinition; +import com.xgh.test.spring.step04.beans.factory.config.ConfigurableBeanFactory; + +import java.util.HashMap; +import java.util.Map; + +/** + * com.xgh.test.spring.step02.support.DefaultListableBeanFactory + * + * @author xgh
+ * @description 核心实现 + * @date 2021年08月17日 + */ +public class DefaultListableBeanFactory extends AbstractAutowireCapableBeanFactory implements BeanDefinitionRegistry, ConfigurableListableBeanFactory { + + private Map beanDefinitionMap = new HashMap<>(); + + + @Override + public void registerBeanDefinition(String beanName, BeanDefinition beanDefinition) { + beanDefinitionMap.put(beanName,beanDefinition); + } + + @Override + public boolean containsBeanDefinition(String beanName) { + return beanDefinitionMap.containsKey(beanName); + } + + @Override + public Map getBeansOfType(Class type) throws BeansException { + Map result = new HashMap<>(); + beanDefinitionMap.forEach((beanName,beanDefinition)->{ + Class beanClass = beanDefinition.getBeanClass(); + if(type.isAssignableFrom(beanClass)){ + result.put(beanName,(T) getBean(beanName)); + } + }); + return result; + } + + @Override + public String[] getBeanDefinitionNames() { + return beanDefinitionMap.keySet().toArray(new String[0]); + } + + @Override + public BeanDefinition getBeanDefinition(String beanName) throws BeansException { + BeanDefinition beanDefinition = beanDefinitionMap.get(beanName); + if(null == beanDefinition){ + throw new BeansException("No bean named '"+ beanName+"' is definded"); + } + return beanDefinition; + } + + +} diff --git a/src/main/java/com/xgh/test/spring/step04/beans/factory/support/DefaultSingletonBeanRegistry.java b/src/main/java/com/xgh/test/spring/step04/beans/factory/support/DefaultSingletonBeanRegistry.java new file mode 100644 index 0000000..2a1dc3d --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step04/beans/factory/support/DefaultSingletonBeanRegistry.java @@ -0,0 +1,27 @@ +package com.xgh.test.spring.step04.beans.factory.support; + +import com.xgh.test.spring.step04.beans.factory.config.SingletonBeanRegistry; + +import java.util.HashMap; +import java.util.Map; + +/** + * com.xgh.test.spring.step02.support.DefaultSingletinBeanRegistry + * + * @author xgh
+ * @description 单例实例的获取接口 + * @date 2021年08月17日 + */ +public class DefaultSingletonBeanRegistry implements SingletonBeanRegistry { + + private Map singletonObjects = new HashMap<>(); + + @Override + public Object getSingleton(String beanName) { + return singletonObjects.get(beanName); + } + + protected void addSingleton(String beanName,Object singletonObject){ + singletonObjects.put(beanName,singletonObject); + } +} diff --git a/src/main/java/com/xgh/test/spring/step04/beans/factory/support/InstantiationStrategy.java b/src/main/java/com/xgh/test/spring/step04/beans/factory/support/InstantiationStrategy.java new file mode 100644 index 0000000..abc3e52 --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step04/beans/factory/support/InstantiationStrategy.java @@ -0,0 +1,17 @@ +package com.xgh.test.spring.step04.beans.factory.support; + +import com.xgh.test.spring.step04.beans.factory.config.BeanDefinition; + +import java.lang.reflect.Constructor; + +/** + * com.xgh.test.spring.step02.support.InstantionStrategy + * + * @author xgh
+ * @description 实例化策略 + * @date 2021年08月17日 + */ +public interface InstantiationStrategy { + + public Object instantiate(BeanDefinition beanDefinition, String beanName, Constructor ctor, Object[] args); +} diff --git a/src/main/java/com/xgh/test/spring/step04/beans/factory/support/SimpleInstantiationStrategy.java b/src/main/java/com/xgh/test/spring/step04/beans/factory/support/SimpleInstantiationStrategy.java new file mode 100644 index 0000000..29c9d6a --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step04/beans/factory/support/SimpleInstantiationStrategy.java @@ -0,0 +1,34 @@ +package com.xgh.test.spring.step04.beans.factory.support; + +import com.xgh.test.spring.step04.beans.BeansException; +import com.xgh.test.spring.step04.beans.factory.config.BeanDefinition; + +import java.lang.reflect.Constructor; +import java.lang.reflect.InvocationTargetException; + +/** + * com.xgh.test.spring.step02.support.SimpleInstantiationStrategy + * + * @author xgh
+ * @description jdk + * @date 2021年08月17日 + */ +public class SimpleInstantiationStrategy implements InstantiationStrategy { + + + @Override + public Object instantiate(BeanDefinition beanDefinition, String beanName, Constructor ctor, Object[] args) { + Class clazz = beanDefinition.getBeanClass(); + try { + if (null != ctor) { + return clazz.getDeclaredConstructor(ctor.getParameterTypes()).newInstance(args); + }else { + return clazz.getDeclaredConstructor().newInstance(); + } + }catch(InstantiationException |IllegalAccessException|InvocationTargetException | NoSuchMethodException e) { + throw new BeansException("Failed to instantiate["+clazz.getName()+"]",e); + } + } + + +} diff --git a/src/main/java/com/xgh/test/spring/step04/beans/factory/xml/XmlBeanDefinitionReader.java b/src/main/java/com/xgh/test/spring/step04/beans/factory/xml/XmlBeanDefinitionReader.java new file mode 100644 index 0000000..b65ad50 --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step04/beans/factory/xml/XmlBeanDefinitionReader.java @@ -0,0 +1,114 @@ +package com.xgh.test.spring.step04.beans.factory.xml; + +import cn.hutool.core.util.StrUtil; +import cn.hutool.core.util.XmlUtil; +import com.xgh.test.spring.step04.beans.BeansException; +import com.xgh.test.spring.step04.beans.PropertyValue; +import com.xgh.test.spring.step04.beans.factory.config.BeanDefinition; +import com.xgh.test.spring.step04.beans.factory.config.BeanReference; +import com.xgh.test.spring.step04.beans.factory.support.AbstractBeanDefinitionReader; +import com.xgh.test.spring.step04.beans.factory.support.BeanDefinitionRegistry; +import com.xgh.test.spring.step04.core.io.Resource; +import com.xgh.test.spring.step04.core.io.ResourceLoader; +import org.w3c.dom.Document; +import org.w3c.dom.Element; +import org.w3c.dom.Node; +import org.w3c.dom.NodeList; + +import java.io.IOException; +import java.io.InputStream; + +/** + * com.xgh.test.spring.step04.beans.factory.xml.XmlBeanDefinitionReader + * + * @author xgh
+ * @description 解析xml处理bean注册 + * @date 2021年08月27日 + */ +public class XmlBeanDefinitionReader extends AbstractBeanDefinitionReader { + + + public XmlBeanDefinitionReader(BeanDefinitionRegistry registry) { + super(registry); + } + + public XmlBeanDefinitionReader(BeanDefinitionRegistry registry, ResourceLoader resourceLoader) { + super(registry, resourceLoader); + } + + @Override + public void loadBeanDefinitions(Resource resource) throws BeansException { + + try { + try (InputStream inputStream = resource.getInputStream()) { + doLoadBeanDefinitions(inputStream); + } + } catch (IOException | ClassNotFoundException e) { + throw new BeansException("IOException parsing XML document form" + resource, e); + } + } + + protected void doLoadBeanDefinitions(InputStream inputStream) throws ClassNotFoundException { + Document doc = XmlUtil.readXML(inputStream); + Element root = doc.getDocumentElement(); + NodeList childNodes = root.getChildNodes(); + for (int i = 0; i < childNodes.getLength(); i++) { + if (!(childNodes.item(i) instanceof Element)) { + continue; + } + if (!"bean".equals(childNodes.item(i).getNodeName())) { + continue; + } + + Element bean = (Element) childNodes.item(i); + + String id = bean.getAttribute("id"); + String name = bean.getAttribute("name"); + String className = bean.getAttribute("class"); + Class clazz = Class.forName(className); + String beanName = StrUtil.isNotEmpty(id) ? id : name; + if (StrUtil.isEmpty(beanName)) { + beanName = StrUtil.lowerFirst(clazz.getSimpleName()); + } + //定义bean + BeanDefinition beanDefinition = new BeanDefinition(clazz); + //读取属性填充 + for (int j = 0; j < bean.getChildNodes().getLength(); j++) { + if (!(bean.getChildNodes().item(j) instanceof Element)) { + continue; + } + if (!"property".equals(bean.getChildNodes().item(j).getNodeName())) { + continue; + } + + Element property = (Element) bean.getChildNodes().item(j); + String attrName = property.getAttribute("name"); + String attrValue = property.getAttribute("value"); + String attrRef = property.getAttribute("ref"); + //获取属性值:引入对象值对象 + Object value = StrUtil.isNotEmpty(attrRef) ? new BeanReference(attrRef) : attrValue; + PropertyValue propertyValue = new PropertyValue(attrName, value); + beanDefinition.getPropertyValues().addPropertyValue(propertyValue); + } + if (getRegistry().containsBeanDefinition(beanName)) { + throw new BeansException("Duplicate beanName[ " + beanName + "] is not allowed"); + } + //注册bd + getRegistry().registerBeanDefinition(beanName, beanDefinition); + } + } + + @Override + public void loadBeanDefinitions(Resource... resources) throws BeansException { + for (Resource resource : resources) { + loadBeanDefinitions(resource); + } + } + + @Override + public void loadBeanDefinitions(String location) throws BeansException { + ResourceLoader resourceLoader = getResourceLoader(); + Resource resource = resourceLoader.getResource(location); + loadBeanDefinitions(resource); + } +} diff --git a/src/main/java/com/xgh/test/spring/step04/core/io/ClassPathResource.java b/src/main/java/com/xgh/test/spring/step04/core/io/ClassPathResource.java new file mode 100644 index 0000000..d4dcad9 --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step04/core/io/ClassPathResource.java @@ -0,0 +1,41 @@ +package com.xgh.test.spring.step04.core.io; + +import cn.hutool.core.lang.Assert; +import com.xgh.test.spring.step04.util.ClassUtils; + +import java.io.FileNotFoundException; +import java.io.IOException; +import java.io.InputStream; + +/** + * com.xgh.test.spring.step04.core.io.ClassPathResource + * + * @author xgh
+ * @description classPath的资源加载器 + * @date 2021年08月27日 + */ +public class ClassPathResource implements Resource{ + + private final String path; + + private ClassLoader classLoader; + + public ClassPathResource(String path) { + this(path,null); + } + + public ClassPathResource(String path, ClassLoader classLoader) { + Assert.notNull(path,"path mast not be null"); + this.path = path; + this.classLoader = (classLoader != null ? classLoader : ClassUtils.getDefaultClassLoader()); + } + + @Override + public InputStream getInputStream() throws IOException { + InputStream is = classLoader.getResourceAsStream(path); + if(is == null){ + throw new FileNotFoundException(this.path + " cannot be opened because ist does not exist"); + } + return is; + } +} diff --git a/src/main/java/com/xgh/test/spring/step04/core/io/DefaultResourceLoader.java b/src/main/java/com/xgh/test/spring/step04/core/io/DefaultResourceLoader.java new file mode 100644 index 0000000..dd66fab --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step04/core/io/DefaultResourceLoader.java @@ -0,0 +1,35 @@ +package com.xgh.test.spring.step04.core.io; + +import cn.hutool.core.lang.Assert; + +import java.net.MalformedURLException; +import java.net.URL; + +/** + * com.xgh.test.spring.step04.core.io.DefaultResourceLoader + * + * @author xgh
+ * @description 获取资源 + * @date 2021年08月27日 + */ +public class DefaultResourceLoader implements ResourceLoader{ + + + + @Override + public Resource getResource(String location) { + Assert.notNull(location,"location mast not be null"); + if(location.startsWith(CLASSPATH_URL_PREFIX)){ + return new ClassPathResource(location.substring(CLASSPATH_URL_PREFIX.length())); + } + else + { + try { + URL url = new URL(location); + return new UrlResource(url); + }catch (MalformedURLException e){ + return new FileSystemResource(location); + } + } + } +} diff --git a/src/main/java/com/xgh/test/spring/step04/core/io/FileSystemResource.java b/src/main/java/com/xgh/test/spring/step04/core/io/FileSystemResource.java new file mode 100644 index 0000000..f8831d7 --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step04/core/io/FileSystemResource.java @@ -0,0 +1,40 @@ +package com.xgh.test.spring.step04.core.io; + +import java.io.File; +import java.io.FileInputStream; +import java.io.IOException; +import java.io.InputStream; + +/** + * com.xgh.test.spring.step04.core.io.FileSystemResource + * + * @author xgh
+ * @description 文件资源加载 + * @date 2021年08月27日 + */ +public class FileSystemResource implements Resource{ + + private final File file; + + private final String path; + + public FileSystemResource(File file) { + this.file = file; + this.path = file.getPath(); + } + + + public FileSystemResource(String path) { + this.file = new File(path); + this.path = path; + } + + @Override + public InputStream getInputStream() throws IOException { + return new FileInputStream(this.file); + } + + public String getPath() { + return path; + } +} diff --git a/src/main/java/com/xgh/test/spring/step04/core/io/Resource.java b/src/main/java/com/xgh/test/spring/step04/core/io/Resource.java new file mode 100644 index 0000000..305f04b --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step04/core/io/Resource.java @@ -0,0 +1,16 @@ +package com.xgh.test.spring.step04.core.io; + +import java.io.IOException; +import java.io.InputStream; + +/** + * com.xgh.test.spring.step04.core.io.Resource + * + * @author xgh
+ * @description 资源加载接口的定义和实现 + * @date 2021年08月27日 + */ +public interface Resource { + + InputStream getInputStream() throws IOException; +} diff --git a/src/main/java/com/xgh/test/spring/step04/core/io/ResourceLoader.java b/src/main/java/com/xgh/test/spring/step04/core/io/ResourceLoader.java new file mode 100644 index 0000000..2268713 --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step04/core/io/ResourceLoader.java @@ -0,0 +1,15 @@ +package com.xgh.test.spring.step04.core.io; + +/** + * com.xgh.test.spring.step04.core.io.ResourceLoader + * + * @author xgh
+ * @description 资源加载器 + * @date 2021年08月27日 + */ +public interface ResourceLoader { + + String CLASSPATH_URL_PREFIX= "classpath:"; + + Resource getResource(String location); +} diff --git a/src/main/java/com/xgh/test/spring/step04/core/io/UrlResource.java b/src/main/java/com/xgh/test/spring/step04/core/io/UrlResource.java new file mode 100644 index 0000000..b5c3cf3 --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step04/core/io/UrlResource.java @@ -0,0 +1,39 @@ +package com.xgh.test.spring.step04.core.io; + +import cn.hutool.core.lang.Assert; + +import java.io.IOException; +import java.io.InputStream; +import java.net.HttpURLConnection; +import java.net.URL; +import java.net.URLConnection; + +/** + * com.xgh.test.spring.step04.core.io.UrlResource + * + * @author xgh
+ * @description + * @date 2021年08月27日 + */ +public class UrlResource implements Resource{ + + private final URL url; + + public UrlResource(URL url) { + Assert.notNull(url,"url mast not be null"); + this.url = url; + } + + @Override + public InputStream getInputStream() throws IOException { + URLConnection con = this.url.openConnection(); + try { + return con.getInputStream(); + }catch (IOException e){ + if(con instanceof HttpURLConnection){ + ((HttpURLConnection) con).disconnect(); + } + throw e; + } + } +} diff --git a/src/main/java/com/xgh/test/spring/step04/util/ClassUtils.java b/src/main/java/com/xgh/test/spring/step04/util/ClassUtils.java new file mode 100644 index 0000000..d102e6a --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step04/util/ClassUtils.java @@ -0,0 +1,24 @@ +package com.xgh.test.spring.step04.util; + +/** + * com.xgh.test.spring.step04.util.ClassUtils + * + * @author xgh
+ * @description + * @date 2021年08月27日 + */ +public class ClassUtils { + + public static ClassLoader getDefaultClassLoader(){ + ClassLoader cl = null; + try{ + cl = Thread.currentThread().getContextClassLoader(); + }catch (Throwable ex){ + System.out.println(ex); + } + if(cl == null){ + cl = ClassUtils.class.getClassLoader(); + } + return cl; + } +} diff --git a/src/test/java/com/xgh/test/spring/test04/Test04.java b/src/test/java/com/xgh/test/spring/test04/Test04.java new file mode 100644 index 0000000..5554c1c --- /dev/null +++ b/src/test/java/com/xgh/test/spring/test04/Test04.java @@ -0,0 +1,55 @@ +package com.xgh.test.spring.test04; + +import cn.hutool.core.io.IoUtil; +import com.xgh.test.spring.step04.beans.factory.support.DefaultListableBeanFactory; +import com.xgh.test.spring.step04.beans.factory.xml.XmlBeanDefinitionReader; +import com.xgh.test.spring.step04.core.io.DefaultResourceLoader; +import com.xgh.test.spring.step04.core.io.Resource; +import org.junit.Before; +import org.junit.Test; + +import java.io.IOException; +import java.io.InputStream; + +/** + * com.xgh.test.spring.test04.Test04 + * + * @author xgh
+ * @description + * @date 2021年08月27日 + */ +public class Test04 { + + private DefaultResourceLoader resourceLoader; + + @Before + public void init(){ + resourceLoader = new DefaultResourceLoader(); + } + + @Test + public void test_classpath() throws IOException{ + Resource resource = resourceLoader.getResource("classpath:important.properties"); + InputStream inputStream = resource.getInputStream(); + String content = IoUtil.readUtf8(inputStream); + System.out.println(content); + } + + @Test + public void test_file()throws IOException{ + Resource resource = resourceLoader.getResource("src/test/resources/important.properties"); + InputStream inputStream = resource.getInputStream(); + String content = IoUtil.readUtf8(inputStream); + System.out.println(content); + } + + @Test + public void test_xml(){ + DefaultListableBeanFactory beanFactory = new DefaultListableBeanFactory(); + XmlBeanDefinitionReader reader = new XmlBeanDefinitionReader(beanFactory); + reader.loadBeanDefinitions("classpath:spring.xml"); + UserService userService = beanFactory.getBean("userService", UserService.class); + System.out.println(userService.queryUserInfo()); + } + +} diff --git a/src/test/java/com/xgh/test/spring/test04/UserDao.java b/src/test/java/com/xgh/test/spring/test04/UserDao.java new file mode 100644 index 0000000..e513f95 --- /dev/null +++ b/src/test/java/com/xgh/test/spring/test04/UserDao.java @@ -0,0 +1,29 @@ +package com.xgh.test.spring.test04; + +import java.util.HashMap; +import java.util.Map; + +/** + * com.xgh.test.spring.test03.UserDao + * + * @author xgh
+ * @description + * @date 2021年08月24日 + */ +public class UserDao { + + + private static Map hashMap = new HashMap<>(); + + + static { + hashMap.put("10001","张三"); + hashMap.put("10002","李四"); + hashMap.put("10003","王五"); + } + + + public String queryUserName(String uId){ + return hashMap.get(uId); + } +} diff --git a/src/test/java/com/xgh/test/spring/test04/UserService.java b/src/test/java/com/xgh/test/spring/test04/UserService.java new file mode 100644 index 0000000..e74a939 --- /dev/null +++ b/src/test/java/com/xgh/test/spring/test04/UserService.java @@ -0,0 +1,36 @@ +package com.xgh.test.spring.test04; + + +/** + * com.xgh.test.spring.test03.UserService + * + * @author xgh
+ * @description + * @date 2021年08月24日 + */ +public class UserService { + + private String uId; + + private UserDao userDao; + + public String queryUserInfo() { + return userDao.queryUserName(uId); + } + + public String getuId() { + return uId; + } + + public void setuId(String uId) { + this.uId = uId; + } + + public UserDao getUserDao() { + return userDao; + } + + public void setUserDao(UserDao userDao) { + this.userDao = userDao; + } +} diff --git a/src/test/resources/Spring.xml b/src/test/resources/Spring.xml new file mode 100644 index 0000000..e6f226e --- /dev/null +++ b/src/test/resources/Spring.xml @@ -0,0 +1,11 @@ + + + + + + + + + + + \ No newline at end of file diff --git a/src/test/resources/important.properties b/src/test/resources/important.properties new file mode 100644 index 0000000..9a02442 --- /dev/null +++ b/src/test/resources/important.properties @@ -0,0 +1 @@ +system.key=OLpj9823dz \ No newline at end of file From 4e5a126d45050b43fcaa37bab6a676782dca74c9 Mon Sep 17 00:00:00 2001 From: xgh <1371818484@qq.com> Date: Tue, 31 Aug 2021 21:52:44 +0800 Subject: [PATCH 16/20] =?UTF-8?q?=E5=BA=94=E7=94=A8=E4=B8=8A=E4=B8=8B?= =?UTF-8?q?=E6=96=8701?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../config/BeanFactoryPostProcessor.java | 14 +++++++++++ .../factory/config/BeanPostProcessor.java | 25 +++++++++++++++++++ .../step04/context/ApplicationContext.java | 6 +++++ 3 files changed, 45 insertions(+) create mode 100644 src/main/java/com/xgh/test/spring/step04/beans/factory/config/BeanFactoryPostProcessor.java create mode 100644 src/main/java/com/xgh/test/spring/step04/beans/factory/config/BeanPostProcessor.java create mode 100644 src/main/java/com/xgh/test/spring/step04/context/ApplicationContext.java diff --git a/src/main/java/com/xgh/test/spring/step04/beans/factory/config/BeanFactoryPostProcessor.java b/src/main/java/com/xgh/test/spring/step04/beans/factory/config/BeanFactoryPostProcessor.java new file mode 100644 index 0000000..55172a4 --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step04/beans/factory/config/BeanFactoryPostProcessor.java @@ -0,0 +1,14 @@ +package com.xgh.test.spring.step04.beans.factory.config; + + +import com.xgh.test.spring.step04.beans.BeansException; +import com.xgh.test.spring.step04.beans.factory.ConfigurableListableBeanFactory; + +public interface BeanFactoryPostProcessor { + + + /** + * 在所有的beanDefinition加载完成后,实例化bean对象之前,提供修改BeanDefinition属性的机制 + */ + void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException; +} diff --git a/src/main/java/com/xgh/test/spring/step04/beans/factory/config/BeanPostProcessor.java b/src/main/java/com/xgh/test/spring/step04/beans/factory/config/BeanPostProcessor.java new file mode 100644 index 0000000..e1e1763 --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step04/beans/factory/config/BeanPostProcessor.java @@ -0,0 +1,25 @@ +package com.xgh.test.spring.step04.beans.factory.config; + +import com.xgh.test.spring.step04.beans.BeansException; + +public interface BeanPostProcessor { + + /** + * 在bean对象执行初始化方法之前,执行此方法 + * @param bean + * @param beanName + * @return + * @throws BeansException + */ + Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException; + + + /** + * 在bean对象执行初始化完成之后执行此方法 + * @param bean + * @param beanName + * @return + * @throws BeansException + */ + Object postProcessAfterInitialization(Object bean,String beanName) throws BeansException; +} diff --git a/src/main/java/com/xgh/test/spring/step04/context/ApplicationContext.java b/src/main/java/com/xgh/test/spring/step04/context/ApplicationContext.java new file mode 100644 index 0000000..098fead --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step04/context/ApplicationContext.java @@ -0,0 +1,6 @@ +package com.xgh.test.spring.step04.context; + +import com.xgh.test.spring.step04.beans.factory.ListableBeanFactory; + +public interface ApplicationContext extends ListableBeanFactory { +} From 9a67d7fc286653af9a46d57fd8f4debfb43aef03 Mon Sep 17 00:00:00 2001 From: xgh <1371818484@qq.com> Date: Tue, 28 Jun 2022 20:12:21 +0800 Subject: [PATCH 17/20] spring --- pom.xml | 6 +- .../java/com/xgh/test/design/Transaction.java | 11 +++ .../redis/lock/RedisWithReentrantLock.java | 83 ++++++++++++++++ .../ConfigurableListableBeanFactory.java | 8 +- .../step04/beans/factory/DisposableBean.java | 8 ++ .../beans/factory/InitializingBean.java | 14 +++ .../beans/factory/config/BeanDefinition.java | 19 ++++ .../config/ConfigurableBeanFactory.java | 7 ++ .../AbstractAutowireCapableBeanFactory.java | 87 +++++++++++++++-- .../factory/support/AbstractBeanFactory.java | 18 +++- .../factory/support/BeanDefinitionReader.java | 2 + .../support/DefaultListableBeanFactory.java | 5 + .../support/DefaultSingletonBeanRegistry.java | 24 +++++ .../support/DisposableBeanAdapter.java | 43 +++++++++ .../factory/xml/XmlBeanDefinitionReader.java | 7 ++ .../ConfigurableApplicationContext.java | 18 ++++ .../support/AbstractApplicationContext.java | 94 +++++++++++++++++++ .../AbstractRefreshableApplication.java | 33 +++++++ .../AbstractXmlApplicationContext.java | 25 +++++ .../ClassPathXmlApplicationContext.java | 27 ++++++ .../test05/MyBeanFactoryPostProcessor.java | 20 ++++ .../spring/test05/MyBeanPostProcessor.java | 20 ++++ .../com/xgh/test/spring/test05/Test05.java | 46 +++++++++ .../com/xgh/test/spring/test05/UserDao.java | 26 +++++ .../xgh/test/spring/test05/UserService.java | 64 +++++++++++++ src/test/resources/Spring.xml | 7 +- 26 files changed, 709 insertions(+), 13 deletions(-) create mode 100644 src/main/java/com/xgh/test/redis/lock/RedisWithReentrantLock.java create mode 100644 src/main/java/com/xgh/test/spring/step04/beans/factory/DisposableBean.java create mode 100644 src/main/java/com/xgh/test/spring/step04/beans/factory/InitializingBean.java create mode 100644 src/main/java/com/xgh/test/spring/step04/beans/factory/support/DisposableBeanAdapter.java create mode 100644 src/main/java/com/xgh/test/spring/step04/context/ConfigurableApplicationContext.java create mode 100644 src/main/java/com/xgh/test/spring/step04/context/support/AbstractApplicationContext.java create mode 100644 src/main/java/com/xgh/test/spring/step04/context/support/AbstractRefreshableApplication.java create mode 100644 src/main/java/com/xgh/test/spring/step04/context/support/AbstractXmlApplicationContext.java create mode 100644 src/main/java/com/xgh/test/spring/step04/context/support/ClassPathXmlApplicationContext.java create mode 100644 src/test/java/com/xgh/test/spring/test05/MyBeanFactoryPostProcessor.java create mode 100644 src/test/java/com/xgh/test/spring/test05/MyBeanPostProcessor.java create mode 100644 src/test/java/com/xgh/test/spring/test05/Test05.java create mode 100644 src/test/java/com/xgh/test/spring/test05/UserDao.java create mode 100644 src/test/java/com/xgh/test/spring/test05/UserService.java diff --git a/pom.xml b/pom.xml index 604a2ba..3a26a3e 100644 --- a/pom.xml +++ b/pom.xml @@ -61,7 +61,11 @@ hutool-core 5.6.3 - + + redis.clients + jedis + 2.7.1 + diff --git a/src/main/java/com/xgh/test/design/Transaction.java b/src/main/java/com/xgh/test/design/Transaction.java index 6de06d7..e32aaee 100644 --- a/src/main/java/com/xgh/test/design/Transaction.java +++ b/src/main/java/com/xgh/test/design/Transaction.java @@ -103,5 +103,16 @@ private boolean getLock(String id) { return true; } + public static void main(String[] args) { + Integer[] a = new Integer[]{1,2,3,4,5}; + + a[2] = 5; + for (Integer integer : a) { + System.out.println(integer); + } + + + } + } diff --git a/src/main/java/com/xgh/test/redis/lock/RedisWithReentrantLock.java b/src/main/java/com/xgh/test/redis/lock/RedisWithReentrantLock.java new file mode 100644 index 0000000..765a50e --- /dev/null +++ b/src/main/java/com/xgh/test/redis/lock/RedisWithReentrantLock.java @@ -0,0 +1,83 @@ +package com.xgh.test.redis.lock; + +import redis.clients.jedis.*; + +import java.util.HashMap; +import java.util.Map; + +public class RedisWithReentrantLock { + + private ThreadLocal> lockers = new ThreadLocal<>(); + + private Jedis jedis; + + public RedisWithReentrantLock(Jedis jedis) { + this.jedis = jedis; + } + + private boolean _lock(String key) { + return jedis.set(key, "", "nx", "ex", 5L) != null; + } + + private void _unlock(String key) { + jedis.del(key); + } + + + private Map currentLockers() { + Map refs = lockers.get(); + if (refs != null) { + return refs; + } + lockers.set(new HashMap<>()); + return lockers.get(); + } + + public boolean lock(String key) { + Map refs = currentLockers(); + Integer refCnt = refs.get(key); + if (refCnt != null) { + refs.put(key, refCnt + 1); + return true; + } + boolean ok = this._lock(key); + if (!ok) { + return false; + } + refs.put(key, 1); + return true; + } + + + public boolean unlock(String key) { + Map refs = currentLockers(); + + Integer refCnt = refs.get(key); + if (refCnt == null) { + return false; + } + refCnt -= 1; + if (refCnt > 0) { + refs.put(key, refCnt); + } else { + refs.remove(key); + this._unlock(key); + } + return true; + } + + public static void main(String[] args) { + Jedis jedis = new Jedis("192.168.220.131", 6379); + jedis.auth("123456"); + RedisWithReentrantLock redisWithReentrantLock = new RedisWithReentrantLock(jedis); + + System.out.println(redisWithReentrantLock.lock("books1")); + System.out.println(redisWithReentrantLock.lock("books1")); + System.out.println(redisWithReentrantLock.unlock("books1")); + System.out.println(redisWithReentrantLock.unlock("books1")); + + + } + + +} diff --git a/src/main/java/com/xgh/test/spring/step04/beans/factory/ConfigurableListableBeanFactory.java b/src/main/java/com/xgh/test/spring/step04/beans/factory/ConfigurableListableBeanFactory.java index 27c3789..c42875d 100644 --- a/src/main/java/com/xgh/test/spring/step04/beans/factory/ConfigurableListableBeanFactory.java +++ b/src/main/java/com/xgh/test/spring/step04/beans/factory/ConfigurableListableBeanFactory.java @@ -3,6 +3,7 @@ import com.xgh.test.spring.step04.beans.BeansException; import com.xgh.test.spring.step04.beans.factory.config.AutowireCapableBeanFactory; import com.xgh.test.spring.step04.beans.factory.config.BeanDefinition; +import com.xgh.test.spring.step04.beans.factory.config.BeanPostProcessor; import com.xgh.test.spring.step04.beans.factory.config.ConfigurableBeanFactory; /** @@ -14,5 +15,10 @@ */ public interface ConfigurableListableBeanFactory extends ListableBeanFactory, AutowireCapableBeanFactory, ConfigurableBeanFactory { - BeanDefinition getBeanDefinition(String beanName)throws BeansException; + BeanDefinition getBeanDefinition(String beanName) throws BeansException; + + void preInstantiateSingletons() throws BeansException; + + + void addBeanPostProcessor(BeanPostProcessor beanPostProcessor); } diff --git a/src/main/java/com/xgh/test/spring/step04/beans/factory/DisposableBean.java b/src/main/java/com/xgh/test/spring/step04/beans/factory/DisposableBean.java new file mode 100644 index 0000000..65a5047 --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step04/beans/factory/DisposableBean.java @@ -0,0 +1,8 @@ +package com.xgh.test.spring.step04.beans.factory; + +public interface DisposableBean { + + void destroy()throws Exception; + + +} diff --git a/src/main/java/com/xgh/test/spring/step04/beans/factory/InitializingBean.java b/src/main/java/com/xgh/test/spring/step04/beans/factory/InitializingBean.java new file mode 100644 index 0000000..b59ab20 --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step04/beans/factory/InitializingBean.java @@ -0,0 +1,14 @@ +package com.xgh.test.spring.step04.beans.factory; + + +/** + * 实现此接口的bean对象,会在beanFactory设置属性后做出相应的处理,儒执行自定义初始化,或者仅仅校验是否设置了所有强制属性 + */ +public interface InitializingBean { + + /** + * 处理了属性填充后调用 + * @throws Exception + */ + void afterPropertiesSet() throws Exception; +} diff --git a/src/main/java/com/xgh/test/spring/step04/beans/factory/config/BeanDefinition.java b/src/main/java/com/xgh/test/spring/step04/beans/factory/config/BeanDefinition.java index 80b9696..bb67517 100644 --- a/src/main/java/com/xgh/test/spring/step04/beans/factory/config/BeanDefinition.java +++ b/src/main/java/com/xgh/test/spring/step04/beans/factory/config/BeanDefinition.java @@ -15,6 +15,9 @@ public class BeanDefinition { private PropertyValues propertyValues; + private String initMethodName; + + private String destroyMethodName; public BeanDefinition(Class beanClass) { this.beanClass = beanClass; @@ -41,4 +44,20 @@ public PropertyValues getPropertyValues() { public void setPropertyValues(PropertyValues propertyValues) { this.propertyValues = propertyValues; } + + public String getInitMethodName() { + return initMethodName; + } + + public void setInitMethodName(String initMethodName) { + this.initMethodName = initMethodName; + } + + public String getDestroyMethodName() { + return destroyMethodName; + } + + public void setDestroyMethodName(String destroyMethodName) { + this.destroyMethodName = destroyMethodName; + } } diff --git a/src/main/java/com/xgh/test/spring/step04/beans/factory/config/ConfigurableBeanFactory.java b/src/main/java/com/xgh/test/spring/step04/beans/factory/config/ConfigurableBeanFactory.java index 4f5cc63..cf9b26b 100644 --- a/src/main/java/com/xgh/test/spring/step04/beans/factory/config/ConfigurableBeanFactory.java +++ b/src/main/java/com/xgh/test/spring/step04/beans/factory/config/ConfigurableBeanFactory.java @@ -14,4 +14,11 @@ public interface ConfigurableBeanFactory extends HierarchicalBeanFactory,Singlet String SCOPE_SINGLETON = "singleton"; String SCOPE_PROTOTYPE= "prototype"; + + void addBeanPostProcessor(BeanPostProcessor beanPostProcessor); + + /** + * 销毁单例bean + */ + void destroySingletons(); } diff --git a/src/main/java/com/xgh/test/spring/step04/beans/factory/support/AbstractAutowireCapableBeanFactory.java b/src/main/java/com/xgh/test/spring/step04/beans/factory/support/AbstractAutowireCapableBeanFactory.java index 9b178df..e1738c4 100644 --- a/src/main/java/com/xgh/test/spring/step04/beans/factory/support/AbstractAutowireCapableBeanFactory.java +++ b/src/main/java/com/xgh/test/spring/step04/beans/factory/support/AbstractAutowireCapableBeanFactory.java @@ -1,13 +1,19 @@ package com.xgh.test.spring.step04.beans.factory.support; import cn.hutool.core.bean.BeanUtil; +import cn.hutool.core.util.StrUtil; import com.xgh.test.spring.step04.beans.BeansException; import com.xgh.test.spring.step04.beans.PropertyValue; import com.xgh.test.spring.step04.beans.PropertyValues; +import com.xgh.test.spring.step04.beans.factory.DisposableBean; +import com.xgh.test.spring.step04.beans.factory.InitializingBean; +import com.xgh.test.spring.step04.beans.factory.config.AutowireCapableBeanFactory; import com.xgh.test.spring.step04.beans.factory.config.BeanDefinition; +import com.xgh.test.spring.step04.beans.factory.config.BeanPostProcessor; import com.xgh.test.spring.step04.beans.factory.config.BeanReference; import java.lang.reflect.Constructor; +import java.lang.reflect.Method; /** * com.xgh.test.spring.step02.support.AbstractAutowireCapableBeanFactory @@ -16,25 +22,92 @@ * @description 实例化bean类 * @date 2021年08月17日 */ -public abstract class AbstractAutowireCapableBeanFactory extends AbstractBeanFactory { +public abstract class AbstractAutowireCapableBeanFactory extends AbstractBeanFactory implements AutowireCapableBeanFactory { private InstantiationStrategy instantiationStrategy = new CglibSubclassingInstantiationStrategy(); - + @Override protected Object creatBean(String beanName, BeanDefinition beanDefinition, Object... args) throws BeansException { Object bean = null; try { - - bean = creatBeanInstance(beanDefinition,beanName,args); - //设置参数 + bean = creatBeanInstance(beanDefinition,beanName,args); + //给bean填充属性 applyPropertyValues(beanName,bean,beanDefinition); - } catch (Exception e) { - throw new BeansException("Instantiation of bean fail",e); + + //执行bean的初始化方法和beanPostProcessor的前置后置方法 + bean = initializeBean(beanName,bean,beanDefinition); + + }catch (Exception e){ + throw new BeansException("instantiation of bean failed",e); } + //注册实现了disposableBean接口的bean + registerDisposableBeanIfNecessary(beanName,bean,beanDefinition); addSingleton(beanName,bean); return bean; } + protected void registerDisposableBeanIfNecessary(String beanName, Object bean, BeanDefinition beanDefinition){ + if(bean instanceof DisposableBean || StrUtil.isNotEmpty(beanDefinition.getDestroyMethodName())){ + registerDisposableBean(beanName,new DisposableBeanAdapter(bean,beanName,beanDefinition)); + } + } + + + protected Object initializeBean(String beanName, Object bean, BeanDefinition beanDefinition){ + //执行beanPostProcessor before处理 + Object wrappedBean = applyBeanPostProcessorsBeforeInitialization(bean,beanName); + try { + //待完成内容invokeInitMethods + invokeInitMethods(beanName,wrappedBean,beanDefinition); + }catch (Exception e){ + throw new BeansException("Invocation of init method of bean[" + beanName + "] failed", e); + } + + //执行beanPostProcessor after + wrappedBean = applyBeanPostProcessorAfterInitialization(bean,beanName); + return wrappedBean; + } + + private Object applyBeanPostProcessorAfterInitialization(Object existingBean, String beanName) { + Object result =existingBean; + for (BeanPostProcessor processor : getBeanPostProcessors()) { + Object current = processor.postProcessAfterInitialization(result,beanName); + if(null == current){ + return result; + } + result = current; + } + return result; + } + + private void invokeInitMethods(String beanName, Object bean, BeanDefinition beanDefinition) throws Exception { + //1实现接口initializingBean + if(bean instanceof InitializingBean){ + ((InitializingBean) bean).afterPropertiesSet(); + } + //注解配置init-method(判断是否为了避免二次执行) + String initMethodName = beanDefinition.getInitMethodName(); + if(StrUtil.isNotEmpty(initMethodName)){ + Method initMethod = beanDefinition.getBeanClass().getMethod(initMethodName); + if(null == initMethod){ + throw new BeansException("Could not find an init method named '" + initMethodName + "' on bean with name '" + beanName + "'"); + } + initMethod.invoke(bean); + } + + } + + public Object applyBeanPostProcessorsBeforeInitialization(Object existingBean, String beanName){ + Object result =existingBean; + for (BeanPostProcessor processor : getBeanPostProcessors()) { + Object current = processor.postProcessBeforeInitialization(result,beanName); + if(null == current){ + return result; + } + result = current; + } + return result; + }; protected Object creatBeanInstance(BeanDefinition beanDefinition, String beanName, Object[] args){ Constructor constructorToUser = null; diff --git a/src/main/java/com/xgh/test/spring/step04/beans/factory/support/AbstractBeanFactory.java b/src/main/java/com/xgh/test/spring/step04/beans/factory/support/AbstractBeanFactory.java index 31673cb..5f96fec 100644 --- a/src/main/java/com/xgh/test/spring/step04/beans/factory/support/AbstractBeanFactory.java +++ b/src/main/java/com/xgh/test/spring/step04/beans/factory/support/AbstractBeanFactory.java @@ -3,6 +3,11 @@ import com.xgh.test.spring.step04.beans.BeansException; import com.xgh.test.spring.step04.beans.factory.BeanFactory; import com.xgh.test.spring.step04.beans.factory.config.BeanDefinition; +import com.xgh.test.spring.step04.beans.factory.config.BeanPostProcessor; +import com.xgh.test.spring.step04.beans.factory.config.ConfigurableBeanFactory; + +import java.util.ArrayList; +import java.util.List; /** * com.xgh.test.spring.step02.support.AbstractBeanFactory @@ -11,7 +16,9 @@ * @description 抽象类定义模板方法 * @date 2021年08月17日 */ -public abstract class AbstractBeanFactory extends DefaultSingletonBeanRegistry implements BeanFactory { +public abstract class AbstractBeanFactory extends DefaultSingletonBeanRegistry implements ConfigurableBeanFactory { + + private final List beanPostProcessors = new ArrayList<>(); @Override public Object getBean(String beanName) throws BeansException { @@ -41,4 +48,13 @@ public T getBean(String beanName, Class requiredType) { protected abstract BeanDefinition getBeanDefinition(String beanName) throws BeansException; + @Override + public void addBeanPostProcessor(BeanPostProcessor beanPostProcessor) { + this.beanPostProcessors.remove(beanPostProcessor); + this.beanPostProcessors.add(beanPostProcessor); + } + + public List getBeanPostProcessors(){ + return this.beanPostProcessors; + } } diff --git a/src/main/java/com/xgh/test/spring/step04/beans/factory/support/BeanDefinitionReader.java b/src/main/java/com/xgh/test/spring/step04/beans/factory/support/BeanDefinitionReader.java index 4afca7f..762f83e 100644 --- a/src/main/java/com/xgh/test/spring/step04/beans/factory/support/BeanDefinitionReader.java +++ b/src/main/java/com/xgh/test/spring/step04/beans/factory/support/BeanDefinitionReader.java @@ -23,4 +23,6 @@ public interface BeanDefinitionReader { void loadBeanDefinitions(String location) throws BeansException; + void loadBeanDefinitions(String... locations)throws BeansException; + } diff --git a/src/main/java/com/xgh/test/spring/step04/beans/factory/support/DefaultListableBeanFactory.java b/src/main/java/com/xgh/test/spring/step04/beans/factory/support/DefaultListableBeanFactory.java index d234eab..5360b2a 100644 --- a/src/main/java/com/xgh/test/spring/step04/beans/factory/support/DefaultListableBeanFactory.java +++ b/src/main/java/com/xgh/test/spring/step04/beans/factory/support/DefaultListableBeanFactory.java @@ -56,5 +56,10 @@ public BeanDefinition getBeanDefinition(String beanName) throws BeansException { return beanDefinition; } + @Override + public void preInstantiateSingletons() throws BeansException { + beanDefinitionMap.keySet().forEach(this::getBean); + } + } diff --git a/src/main/java/com/xgh/test/spring/step04/beans/factory/support/DefaultSingletonBeanRegistry.java b/src/main/java/com/xgh/test/spring/step04/beans/factory/support/DefaultSingletonBeanRegistry.java index 2a1dc3d..bdafd82 100644 --- a/src/main/java/com/xgh/test/spring/step04/beans/factory/support/DefaultSingletonBeanRegistry.java +++ b/src/main/java/com/xgh/test/spring/step04/beans/factory/support/DefaultSingletonBeanRegistry.java @@ -1,9 +1,12 @@ package com.xgh.test.spring.step04.beans.factory.support; +import com.xgh.test.spring.step04.beans.BeansException; +import com.xgh.test.spring.step04.beans.factory.DisposableBean; import com.xgh.test.spring.step04.beans.factory.config.SingletonBeanRegistry; import java.util.HashMap; import java.util.Map; +import java.util.Set; /** * com.xgh.test.spring.step02.support.DefaultSingletinBeanRegistry @@ -16,6 +19,8 @@ public class DefaultSingletonBeanRegistry implements SingletonBeanRegistry { private Map singletonObjects = new HashMap<>(); + private final Map disposableBeans = new HashMap<>(); + @Override public Object getSingleton(String beanName) { return singletonObjects.get(beanName); @@ -24,4 +29,23 @@ public Object getSingleton(String beanName) { protected void addSingleton(String beanName,Object singletonObject){ singletonObjects.put(beanName,singletonObject); } + + public void registerDisposableBean(String beanName,DisposableBean bean){ + disposableBeans.put(beanName,bean); + } + + + public void destroySingletons(){ + Set keySet= this.disposableBeans.keySet(); + Object[] disposableBeanNames = keySet.toArray(); + for (int i = disposableBeanNames.length-1 ; i >= 0; i--){ + Object beanName = disposableBeanNames[i]; + DisposableBean disposableBean = disposableBeans.remove(beanName); + try { + disposableBean.destroy(); + }catch (Exception e){ + throw new BeansException("Destroy method on bean with name '" + beanName + "' threw an exception", e); + } + } + } } diff --git a/src/main/java/com/xgh/test/spring/step04/beans/factory/support/DisposableBeanAdapter.java b/src/main/java/com/xgh/test/spring/step04/beans/factory/support/DisposableBeanAdapter.java new file mode 100644 index 0000000..3d719bb --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step04/beans/factory/support/DisposableBeanAdapter.java @@ -0,0 +1,43 @@ +package com.xgh.test.spring.step04.beans.factory.support; + +import cn.hutool.core.util.StrUtil; +import com.xgh.test.spring.step04.beans.BeansException; +import com.xgh.test.spring.step04.beans.factory.DisposableBean; +import com.xgh.test.spring.step04.beans.factory.config.BeanDefinition; + +import java.lang.reflect.Method; + + +/** + * bean 实例销毁步鄹 + */ +public class DisposableBeanAdapter implements DisposableBean { + + private final Object bean; + + private final String beanName; + + private String destroyMethodName; + + public DisposableBeanAdapter(Object bean, String beanName, BeanDefinition beanDefinition) { + this.bean = bean; + this.beanName = beanName; + this.destroyMethodName = beanDefinition.getDestroyMethodName(); + } + + @Override + public void destroy() throws Exception { + if(bean instanceof DisposableBean){ + ((DisposableBean) bean).destroy(); + } + //注解配置destroy-method(判断是为了必维二次执行销毁) + if(StrUtil.isNotEmpty(destroyMethodName) && !(bean instanceof DisposableBean && "destroy".equals(this.destroyMethodName))){ + Method destroyMethod = bean.getClass().getMethod(destroyMethodName); + if(null == destroyMethod){ + throw new BeansException("Couldn't find a destroy method named '" + destroyMethodName + "' on bean with name '" + beanName + "'"); + } + destroyMethod.invoke(bean); + } + + } +} diff --git a/src/main/java/com/xgh/test/spring/step04/beans/factory/xml/XmlBeanDefinitionReader.java b/src/main/java/com/xgh/test/spring/step04/beans/factory/xml/XmlBeanDefinitionReader.java index b65ad50..e845d5b 100644 --- a/src/main/java/com/xgh/test/spring/step04/beans/factory/xml/XmlBeanDefinitionReader.java +++ b/src/main/java/com/xgh/test/spring/step04/beans/factory/xml/XmlBeanDefinitionReader.java @@ -111,4 +111,11 @@ public void loadBeanDefinitions(String location) throws BeansException { Resource resource = resourceLoader.getResource(location); loadBeanDefinitions(resource); } + + @Override + public void loadBeanDefinitions(String... locations) throws BeansException { + for (String location : locations) { + loadBeanDefinitions(location); + } + } } diff --git a/src/main/java/com/xgh/test/spring/step04/context/ConfigurableApplicationContext.java b/src/main/java/com/xgh/test/spring/step04/context/ConfigurableApplicationContext.java new file mode 100644 index 0000000..e01d382 --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step04/context/ConfigurableApplicationContext.java @@ -0,0 +1,18 @@ +package com.xgh.test.spring.step04.context; + +import com.xgh.test.spring.step04.beans.BeansException; +import com.xgh.test.spring.step04.beans.factory.ListableBeanFactory; + +public interface ConfigurableApplicationContext extends ListableBeanFactory { + + + /** + * 刷新容器 + * @throws BeansException + */ + void refresh() throws BeansException; + + void registerShutdownHook(); + + void close(); +} diff --git a/src/main/java/com/xgh/test/spring/step04/context/support/AbstractApplicationContext.java b/src/main/java/com/xgh/test/spring/step04/context/support/AbstractApplicationContext.java new file mode 100644 index 0000000..eda5539 --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step04/context/support/AbstractApplicationContext.java @@ -0,0 +1,94 @@ +package com.xgh.test.spring.step04.context.support; + +import com.xgh.test.spring.step04.beans.BeansException; +import com.xgh.test.spring.step04.beans.factory.ConfigurableListableBeanFactory; +import com.xgh.test.spring.step04.beans.factory.config.BeanFactoryPostProcessor; +import com.xgh.test.spring.step04.beans.factory.config.BeanPostProcessor; +import com.xgh.test.spring.step04.context.ConfigurableApplicationContext; +import com.xgh.test.spring.step04.core.io.DefaultResourceLoader; + +import java.util.Map; + + +/** + * 应用上下文抽象类实现 + */ +public abstract class AbstractApplicationContext extends DefaultResourceLoader implements ConfigurableApplicationContext { + + + @Override + public void refresh() throws BeansException { + //1创建beanFactory,加载BeanDefinition + refreshFactory(); + + //2.获取beanFactory + ConfigurableListableBeanFactory beanFactory = getBeanFactory(); + + //3在bean实例化之前执行beanFactoryPostProcess + invokeBeanFactoryPostProcessors(beanFactory); + //4.beanPostProcessor需要提前雨其他bean对象实例化之前注册操作 + registerBeanPostProcessors(beanFactory); + //5提前实例化单例bean对象 + beanFactory.preInstantiateSingletons(); + + + } + protected abstract ConfigurableListableBeanFactory getBeanFactory() throws BeansException; + + + protected abstract void refreshFactory() ; + + private void registerBeanPostProcessors(ConfigurableListableBeanFactory beanFactory){ + Map beanPostProcessorMap = beanFactory.getBeansOfType(BeanPostProcessor.class); + for (BeanPostProcessor beanPostProcessor : beanPostProcessorMap.values()) { + beanFactory.addBeanPostProcessor(beanPostProcessor); + } + //... getBean getBeansOfType getBeanDefinitionNames; + } + + @Override + public Object getBean(String beanName) throws BeansException { + return getBeanFactory().getBean(beanName); + } + + @Override + public T getBean(String beanName, Class requiredType) { + return getBeanFactory().getBean(beanName,requiredType); + } + + @Override + public Object getBean(String beanName, Object... args) throws BeansException { + return getBeanFactory().getBean(beanName,args); + } + + @Override + public Map getBeansOfType(Class type) throws BeansException { + return getBeanFactory().getBeansOfType(type); + } + + @Override + public String[] getBeanDefinitionNames() { + return getBeanFactory().getBeanDefinitionNames(); + } + + private void invokeBeanFactoryPostProcessors(ConfigurableListableBeanFactory beanFactory){ + Map beanFactoryPostProcessorMap = beanFactory.getBeansOfType(BeanFactoryPostProcessor.class); + for (BeanFactoryPostProcessor beanFactoryPostProcessor : beanFactoryPostProcessorMap.values()) { + beanFactoryPostProcessor.postProcessBeanFactory(beanFactory); + } + } + + public void registerShutdownHook(){ + Runtime.getRuntime().addShutdownHook(new Thread(this::close)); + } + + public void close(){ + getBeanFactory().destroySingletons(); + } + + + + + + +} diff --git a/src/main/java/com/xgh/test/spring/step04/context/support/AbstractRefreshableApplication.java b/src/main/java/com/xgh/test/spring/step04/context/support/AbstractRefreshableApplication.java new file mode 100644 index 0000000..b313493 --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step04/context/support/AbstractRefreshableApplication.java @@ -0,0 +1,33 @@ +package com.xgh.test.spring.step04.context.support; + +import com.xgh.test.spring.step04.beans.BeansException; +import com.xgh.test.spring.step04.beans.factory.ConfigurableListableBeanFactory; +import com.xgh.test.spring.step04.beans.factory.support.DefaultListableBeanFactory; + +/** + * 获取bean工厂和加载资源 + */ +public abstract class AbstractRefreshableApplication extends AbstractApplicationContext { + + private DefaultListableBeanFactory beanFactory; + + + @Override + protected void refreshFactory() throws BeansException { + DefaultListableBeanFactory beanFactory = creatBeanFactory(); + //加载spring.xml配置文件 + loadBeanDefinitions(beanFactory); + this.beanFactory = beanFactory; + } + + protected abstract void loadBeanDefinitions(DefaultListableBeanFactory beanFactory); + + private DefaultListableBeanFactory creatBeanFactory() { + return new DefaultListableBeanFactory(); + } + + @Override + protected ConfigurableListableBeanFactory getBeanFactory() throws BeansException { + return beanFactory; + } +} diff --git a/src/main/java/com/xgh/test/spring/step04/context/support/AbstractXmlApplicationContext.java b/src/main/java/com/xgh/test/spring/step04/context/support/AbstractXmlApplicationContext.java new file mode 100644 index 0000000..7137aea --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step04/context/support/AbstractXmlApplicationContext.java @@ -0,0 +1,25 @@ +package com.xgh.test.spring.step04.context.support; + +import com.xgh.test.spring.step04.beans.factory.support.DefaultListableBeanFactory; +import com.xgh.test.spring.step04.beans.factory.xml.XmlBeanDefinitionReader; + +/** + * 上下文对配置信息的加载 + */ +public abstract class AbstractXmlApplicationContext extends AbstractRefreshableApplication { + + + @Override + protected void loadBeanDefinitions(DefaultListableBeanFactory beanFactory) { + XmlBeanDefinitionReader beanDefinitionReader = new XmlBeanDefinitionReader(beanFactory, this); + + String[] configLocations = getConfigLocations(); + if( null != configLocations){ + beanDefinitionReader.loadBeanDefinitions(configLocations); + } + } + + protected abstract String[] getConfigLocations(); + + +} diff --git a/src/main/java/com/xgh/test/spring/step04/context/support/ClassPathXmlApplicationContext.java b/src/main/java/com/xgh/test/spring/step04/context/support/ClassPathXmlApplicationContext.java new file mode 100644 index 0000000..55c4c46 --- /dev/null +++ b/src/main/java/com/xgh/test/spring/step04/context/support/ClassPathXmlApplicationContext.java @@ -0,0 +1,27 @@ +package com.xgh.test.spring.step04.context.support; + +/** + * 应用上下文实现类 + */ + +public class ClassPathXmlApplicationContext extends AbstractXmlApplicationContext{ + + private String[] configLocations; + + public ClassPathXmlApplicationContext() { + } + + public ClassPathXmlApplicationContext(String configLocation){ + this(new String[]{configLocation}); + } + + public ClassPathXmlApplicationContext(String[] configLocations) { + this.configLocations = configLocations; + refresh(); + } + + @Override + protected String[] getConfigLocations() { + return configLocations; + } +} diff --git a/src/test/java/com/xgh/test/spring/test05/MyBeanFactoryPostProcessor.java b/src/test/java/com/xgh/test/spring/test05/MyBeanFactoryPostProcessor.java new file mode 100644 index 0000000..d8d69b0 --- /dev/null +++ b/src/test/java/com/xgh/test/spring/test05/MyBeanFactoryPostProcessor.java @@ -0,0 +1,20 @@ +package com.xgh.test.spring.test05; + +import com.xgh.test.spring.step04.beans.BeansException; +import com.xgh.test.spring.step04.beans.PropertyValue; +import com.xgh.test.spring.step04.beans.PropertyValues; +import com.xgh.test.spring.step04.beans.factory.ConfigurableListableBeanFactory; +import com.xgh.test.spring.step04.beans.factory.config.BeanDefinition; +import com.xgh.test.spring.step04.beans.factory.config.BeanFactoryPostProcessor; + +public class MyBeanFactoryPostProcessor implements BeanFactoryPostProcessor { + @Override + public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) + throws BeansException { + BeanDefinition beanDefinition = beanFactory.getBeanDefinition("userService" + ); + PropertyValues propertyValues = beanDefinition.getPropertyValues(); + propertyValues.addPropertyValue(new PropertyValue("company", "改为:字节跳动 ")); + } + +} diff --git a/src/test/java/com/xgh/test/spring/test05/MyBeanPostProcessor.java b/src/test/java/com/xgh/test/spring/test05/MyBeanPostProcessor.java new file mode 100644 index 0000000..becf977 --- /dev/null +++ b/src/test/java/com/xgh/test/spring/test05/MyBeanPostProcessor.java @@ -0,0 +1,20 @@ +package com.xgh.test.spring.test05; + +import com.xgh.test.spring.step04.beans.BeansException; +import com.xgh.test.spring.step04.beans.factory.config.BeanPostProcessor; + +public class MyBeanPostProcessor implements BeanPostProcessor { + @Override + public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException { + if("userService".equals(beanName)){ + UserService userService = (UserService) bean; + userService.setLocation("改为北京"); + } + return bean; + } + + @Override + public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException { + return bean; + } +} diff --git a/src/test/java/com/xgh/test/spring/test05/Test05.java b/src/test/java/com/xgh/test/spring/test05/Test05.java new file mode 100644 index 0000000..bd3d9eb --- /dev/null +++ b/src/test/java/com/xgh/test/spring/test05/Test05.java @@ -0,0 +1,46 @@ +package com.xgh.test.spring.test05; + +import com.xgh.test.spring.step04.beans.factory.support.DefaultListableBeanFactory; +import com.xgh.test.spring.step04.beans.factory.xml.XmlBeanDefinitionReader; +import com.xgh.test.spring.step04.context.support.ClassPathXmlApplicationContext; +import org.junit.Test; + +public class Test05 { + + + @Test + public void test_xml() { + // 1.初始化 BeanFactory + ClassPathXmlApplicationContext applicationContext = new ClassPathXmlApplicationContext("classpath:spring.xml"); + applicationContext.registerShutdownHook(); + + // 2. 获取Bean对象调用方法 + UserService userService = applicationContext.getBean("userService", UserService.class); + String result = userService.queryUserInfo(); + System.out.println("测试结果:" + result); + } + + @Test + public void test_hook() { + Runtime.getRuntime().addShutdownHook(new Thread(() -> System.out.println("close!"))); + } + + @Test + public void test_BeanFactoryPostProcessorAndBeanPostProcessor() { +// 1.初始化 BeanFactory + DefaultListableBeanFactory beanFactory = new DefaultListableBeanFactory(); +// 2. 读取配置文件&注册 Bean + XmlBeanDefinitionReader reader = new XmlBeanDefinitionReader(beanFactory); + reader.loadBeanDefinitions("classpath:spring.xml"); +// 3. BeanDefinition 加载完成 & Bean 实例化之前,修改 BeanDefinition 的属性值 + MyBeanFactoryPostProcessor beanFactoryPostProcessor = new MyBeanFactoryPostProcessor(); + beanFactoryPostProcessor.postProcessBeanFactory(beanFactory); +// 4. Bean 实例化之后,修改 Bean 属性信息 + MyBeanPostProcessor beanPostProcessor = new MyBeanPostProcessor(); + beanFactory.addBeanPostProcessor(beanPostProcessor); +// 5. 获取 Bean 对象调用方法 + UserService userService = beanFactory.getBean("userService", UserService.class); + String result = userService.queryUserInfo(); + System.out.println("测试结果: " + result); + } +} diff --git a/src/test/java/com/xgh/test/spring/test05/UserDao.java b/src/test/java/com/xgh/test/spring/test05/UserDao.java new file mode 100644 index 0000000..24c0967 --- /dev/null +++ b/src/test/java/com/xgh/test/spring/test05/UserDao.java @@ -0,0 +1,26 @@ +package com.xgh.test.spring.test05; + +import java.util.HashMap; +import java.util.Map; + +public class UserDao { + + private static Map hashMap = new HashMap<>(); + + public void initDataMethod(){ + System.out.println("执行:init-method"); + hashMap.put("10001", "小傅哥"); + hashMap.put("10002", "八杯水"); + hashMap.put("10003", "阿毛"); + } + + public void destroyDataMethod(){ + System.out.println("执行:destroy-method"); + hashMap.clear(); + } + + public String queryUserName(String uId) { + return hashMap.get(uId); + } + +} diff --git a/src/test/java/com/xgh/test/spring/test05/UserService.java b/src/test/java/com/xgh/test/spring/test05/UserService.java new file mode 100644 index 0000000..2934a78 --- /dev/null +++ b/src/test/java/com/xgh/test/spring/test05/UserService.java @@ -0,0 +1,64 @@ +package com.xgh.test.spring.test05; + + +import com.xgh.test.spring.step04.beans.factory.DisposableBean; +import com.xgh.test.spring.step04.beans.factory.InitializingBean; + +/** + * + */ +public class UserService implements InitializingBean, DisposableBean { + + private String uId; + private String company; + private String location; + private UserDao userDao; + + @Override + public void destroy() throws Exception { + System.out.println("执行:UserService.destroy"); + } + + @Override + public void afterPropertiesSet() throws Exception { + System.out.println("执行:UserService.afterPropertiesSet"); + } + + public String queryUserInfo() { + return userDao.queryUserName(uId) + "," + company + "," + location; + } + + public String getuId() { + return uId; + } + + public void setuId(String uId) { + this.uId = uId; + } + + public String getCompany() { + return company; + } + + public void setCompany(String company) { + this.company = company; + } + + public String getLocation() { + return location; + } + + public void setLocation(String location) { + this.location = location; + } + + public UserDao getUserDao() { + return userDao; + } + + public void setUserDao(UserDao userDao) { + this.userDao = userDao; + } + + +} diff --git a/src/test/resources/Spring.xml b/src/test/resources/Spring.xml index e6f226e..203ea83 100644 --- a/src/test/resources/Spring.xml +++ b/src/test/resources/Spring.xml @@ -1,11 +1,12 @@ - + - + + + - \ No newline at end of file From 5862a2f0c69c0f6db0e6033b3b1e9b33f91a3764 Mon Sep 17 00:00:00 2001 From: xgh Date: Tue, 26 Jul 2022 11:25:14 +0800 Subject: [PATCH 18/20] =?UTF-8?q?feat=EF=BC=9A=E9=93=BE=E8=A1=A8?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../com/xgh/test/algorithms/LinkedList.java | 164 ++++++++++++++++++ .../java/com/xgh/test/algorithms/List.java | 18 ++ 2 files changed, 182 insertions(+) create mode 100644 src/main/java/com/xgh/test/algorithms/LinkedList.java create mode 100644 src/main/java/com/xgh/test/algorithms/List.java diff --git a/src/main/java/com/xgh/test/algorithms/LinkedList.java b/src/main/java/com/xgh/test/algorithms/LinkedList.java new file mode 100644 index 0000000..f3a9476 --- /dev/null +++ b/src/main/java/com/xgh/test/algorithms/LinkedList.java @@ -0,0 +1,164 @@ +package com.xgh.test.algorithms; + + +public class LinkedList implements List { + + transient int size = 0; + + transient Node first; + + transient Node last; + + public LinkedList() { + + + } + + void linkFirst(E e) { + final Node f = first; + final Node newNode = new Node<>(null, e, f); + first = newNode; + if (f == null) { + last = newNode; + } else { + f.prev = newNode; + } + size++; + } + + void linkLast(E e) { + final Node l = last; + final Node newNode = new Node<>(l, e, null); + last = newNode; + if (l == null) { + first = newNode; + } else { + l.next = newNode; + } + size++; + } + + @Override + public boolean add(E e) { + linkLast(e); + return true; + } + + @Override + public boolean addFirst(E e) { + linkFirst(e); + return true; + } + + @Override + public boolean addLast(E e) { + linkLast(e); + return true; + } + + @Override + public boolean remove(Object o) { + if (o == null) { + for (Node x = first; x != null; x = x.next) { + if (x.item == null) { + unLink(x); + return true; + } + } + } else { + for (Node x = first; x != null; x = x.next) { + if (o.equals(x.item)) { + unLink(x); + return true; + } + } + } + return false; + } + + @Override + public E get(int index) { + return node(index).item; + } + + + @Override + public void printLinkList() { + if (this.size == 0) { + System.out.println("链表为空"); + } else { + Node temp = first; + System.out.print("目前的列表,头节点:" + first.item + " 尾节点:" + last.item + " 整体:"); + while (temp != null) { + System.out.print(temp.item + ","); + temp = temp.next; + } + System.out.println(); + } + } + + E unLink(Node x) { + final E element = x.item; + final Node prev = x.prev; + final Node next = x.next; + if (prev == null) { + first = next; + } else { + prev.next = next; + x.prev = null; + } + if (next == null) { + last = prev; + } else { + next.prev = prev; + x.next = null; + } + x.item = null; + size--; + return element; + } + + private Node node(int index) { + Node x; + if (index < (size >> 1)) { + x = first; + for (int i = 0; i < index; i++) { + x = x.next; + } + } else { + x = last; + for (int i = size - 1; i > index; i--) { + x = x.prev; + } + } + return x; + } + + private static class Node { + E item; + Node next; + Node prev; + + public Node(Node prev, E item, Node next) { + this.item = item; + this.next = next; + this.prev = prev; + } + } + + public static void main(String[] args) { + List list = new LinkedList<>(); + // 添加元素 + list.add("a"); + list.addFirst("b"); + list.addLast("c"); + // 打印列表 + list.printLinkList(); + // 头插元素 + list.addFirst("d"); + // 删除元素 + list.remove("b"); + // 打印列表 + list.printLinkList(); + } +} diff --git a/src/main/java/com/xgh/test/algorithms/List.java b/src/main/java/com/xgh/test/algorithms/List.java new file mode 100644 index 0000000..d861e45 --- /dev/null +++ b/src/main/java/com/xgh/test/algorithms/List.java @@ -0,0 +1,18 @@ +package com.xgh.test.algorithms; + +public interface List { + + boolean add(E e); + + boolean addFirst(E e); + + boolean addLast(E e); + + boolean remove(Object o); + + E get(int index); + + void printLinkList(); + + +} From 90106511142b80129d977e797954f48ff549324d Mon Sep 17 00:00:00 2001 From: xgh <1371818484@qq.com> Date: Tue, 2 Aug 2022 14:55:11 +0800 Subject: [PATCH 19/20] =?UTF-8?q?=E6=95=B0=E7=BB=84?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../xgh/test/algorithms/array/ArrayList.java | 65 +++++++++++++++++++ .../com/xgh/test/algorithms/array/List.java | 10 +++ .../algorithms/{ => link}/LinkedList.java | 2 +- .../xgh/test/algorithms/{ => link}/List.java | 2 +- 4 files changed, 77 insertions(+), 2 deletions(-) create mode 100644 src/main/java/com/xgh/test/algorithms/array/ArrayList.java create mode 100644 src/main/java/com/xgh/test/algorithms/array/List.java rename src/main/java/com/xgh/test/algorithms/{ => link}/LinkedList.java (98%) rename src/main/java/com/xgh/test/algorithms/{ => link}/List.java (83%) diff --git a/src/main/java/com/xgh/test/algorithms/array/ArrayList.java b/src/main/java/com/xgh/test/algorithms/array/ArrayList.java new file mode 100644 index 0000000..e43c2fb --- /dev/null +++ b/src/main/java/com/xgh/test/algorithms/array/ArrayList.java @@ -0,0 +1,65 @@ +package com.xgh.test.algorithms.array; + +import java.util.Arrays; + +public class ArrayList implements List { + + + /** + * 默认初始化空间 + */ + private static final int DEFAULT_CAPACITY = 10; + + /** + * 空元素 + */ + private static final Object[] DEFAULTCAPACITY_EMPTY_ELEMENTDATA = {}; + + /** + * 元素数组缓冲区 + */ + transient Object[] elementData; + + /** + * 元素个数 + */ + private int size; + + + @Override + public boolean add(E e) { + //确保内部容量 + int minCapacity = size + 1; + if (elementData == DEFAULTCAPACITY_EMPTY_ELEMENTDATA) { + minCapacity = Math.max(DEFAULT_CAPACITY, minCapacity); + } + //判断扩容 + if (minCapacity - elementData.length > 0) { + int oldCapacity = elementData.length; + int newCapacity = oldCapacity + (oldCapacity >> 1); + if(newCapacity -minCapacity < 0){ + newCapacity = minCapacity; + } + elementData = Arrays.copyOf(elementData,newCapacity); + } + //添加原始 + elementData[size++] = e; + return true; + } + + @Override + public E remove(int index) { + E oldElement =(E) elementData[index]; + int numMoved = size - index - 1; + if(numMoved > 0){ + System.arraycopy(elementData,index+1,elementData,index,numMoved); + } + elementData[--size] = null; + return oldElement; + } + + @Override + public E get(int index) { + return (E) elementData[index]; + } +} diff --git a/src/main/java/com/xgh/test/algorithms/array/List.java b/src/main/java/com/xgh/test/algorithms/array/List.java new file mode 100644 index 0000000..98d4573 --- /dev/null +++ b/src/main/java/com/xgh/test/algorithms/array/List.java @@ -0,0 +1,10 @@ +package com.xgh.test.algorithms.array; + +public interface List{ + + boolean add(E e); + + E remove(int index); + + E get(int index); +} diff --git a/src/main/java/com/xgh/test/algorithms/LinkedList.java b/src/main/java/com/xgh/test/algorithms/link/LinkedList.java similarity index 98% rename from src/main/java/com/xgh/test/algorithms/LinkedList.java rename to src/main/java/com/xgh/test/algorithms/link/LinkedList.java index f3a9476..50b7e0f 100644 --- a/src/main/java/com/xgh/test/algorithms/LinkedList.java +++ b/src/main/java/com/xgh/test/algorithms/link/LinkedList.java @@ -1,4 +1,4 @@ -package com.xgh.test.algorithms; +package com.xgh.test.algorithms.link; public class LinkedList implements List { diff --git a/src/main/java/com/xgh/test/algorithms/List.java b/src/main/java/com/xgh/test/algorithms/link/List.java similarity index 83% rename from src/main/java/com/xgh/test/algorithms/List.java rename to src/main/java/com/xgh/test/algorithms/link/List.java index d861e45..2670fca 100644 --- a/src/main/java/com/xgh/test/algorithms/List.java +++ b/src/main/java/com/xgh/test/algorithms/link/List.java @@ -1,4 +1,4 @@ -package com.xgh.test.algorithms; +package com.xgh.test.algorithms.link; public interface List { From 2fa97dce99525336cfd348e49f8a5890564852ea Mon Sep 17 00:00:00 2001 From: xgh <1371818484@qq.com> Date: Thu, 18 Aug 2022 11:35:56 +0800 Subject: [PATCH 20/20] =?UTF-8?q?api=E7=BD=91=E5=85=B301?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- gateway/pom.xml | 96 +++++++++++++++++++ .../com/xgh/gateway/day01/BaseHandler.java | 19 ++++ .../day01/SessionChannelInitializer.java | 24 +++++ .../com/xgh/gateway/day01/SessionServer.java | 54 +++++++++++ .../day01/handlers/SessionServerHandler.java | 42 ++++++++ .../day02/bind/GenericReferenceProxy.java | 45 +++++++++ .../bind/GenericReferenceProxyFactory.java | 51 ++++++++++ .../day02/bind/GenericReferenceRegistry.java | 52 ++++++++++ .../gateway/day02/bind/IGenericReference.java | 9 ++ .../gateway/day02/session/BaseHandler.java | 20 ++++ .../gateway/day02/session/Configuration.java | 75 +++++++++++++++ ...GenericReferenceSessionFactoryBuilder.java | 25 +++++ .../IGenericReferenceSessionFactory.java | 17 ++++ .../session/SessionChannelInitializer.java | 28 ++++++ .../gateway/day02/session/SessionServer.java | 59 ++++++++++++ .../GenericReferenceSessionFactory.java | 39 ++++++++ .../handlers/SessionServerHandler.java | 58 +++++++++++ .../java/com/xgh/gateway/day01/ApiTest.java | 37 +++++++ .../java/com/xgh/gateway/day02/ApiTest.java | 30 ++++++ .../java/com/xgh/gateway/day02/CglibTest.java | 43 +++++++++ .../java/com/xgh/gateway/day02/RPCTest.java | 47 +++++++++ pom.xml | 4 + 22 files changed, 874 insertions(+) create mode 100644 gateway/pom.xml create mode 100644 gateway/src/main/java/com/xgh/gateway/day01/BaseHandler.java create mode 100644 gateway/src/main/java/com/xgh/gateway/day01/SessionChannelInitializer.java create mode 100644 gateway/src/main/java/com/xgh/gateway/day01/SessionServer.java create mode 100644 gateway/src/main/java/com/xgh/gateway/day01/handlers/SessionServerHandler.java create mode 100644 gateway/src/main/java/com/xgh/gateway/day02/bind/GenericReferenceProxy.java create mode 100644 gateway/src/main/java/com/xgh/gateway/day02/bind/GenericReferenceProxyFactory.java create mode 100644 gateway/src/main/java/com/xgh/gateway/day02/bind/GenericReferenceRegistry.java create mode 100644 gateway/src/main/java/com/xgh/gateway/day02/bind/IGenericReference.java create mode 100644 gateway/src/main/java/com/xgh/gateway/day02/session/BaseHandler.java create mode 100644 gateway/src/main/java/com/xgh/gateway/day02/session/Configuration.java create mode 100644 gateway/src/main/java/com/xgh/gateway/day02/session/GenericReferenceSessionFactoryBuilder.java create mode 100644 gateway/src/main/java/com/xgh/gateway/day02/session/IGenericReferenceSessionFactory.java create mode 100644 gateway/src/main/java/com/xgh/gateway/day02/session/SessionChannelInitializer.java create mode 100644 gateway/src/main/java/com/xgh/gateway/day02/session/SessionServer.java create mode 100644 gateway/src/main/java/com/xgh/gateway/day02/session/defaults/GenericReferenceSessionFactory.java create mode 100644 gateway/src/main/java/com/xgh/gateway/day02/session/handlers/SessionServerHandler.java create mode 100644 gateway/src/test/java/com/xgh/gateway/day01/ApiTest.java create mode 100644 gateway/src/test/java/com/xgh/gateway/day02/ApiTest.java create mode 100644 gateway/src/test/java/com/xgh/gateway/day02/CglibTest.java create mode 100644 gateway/src/test/java/com/xgh/gateway/day02/RPCTest.java diff --git a/gateway/pom.xml b/gateway/pom.xml new file mode 100644 index 0000000..7ddac45 --- /dev/null +++ b/gateway/pom.xml @@ -0,0 +1,96 @@ + + + + test + com.xgh + 0.0.1-SNAPSHOT + + 4.0.0 + + gateway + + + 8 + 8 + UTF-8 + + + + io.netty + netty-all + 4.1.77.Final + + + com.alibaba + fastjson + 2.0.7 + + + org.slf4j + slf4j-api + 1.7.36 + jar + compile + + + ch.qos.logback + logback-core + 1.2.11 + jar + + + ch.qos.logback + logback-classic + 1.2.11 + jar + + + junit + junit + 4.13.2 + test + + + org.apache.dubbo + dubbo + 2.7.5 + + + org.apache.zookeeper + zookeeper + 3.4.13 + + + org.apache.curator + curator-framework + 4.0.1 + + + org.apache.curator + curator-recipes + 4.0.1 + + + cglib + cglib + 3.3.0 + + + + + + + org.apache.maven.plugins + maven-compiler-plugin + 3.3 + + ${maven.compiler.source} + ${maven.compiler.target} + ${project.build.sourceEncoding} + + + + + \ No newline at end of file diff --git a/gateway/src/main/java/com/xgh/gateway/day01/BaseHandler.java b/gateway/src/main/java/com/xgh/gateway/day01/BaseHandler.java new file mode 100644 index 0000000..b583402 --- /dev/null +++ b/gateway/src/main/java/com/xgh/gateway/day01/BaseHandler.java @@ -0,0 +1,19 @@ +package com.xgh.gateway.day01; + +import io.netty.channel.Channel; +import io.netty.channel.ChannelHandlerContext; +import io.netty.channel.SimpleChannelInboundHandler; + +/** + * 数据处理器 基类 + * @param + */ +public abstract class BaseHandler extends SimpleChannelInboundHandler { + + @Override + protected void channelRead0(ChannelHandlerContext ctx, T msg) throws Exception { + session(ctx,ctx.channel(),msg); + } + + protected abstract void session(ChannelHandlerContext ctx, final Channel channel, T request); +} diff --git a/gateway/src/main/java/com/xgh/gateway/day01/SessionChannelInitializer.java b/gateway/src/main/java/com/xgh/gateway/day01/SessionChannelInitializer.java new file mode 100644 index 0000000..c8f3477 --- /dev/null +++ b/gateway/src/main/java/com/xgh/gateway/day01/SessionChannelInitializer.java @@ -0,0 +1,24 @@ +package com.xgh.gateway.day01; + +import com.xgh.gateway.day01.handlers.SessionServerHandler; +import io.netty.channel.ChannelHandler; +import io.netty.channel.ChannelHandlerContext; +import io.netty.channel.ChannelInitializer; +import io.netty.channel.ChannelPipeline; +import io.netty.channel.socket.SocketChannel; +import io.netty.handler.codec.http.HttpObjectAggregator; +import io.netty.handler.codec.http.HttpRequestDecoder; +import io.netty.handler.codec.http.HttpResponseEncoder; + + +public class SessionChannelInitializer extends ChannelInitializer { + + @Override + protected void initChannel(SocketChannel channel) throws Exception { + ChannelPipeline line = channel.pipeline(); + line.addLast(new HttpRequestDecoder()) + .addLast(new HttpResponseEncoder()) + .addLast(new HttpObjectAggregator(1024*1024)) + .addLast(new SessionServerHandler()); + } +} diff --git a/gateway/src/main/java/com/xgh/gateway/day01/SessionServer.java b/gateway/src/main/java/com/xgh/gateway/day01/SessionServer.java new file mode 100644 index 0000000..5c4e0a4 --- /dev/null +++ b/gateway/src/main/java/com/xgh/gateway/day01/SessionServer.java @@ -0,0 +1,54 @@ +package com.xgh.gateway.day01; + +import io.netty.bootstrap.ServerBootstrap; +import io.netty.channel.Channel; +import io.netty.channel.ChannelFuture; +import io.netty.channel.ChannelOption; +import io.netty.channel.EventLoopGroup; +import io.netty.channel.nio.NioEventLoopGroup; +import io.netty.channel.socket.nio.NioServerSocketChannel; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + + +import java.util.concurrent.Callable; + + +/** + * 网关会话服务 + * @author xgh + */ +public class SessionServer implements Callable { + + private final Logger logger = LoggerFactory.getLogger(SessionServer.class); + + private final EventLoopGroup boss = new NioEventLoopGroup(1); + + private final EventLoopGroup work = new NioEventLoopGroup(); + + private Channel channel; + + + @Override + public Channel call() throws Exception { + ChannelFuture channelFuture = null; + try { + ServerBootstrap b = new ServerBootstrap(); + b.group(boss,work) + .channel(NioServerSocketChannel.class) + .option(ChannelOption.SO_BACKLOG,128) + .childHandler(new SessionChannelInitializer()); + channelFuture = b.bind(7387).syncUninterruptibly(); + this.channel = channelFuture.channel(); + }catch (Exception e){ + logger.error("socket server start error.",e); + }finally { + if(null != channelFuture || channelFuture.isSuccess()){ + logger.error("socket server start done."); + }else { + logger.error("socket server start error."); + } + } + return channel; + } +} diff --git a/gateway/src/main/java/com/xgh/gateway/day01/handlers/SessionServerHandler.java b/gateway/src/main/java/com/xgh/gateway/day01/handlers/SessionServerHandler.java new file mode 100644 index 0000000..2ef899d --- /dev/null +++ b/gateway/src/main/java/com/xgh/gateway/day01/handlers/SessionServerHandler.java @@ -0,0 +1,42 @@ +package com.xgh.gateway.day01.handlers; + +import com.alibaba.fastjson.JSON; +import com.alibaba.fastjson.serializer.SerializerFeature; +import com.xgh.gateway.day01.BaseHandler; +import io.netty.channel.Channel; +import io.netty.channel.ChannelHandlerContext; +import io.netty.handler.codec.http.*; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +/** + * 会话处理器 + */ +public class SessionServerHandler extends BaseHandler { + + private final Logger logger = LoggerFactory.getLogger(SessionServerHandler.class); + + @Override + protected void session(ChannelHandlerContext ctx, Channel channel, FullHttpRequest request) { + logger.info("网关接收请求uri:{} method:{}",request.uri(),request.method()); + //返回信息处理 + DefaultFullHttpResponse response = new DefaultFullHttpResponse(HttpVersion.HTTP_1_1, HttpResponseStatus.OK); + //返回信息控制 + response.content().writeBytes(JSON.toJSONBytes("你访问的路径被网关管理了uri:"+request.uri(), SerializerFeature.PrettyFormat)); + //设置头部信息 + HttpHeaders headers = response.headers(); + //返回内容类型 + headers.add(HttpHeaderNames.CONTENT_TYPE,HttpHeaderValues.APPLICATION_JSON+";charset=UTF-8"); + //响应体长度 + headers.add(HttpHeaderNames.CONTENT_LENGTH,response.content().readableBytes()); + //配置链接持久 + headers.add(HttpHeaderNames.CONNECTION,HttpHeaderValues.KEEP_ALIVE); + //配置跨域 + headers.add(HttpHeaderNames.ACCESS_CONTROL_ALLOW_ORIGIN,"*"); + headers.add(HttpHeaderNames.ACCESS_CONTROL_ALLOW_HEADERS,"*"); + headers.add(HttpHeaderNames.ACCESS_CONTROL_ALLOW_METHODS,"GET, POST, PUT, DELETE"); + headers.add(HttpHeaderNames.ACCESS_CONTROL_ALLOW_CREDENTIALS,"true"); + + channel.writeAndFlush(response); + } +} diff --git a/gateway/src/main/java/com/xgh/gateway/day02/bind/GenericReferenceProxy.java b/gateway/src/main/java/com/xgh/gateway/day02/bind/GenericReferenceProxy.java new file mode 100644 index 0000000..92c8abb --- /dev/null +++ b/gateway/src/main/java/com/xgh/gateway/day02/bind/GenericReferenceProxy.java @@ -0,0 +1,45 @@ +package com.xgh.gateway.day02.bind; + +import net.sf.cglib.proxy.MethodInterceptor; +import net.sf.cglib.proxy.MethodProxy; +import org.apache.dubbo.rpc.service.GenericService; + +import java.lang.reflect.Method; + +/** + * 泛化调用静态代理:方便做一些拦截处理,给http对应rpc调用,做一层代理控制 + * 每调用到一个http对应的网关方法,就会代理的方式调用到rpc对应的泛化方法上 + */ +public class GenericReferenceProxy implements MethodInterceptor { + + /** + * rpc泛化调用服务 + */ + private final GenericService genericService; + + /** + * rpc 泛化调用方法 + */ + private final String methodName; + + public GenericReferenceProxy(GenericService genericService, String methodName) { + this.genericService = genericService; + this.methodName = methodName; + } + + + /** + * + * 做一层代理控制,后续不止是可以使用 Dubbo 泛化调用,也可以是其他服务的泛化调用 + * 泛化调用文档:https://dubbo.apache.org/zh/docsv2.7/user/examples/generic-reference/ + */ + @Override + public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable { + Class[] parameterTypes = method.getParameterTypes(); + String[] parameters = new String[parameterTypes.length]; + for (int i = 0; i < parameterTypes.length; i++) { + parameters[i] = parameterTypes[i].getName(); + } + return genericService.$invoke(methodName,parameters,args); + } +} diff --git a/gateway/src/main/java/com/xgh/gateway/day02/bind/GenericReferenceProxyFactory.java b/gateway/src/main/java/com/xgh/gateway/day02/bind/GenericReferenceProxyFactory.java new file mode 100644 index 0000000..62fe409 --- /dev/null +++ b/gateway/src/main/java/com/xgh/gateway/day02/bind/GenericReferenceProxyFactory.java @@ -0,0 +1,51 @@ +package com.xgh.gateway.day02.bind; + + +import net.sf.cglib.core.Signature; +import net.sf.cglib.proxy.Enhancer; +import net.sf.cglib.proxy.InterfaceMaker; +import org.apache.dubbo.rpc.service.GenericService; +import org.objectweb.asm.Type; + +import java.util.Map; +import java.util.concurrent.ConcurrentHashMap; + +/** + * 泛化调用静态代理工厂 + */ +public class GenericReferenceProxyFactory { + + /** + * rpc 泛化调用服务 + */ + private final GenericService genericService; + + + private final Map genericReferenceCache = new ConcurrentHashMap<>(); + + + public GenericReferenceProxyFactory(GenericService genericService) { + this.genericService = genericService; + } + + public IGenericReference newInstance(String method){ + return genericReferenceCache.computeIfAbsent(method,k->{ + //泛化调用 + GenericReferenceProxy genericReferenceProxy = new GenericReferenceProxy(genericService, method); + //创建接口 + InterfaceMaker interfaceMaker = new InterfaceMaker(); + interfaceMaker.add(new Signature(method, Type.getType(String.class),new Type[]{ Type.getType(String.class)}),null); + Class interfaceClass = interfaceMaker.create(); + //代理对象 + Enhancer enhancer = new Enhancer(); + enhancer.setSuperclass(Object.class); + //IGenericReference 统一泛化接口 + //interfaceClass 根据泛化调用注册信息创建的接口,建立http->rpc关联 + enhancer.setInterfaces(new Class[]{IGenericReference.class,interfaceClass}); + enhancer.setCallback(genericReferenceProxy); + return (IGenericReference) enhancer.create(); + }); + } + + +} diff --git a/gateway/src/main/java/com/xgh/gateway/day02/bind/GenericReferenceRegistry.java b/gateway/src/main/java/com/xgh/gateway/day02/bind/GenericReferenceRegistry.java new file mode 100644 index 0000000..c569d31 --- /dev/null +++ b/gateway/src/main/java/com/xgh/gateway/day02/bind/GenericReferenceRegistry.java @@ -0,0 +1,52 @@ +package com.xgh.gateway.day02.bind; + +import com.xgh.gateway.day02.session.Configuration; +import org.apache.dubbo.config.ApplicationConfig; +import org.apache.dubbo.config.ReferenceConfig; +import org.apache.dubbo.config.RegistryConfig; +import org.apache.dubbo.config.bootstrap.DubboBootstrap; +import org.apache.dubbo.config.utils.ReferenceConfigCache; +import org.apache.dubbo.rpc.service.GenericService; + +import java.util.HashMap; +import java.util.Map; + +/** + * 泛化调用注册器 + */ +public class GenericReferenceRegistry { + + private final Configuration configuration; + + + public GenericReferenceRegistry(Configuration configuration) { + this.configuration = configuration; + } + + private final Map knownGenericReferences = new HashMap<>(); + + public IGenericReference getGenericReference(String methodName){ + GenericReferenceProxyFactory genericReferenceProxyFactory = knownGenericReferences.get(methodName); + if(genericReferenceProxyFactory == null){ + throw new RuntimeException("Type "+methodName +"is not know to the GenericReferenceRegistry."); + } + return genericReferenceProxyFactory.newInstance(methodName); + } + + + public void addGenericReference(String application,String interfaceName,String methodName){ + //获取基础服务 + ApplicationConfig applicationConfig = configuration.getApplicationConfig(application); + RegistryConfig registryConfig = configuration.getRegistryConfig(application); + ReferenceConfig referenceConfig = configuration.getReferenceConfig(interfaceName); + //构建dubbo服务 + DubboBootstrap bootstrap = DubboBootstrap.getInstance(); + bootstrap.application(applicationConfig).registry(registryConfig).reference(referenceConfig).start(); + //调用泛化服务 + ReferenceConfigCache cache = ReferenceConfigCache.getCache(); + GenericService genericService = cache.get(referenceConfig); + //创建并保存到工厂 + knownGenericReferences.put(methodName,new GenericReferenceProxyFactory(genericService)); + } + +} diff --git a/gateway/src/main/java/com/xgh/gateway/day02/bind/IGenericReference.java b/gateway/src/main/java/com/xgh/gateway/day02/bind/IGenericReference.java new file mode 100644 index 0000000..a71538f --- /dev/null +++ b/gateway/src/main/java/com/xgh/gateway/day02/bind/IGenericReference.java @@ -0,0 +1,9 @@ +package com.xgh.gateway.day02.bind; + +/** + * 统一泛化接口 + */ +public interface IGenericReference { + + String $invoke(String args); +} diff --git a/gateway/src/main/java/com/xgh/gateway/day02/session/BaseHandler.java b/gateway/src/main/java/com/xgh/gateway/day02/session/BaseHandler.java new file mode 100644 index 0000000..3253e05 --- /dev/null +++ b/gateway/src/main/java/com/xgh/gateway/day02/session/BaseHandler.java @@ -0,0 +1,20 @@ +package com.xgh.gateway.day02.session; + +import io.netty.channel.Channel; +import io.netty.channel.ChannelHandlerContext; +import io.netty.channel.SimpleChannelInboundHandler; + +/** + * 数据处理基类 + * @param + */ +public abstract class BaseHandler extends SimpleChannelInboundHandler { + + + @Override + protected void channelRead0(ChannelHandlerContext ctx, T msg) throws Exception { + session(ctx,ctx.channel(),msg); + } + + protected abstract void session(ChannelHandlerContext ctx, Channel channel, T request); +} diff --git a/gateway/src/main/java/com/xgh/gateway/day02/session/Configuration.java b/gateway/src/main/java/com/xgh/gateway/day02/session/Configuration.java new file mode 100644 index 0000000..9790beb --- /dev/null +++ b/gateway/src/main/java/com/xgh/gateway/day02/session/Configuration.java @@ -0,0 +1,75 @@ +package com.xgh.gateway.day02.session; + + +import com.xgh.gateway.day02.bind.GenericReferenceRegistry; +import com.xgh.gateway.day02.bind.IGenericReference; +import org.apache.dubbo.config.ApplicationConfig; +import org.apache.dubbo.config.ReferenceConfig; +import org.apache.dubbo.config.RegistryConfig; +import org.apache.dubbo.rpc.service.GenericService; + +import java.util.HashMap; +import java.util.Map; + +/** + * 会话生命周期配置类 + */ +public class Configuration { + + private final GenericReferenceRegistry registry = new GenericReferenceRegistry(this); + + //rpc 应用服务配置项api-gateway-test + private final Map applicationConfigMap = new HashMap<>(); + + /** + * 注册中心配置项 zookeeper:127.0.0.1:2181 + */ + private final Map registryConfigMap = new HashMap<>(); + + /** + * rpc 泛化服务配置项目 cn.gateway.rpc.IActivityBooth + */ + private final Map> referenceConfigMap = new HashMap<>(); + + + public Configuration(){ + ApplicationConfig application = new ApplicationConfig(); + application.setName("api-gateway-test"); + application.setQosEnable(false); + + RegistryConfig registryConfig = new RegistryConfig(); + registryConfig.setAddress("zookeeper://127.0.0.1:2181"); + registryConfig.setRegister(false); + + ReferenceConfig reference = new ReferenceConfig<>(); + reference.setInterface(""); + reference.setVersion("1.0.1"); + reference.setGeneric("true"); + + applicationConfigMap.put("api-gateway-test",application); + registryConfigMap.put("api-gateway-test",registryConfig); + referenceConfigMap.put("",reference); + } + + public ApplicationConfig getApplicationConfig(String applicationName){ + return applicationConfigMap.get(applicationName); + } + + public RegistryConfig getRegistryConfig(String applicationName){ + return registryConfigMap.get(applicationName); + } + + public ReferenceConfig getReferenceConfig(String interfaceName){ + return referenceConfigMap.get(interfaceName); + } + + + public void addGenericReference(String application,String interfaceName,String methodName){ + registry.addGenericReference(application,interfaceName,methodName); + } + + public IGenericReference getGenericReference(String methodName){ + return registry.getGenericReference(methodName); + } + +} diff --git a/gateway/src/main/java/com/xgh/gateway/day02/session/GenericReferenceSessionFactoryBuilder.java b/gateway/src/main/java/com/xgh/gateway/day02/session/GenericReferenceSessionFactoryBuilder.java new file mode 100644 index 0000000..5f9165b --- /dev/null +++ b/gateway/src/main/java/com/xgh/gateway/day02/session/GenericReferenceSessionFactoryBuilder.java @@ -0,0 +1,25 @@ +package com.xgh.gateway.day02.session; + + +import com.xgh.gateway.day02.session.defaults.GenericReferenceSessionFactory; +import io.netty.channel.Channel; + +import java.util.concurrent.ExecutionException; +import java.util.concurrent.Future; + +/** + * 会话工厂构建类 + */ +public class GenericReferenceSessionFactoryBuilder { + + public Future build(Configuration configuration) { + IGenericReferenceSessionFactory genericReferenceSessionFactory = new GenericReferenceSessionFactory(configuration); + try { + return genericReferenceSessionFactory.openSession(); + } catch (ExecutionException | InterruptedException e) { + throw new RuntimeException(e); + } + } + + +} diff --git a/gateway/src/main/java/com/xgh/gateway/day02/session/IGenericReferenceSessionFactory.java b/gateway/src/main/java/com/xgh/gateway/day02/session/IGenericReferenceSessionFactory.java new file mode 100644 index 0000000..d0cc1b0 --- /dev/null +++ b/gateway/src/main/java/com/xgh/gateway/day02/session/IGenericReferenceSessionFactory.java @@ -0,0 +1,17 @@ +package com.xgh.gateway.day02.session; + +import io.netty.channel.Channel; + +import java.util.concurrent.ExecutionException; +import java.util.concurrent.Future; + + +/** + * 泛化调用工厂接口 + */ +public interface IGenericReferenceSessionFactory { + + Future openSession() throws ExecutionException, InterruptedException; + + +} diff --git a/gateway/src/main/java/com/xgh/gateway/day02/session/SessionChannelInitializer.java b/gateway/src/main/java/com/xgh/gateway/day02/session/SessionChannelInitializer.java new file mode 100644 index 0000000..e0e9141 --- /dev/null +++ b/gateway/src/main/java/com/xgh/gateway/day02/session/SessionChannelInitializer.java @@ -0,0 +1,28 @@ +package com.xgh.gateway.day02.session; + +import com.xgh.gateway.day02.session.handlers.SessionServerHandler; +import io.netty.channel.ChannelInitializer; +import io.netty.channel.socket.SocketChannel; +import io.netty.handler.codec.http.HttpObjectAggregator; +import io.netty.handler.codec.http.HttpRequestDecoder; +import io.netty.handler.codec.http.HttpResponseEncoder; + +/** + * 会话管道初始化类 + */ +public class SessionChannelInitializer extends ChannelInitializer { + + private final Configuration configuration; + + public SessionChannelInitializer(Configuration configuration) { + this.configuration = configuration; + } + + @Override + protected void initChannel(SocketChannel channel) throws Exception { + channel.pipeline().addLast(new HttpRequestDecoder()) + .addLast(new HttpResponseEncoder()) + .addLast(new HttpObjectAggregator(1024*1024)) + .addLast(new SessionServerHandler(configuration)); + } +} diff --git a/gateway/src/main/java/com/xgh/gateway/day02/session/SessionServer.java b/gateway/src/main/java/com/xgh/gateway/day02/session/SessionServer.java new file mode 100644 index 0000000..71fda8e --- /dev/null +++ b/gateway/src/main/java/com/xgh/gateway/day02/session/SessionServer.java @@ -0,0 +1,59 @@ +package com.xgh.gateway.day02.session; + + +import io.netty.bootstrap.ServerBootstrap; +import io.netty.channel.Channel; +import io.netty.channel.ChannelFuture; +import io.netty.channel.ChannelOption; +import io.netty.channel.EventLoopGroup; +import io.netty.channel.nio.NioEventLoopGroup; +import io.netty.channel.socket.nio.NioServerSocketChannel; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.net.InetSocketAddress; +import java.util.concurrent.Callable; + +/** + * 网关会话服务 + */ +public class SessionServer implements Callable { + + private final Logger logger = LoggerFactory.getLogger(SessionServer.class); + + private final Configuration configuration; + + private final EventLoopGroup boss = new NioEventLoopGroup(1); + + private final EventLoopGroup work = new NioEventLoopGroup(); + + private Channel channel; + + public SessionServer(Configuration configuration) { + this.configuration = configuration; + } + + + @Override + public Channel call() throws Exception { + ChannelFuture channelFuture = null; + try { + ServerBootstrap b =new ServerBootstrap(); + b.group(boss,work) + .channel(NioServerSocketChannel.class) + .option(ChannelOption.SO_BACKLOG,128) + .childHandler(new SessionChannelInitializer(configuration)); + channelFuture = b.bind(new InetSocketAddress(7397)).syncUninterruptibly(); + this.channel = channelFuture.channel(); + }catch (Exception e){ + logger.error("socket server start error .",e); + }finally { + if(null != channelFuture && channelFuture.isSuccess()){ + logger.info("socket server start done."); + }else { + logger.error("socket server start error."); + } + } + return channel; + } +} diff --git a/gateway/src/main/java/com/xgh/gateway/day02/session/defaults/GenericReferenceSessionFactory.java b/gateway/src/main/java/com/xgh/gateway/day02/session/defaults/GenericReferenceSessionFactory.java new file mode 100644 index 0000000..92135d6 --- /dev/null +++ b/gateway/src/main/java/com/xgh/gateway/day02/session/defaults/GenericReferenceSessionFactory.java @@ -0,0 +1,39 @@ +package com.xgh.gateway.day02.session.defaults; + +import com.xgh.gateway.day02.session.Configuration; +import com.xgh.gateway.day02.session.IGenericReferenceSessionFactory; +import com.xgh.gateway.day02.session.SessionServer; +import io.netty.channel.Channel; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.util.concurrent.ExecutionException; +import java.util.concurrent.Executors; +import java.util.concurrent.Future; + +public class GenericReferenceSessionFactory implements IGenericReferenceSessionFactory { + + private final Logger logger = LoggerFactory.getLogger(GenericReferenceSessionFactory.class); + + private final Configuration configuration; + + public GenericReferenceSessionFactory(Configuration configuration) { + this.configuration = configuration; + } + + @Override + public Future openSession() throws ExecutionException, InterruptedException { + SessionServer server = new SessionServer(configuration); + Future future = Executors.newFixedThreadPool(2).submit(server); + Channel channel = future.get(); + if(null == channel){ + throw new RuntimeException("netty server start error channel is null"); + } + while (!channel.isActive()){ + logger.info("netty server gateway start ing ..."); + Thread.sleep(500); + } + logger.info("netty server gateway start Done:{}",channel.localAddress()); + return future; + } +} diff --git a/gateway/src/main/java/com/xgh/gateway/day02/session/handlers/SessionServerHandler.java b/gateway/src/main/java/com/xgh/gateway/day02/session/handlers/SessionServerHandler.java new file mode 100644 index 0000000..b29226a --- /dev/null +++ b/gateway/src/main/java/com/xgh/gateway/day02/session/handlers/SessionServerHandler.java @@ -0,0 +1,58 @@ +package com.xgh.gateway.day02.session.handlers; + +import com.alibaba.fastjson.JSON; +import com.alibaba.fastjson.serializer.SerializerFeature; +import com.xgh.gateway.day02.bind.IGenericReference; +import com.xgh.gateway.day02.session.BaseHandler; +import com.xgh.gateway.day02.session.Configuration; +import io.netty.channel.Channel; +import io.netty.channel.ChannelHandlerContext; +import io.netty.handler.codec.http.*; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +/** + * 会话服务处理器 + */ +public class SessionServerHandler extends BaseHandler { + + private final Logger logger = LoggerFactory.getLogger(SessionServerHandler.class); + + private final Configuration configuration; + + public SessionServerHandler(Configuration configuration) { + this.configuration = configuration; + } + + + @Override + protected void session(ChannelHandlerContext ctx, Channel channel, FullHttpRequest request) { + logger.info("网关接收请求uri:{},method:{}",request.uri(),request.method()); + //返回信息控制 + String methodName = request.uri().substring(1); + if(methodName.equals("favicon.ico")){return;} + //返回信息处理 + DefaultFullHttpResponse response = new DefaultFullHttpResponse(HttpVersion.HTTP_1_1, HttpResponseStatus.OK); + //服务泛化调用 + IGenericReference reference = configuration.getGenericReference("sayHi"); + String result = reference.$invoke("test)" + " " + System.currentTimeMillis()); + + //设置回写数据 + response.content().writeBytes(JSON.toJSONBytes(result, SerializerFeature.PrettyFormat)); + //头部信息设置 + HttpHeaders heads = response.headers(); + //返回内容类型 + heads.add(HttpHeaderNames.CONTENT_TYPE,HttpHeaderValues.APPLICATION_JSON+";charset=UTF-8"); + //响应体长度 + heads.add(HttpHeaderNames.CONTENT_LENGTH,response.content().readableBytes()); + //配置持久化 + heads.add(HttpHeaderNames.CONNECTION,HttpHeaderValues.KEEP_ALIVE); + //配置跨域 + heads.add(HttpHeaderNames.ACCESS_CONTROL_ALLOW_ORIGIN, "*"); + heads.add(HttpHeaderNames.ACCESS_CONTROL_ALLOW_HEADERS, "*"); + heads.add(HttpHeaderNames.ACCESS_CONTROL_ALLOW_METHODS, "GET, POST, PUT, DELETE"); + heads.add(HttpHeaderNames.ACCESS_CONTROL_ALLOW_CREDENTIALS, "true"); + + channel.writeAndFlush(response); + } +} diff --git a/gateway/src/test/java/com/xgh/gateway/day01/ApiTest.java b/gateway/src/test/java/com/xgh/gateway/day01/ApiTest.java new file mode 100644 index 0000000..d2b54f0 --- /dev/null +++ b/gateway/src/test/java/com/xgh/gateway/day01/ApiTest.java @@ -0,0 +1,37 @@ +package com.xgh.gateway.day01; + +import io.netty.channel.Channel; +import org.junit.Test; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.util.concurrent.ExecutionException; +import java.util.concurrent.Executors; +import java.util.concurrent.Future; + +/** + * @description 测试 + */ +public class ApiTest { + + private final Logger logger = LoggerFactory.getLogger(ApiTest.class); + + @Test + public void test() throws ExecutionException, InterruptedException { + SessionServer server = new SessionServer(); + + Future future = Executors.newFixedThreadPool(2).submit(server); + Channel channel = future.get(); + + if (null == channel) throw new RuntimeException("netty server start error channel is null"); + + while (!channel.isActive()) { + logger.info("NettyServer启动服务 ..."); + Thread.sleep(500); + } + logger.info("NettyServer启动服务完成 {}", channel.localAddress()); + + Thread.sleep(Long.MAX_VALUE); + } + +} \ No newline at end of file diff --git a/gateway/src/test/java/com/xgh/gateway/day02/ApiTest.java b/gateway/src/test/java/com/xgh/gateway/day02/ApiTest.java new file mode 100644 index 0000000..aa59e98 --- /dev/null +++ b/gateway/src/test/java/com/xgh/gateway/day02/ApiTest.java @@ -0,0 +1,30 @@ +package com.xgh.gateway.day02; + +import com.xgh.gateway.day02.session.Configuration; +import com.xgh.gateway.day02.session.GenericReferenceSessionFactoryBuilder; +import io.netty.channel.Channel; +import org.junit.Test; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.util.concurrent.ExecutionException; +import java.util.concurrent.Future; + + +public class ApiTest { + + private final Logger logger = LoggerFactory.getLogger(ApiTest.class); + + @Test + public void test_GenericReference() throws InterruptedException, ExecutionException { + Configuration configuration = new Configuration(); + configuration.addGenericReference("api-gateway-test", "cn.bugstack.gateway.rpc.IActivityBooth", "sayHi"); + + GenericReferenceSessionFactoryBuilder builder = new GenericReferenceSessionFactoryBuilder(); + Future future = builder.build(configuration); + + logger.info("服务启动完成 {}", future.get().id()); + + Thread.sleep(Long.MAX_VALUE); + } +} \ No newline at end of file diff --git a/gateway/src/test/java/com/xgh/gateway/day02/CglibTest.java b/gateway/src/test/java/com/xgh/gateway/day02/CglibTest.java new file mode 100644 index 0000000..98b4361 --- /dev/null +++ b/gateway/src/test/java/com/xgh/gateway/day02/CglibTest.java @@ -0,0 +1,43 @@ +package com.xgh.gateway.day02; + +import com.alibaba.fastjson.JSON; +import net.sf.cglib.core.Signature; +import net.sf.cglib.proxy.Enhancer; +import net.sf.cglib.proxy.InterfaceMaker; +import net.sf.cglib.proxy.MethodInterceptor; +import net.sf.cglib.proxy.MethodProxy; +import org.junit.Test; +import org.objectweb.asm.Type; + +import java.lang.reflect.InvocationTargetException; +import java.lang.reflect.Method; + +public class CglibTest implements MethodInterceptor { + + @Override + public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable { + return JSON.toJSONString(objects); + } + + @Test + public void test_cglib() throws NoSuchMethodException, InvocationTargetException, IllegalAccessException, NoSuchFieldException { + // 定义接口 + InterfaceMaker interfaceMaker = new InterfaceMaker(); + interfaceMaker.add(new Signature("sayHi", Type.getType(String.class), new Type[]{Type.getType(String.class)}), null); + Class interfaceClass = interfaceMaker.create(); + + // 创建代理 + Enhancer enhancer = new Enhancer(); + enhancer.setSuperclass(Object.class); + enhancer.setInterfaces(new Class[]{interfaceClass}); + enhancer.setCallback(this); + Object obj = enhancer.create(); + + // 调用方法 + Method method = obj.getClass().getMethod("sayHi", String.class); + Object result = method.invoke(obj,"hi xiaofuge"); + + System.out.println(result); + } + +} \ No newline at end of file diff --git a/gateway/src/test/java/com/xgh/gateway/day02/RPCTest.java b/gateway/src/test/java/com/xgh/gateway/day02/RPCTest.java new file mode 100644 index 0000000..c74b0b0 --- /dev/null +++ b/gateway/src/test/java/com/xgh/gateway/day02/RPCTest.java @@ -0,0 +1,47 @@ +package com.xgh.gateway.day02; + +import org.apache.dubbo.config.ApplicationConfig; +import org.apache.dubbo.config.ReferenceConfig; +import org.apache.dubbo.config.RegistryConfig; +import org.apache.dubbo.config.bootstrap.DubboBootstrap; +import org.apache.dubbo.config.utils.ReferenceConfigCache; +import org.apache.dubbo.rpc.service.GenericService; +import org.junit.Test; + +/** + * 泛化调用测试 + * 官网案例:https://dubbo.apache.org/zh/docs/advanced/generic-reference/ + */ +public class RPCTest { + + @Test + public void test_rpc() { + + ApplicationConfig application = new ApplicationConfig(); + application.setName("api-gateway-test"); + application.setQosEnable(false); + + RegistryConfig registry = new RegistryConfig(); + registry.setAddress("zookeeper://127.0.0.1:2181"); + registry.setRegister(false); + + ReferenceConfig reference = new ReferenceConfig<>(); + reference.setInterface("cn.bugstack.gateway.rpc.IActivityBooth"); + reference.setVersion("1.0.0"); + reference.setGeneric("true"); + + DubboBootstrap bootstrap = DubboBootstrap.getInstance(); + bootstrap.application(application) + .registry(registry) + .reference(reference) + .start(); + + ReferenceConfigCache cache = ReferenceConfigCache.getCache(); + GenericService genericService = cache.get(reference); + + Object result = genericService.$invoke("sayHi", new String[]{"java.lang.String"}, new Object[]{"world"}); + + System.out.println(result); + } + +} diff --git a/pom.xml b/pom.xml index 3a26a3e..9e0ee98 100644 --- a/pom.xml +++ b/pom.xml @@ -2,6 +2,10 @@ 4.0.0 + pom + + gateway + org.springframework.boot spring-boot-starter-parent