详细Android连接远程的MySQL数据库实例
mhr18 2024-09-18 20:18 20 浏览 0 评论
前言
第一次连接远程MySQL数据,在此留下笔记,针对遇到的总是也一并给出,在此说明,连接远程数据库在我没遇到的问题暂时无法给出,我尽量按写程的时间最全的给出代码。连接的方法是采用mysql-connector-java-5.1.30-bin.jar包进行连接。欢迎讨论。
说明:
- 以下记录按时间先后顺利给出,采用同样的环境应该可以重现。
- 代码部分也由顺序给出,如:某一个方法或者类后面的代码可能比前面贴出的全,也可能是增加了内容,也可能是解决了前面的某个问题。
- 为了程序的可扩展性,本示例采用数据与UI分离写法,自定义了页面的基类BasePager,initView()和initData()
环境配置
服务器:阿里云服务器 ECS
操作系统:centos 7
MySQL:version: 5.7.27
连接服务器软件:FinalShell
MySQL管理软件:RoboDB MySQL Manager
安卓开发软件:Androi Studio 3.5.2
查看MySQL:version方法
在登陆服务器后输入mysql -uroot -p再输入密码。
[root@njzhw ~]# mysql -uroot -p
Enter password:
Welcome to the MySQL monitor. Commands end with ; or \g.
Your MySQL connection id is 1234354
Server version: 5.7.27 MySQL Community Server (GPL)
Copyright (c) 2000, 2019, Oracle and/or its affiliates. All rights reserved.
Oracle is a registered trademark of Oracle Corporation and/or its
affiliates. Other names may be trademarks of their respective
owners.
Type 'help;' or '\h' for help. Type '\c' to clear the current input statement.
mysql>
新建Android工程
新建一个空白Activity的工程文,目前采用最新的各类文件版本,
build.gradle
app的build.gradle文件如下:所有支持均已最新版本。minSdkVersion 19因受我电脑外挂android模拟器的限制,采用了19。高了模拟器运行不起来。
apply plugin: 'com.android.application'
android {
compileSdkVersion 29
buildToolsVersion "29.0.3"
defaultConfig {
applicationId "cn.plczl.myapplication"
minSdkVersion 19
targetSdkVersion 29
versionCode 1
versionName "1.0"
testInstrumentationRunner "androidx.test.runner.AndroidJUnitRunner"
}
buildTypes {
release {
minifyEnabled false
proguardFiles getDefaultProguardFile('proguard-android-optimize.txt'), 'proguard-rules.pro'
}
}
}
dependencies {
implementation fileTree(dir: 'libs', include: ['*.jar'])
implementation 'androidx.appcompat:appcompat:1.1.0'
implementation 'androidx.constraintlayout:constraintlayout:1.1.3'
testImplementation 'junit:junit:4.13'
androidTestImplementation 'androidx.test.ext:junit:1.1.1'
androidTestImplementation 'androidx.test.espresso:espresso-core:3.2.0'
}
activity_main.xml布局文件
MainActivity所对应的activity_main.xml的布局文只有一个FrameLayout,所有数据显示采用另外的代码实现。
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
tools:context=".MainActivity">
<FrameLayout
android:id="@+id/fl_main"
android:layout_width="match_parent"
android:layout_height="match_parent" />
</RelativeLayout>
MainActivity类修改为继承于Activity
新建好的MainActivity是继承于AppCompatActivity。区别是软件运行时不需要其默认的标题栏
百度的区别:
Activity 是其它 Activity 的基类,包括 AppCompatActivity。默认带标题栏,从字面理解,App兼容Activity,意思是允许我们轻易地将API 21+的特性应用到之前的那些老的,不兼容的Activity上面。例如很容易地将Toolbar添加到Activity上
代码如下:
package cn.plczl.mysql;
import android.app.Activity;
import android.os.Bundle;
import androidx.appcompat.app.AppCompatActivity;
public class MainActivity extends Activity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
}
}
新建一个基类
用于子页面来继承,这样可以有多个页面采用自定义的一类进行界面、数据分离处理。
在工程中新建一个base的包,包中建立一个BasePager的类。代码如下:本基类无需要继承,
主要代码如下:
- 其中有一个构造函数public BasePager(Context context),用于获取上下文和初始化视图(rootView);
- 一个抽象的initView方法,强制由调用的页面来实现,以实现不同页面有不同的UI结构;
- 一个初始化数据的方法(initData),本方法为一个空方法,由继承它的子页面来实现数据填充。在子页面中还可以在本方法中进行耗时的联网,数据请求,数据显示的操作。
注意: - 因public abstract View initView();这个是一个抽象的方法,所对应的public abstract class BasePager 类也必须是抽象的,也就是由abstract关键字。
- public Context conetxt;要修改为public,如果自生成时可能是这样的,private final Context conetxt;因为这个conetxt要被其它的类调用。
package com.atguigu.mysql_dome.Base;
import android.content.Context;
import android.view.View;
public abstract class BasePager {
public Context conetxt;
public View rootview;
//本类的构造函数
public BasePager(Context context){
this.conetxt=context;
rootview=initView();
}
//初始化视图的抽象类。强制由继承本类的其它类来实现
public abstract View initView();
//初始化视图的空方法,执行联网、数据请求、数据显示操作,本空方法不能缺少。
public void initData(){
}
}
新建一个子页面
在工程中新建一个包(pager)来存放各子页面,在其包下新建一个EthMainPager的子页面,
代码如下:初步的实现了构造方法,initView()方法和initData方法。
其在程序中显示“初始化了EthMainPager”的文字。
package cn.plczl.mysql.pager;
import android.content.Context;
import android.view.Gravity;
import android.view.View;
import android.widget.TextView;
import cn.plczl.mysql.base.BasePager;
import cn.plczl.mysql.utils.LogUtil;
public class EthMainPager extends BasePager {
private TextView textView;
public EthMainPager(Context context) {
super(context);
}
@Override
public View initView() {
textView=new TextView(conetxt);
textView.setGravity(Gravity.CENTER);
textView.setTextSize(20);
return textView;
}
@Override
public void initData() {
super.initData();
LogUtil.e("初始化了EthMainPager");
textView.setText("初始化了EthMainPager");
}
}
将EthMainPager页面放入MainActivity中并显示。
修改MainActivity,为程序扩展,将EthMainPager放入一个basePagers的集合中,自定义了setFragment、getEthMainPager方法,新建了一个ReplacePager类
public class MainActivity extends FragmentActivity {
private ArrayList<BasePager> basePagers;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
basePagers = new ArrayList<>();
basePagers.add(new EthMainPager(this));
setFragment();
}
private void setFragment() {
FragmentManager fragmentManager = getSupportFragmentManager();
FragmentTransaction fragmentTransaction = fragmentManager.beginTransaction();
//这里采用了replace方法,如果只是一个页面,也可以采用add方法。
fragmentTransaction.replace(R.id.fl_main,new ReplacePager(getEthMainPager()));
fragmentTransaction.commit();
}
private BasePager getEthMainPager() {
BasePager basePager=basePagers.get(0);
if(basePager!=null){
basePager.initData();
}
return basePager;
}
public static class ReplacePager extends Fragment {
public BasePager currPager;
public ReplacePager(BasePager basePager) {
this.currPager=basePager;
}
@Override
public void onCreate(@Nullable Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
}
@Nullable
@Override
public View onCreateView(@NonNull LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {
return currPager.rootView;
}
}
}
为使程序看起来简捷,将public static class ReplacePager extends Fragment抽取出来形成一个单独的类。
将其抽取至base目录中。文件名为ReplacePager
在MainActivity加入包含import cn.plczl.mysql.base.ReplacePager;
package cn.plczl.mysql.base;
import android.os.Bundle;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import androidx.annotation.NonNull;
import androidx.annotation.Nullable;
import androidx.fragment.app.Fragment;
public class ReplacePager extends Fragment {
public BasePager currPager;
public ReplacePager(BasePager basePager) {
this.currPager=basePager;
}
@Override
public void onCreate(@Nullable Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
}
@Nullable
@Override
public View onCreateView(@NonNull LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {
return currPager.rootView;
}
}
初步运行界面
以上代码完成初步的界面搭建,初步运行起来的画面如下:
子页面的完善编写
接下来的基本操作都将在EthMainPager子页面中去实现界面,联网和数据请示以及数据显示的操作。
子页面的布局
在EthMainPager的public View initView() 加入一条View view=View.inflate(conetxt, R.layout.ethmainpager,null);,用于加载ethmainpager.xml布局文,不再采用上面程序中的如下代码来初始化子页面了,initView体现了页面的初始化,
textView=new TextView(conetxt);
textView.setGravity(Gravity.CENTER);
textView.setTextSize(20);
return textView;
@Override
public View initView() {
View view=View.inflate(conetxt, R.layout.ethmainpager,null);
return view;
}
ethmainpager.xml文件
这里采用简单的一个线性布局,里面有四个TextView,这个程序是Dome性质,只要参调出MySQL数据一张表中库中的两个数据和另一张表中的一个数据,
这里设定能调出数据库最近更新的日期和时间就完成任务。
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:orientation="vertical"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:background="#23D3FE">
<RelativeLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginTop="5dp"
android:layout_marginBottom="5dp"
android:layout_marginRight="20dp"
android:layout_marginLeft="20dp">
<TextView
android:id="@+id/tv_1"
android:layout_width="wrap_content"
android:layout_height="25dp"
android:text="数据更新时间"
android:padding="2dp"
android:textSize="16sp"
android:gravity="left"
android:layout_marginLeft="5dp"/>
<TextView
android:id="@+id/tv_data1"
android:layout_width="wrap_content"
android:layout_height="25dp"
android:text="2020-01-27"
android:padding="2dp"
android:textSize="16sp"
android:gravity="left"
android:layout_toRightOf="@id/tv_1"
android:layout_marginLeft="5dp"/>
<TextView
android:id="@+id/tv_time"
android:layout_width="wrap_content"
android:layout_height="25dp"
android:text="21:00:00"
android:padding="2dp"
android:textSize="16sp"
android:gravity="left"
android:layout_toRightOf="@id/tv_data1"
android:layout_marginRight="20dp"/>
</RelativeLayout>
<TextView
android:id="@+id/tv_version"
android:layout_width="wrap_content"
android:layout_height="25dp"
android:text="21:00:00"
android:padding="2dp"
android:textSize="16sp"
android:gravity="left"
android:layout_marginLeft="22dp"
android:layout_toRightOf="@id/tv_data"
android:layout_marginRight="20dp"/>
</LinearLayout>
实例化控件
将用于要显进日期和时间的两个TextView实例化。
private TextView tv_data;
private TextView tv_time;
private TextView tv_version;
public EthMainPager(Context context) {
super(context);
}
@Override
public View initView() {
View view = View.inflate(conetxt, R.layout.ethmainpager,null);
tv_data = view.findViewById(R.id.tv_data);
tv_time = view.findViewById(R.id.tv_time);
tv_version = view.findViewById(R.id.tv_version);
return view;
}
导入jar包
采用mysql-connector-java-5.1.30-bin.jar进行MySQL连接,
将下载好的mysql-connector-java-5.1.30-bin.jar复制到工程的libs文件夹下,然点在mysql-connector-java-5.1.30-bin.jar点右键——>点击Add As Library...
完成后build.gradle文件中有这个包的加入。
导入后的build.gradle如下,最后一行代码就是导入了mysql-connector-java-5.1.30-bin.jar
apply plugin: 'com.android.application'
android {
compileSdkVersion 29
buildToolsVersion "29.0.3"
defaultConfig {
applicationId "cn.plczl.myapplication"
minSdkVersion 19
targetSdkVersion 29
versionCode 1
versionName "1.0"
testInstrumentationRunner "androidx.test.runner.AndroidJUnitRunner"
}
buildTypes {
release {
minifyEnabled false
proguardFiles getDefaultProguardFile('proguard-android-optimize.txt'), 'proguard-rules.pro'
}
}
}
dependencies {
implementation fileTree(include: ['*.jar'], dir: 'libs')
implementation 'androidx.appcompat:appcompat:1.1.0'
implementation 'androidx.constraintlayout:constraintlayout:1.1.3'
testImplementation 'junit:junit:4.13'
androidTestImplementation 'androidx.test.ext:junit:1.1.1'
androidTestImplementation 'androidx.test.espresso:espresso-core:3.2.0'
implementation files('libs/mysql-connector-java-5.1.30-bin.jar')
}
数据库连接工具类
编写了连接数据库的工具类DBUtils.jave。用于要对数据操作的调用,代码如下:
其中有三个方法:
- getConn用于连接数据,其中String dbName参数为要连接的数据名字,
- getEthData方法是获得eth数据库中的apidate表的最新一行的数据方法。
- getVersion方法是eth数据库中的version表中的version字段的的最新一行的数据方法。
调用的方法分别以一个数组和字符串返回。
package cn.plczl.mysql.utils;
import android.util.Log;
import java.sql.Connection;
import java.sql.DriverManager;
import java.sql.ResultSet;
import java.sql.SQLException;
//连接mySql数据库方法
public class DBUtils {
private static final String TAG ="Dbutils";
private static String driver="com.mysql.jdbc.Driver";
//这里是MySQL的用户名
private static String user="root";
//这里是我的MySQL密码,本文作了隐藏处理,
private static String password = "xxxxxxx";
public static Connection getConn(String dbName){
Connection connection = null;
try{
Class.forName(driver);
//数据的IP地址,本文中的地址不是我的真实地址,请换为你的真实IP地址。
String ip="39.10.011";
String port="3306";
String url = "jdbc:mysql://" + ip + ":" + port +"/" + dbName;
connection= DriverManager.getConnection(url, user, password);
Log.e("数据库连接", "成功!");
} catch (Exception e) {
Log.e("数据库连接", "失败!");
e.printStackTrace();
}
return connection;
}
public static String[] getEthData() {
String rsdata[]=new String[24];
Connection connection = getConn("eth");
if (connection!=null){
String sql="select * from apidate order by id DESC limit 1";
try{
java.sql.Statement statement = connection.createStatement();
ResultSet rSet = statement.executeQuery(sql);
while (rSet.next()) {
rsdata[0]=rSet.getString("date");//日期
rsdata[1]=rSet.getString("time");//时间
rsdata[2]=rSet.getString("capital").substring(0,4);//人民币总收益
rsdata[3]=rSet.getString("worker_length");//矿机总数
rsdata[4]=rSet.getString("worker_lenght_online");//在线矿机数
rsdata[5]=rSet.getString("worker_length_offline");//离线矿机数
rsdata[6]=rSet.getString("dead_sl");//失线矿机数
rsdata[7]=rSet.getString("hash_24_hour");//24小时平均算力
rsdata[8]=rSet.getString("last_day_value").substring(0,9);//过去24小时收益
rsdata[9]=rSet.getString("hash_15");//15分钟平均算力
rsdata[10]=rSet.getString("balance").substring(0,9);//当前余额
rsdata[11]=rSet.getString("local_hash");//本地算力
rsdata[12]=rSet.getString("value").substring(0,9);//总收益
rsdata[13]=rSet.getString("quanwangsl")+" TH/s";//全网算力
rsdata[14]=rSet.getString("f2poolsl")+" TH/s";//全网算力
rsdata[15]=rSet.getString(
"meiM_Eth")+" ETH ≈ ¥"
+String.valueOf(Double.valueOf(rSet.getString("meiM_Eth"))
*Double.valueOf(rSet.getString("meiyuanzhisu"))
*Double.valueOf(rSet.getString("usdCnyRate")))
.substring(0,5);//每M收益
rsdata[16]= String.valueOf(
Double.valueOf(rSet.getString("waikuangnandu"))/10000/10000/10000/1000)
.substring(0,5)+" P";//当前难度
rsdata[17]="#34;+rSet.getString(
"price")+" ≈ ¥"
+String.valueOf(Double.valueOf(rSet.getString("price"))
*Double.valueOf(rSet.getString("usdCnyRate")))
.substring(0,4);//币价usdCnyRate
rsdata[18]="¥"+rSet.getString("ETC_index");//etc指数
rsdata[19]=rSet.getString("ETC_zhangdie");//etc涨跌
rsdata[20]=rSet.getString("ETH_zuigaojia");//eth最高价
rsdata[21]=rSet.getString("ETH_zuidijia");//eth最低价
rsdata[22]=rSet.getString("riseandfall");//ETH涨跌
rsdata[23]=rSet.getString("id");//数据库数据条数
Log.e(TAG,"数组组装成功");
}
connection.close();
} catch (SQLException e) {
e.printStackTrace();
}
};
Log.e(TAG,"数组已返回");
return rsdata;
}
public static String getVersion() {
String version=null;
Connection connection = getConn("eth");
if (connection!=null){
String sql="select version from version";
try{
java.sql.Statement statement = connection.createStatement();
ResultSet rSet = statement.executeQuery(sql);
while (rSet.next()) {
version=rSet.getString("version");//日期
Log.e(TAG,"数组组装成功==="+version);
}
connection.close();
} catch (SQLException e) {
e.printStackTrace();
}
};
Log.e(TAG,"数组已返回==========");
return version;
}
}
获得数据更新
在EthMainPagerr的initData()方法中写了两个方法,分别连接两个数据库获得数据并显示。
getData()方法是调用DBUtils数据库连接工具中的getEthData()
getVersion();方法是调用DBUtils数据库连接工具中的getVersion()
采用在子线程中连接数据库和Handler切换到主线程进行数据更新。
@Override
public void initData() {
super.initData();
LogUtil.e("初始化了EthMainPager");
getData();
getVersion();
}
getData()
简单的new了一个Thread子线程,用于调用 ethdata=DBUtils.getEthData();方法,将数据存入ethdata数组中。通过handler.sendEmptyMessage(0);返回主线程。
private void getData() {
new Thread(){
@Override
public void run() {
super.run();
ethdata=DBUtils.getEthData();
handler.sendEmptyMessage(0);
}
}.start();
}
handler
在handler中更换tv_data.setText(ethdata[0]);tv_time.setText(ethdata[1]);获得的数据。
private Handler handler = new Handler(){
@Override
public void handleMessage(@NonNull Message msg) {
super.handleMessage(msg);
LogUtil.e(String.valueOf(ethdata.length));
tv_data.setText(ethdata[0]);
tv_time.setText(ethdata[1]);
}
};
getVersion()
与 getData()类似。
version=DBUtils.getVersion();方法,将数据存入version数组中。通过versionhandle.sendEmptyMessage(1);返回主线程。
private void getVersion() {
new Thread(){
@Override
public void run() {
super.run();
version=DBUtils.getVersion();
versionhandle.sendEmptyMessage(1);
}
}.start();
}
versionhandle
在versionhandle中更新tv_version.setText(version);获得的数据。
private Handler versionhandle = new Handler(){
@Override
public void handleMessage(@NonNull Message msg) {
super.handleMessage(msg);
LogUtil.e(String.valueOf(ethdata.length));
tv_version.setText(version);
}
};
最终运行的结果
软件运行结果
数据库中的数据
相关推荐
- 【预警通报】关于WebLogic存在远程代码执行高危漏洞的预警通报
-
近日,Oracle官方发布了2021年1月关键补丁更新公告CPU(CriticalPatchUpdate),共修复了包括CVE-2021-2109(WeblogicServer远程代码执行漏洞)...
- 医院信息系统突发应急演练记录(医院信息化应急演练)
-
信息系统突发事件应急预案演练记录演练内容信息系统突发事件应急预案演练参与人员信息科参与科室:全院各部门日期xxxx-xx-xx时间20:00至24:00地点信息科记录:xxx1、...
- 一文掌握怎么利用Shell+Python实现完美版的多数据源备份程序
-
简介:在当今数字化时代,无论是企业还是个人,数据的安全性和业务的连续性都是至关重要的。数据一旦丢失,可能会造成无法估量的损失。因此,如何有效地对分布在不同位置的数据进行备份,尤其是异地备份,成为了一个...
- docker搭建系统环境(docker搭建centos)
-
Docker安装(CentOS7)1.卸载旧版Docker#检查已安装版本yumlistinstalled|grepdocker#卸载旧版本yumremove-ydocker.x...
- 基础篇:数据库 SQL 入门教程(sql数据库入门书籍推荐)
-
SQL介绍什么是SQLSQL指结构化查询语言,是用于访问和处理数据库的标准的计算机语言。它使我们有能力访问数据库,可与多种数据库程序协同工作,如MSAccess、DB2、Informix、M...
- Java21杀手级新特性!3行代码性能翻倍
-
导语某券商系统用这招,交易延迟从12ms降到0.8ms!本文揭秘Oracle官方未公开的Record模式匹配+虚拟线程深度优化+向量API神操作,代码量直降70%!一、Record模式匹配(代码量↓8...
- 一文读懂JDK21的虚拟线程(java虚拟线程)
-
概述JDK21已于2023年9月19日发布,作为Oracle标准Java实现的一个LTS版本发布,发布了15想新特性,其中虚拟线程呼声较高。虚拟线程是JDK21中引入的一项重要特性,它是一种轻量级的...
- 效率!MacOS下超级好用的Linux虚拟工具:Lima
-
对于MacOS用户来说,搭建Linux虚拟环境一直是件让人头疼的事。无论是VirtualBox还是商业的VMware,都显得过于笨重且配置复杂。今天,我们要介绍一个轻巧方便的纯命令行Linux虚拟工具...
- 所谓SaaS(所谓三维目标一般都应包括)
-
2010年前后,一个科技媒体的主编写一些关于云计算的概念性问题,就可以作为头版头条了。那时候的云计算,更多的还停留在一些概念性的问题上。而基于云计算而生的SaaS更是“养在深闺人未识”,一度成为被IT...
- ORA-00600 「25027」 「x」报错(报错0xc0000001)
-
问题现象:在用到LOB大对象的业务中,进行数据的插入,失败了,在报警文件中报错:ORA-00600:内部错误代码,参数:[25027],[10],[0],[],[],[],[],[...
- 安卓7源码编译(安卓源码编译环境lunch失败,uname命令找不到)
-
前面已经下载好源码了,接下来是下载手机对应的二进制驱动执行编译源码命令下载厂商驱动https://developers.google.com/android/drivers?hl=zh-cn搜索NGI...
- 编译安卓源码(编译安卓源码 电脑配置)
-
前面已经下载好源码了,接下来是下载手机对应的二进制驱动执行编译源码命令下载厂商驱动https://developers.google.com/android/drivers?hl=zh-cn搜索NGI...
- 360 Vulcan Team首战告捷 以17.5万美金强势领跑2019“天府杯“
-
2019年11月16日,由360集团、百度、腾讯、阿里巴巴、清华大学与中科院等多家企业和研究机构在成都联合主办了2019“天府杯”国际网络安全大赛暨2019天府国际网络安全高峰论坛。而开幕当日最激荡人...
- Syslog 日志分析与异常检测技巧(syslog发送日志配置)
-
系统日志包含有助于分析网络设备整体运行状况的重要信息。然而,理解并从中提取有效数据往往颇具挑战。本文将详解从基础命令行工具到专业日志管理软件的全流程分析技巧,助你高效挖掘Syslog日志价值。Gr...
- 从Oracle演进看数据库技术的发展(从oracle演进看数据库技术的发展的过程)
-
数据库技术发展本质上是应用需求驱动与基础架构演进的双向奔赴,如何分析其技术发展的脉络和方向?考虑到oracle数据库仍然是这个领域的王者,以其为例,管中窥豹,对其从Oracle8i到23ai版本的核...
你 发表评论:
欢迎- 一周热门
- 最近发表
- 标签列表
-
- oracle位图索引 (74)
- oracle基目录 (50)
- oracle批量插入数据 (65)
- oracle事务隔离级别 (53)
- oracle主从同步 (55)
- oracle 乐观锁 (51)
- redis 命令 (78)
- php redis (88)
- redis 存储 (66)
- redis 锁 (69)
- 启动 redis (66)
- redis 时间 (56)
- redis 删除 (67)
- redis内存 (57)
- redis并发 (52)
- redis 主从 (69)
- redis 订阅 (51)
- redis 登录 (54)
- redis 面试 (58)
- 阿里 redis (59)
- redis 搭建 (53)
- redis的缓存 (55)
- lua redis (58)
- redis 连接池 (61)
- redis 限流 (51)