为什么使用MVP,
MVP框架相对于MVC框架来说相对复杂一些,代码量相对也要更大一些。但是MVP框架使得model层和view层之间分割开来,使用presenter作为两者之间交互的桥梁。耦合度更低更方便后期维护。
最主要的原因是:使用MVP框架即使是项目开发一般让另一个人进行开发也可以很好的入手,而不会出现MVC那种Activity分工不明确,一个类中调用几百个方法作者自己都难以搞清的情况。

下面这个是契约类,model层负责持有数据,presenter层持有数据,view层对数据进行展示。

public interface IHomeContract {interface IModel extends IBaseModel{void getFlowdata(String url,ModelCallback modelCallback);void getProductdata(String url,ModelCallback modelCallback);interface ModelCallback{void success(Object data);void error(Throwable throwable);}}interface IView extends IBaseView {void success(Object data);void error(Throwable throwable);}interface IPresenter {void getFlowdata(String url);void getProductdata(String url);}}

下边开始基类的抽取,首先抽取Presenter层的基类。我们在 Presenter 层直接绑定了 View 才可以拿到 View 层的引用,它们之间是强引用的关系,如果不进行解绑的话,那就会造成内存泄漏的情况发生。为了防止内存泄漏,我们在最后要解绑view。

public abstract class BasePresenter<M extends IBaseModel,V extends IBaseView> {public M model;public WeakReference<V> weakReference;public BasePresenter(){model = initModel();}/*** 绑定vieww* @param v*/public void attach(V v){weakReference = new WeakReference<>(v);}protected abstract M initModel();/*** 解绑view,解决内存泄漏问题*/public void dettach(){if (weakReference!=null){weakReference.clear();weakReference = null;}}public V getView(){return weakReference.get();}}

接下来对Activity进行基类抽取,封装泛型同时解决内存泄漏问题。

public abstract class BaseActivity<P extends BasePresenter> extends AppCompatActivity implements IBaseView{public P presenter;@Overrideprotected void onCreate(@Nullable Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(layoutId());presenter = initPresenter();if (presenter!=null){presenter.attach(this);//绑定view}initView();initData();}//让子类创建protected abstract P initPresenter();protected abstract void initData();protected abstract void initView();protected abstract int layoutId();@Overrideprotected void onDestroy() {super.onDestroy();if (presenter!=null){presenter.dettach();//解决内存泄漏}}
}

同上,抽取ragment对泛型进行封装,同时解决内存泄漏问题

public abstract class BaseFragment<P extends BasePresenter> extends Fragment implements IBaseView{public P presenter;@Nullable@Overridepublic View onCreateView(@NonNull LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {View view = inflater.inflate(LayoutId(),container,false);presenter = initPresenter();if (presenter!=null){presenter.attach(this);//绑定view}initView(view);return view;}protected abstract P initPresenter();protected abstract void initView(View view);protected abstract int LayoutId();@Overridepublic void onActivityCreated(@Nullable Bundle savedInstanceState) {super.onActivityCreated(savedInstanceState);initData();}protected abstract void initData();@Overridepublic void onDestroy() {super.onDestroy();if (presenter!=null){presenter.dettach();//解决内存泄漏}}
}

下面对model层进行封装,基类的作用就是对公共的代码进行封装,减少代码的重复性,增强代码的维护性,但是model层没有公共的方法,所以model层为空。

public interface IBaseModel {/*没用公共的方法*/
}

同上,view层只封装一个显示和一个隐藏即可。

public interface IBaseView {void showLoading();void hideLoading();
}

下面对工具类进行封装,为了解决多线程同时也是为了安全,这里使用了双重检验锁的单例模式,同时建议对属性进行私有化设置,这样可以有效防止外界new出新的对象。这里同时写了get请求和post请求,可以根据情况需要进行选择。

public class VolleyUtils {private RequestQueue requestQueue;//volley请求队列//双重检验锁的单例模式//属性私有private static VolleyUtils instance;//构造方法私有,防止外界(调用者)new 出新的对象private VolleyUtils() {requestQueue = Volley.newRequestQueue(App.getContext());}/*** 暴露公共方法,创建私有对象,供外部调用,双重:两次判断,检验锁:同步锁** @return*/public static VolleyUtils getInstance() {//第一重if (instance == null) {//加锁:为了解决多线程并发安全synchronized (VolleyUtils.class) {//二重if (instance == null) {instance = new VolleyUtils();}}}return instance;}/*** volley的get请求*/public void doGet( String url, final VolleyCallback volleyCallback) {//第二步StringRequest stringRequest = new StringRequest(StringRequest.Method.GET, url, new Response.Listener<String>() {@Overridepublic void onResponse(String response) {if (volleyCallback != null) {volleyCallback.success(response);}}}, new Response.ErrorListener() {@Overridepublic void onErrorResponse(VolleyError error) {if (volleyCallback != null) {volleyCallback.failure(error);}}});//第三步requestQueue.add(stringRequest);}/*** volley的post请求*/public void doPost(final Map<String, String> params, String url, final VolleyCallback volleyCallback) {//第二步StringRequest stringRequest = new StringRequest(StringRequest.Method.POST, url, new Response.Listener<String>() {@Overridepublic void onResponse(String response) {if (volleyCallback != null) {volleyCallback.success(response);}}}, new Response.ErrorListener() {@Overridepublic void onErrorResponse(VolleyError error) {if (volleyCallback != null) {volleyCallback.failure(error);}}}) {@Overrideprotected Map<String, String> getParams() throws AuthFailureError {return params;}};//第三步questQueue.add(stringRequest);}public interface VolleyCallback {void success(String response);void failure(Throwable error);}
}

在完成基类的封装和工具类以后我们要完成MVP框架的搭建。首先是model层的代码
model层主要就是请求数据并把数据传给p层,最后由P层传给view层进行展示。


public class HomeModel implements IHomeContract.IModel {@Overridepublic void getFlowdata(String url, final ModelCallback modelCallback) {VolleyUtils.getInstance().doGet(url, new VolleyUtils.VolleyCallback() {@Overridepublic void success(String response) {FlowEntity flowEntity = new Gson().fromJson(response,FlowEntity.class);modelCallback.success(flowEntity);}@Overridepublic void failure(Throwable error) {modelCallback.error(error);}});}@Overridepublic void getProductdata(String url, final ModelCallback modelCallback) {VolleyUtils.getInstance().doGet(url, new VolleyUtils.VolleyCallback() {@Overridepublic void success(String response) {System.out.println("resonse:"+response);ProductEntity productEntity = new Gson().fromJson(response,ProductEntity.class);modelCallback.success(productEntity);}@Overridepublic void failure(Throwable error) {System.out.println("resonseerror:"+error.getMessage());modelCallback.error(error);}});}
}

这个是presenter层的代码,presenter主要是model层和view层之间通讯的桥梁。


public class HomePresenter extends BasePresenter<HomeModel, IHomeContract.IView> implements IHomeContract.IPresenter {@Overrideprotected HomeModel initModel() {return new HomeModel();}@Overridepublic void getFlowdata(String url) {model.getFlowdata(url, new IHomeContract.IModel.ModelCallback() {@Overridepublic void success(Object data) {getView().success(data);}@Overridepublic void error(Throwable throwable) {getView().error(throwable);}});}@Overridepublic void getProductdata(String url) {model.getProductdata(url, new IHomeContract.IModel.ModelCallback() {@Overridepublic void success(Object data) {getView().success(data);}@Overridepublic void error(Throwable throwable) {getView().error(throwable);}});}
}

接下来的是首页面的布局,由fragment和viewpager联合组成,确保用户无论是点击下方的按钮还是单纯的滑动,都可以对页面进行切换。

《MainActivity---------首页的布局文件》
<LinearLayout 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"android:orientation="vertical"tools:context=".view.activity.MainActivity"><androidx.viewpager.widget.ViewPagerandroid:id="@+id/viewpager"android:layout_width="match_parent"android:layout_height="0dp"android:layout_weight="1"/><RadioGroupandroid:id="@+id/radiogroup"android:orientation="horizontal"android:layout_width="match_parent"android:layout_height="wrap_content"><RadioButtonandroid:padding="10dp"android:text="首页"android:gravity="center"android:id="@+id/radio_home"android:layout_width="0dp"android:layout_weight="1"android:button="@null"android:layout_height="wrap_content"/><RadioButtonandroid:padding="10dp"android:gravity="center"android:text="列表"android:button="@null"android:id="@+id/radio_news"android:layout_width="0dp"android:layout_weight="1"android:layout_height="wrap_content"/><RadioButtonandroid:padding="10dp"android:button="@null"android:gravity="center"android:text="我的"android:id="@+id/radio_my"android:layout_weight="1"android:layout_width="0dp"android:layout_height="wrap_content"/></RadioGroup>
</LinearLayout>

接下来是首页面的Activity代码部分,主要是viewpager和fragment进行展示,所有的fragment对象都添加进同一个队列中,方便进行切换。点击按钮切换通过RadioGroup和RadioButton的点击监听以及多分支选完成。

public class MainActivity extends BaseActivity {private ViewPager viewPager;private RadioGroup radioGroup;private List<Fragment> fragmentList ;@Overrideprotected BasePresenter initPresenter() {return null;}@Overrideprotected void initData() {fragmentList = new ArrayList<>();fragmentList.add(new HomeFragment());fragmentList.add(new BjFragment());fragmentList.add(new MyFragment());viewPager.setAdapter(new FragmentPagerAdapter(getSupportFragmentManager()) {@NonNull@Overridepublic Fragment getItem(int position) {return fragmentList.get(position);}@Overridepublic int getCount() {return fragmentList.size();}});}@Overrideprotected void initView() {viewPager = findViewById(R.id.viewpager);radioGroup = findViewById(R.id.radiogroup);viewPager.addOnPageChangeListener(new ViewPager.OnPageChangeListener() {@Overridepublic void onPageScrolled(int position, float positionOffset, int positionOffsetPixels) {}@Overridepublic void onPageSelected(int position) {RadioButton radioButton = (RadioButton) radioGroup.getChildAt(position);radioButton.setChecked(true);}@Overridepublic void onPageScrollStateChanged(int state) {}});radioGroup.setOnCheckedChangeListener(new RadioGroup.OnCheckedChangeListener() {@Overridepublic void onCheckedChanged(RadioGroup group, int checkedId) {switch (checkedId){case R.id.radio_home:viewPager.setCurrentItem(0);break;case R.id.radio_news:viewPager.setCurrentItem(1);break;case  R.id.radio_my:viewPager.setCurrentItem(2);break;}}});}@Overrideprotected int layoutId() {return R.layout.activity_main;}@Overridepublic void showLoading() {}@Overridepublic void hideLoading() {}
}

为了是页面开起来整体性更高,页面更加平滑。这里推荐对展示的数据使用流式布局


public class FlowLayout extends ViewGroup {public FlowLayout(Context context) {super(context);}public FlowLayout(Context context, AttributeSet attrs) {super(context, attrs);}public FlowLayout(Context context, AttributeSet attrs, int defStyleAttr) {super(context, attrs, defStyleAttr);}@Overrideprotected void onLayout(boolean changed, int l, int t, int r, int b) {int left = 0;int top = 0;int right = 0;int bottom = 0;int count = getChildCount();//得到子控件的数量if (count>0){for (int i = 0; i < count; i++) {//循环每一个子控件View view = getChildAt(i);//子view添加进去的,所以要让系统测量一下每个子控件的大小view.measure(0,0);int childWidth = view.getMeasuredWidth();//view的宽int childHeight = view.getMeasuredHeight();//view的高//累加rightright =left+childWidth;//屏幕宽度,px像素单位int widthPixels = getResources().getDisplayMetrics().widthPixels;//如果每一行的right大于屏幕宽度,开始折行判断if (right>widthPixels){left = 0;//折行后第一个控件距离左边距离为0right = left+childWidth;top = bottom+30;}bottom = top+childHeight;//就是对view进行摆放view.layout(left,top,right,bottom);left+=childWidth+30;}}}/*** 加入数据*/public void add(List<String> tags){if (tags!=null&&tags.size()>0){for (String tag : tags) {//动态创建textviewfinal TextView textView = new TextView(getContext());textView.setText(tag);//设置文本//添加子控件到当前流式布局中addView(textView);textView.setOnClickListener(new OnClickListener() {@Overridepublic void onClick(View v) {flowCallback.flowClick(textView.getText().toString());}});}}}/*** 添加单个view到流式布局*/public void addTextView(String name){final TextView textView = new TextView(getContext());textView.setText(name);textView.setOnClickListener(new OnClickListener() {@Overridepublic void onClick(View v) {Toast.makeText(getContext(), textView.getText().toString(), Toast.LENGTH_SHORT).show();flowCallback.flowClick(textView.getText().toString());}});//加入到流式布局中addView(textView);}private FlowCallback flowCallback;public void setFlowCallback(FlowCallback flowCallback) {this.flowCallback = flowCallback;}public interface FlowCallback{void flowClick(String name);}}

下面是流式布局的的布局文件

《HomeFragment——layout布局文件》
<?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"><LinearLayoutandroid:layout_width="match_parent"android:layout_height="wrap_content" android:orientation="horizontal"><EditTextandroid:id="@+id/et_keyword"android:hint="请输入搜索关键词"android:layout_width="wrap_content"android:layout_height="wrap_content"/><Buttonandroid:id="@+id/btn_search"android:text="搜索"android:layout_width="wrap_content"android:layout_height="wrap_content"/></LinearLayout><com.laoxu.yuekao.view.widgets.FlowLayoutandroid:id="@+id/flowlayout"android:layout_width="match_parent"android:layout_height="150dp"/><androidx.recyclerview.widget.RecyclerViewandroid:id="@+id/rv"android:layout_width="match_parent"android:layout_height="wrap_content"/>
</LinearLayout>

电商功能要求根据用户输入的关键词进行索引相关内容,为了方便用户使用记录用户的搜索记录,添加用户每一次搜索的关键词,在用户第二次点击时可以直接进行跳转,这里使用了网格布局。先写适配器,用户方法的回调使用接口回调进行。
列表中图片的展示使用Glide加载,完成错位图、占位图、原型的加载。
先是一个设定一个布局,设定列表展示的样子。

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"android:orientation="vertical" android:layout_width="match_parent"android:layout_height="match_parent"><ImageViewandroid:id="@+id/iv"android:layout_width="80dp"android:layout_height="80dp"/><TextViewandroid:id="@+id/name"android:layout_width="wrap_content"android:layout_height="wrap_content"/>
</LinearLayout>

接下来是适配器部分。

public class ProductAdapter extends RecyclerView.Adapter<ProductAdapter.MyViewHolder> {private Context context;private List<ProductEntity.Product> list;public ProductAdapter(Context context, List<ProductEntity.Product> list) {this.context = context;this.list = list;}@NonNull@Overridepublic MyViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int viewType) {View view = View.inflate(context,R.layout.product_item_layout,null);MyViewHolder myViewHolder = new MyViewHolder(view);return myViewHolder;}@Overridepublic void onBindViewHolder(@NonNull MyViewHolder holder, final int position) {holder.nameTv.setText(list.get(position).commodityName);Glide.with(context).load(list.get(position).masterPic).placeholder(R.mipmap.ic_launcher).error(R.mipmap.ic_launcher).circleCrop().into(holder.iconIv);//点击事件,通过下面声明的接口,发送商品标题holder.itemView.setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View v) {rvItemClick.onClick(list.get(position).commodityName);}});}@Overridepublic int getItemCount() {return list.size();}class MyViewHolder extends RecyclerView.ViewHolder{private ImageView iconIv;private TextView nameTv;public MyViewHolder(@NonNull View itemView) {super(itemView);iconIv = itemView.findViewById(R.id.iv);nameTv = itemView.findViewById(R.id.name);}}//声明接口private RvItemClick rvItemClick;public void setRvItemClick(RvItemClick rvItemClick) {this.rvItemClick = rvItemClick;}public interface RvItemClick{void onClick(String name);}
}

下面是本次项目的主体部分,同时也是整个项目的核心,是我们之前的布局要展示的内容。同时也是我们项目最核心的代码。在用户点击搜索按钮一个获取关键字,根据关键词来进行搜索,若是关键字为空则对用户进行输入提示。根据关键词进行点击搜索。点击这一部分使用接口回调完成。
点击商品详情后Toast提示,同时跳转到第二个页面。


public class HomeFragment extends BaseFragment<HomePresenter> implements IHomeContract.IView {private FlowLayout flowLayout;private EditText editText;private Button btn;private RecyclerView rv;@Overrideprotected HomePresenter initPresenter() {return new HomePresenter();}@Overrideprotected void initView(View view) {flowLayout =view.findViewById(R.id.flowlayout);rv =view.findViewById(R.id.rv);rv.setLayoutManager(new GridLayoutManager(getActivity(),2));editText =view.findViewById(R.id.et_keyword);btn =view.findViewById(R.id.btn_search);//点击流式布局的tag,请求搜索接口flowLayout.setFlowCallback(new FlowLayout.FlowCallback() {@Overridepublic void flowClick(String name) {String url = "http://172.17.8.100/small/commodity/v1/findCommodityByKeyword?keyword="+URLEncoder.encode(name)+"&count=10&page=1";presenter.getProductdata(url);}});btn.setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View v) {if (TextUtils.isEmpty(editText.getText().toString())){Toast.makeText(getActivity(), "不能为空", Toast.LENGTH_SHORT).show();return;}flowLayout.addTextView(editText.getText().toString());String url = "http://172.17.8.100/small/commodity/v1/findCommodityByKeyword?keyword="+URLEncoder.encode(editText.getText().toString())+"&count=10&page=1";presenter.getProductdata(url);}});}@Overrideprotected int LayoutId() {return R.layout.fragment_home_layout;}@Overrideprotected void initData() {String url = "http://blog.zhaoliang5156.cn/baweiapi/"+ URLEncoder.encode("手机");presenter.getFlowdata(url);}@Overridepublic void showLoading() {}@Overridepublic void hideLoading() {}/*** 成功* @param data*/@Overridepublic void success(Object data) {if (data instanceof FlowEntity){//instance//FlowEntity flowEntity = (FlowEntity) data;flowLayout.add(flowEntity.tags);}else if (data instanceof ProductEntity){ProductEntity productEntity = (ProductEntity) data;//适配器,recyclerviewProductAdapter productAdapter = new ProductAdapter(getActivity(),productEntity.result);rv.setAdapter(productAdapter);//接收接口传过来的数据productAdapter.setRvItemClick(new ProductAdapter.RvItemClick() {@Overridepublic void onClick(String name) {Toast.makeText(getActivity(), name, Toast.LENGTH_SHORT).show();startActivity(new Intent(getActivity(), SecondActivity.class));}});}}/*** 失败* @param throwable*/@Overridepublic void error(Throwable throwable) {}
}

下边是点击以后跳转的第二个activity的布局情况

<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"><WebViewandroid:id="@+id/webview"android:layout_width="match_parent"android:layout_height="wrap_content"/><Buttonandroid:id="@+id/btn"android:text="去调用js空参方法"android:layout_alignParentRight="true"android:layout_alignParentBottom="true"android:layout_width="wrap_content"android:layout_height="wrap_content"/></RelativeLayout>

下边是代码主要实现的功能:
主要功能是应用内加载本地图片,Android和JavaScript进行交互,点击以后进传值并吐司。


public class SecondActivity extends BaseActivity {private WebView webView;private Button button;@Overrideprotected BasePresenter initPresenter() {return null;}@Overrideprotected void initData() {}@Overrideprotected void initView() {webView  = findViewById(R.id.webview);webView.getSettings().setJavaScriptEnabled(true);webView.setWebViewClient(new WebViewClient());webView.loadUrl("file:///android_asset/h.html");//把android的user的对象映射到webview里面User user = new User();webView.addJavascriptInterface(user,"ccc");button = findViewById(R.id.btn);button.setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View v) {String s = "我来自于adnroid的世界";webView.loadUrl("javascript:jsFunction2('"+s+"')");}});}@Overrideprotected int layoutId() {return R.layout.activity_second;}@Overridepublic void showLoading() {}@Overridepublic void hideLoading() {}
}

这个是assets文件下要加载的本地网页的代码

<html>
<head><meta content="charset=utf-8" /><script type="text/javascript">function jsFunction2(hello){document.getElementsByClassName("name").innerHTML = hello;alert(hello);}</script></head><body>
<p>我是一个段落</p>
<p class="name">我是从android过来的</p>
<button onclick="window.ccc.aaa()" >无参调用android</button>
<button onclick="window.ccc.bbb('电脑')">有参调用android</button>
</body>
</html>

下面是android和js交互的部分


public class User {/*** 注解*/@JavascriptInterfacepublic void aaa(){Log.e("aaa","js无参数传递过来的方法");Toast.makeText(App.getContext(), "js无参数传递过来的方法", Toast.LENGTH_SHORT).show();}@JavascriptInterfacepublic void bbb(String name){Log.e("aaa",name);Toast.makeText(App.getContext(), name, Toast.LENGTH_SHORT).show();}
}

最后是一个缩减版的全局异常捕获类,因完整版代码太长,这里进行了一定删减。
仅仅需要实现一个接口即可。

public class MyCrashHandler implements Thread.UncaughtExceptionHandler {@Overridepublic void uncaughtException(Thread t, Throwable e) {Log.e("bug====",e.getMessage());}
}

最后把异常捕获和上下文传递在一起。最后不要忘了在清单文件中进行添加。
最后再写全局异常捕获是为了防止在编译的时候报错被捕获,导致我们不知道自己的代码出现了问题。所在建议全局异常捕获类在我们的所有功能都实现以后再进行添加。


public class App extends Application {private static Context context;@Overridepublic void onCreate() {super.onCreate();context = getApplicationContext();//        MyCrashHandler myCrashHandler = new MyCrashHandler();
//        Thread.setDefaultUncaughtExceptionHandler(myCrashHandler);}public static Context getContext() {return context;}
}

Android电商MVP框架相关推荐

  1. android电商平台,基于Android的电商平台通用客户端的设计与实现

    摘要: 随着移动互联网的高速发展,电子商务已经由过去的PC端逐渐进入了移动时代.一方面,对于处于创业阶段的电商平台运营者来说,由于资金不足,开发人员不足,很难在短时间内拥有电商平台的手机客户端.另一方 ...

  2. WordPress电商主题框架Storefront 2.5.0发布

    近期发现百度对二级域名文章的收录似乎不是很快,未来将把WP站长网的一些文章发布到一服客站点来,一服客站点的流量相对大点,这样可将更多的流量引入一服客站点.而WP站长网虽然建立了差不对有一年,但最近才开 ...

  3. python写电商网站框架_Python学员感言:电商项目要先把框架搭起来

    Python学员感言:电商项目要先把框架搭起来 来源:奇酷学院 发表于:2018-11-08 17:36:30 难的是开头,如果一开始,连项目的需求分析都分析不好,框架都搭建不起来,那么这个项目根本没 ...

  4. android电商闹钟,AndroidNativeEmu模拟执行计算出某电商App sign

    一.目标 这几天写代码写的很爽,因为经过几天没日没夜的调试,终于成功的把某电商App的sign用 AndroidNativeEmu 跑出来了,填了无数的坑,跑出正确结果的那一刻,内牛满面呀,心里充满了 ...

  5. 生鲜电商运营框架的对比和分析

    生鲜行业大赛道,高频刚需,低毛利高损耗,近年来科技进步(电子支付,物流体系,数据支持及手机的普及)四因子推动供给, 消费升级(多,快,好,省)带动需求.赛道的核心就是如何通过提升效率减少成本来精细地运 ...

  6. android电商练手项目,非常棒的练手开源电商项目

    大家好,我是小编南风吹,每天推荐一个小工具/源码,装满你的收藏夹,让你轻松节省开发效率,实现不加班不熬夜不掉头发! 今天小编推荐一套开源电商系统,包括前台商城系统及后台管理系统,基于SpringBoo ...

  7. android电商评论,三步教你获取电商评论数据

    现在的电商平台的商品琳琅满目,咱们足不出户就可以淘到性价比很好的尖货.但是东西多了大家不免要比较一番,这个时候看看商品粉丝的评论就尤其重要. 接下来一步步给大家介绍怎样获取评论的数据来供我们分析,以天 ...

  8. Android电商抢购倒计时,Android限时抢购倒计时实现代码

    限时抢购倒计时实现效果图 布局: android:id="@+id/ll_xsqg" android:layout_width="match_parent" a ...

  9. php电商开源框架,Sylius 开源PHP电商解决方案

    开源 PHP 电子商务网站框架 Sylius 发布了 1.0.8 版本,该版本最值得关注的更新就是正式支持 PHP 7.2,此外,还包含各种 bug 的修复,完整更新如下:#9101 修正汇率实体中错 ...

最新文章

  1. Docker 入门系列(5)- Docker 端口映射(映射所有IP地址、映射到指定地址和指定端口、映射指定地址任意端口、查看映射端口配置)
  2. MyEclipse 中配置struts2.2.1的方法
  3. 自学Zabbix3.0版本以上资产清单inventory
  4. MyBatis-16MyBatis动态SQL之【支持多种数据库】
  5. (数据结构与算法)稀疏数组案例
  6. Linux下判断cpu物理个数、几核
  7. ***error*** (zip#Browse) unzip not available on your system
  8. 下载腾讯视频里的视频_手机腾讯视频如何升级新版本
  9. mysql 正则替换 换行,MySQL中使用replace、regexp进行正则表达式替换的用法分析
  10. 信息系统项目管理05——项目范围管理
  11. BZOJ 1878: [SDOI2009]HH的项链【莫队】
  12. 经典算法-(六)老鼠走迷宫
  13. kaldi运行thchs30例子
  14. Android网络小说阅读器的实现
  15. JDownloader 突破百度网盘下载限速
  16. 【地震数据处理】GAN网络基础知识
  17. Tampermonkey油猴脚本 jquery 常用组件
  18. FOI 冬令营 Day6
  19. 思科模拟器Cisco Packet Tracer语言汉化包设置(附下载链接)
  20. 深度剖析mongos连接池

热门文章

  1. citrix(citrix是什么软件)
  2. 『树同构的判定(树Hash)』CF718D:Andrew and Chemistry
  3. Directory Opus12.6在win11下右键菜单显示问题
  4. python paramiko_python paramiko
  5. python中eps参数_Matplotlib简介和pyplot的简单使用——输出eps格式图像
  6. Python进阶(一)(MySQL,Navicat16免费安装)
  7. 《软件工程》第九章 面向对象方法学引论 作业
  8. 闪电博尔特100枪1000环夺冠,超人100枪10环垫底出局 程序中少了个大于号 覆盖结果
  9. 美国学生如何撰写“个人陈述”以争取被大学录取
  10. 100天精通Python丨办公效率篇 —— 11、Python自动化操作 Email(发送邮件、收邮件、邮箱客户端)